自制miniwol程序

背景

接上篇(记一次好玩的操作(内网远程控制主机启动))想一探究,自己做一个wol远程控制主机启动的小程序。

获取wol发出的数据包

wol 的全称是wake on lan,顾名思义,这肯定是一个网络程序,那么抓个包看看吧(由于不知道其用的什么协议,所以只能盲人摸象)。启动wireshark之后立刻执行 wol 00:24:1d:b2:3c:ca 命令,然后在wireshark界面中找到了如下标蓝的那条数据包(因为看他里边有个WOL字样=、=)

至此,得到的信息是:原来WOL不光是一个程序,它还是一个协议啊。

协议分析

获取到数据包之后开始分析其包结构,如下图:
这里写图片描述
找到我们需要的信息如下:

  • 运输层协议:UDP
  • IP :255.255.255.255
  • 端口:40000(实验证明,这里只要是合法端口都可以使用,维基百科上说一般使用端口7或者端口9)
  • 协议内容:前六个字节为 ffffffffffff,紧接着重复十六次对应机器的MAC地址即可。

代码实现

有了协议,代码实现起来就简单多了(代码比较简单,这里就不做解释了)

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

// 将字符串形式的MAC地址转化为16进制时使用的辅助函数
char switch_char(char ch)
{
    if ('0' <= ch && ch <= '9') {
        return ch - '0';
    } else if ('a' <= ch <= 'f') {
        return ch - 'a' + 10;
    } else if ('A' <= ch <= 'F') {
        return ch - 'A' + 10;
    } else {
        fprintf(stderr, "mac invalid!\n");
        exit(1);
    }
}

// 将以字符串表示的MAC地址转换为6个字节的16进制数
void convert(char *dst, char *src)
{
    char *tmp = strtok(src, ":");
    for (int i = 0; tmp != NULL; ++i) {
        dst[i] = (switch_char(*tmp) << 4) | (switch_char(*(tmp + 1)));
        tmp = strtok(NULL, ":");
    }
}

// 构造包
void make_pack(char *data, char *dst)
{
    char *tmp = data;
    // 组织数据包中的数据
    memset(data, 0xff, 6);
    tmp += 6;
    for (int i = 0; i < 16; ++i) {
        memcpy(tmp, dst, 6);
        tmp += 6;
    }
}

// 发送数据包
int send_wol(char *dst)
{
    char data[102];
    int on = 1;
    int fd;

    // 创建套接字
    if ((fd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket");
        return 0;
    }
    setsockopt(fd, SOL_SOCKET, SO_REUSEADDR | SO_BROADCAST, &on, sizeof(on));

    // 地址的相关初始化
    struct sockaddr_in addr;
    memset(&addr, 0, sizeof(addr));
    addr.sin_family = AF_INET;
    addr.sin_addr.s_addr = inet_addr("255.255.255.255");
    addr.sin_port = htons(9);

    make_pack(data, dst);

    int len = sizeof(struct sockaddr_in);
    if (sendto(fd, data, sizeof(data), 0, (struct sockaddr *)&addr, len) < 0) {
        perror("sendto");
        return 0;
    }

    return 1;
}


int main(int argc, char *argv[])
{
    int fd;
    char dst[6], tmp[18];

    if ((argc != 2) || (strlen(argv[1]) != 17)) {
        fprintf(stderr, "%s mac address\n", argv[0]);
        return EXIT_FAILURE;
    }

    strcpy(tmp, argv[1]);
    convert(dst, tmp);
    if (send_wol(dst)) {
        printf("miniwol : %s\n", argv[1]);
    } else {
        return EXIT_FAILURE;
    }

    return EXIT_SUCCESS;
}

扩展资料

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值