linux串口编程实例_Linux 网络编程——原始套接字实例:发送 UDP 数据包
生活随笔
收集整理的這篇文章主要介紹了
linux串口编程实例_Linux 网络编程——原始套接字实例:发送 UDP 数据包
小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
以太網報文格式:
IP 報文格式:
UDP 報文格式:
校驗和函數:
/*******************************************************功能:校驗和函數參數:buf: 需要校驗數據的首地址nword: 需要校驗數據長度的一半返回值:校驗和*******************************************************/unsigned short checksum(unsigned short *buf, int nword){unsigned long sum;for(sum = 0; nword > 0; nword--){sum += htons(*buf);buf++;}sum = (sum>>16) + (sum&0xffff);sum += (sum>>16);return ~sum;}需要C/C++ Linux服務器架構師學習資料私信“資料”(資料包括C/C++,Linux,golang技術,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒體,CDN,P2P,K8S,Docker,TCP/IP,協程,DPDK,ffmpeg等),免費分享
這里是在linux下通過原始套接字組一個 udp 數據包,給 PC 機的網絡調試助手發送信息:
#include #include #include #include //struct ifreq#include //ioctl、SIOCGIFADDR#include #include //ETH_P_ALL#include //struct sockaddr_ll unsigned short checksum(unsigned short *buf, int nword);//校驗和函數int main(int argc, char *argv[]){//1.創建通信用的原始套接字int sock_raw_fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));//2.根據各種協議首部格式構建發送數據報unsigned char send_msg[1024] = {//--------------組MAC--------14------0x74, 0x27, 0xea, 0xb5, 0xef, 0xd8, //dst_mac: 74-27-EA-B5-FF-D80xc8, 0x9c, 0xdc, 0xb7, 0x0f, 0x19, //src_mac: c8:9c:dc:b7:0f:190x08, 0x00, //類型:0x0800 IP協議//--------------組IP---------20------0x45, 0x00, 0x00, 0x00, //版本號:4, 首部長度:20字節, TOS:0, --總長度--:0x00, 0x00, 0x00, 0x00,//16位標識、3位標志、13位片偏移都設置00x80, 17, 0x00, 0x00,//TTL:128、協議:UDP(17)、16位首部校驗和10, 221, 20, 11,//src_ip: 10.221.20.1110, 221, 20, 10,//dst_ip: 10.221.20.10//--------------組UDP--------8+78=86------0x1f, 0x90, 0x1f, 0x90, //src_port:0x1f90(8080), dst_port:0x1f90(8080)0x00, 0x00, 0x00, 0x00, //#--16位UDP長度--30個字節、#16位校驗和};int len = sprintf(send_msg+42, "%s", "this is for the udp test");if(len % 2 == 1)//判斷len是否為奇數{len++;//如果是奇數,len就應該加1(因為UDP的數據部分如果不為偶數需要用0填補)}*((unsigned short *)&send_msg[16]) = htons(20+8+len);//IP總長度 = 20 + 8 + len*((unsigned short *)&send_msg[14+20+4]) = htons(8+len);//udp總長度 = 8 + len//3.UDP偽頭部unsigned char pseudo_head[1024] = {//------------UDP偽頭部--------12--10, 221, 20, 11,//src_ip: 10.221.20.1110, 221, 20, 10,//dst_ip: 10.221.20.100x00, 17, 0x00, 0x00, //0,17,#--16位UDP長度--20個字節};*((unsigned short *)&pseudo_head[10]) = htons(8 + len);//為頭部中的udp長度(和真實udp長度是同一個值)//4.構建udp校驗和需要的數據報 = udp偽頭部 + udp數據報memcpy(pseudo_head+12, send_msg+34, 8+len);//--計算udp校驗和時需要加上偽頭部--//5.對IP首部進行校驗*((unsigned short *)&send_msg[24]) = htons(checksum((unsigned short *)(send_msg+14),20/2));//6.--對UDP數據進行校驗--*((unsigned short *)&send_msg[40]) = htons(checksum((unsigned short *)pseudo_head,(12+8+len)/2));//6.發送數據struct sockaddr_ll sll;//原始套接字地址結構struct ifreq req;//網絡接口地址strncpy(req.ifr_name, "eth0", IFNAMSIZ);//指定網卡名稱if(-1 == ioctl(sock_raw_fd, SIOCGIFINDEX, &req))//獲取網絡接口{perror("ioctl");close(sock_raw_fd);exit(-1);}/*將網絡接口賦值給原始套接字地址結構*/bzero(&sll, sizeof(sll));sll.sll_ifindex = req.ifr_ifindex;len = sendto(sock_raw_fd, send_msg, 14+20+8+len, 0 , (struct sockaddr *)&sll, sizeof(sll));if(len == -1){perror("sendto");}return 0;} unsigned short checksum(unsigned short *buf, int nword){unsigned long sum;for(sum = 0; nword > 0; nword--){sum += htons(*buf);buf++;}sum = (sum>>16) + (sum&0xffff);sum += (sum>>16);return ~sum;}運行結果如下:
總結
以上是生活随笔為你收集整理的linux串口编程实例_Linux 网络编程——原始套接字实例:发送 UDP 数据包的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 美国5月通胀再创新高 “背锅侠”石油公司
- 下一篇: 【转】matlab与C/C++混合编程—