diff --git a/day6/homework/fake_feiqiu.c b/day6/homework/fake_feiqiu.c index 4e7e370..1de635f 100644 --- a/day6/homework/fake_feiqiu.c +++ b/day6/homework/fake_feiqiu.c @@ -19,6 +19,7 @@ #include #include #include +#include // system ssize_t send_datapacket(int fd, unsigned char *buf, ssize_t buf_size, const char *ether_name); unsigned short checksum(unsigned short *buf, int len); @@ -58,13 +59,35 @@ int main(int argc, char const *argv[]) bzero(&sll, sizeof(sll)); sll.sll_ifindex = ether_req.ifr_ifindex; // 将网卡的接口类型赋值给发送接口 */ + char sender_ip[128] = ""; + printf("请输入要伪装的发送者IP: "); + scanf("%s", sender_ip); + + char sender_mac[128] = ""; + printf("请输入要伪装的发送者MAC: "); + scanf("%s", sender_mac); + + char receiver_ip[128] = ""; + printf("请输入要伪装的接收者IP: "); + scanf("%s", receiver_ip); + + char recv_mac[128] = ""; + printf("请输入要伪装的接收者MAC: "); + scanf("%s", recv_mac); + char sender_name[128] = ""; - printf("请输入要伪装的发送者名字:"); + printf("请输入要伪装的发送者名字: "); scanf("%s", sender_name); char sender_pc_name[128] = ""; - printf("请输入要伪装的发送者电脑名字:"); + printf("请输入要伪装的发送者电脑名字: "); scanf("%s", sender_pc_name); + // mac 地址格式化 + unsigned char dst_mac[8] = {0}; // 目的 MAC + unsigned char src_mac[8] = {0}; // 发送者 MAC + sscanf(sender_mac, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx", &src_mac[0], &src_mac[1], &src_mac[2], &src_mac[3], &src_mac[4], &src_mac[5]); + sscanf(recv_mac, "%hhx:%hhx:%hhx:%hhx:%hhx:%hhx", &dst_mac[0], &dst_mac[1], &dst_mac[2], &dst_mac[3], &dst_mac[4], &dst_mac[5]); + while (1) { @@ -82,11 +105,35 @@ int main(int argc, char const *argv[]) char msg_data[1024] = ""; char say_what[1024] = ""; printf("请输入要发送的内容:"); - scanf("%s", say_what); + // scanf("%s", say_what); + fgets(say_what, sizeof(say_what), stdin); // 比 scanf 的优点是可以输入空格 + say_what[strlen(say_what) - 1] = '\0'; // 去掉最后的换行符 if (strcmp(say_what, "exit") == 0) { break; } + if (strcmp(say_what, "clear") == 0) + { + system("clear"); + continue; + } +/* if (strncpy(say_what, "change", 6) == 0) + { + printf("请输入要伪装的发送者名字:"); + scanf("%s", sender_name); + printf("请输入要伪装的发送者电脑名字:"); + scanf("%s", sender_pc_name); + continue; + } + if (strncmp(say_what, "help", 4) == 0) + { + printf("请输入要发送的内容:\n"); + printf("1. 输入exit退出\n"); + printf("2. 输入clear清屏\n"); + printf("3. 输入change更改伪装信息\n"); + continue; + } */ + sprintf(msg_data, "1:%d:%s:%s:%d:%s", 123, sender_name, sender_pc_name, 32, say_what); int msg_data_len = strlen(msg_data) + strlen(msg_data) % 2; // 整数补齐偶数位,strlen(msg_data)%2 : 偶数+0,奇数+1 printf("msg_data_len = %d ---> %s\n", msg_data_len, msg_data); @@ -104,13 +151,16 @@ int main(int argc, char const *argv[]) UDP 校验中的伪头部(pseudo header)是在计算 UDP 校验和时使用的辅助数据。伪头部包含了源 IP 地址、目的 IP 地址、协议类型(通常是 UDP)和 UDP 报文长度等信息。 在计算 UDP 校验和时,将伪头部和 UDP 报文的内容拼接在一起,然后计算校验和。这样做的目的是增加校验和的安全性,使其更具可靠性。 */ + // 196: 4c:e1:73:47:16:3a + // 158: e8:d8:d1:48:46:ae + unsigned char pseudo_udp_buf[1056] = ""; // 伪头部 PSEUDO_UDP_HEAD *pseudo_udp_head = (PSEUDO_UDP_HEAD *)pseudo_udp_buf; - pseudo_udp_head->saddr = inet_addr("192.168.31.155"); // 伪装源IP - pseudo_udp_head->daddr = inet_addr("192.168.31.147"); // 目的IP - pseudo_udp_head->flag = 0; // 0 - pseudo_udp_head->protocol = 17; // 协议类型 UDP - pseudo_udp_head->len = htons(8 + msg_data_len); // UDP 首部长度 + 数据部分长度 + pseudo_udp_head->saddr = inet_addr(sender_ip); // 伪装源IP + pseudo_udp_head->daddr = inet_addr(receiver_ip); // 目的IP + pseudo_udp_head->flag = 0; // 0 + pseudo_udp_head->protocol = 17; // 协议类型 UDP + pseudo_udp_head->len = htons(8 + msg_data_len); // UDP 首部长度 + 数据部分长度 memcpy(pseudo_udp_buf + 12, udp_head, 8); // 拷贝 UDP 首部到伪头部 memcpy(pseudo_udp_buf + 12 + 8, msg_data, msg_data_len); // 拷贝数据部分到伪头部 @@ -130,14 +180,14 @@ int main(int argc, char const *argv[]) ip_head->protocol = 17; // 协议类型 UDP // ip 校验时不需要伪头部 ip_head->check = htons(0); // 原始校验和(暂时未知,赋值0) - ip_head->saddr = inet_addr("192.168.31.155"); // 伪装源IP - ip_head->daddr = inet_addr("192.168.31.147"); // 目的IP + ip_head->saddr = inet_addr(sender_ip); // 伪装源IP + ip_head->daddr = inet_addr(receiver_ip); // 目的IP ip_head->check = checksum((unsigned short *)(udp_buf + 14), 20); // IP 首部校验和(计算 IP 首部的校验和) /* ---------------------组装 UDP 数据报的 MAC 首部--------------------- */ - struct ether_header *mac_head = (struct ether_header *)udp_buf; // MAC 首部开始位置 - unsigned char dst_mac[8] = {0x00, 0xa5, 0x54, 0x89, 0xb5, 0xc5}; // 主机 MAC (目的 MAC) - unsigned char src_mac[8] = {0xb8, 0x8a, 0x60, 0xa0, 0x49, 0x2f}; // 虚拟机 MAC (源 MAC) + struct ether_header *mac_head = (struct ether_header *)udp_buf; // MAC 首部开始位置 + // unsigned char dst_mac[8] = {0xe8, 0x6a, 0x64, 0x6e, 0x93, 0x28}; // 主机 MAC (目的 MAC) + // unsigned char src_mac[8] = {0x4c, 0xe1, 0x73, 0x47, 0x16, 0x3a}; // 虚拟机 MAC (源 MAC) // 使用 memcpy 函数将 dst_mac 的内容拷贝到 mac_head.ether_dhost 中 memcpy(mac_head->ether_dhost, dst_mac, 6); // 目的 MAC memcpy(mac_head->ether_shost, src_mac, 6); // 源 MAC @@ -145,7 +195,7 @@ int main(int argc, char const *argv[]) /* ---------------------发送数据--------------------- */ // int send_len = sendto(sock_fd, udp_buf, 14 + 20 + 8 + msg_data_len, 0, (struct sockaddr *)&sll, sizeof(sll)); - int send_len = send_datapacket(sock_fd, udp_buf, 14 + 20 + 8 + msg_data_len, "ens34"); + int send_len = send_datapacket(sock_fd, udp_buf, 14 + 20 + 8 + msg_data_len, "ens38"); printf("send_len = %d\n", send_len); }