linux下c语言使用mac地址进行socket通信

server端:

#include <unistd.h>
#include <stdio.h>
#include <sys/socket.h>
#include <netinet/ip.h>
#include <netinet/udp.h>
#include<memory.h>
#include <sys/ioctl.h>
#include<stdlib.h>
#include <linux/if_ether.h>
#include <linux/if_packet.h>
#include<arpa/inet.h>
#include<netinet/if_ether.h>
#include<errno.h>
#include<netinet/ether.h>
#include<net/if.h>
#include<string.h>
int main(int argc, char **argv) {int sock, n;char buffer[1024];unsigned char send_msg[1024] = {//--------------组MAC--------14------0x00, 0x0c, 0x29, 0x0d, 0xe8, 0x83, 0x00, 0x0c, 0x29, 0x56, 0xd5, 0xc7, //dst_mac: E8-6A-64-17-CB-5C目的地址//src_mac: 8c:ec:4b:73:79:b5源mac地址0x08, 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位校验和*/};if ((sock = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL))) < 0){perror("socket");exit(1);}struct sockaddr_ll client;socklen_t addr_length = sizeof(struct sockaddr_ll);uint8_t sendbuffer[1024]; while (1) {n = recvfrom(sock, buffer,1024,0, (struct sockaddr *)&client, &addr_length);if (n < 14) {continue;}int num = n-14;memcpy(sendbuffer, buffer, n);char data[1024];data[24]='\0';memcpy(data,sendbuffer+14,num);if(strcmp(data,"test")==0){int len = sprintf(send_msg+14, "%s", "shou");struct sockaddr_ll sll;                 //原始套接字地址结构struct ifreq req;                    //网络接口地址    strncpy(req.ifr_name, "ens33", IFNAMSIZ);         //指定网卡名称if(-1 == ioctl(sock, SIOCGIFINDEX, &req)) //获取网络接口{perror("ioctl");close(sock);exit(-1);}   /*将网络接口赋值给原始套接字地址结构*/bzero(&sll, sizeof(sll));sll.sll_ifindex = req.ifr_ifindex;len = sendto(sock, send_msg, 14+len, 0 , (struct sockaddr *)&sll, sizeof(sll));if(len == -1){perror("sendto");}  }printf("%s\n",data);}
}

socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL))
链路层原始套接字调用socket()函数创建。第一个参数指定协议族类型为PF_PACKET,第二个参数type可以设置为SOCK_RAW或SOCK_DGRAM,第三个参数是协议类型(该参数只对报文接收有意义)。参数type设置为SOCK_RAW时,套接字接收和发送的数据都是从MAC首部开始的。ETH_P_ALL 报收本机收到的所有二层报文。
数据链路层的头信息通常定义在 sockaddr_ll 的结构体中,protocol是按照网络字节顺序(network byte order),大部分定义在头文件中,设置协议时,例如 htons(ETH_P_ALL)来接收所有的数据包;
如果要获取从指定以太网接口卡上的数据包时,在 struct sockaddr_ll中指定网络接口卡,绑定(bind)数据包到该interface上。只有sll_protocol和 sll_ifindex这两个地址字段是用来bind的。

struct sockaddr_ll {
unsigned short sll_family; /* Always AF_PACKET /
unsigned short sll_protocol; /
Physical-layer protocol /
int sll_ifindex; /
Interface number /
unsigned short sll_hatype; /
ARP hardware type /
unsigned char sll_pkttype; /
Packet type /
unsigned char sll_halen; /
Length of address /
unsigned char sll_addr[8]; /
Physical-layer address */
};
sll_protocol : 标准以太网协议类型,按网络字节顺序。定义在中。
sll_ifindex: interface索引,0 匹配所有的网络接口卡;
sll_hatype: ARP 硬件地址类型(hardware address type) 定义在中,常用 ARPHRD_ETHER
sll_pkttype: 包含了packet类型。
PACK_HOST 包地址为本地主机地址。
PACK_BROADCAST 物理层广播包。
PACK_MULTICAST 发送到物理层多播地址的包。
PACK_OTHERHOST 发往其它在混杂模式下被设备捕获的主机的包。
PACK_OUTGOING 本地回环包;
sll_addr 和 ssl_halen 包含了物理层地址和其长度;

    当发送数据包时,指定 sll_family, sll_addr, sll_halen, sll_ifindex, sll_protocol 就足够了。其它字段设置为0; sll_hatype和 sll_pkttype是在接收数据包时使用的; 如果要bind, 只需要使用 sll_protocol和 sll_ifindex;

client端

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <netinet/ether.h>
#include <netpacket/packet.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include<unistd.h>    //close
int main(int argc, char *argv[])
{//1.创建通信用的原始套接字int n;char buffer[1024];int sock_raw_fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));uint8_t sendbuffer[1024];  //2.根据各种协议首部格式构建发送数据报unsigned char send_msg[1024] = {//--------------组MAC--------14------0x00, 0x0c, 0x29, 0x56, 0xd5, 0xc7, //dst_mac: E8-6A-64-17-CB-5C目的地址0x00, 0x0c, 0x29, 0x0d, 0xe8, 0x83, //src_mac: 8c:ec:4b:73:79:b5源mac地址0x08, 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位校验和*/};   while(1){int len = sprintf(send_msg+14, "%s", "test");struct sockaddr_ll sll;                 //原始套接字地址结构struct ifreq req;                    //网络接口地址    strncpy(req.ifr_name, "ens33", 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+len, 0 , (struct sockaddr *)&sll, sizeof(sll));if(len == -1){perror("sendto");}   /socklen_t addr_length = sizeof(struct sockaddr_ll);n = recvfrom(sock_raw_fd, buffer,1024,0, (struct sockaddr *)&sll, &addr_length);if (n < 14) {continue;}int num = n-14;memcpy(sendbuffer, buffer, n);char data[1024];data[24]='\0';memcpy(data,sendbuffer+14,num);printf("%s\n",data);}return 0;
}

使用时注意修改网卡名称,目的地址和源地址。

linux下c语言使用mac地址进行socket通信相关推荐

  1. linux内核设置mac地址,Linux下如何修改网卡MAC地址

    Red Hat Linux下如何修改网卡MAC地址 OS版本:Red Hat Enterprise Linux AS4 核心:Kernel 2.6.9-42 网上有很多关于linux下修改MAC地址的 ...

  2. linux改mac地址目录,Linux下如何修改网卡MAC地址

    网上有很多关于linux下修改MAC地址的方法,大多依葫芦画瓢,似乎都没验证过,达不到修改的目的. 经过我的详细测试,最终成功解决了这个问题. 误区一: #ifconfig eth0 down #if ...

  3. linux获取网卡协议地址,读取linux下的网络设备的mac地址与发送原始数据包 (2011-11-23 20:11)...

    一:linux下的网络设备 linux的网络设备信息都在/proc/net/dev,从这里我们可以得到所有网卡的名字,如eth0, eth1等等 root@dlrc-desktop:/home/dlr ...

  4. 在Windows下和Linux下如何查看网卡MAC地址、以及修改MAC地址

    在Windows下 利用DOS命令打开,弹出命令窗口. 输入命令ipconfig/all,回车. 其中,物理地址为本机的MAC地址. 在Linux下 查看MAC地址的方法有好多种,下面给出四种常见的方 ...

  5. linux c 获取mac地址吗,Linux系统下用C语言获取MAC地址

    最近在做一个小程序,需要用到在linux系统里编写C程序从而获取MAC地址,从网上搜了一遍,想总结一下.如果你就只需要单个功能的程序,可以采用方法一,见代码1,一般最好能够封装起来,写成获取MAC地址 ...

  6. linux临时配置mac地址,Linux获取网卡型号、mac地址、修改IP地址的几种方法

    1.获取所有有(无)网卡型号 方法一.ifconfig 用法:ifconfig | grep | cut -d ':' -f 1 ubuntu root@ubuntu:~# ifconfig | gr ...

  7. Linux 应用程序 嵌汇编,Linux下C语言嵌汇编

    Using Assembly Language in Linux. Intel和AT&T汇编语法差异: 1.前缀: Intel汇编寄存器和立即数无需前缀.后者寄存器前缀为%,立即数前缀为$. ...

  8. Linux下C语言编程-进程的创建

    Linux下C语言编程-进程的创建 作者:hoyt 1.进程的概念 Linux操作系统是面向多用户的.在同一时间可以有许多用户向操作系统发出各种命令.那么操作系统是怎么实现多用户的环境呢?在现代的操作 ...

  9. linux+下c语言编程项目,精通UNIX下C语言编程与项目实践

    cc -I  //include 目录 -L //静态库目录?动态也可以 -l //小写L,接静态库名称?动态也可以 -DXXX='"XXFF"' //-D直接定义宏 -c 只编译 ...

最新文章

  1. linux安装sphinx
  2. 每日一皮:从项目的开始到结束,开发人员的变化...
  3. python360百科_python抓取360百科踩过的坑!
  4. flink的Table类型的变量两种输出的形式
  5. 洛谷P2587 [ZJOI2008] 泡泡堂
  6. HttpReqeust对象
  7. 测试场景组件化轮子——用例元
  8. Python 基础---列表
  9. Git 和 Github的关系
  10. Linux7安装硬盘显示错误,【原创文章】centos7 badblocks检测硬盘出现Value too large for defined data type错误的原因和解决办法...
  11. BAT判断一个目录是否存在
  12. ArcGIS(ArcMap)进行缓存切片
  13. 利用FinalData恢复shift+delete误删的文件
  14. js使用正则表达式验证身份证格式
  15. Nature communications
  16. snownlp抛出错误_网易云评论爬虫及情感分析
  17. Evolutionary algorithm (遗传算法)介绍
  18. 美式英语口语中连读、略读,音变的技巧
  19. 《OKR源于英特尔和谷歌的管理利器》阅读总结
  20. 自制随机小姐姐摄影api

热门文章

  1. 将多张图片无缝拼接方法
  2. 3dsMax里GLTF插件的安装和使用
  3. 教学|ZBrush纹理绘制之,如何渲染逼真的皮肤纹理
  4. 如何使用Tika提取文件内容
  5. Qt开发 之 快速查找 应用程序输出(适合很多IDE,解决查找眼瞎的问题)
  6. 手机邮箱格式怎么写你知道吗?带你注册个人邮箱账号
  7. PHP 给图片加边框
  8. 达梦逻辑备份还原:dimp和dexp
  9. 使用Word对图片进行圆形剪裁
  10. 内核中的死锁问题--当UHCI遇上OHCI .