linux系统中使用socket直接发送ARP数据
这个重点是如这样创建socket: sock_send = socket ( PF_PACKET , SOCK_PACKET , htons ( ETH_P_ARP) ) ;
其后所有收发的数据都是原始的网络数据包。
代码如下:在X86和ARM平台上都测试通过。调用arp_scaner_init之后 ,调用send_arp来发送ARP数据包,thread_read_arp中就会收到对端的反馈,并将其保存。
在此非常感谢其他同仁的分享,使我学会了这个用法。
#include <unistd.h>
#include <errno.h>
#include <netdb.h>
#include <signal.h>
#include <sys/socket.h>
#include <sys/poll.h>
#include <sys/ioctl.h>
#include <netinet/if_ether.h>
#include <net/if_arp.h>
#include <netinet/udp.h>
#include <netinet/ip.h>
#include <stdio.h>
#include <stdarg.h>
#include <net/if.h>
#include <arpa/inet.h>
#include <string.h>
#include "adapter.h"
struct arp_msg {
/* Ethernet header */
uint8_t h_dest[ 6] ; /* 00 destination ether addr */
uint8_t h_source[ 6] ; /* 06 source ether addr */
uint16_t h_proto; /* 0c packet type ID field */
/* ARP packet */
uint16_t htype; /* 0e hardware type (must be ARPHRD_ETHER) */
uint16_t ptype; /* 10 protocol type (must be ETH_P_IP) */
uint8_t hwAddrLen; /* 12 hardware address length (must be 6) */
uint8_t protoAddrLen; /* 13 protocol address length (must be 4) */
uint16_t operation; /* 14 ARP opcode */
uint8_t srcMac[ 6] ; /* 16 sender's hardware address */
uint8_t srcIP[ 4] ; /* 1c sender's IP address */
uint8_t dstMac[ 6] ; /* 20 target's hardware address */
uint8_t dstIP[ 4] ; /* 26 target's IP address */
// uint8_t pad[ 18] ; /* 2a pad for min. ethernet payload (60 bytes) */
}PACKED;
bool g_system_running=true;
uint32_t local_ip;
uint8_t local_mac[ 6 ];
pthread_mutex_t g_cfglock;
volatile uint32_t g_mac_count=0;
char (*mac_list)[18];
int mac_list_max=0;
uint8_t raw_mac[64][6];
int setsockopt_broadcast( int fd)
{
const int const_int_1 =1;
return setsockopt ( fd,SOL_SOCKET,SO_BROADCAST, & const_int_1, sizeof ( const_int_1) ) ;
}
void prmac(u_char *ptr)
{
printf("MAC is: %02x:%02x:%02x:%02x:%02x:%02x\n",*ptr,*(ptr+1),*(ptr+2),*(ptr+3),*(ptr+4),*(ptr+5));
}
void printIP(const char *msg, uint8_t* ip)
{
printf("%s IP:%d.%d.%d.%d\n",msg,ip[0],ip[1],ip[2],ip[3]);
}
int send_arp( uint32_t test_ip, uint32_t from_ip, uint8_t * my_mac, const char * interface)
{
int sock_send;
int rv =1; /* "no reply received" yet */
struct sockaddr addr; /* for interface name */
struct arp_msg arp;
sock_send = socket ( PF_PACKET , SOCK_PACKET , htons ( ETH_P_ARP) ) ;
if ( sock_send == - 1) {
perror ( "raw_socket" ) ;
return - 1;
}
if ( setsockopt_broadcast( sock_send ) == - 1) {
perror ( "cannot enable bcast on raw socket" ) ;
goto ret;
}
/* send arp request */
memset ( & arp,0, sizeof ( arp) ) ;
memset ( arp. h_dest,0xff,6) ; /* MAC DA */
memcpy ( arp. h_source,my_mac,6) ; /* MAC SA */
arp. h_proto = htons ( ETH_P_ARP) ; /* protocol type (Ethernet) */
arp. htype = htons ( ARPHRD_ETHER) ; /* hardware type */
arp. ptype = htons ( ETH_P_IP) ; /* protocol type (ARP message) */
arp. hwAddrLen =6; /* hardware address length */
arp. protoAddrLen =4; /* protocol address length */
arp. operation = htons ( ARPOP_REQUEST) ; /* ARP op code */
memcpy ( arp. srcMac,my_mac,6) ; /* source hardware address */
memcpy ( arp. srcIP, & from_ip, sizeof ( from_ip) ) ; /* source IP address */
memcpy ( arp. dstIP, & test_ip, sizeof ( test_ip) ) ; /* target IP address */
memset ( & addr,0, sizeof ( addr) ) ;
strncpy( addr. sa_data,interface, sizeof ( addr. sa_data) ) ;
if ( sendto ( sock_send, & arp, sizeof ( arp) ,0, & addr, sizeof ( addr) ) <0) {
perror("sendto error");
}
ret:
close (sock_send) ;
return rv;
}
int read_interface( const char * interface, int * i, uint32_t * addr, uint8_t * arp)
{
int fd;
struct ifreq ifr;
struct sockaddr_in * our_ip;
memset ( & ifr,0, sizeof ( ifr) ) ;
fd = socket(AF_INET, SOCK_DGRAM, 0);
ifr. ifr_addr. sa_family = AF_INET ;
strncpy( ifr.ifr_name,interface, sizeof(ifr.ifr_name) );
if ( arp) {
if ( ioctl( fd, SIOCGIFHWADDR , & ifr) !=0) {
printf("SIOCGIFHWADDR error\n");
close ( fd) ;
return - 1;
}
memcpy ( arp,ifr. ifr_hwaddr. sa_data,6) ;
prmac ( arp ) ;
}
if ( addr) {
if ( ioctl( fd, SIOCGIFADDR , & ifr) !=0) {
perror ( "ioctl" ) ;
close ( fd) ;
return - 1;
}
our_ip = ( struct sockaddr_in * ) & ifr. ifr_addr;
* addr =our_ip-> sin_addr. s_addr;
}
close ( fd);
return 0;
}
void* thread_read_arp(void* param)
{
struct sockaddr repsa;
socklen_t salen, recved_len ;
struct arp_msg response;
int recvfd = socket(PF_PACKET, SOCK_PACKET, htons(ETH_P_ARP));
bzero(&response, sizeof(response));
bzero(&repsa, sizeof(repsa));
salen = sizeof(repsa);
while(g_system_running){
if((recved_len = recvfrom(recvfd, &response, sizeof(response), 0, (struct sockaddr *)&repsa, &salen)) <= 0) {
perror("Recvfrom error");
break;
}
printf("Response %d bytes received\n",recved_len);
printIP("response", response.srcIP);
if( memcmp(response.dstIP ,&local_ip, 4 ) ==0
&& memcmp(response.dstMac,local_mac, 6 ) ==0){
printf("get a data\n");
}
}
close(recvfd);
}
int arp_scaner_init( const char * interface )
{
pthread_t tid;
read_interface( interface, NULL,&local_ip,local_mac);
printf("local IP");
printIP("", (uint8_t*)&local_ip );
pthread_mutex_init(&g_cfglock, NULL);
ipc_createthread(&tid, thread_read_arp, NULL, "arp-reading");
send_arp( find_ip,local_ip,local_mac,interface);
}
linux系统中使用socket直接发送ARP数据的更多相关文章
- 如何从Linux系统中获取带宽、流量网络数据
引入 国外的云主机厂商,例如AWS提供的网络数据是以流量为单位的,例如下面的图片: 从上图来看,其取值方式为 每隔5分钟取值1次,(每次)每个点显示为1分钟内的流量字节数(Bytes) 带宽与流量 我 ...
- linux系统中的进程状态分析
转载地址:https://blog.csdn.net/shenwansangz/article/details/51981459 linux是一个多用户,多任务的系统,可以同时运行多个用户的多个程序, ...
- Linux 系统中僵尸进程
Linux 系统中僵尸进程和现实中僵尸(虽然我也没见过)类似,虽然已经死了,但是由于没人给它们收尸,还能四处走动.僵尸进程指的是那些虽然已经终止的进程,但仍然保留一些信息,等待其父进程为其收尸.配图源 ...
- 获得Unix/Linux系统中的IP、MAC地址等信息
获得Unix/Linux系统中的IP.MAC地址等信息 中高级 | 2010-07-13 16:03 | 分类:①C语言. Unix/Linux. 网络编程 ②手册 | 4,471 次阅读 ...
- 为什么在 Linux 系统中,不建议超频
CPU 是一部计算机内的心脏啦!因为不论你做什么事情都需要 CPU 来加以运作的!(虽然有时工作量大.有时工作量小!),在 586 以前的计算机( 包含 386, 486, 与 586 ) ,CPU ...
- (转)浅谈 Linux 系统中的 SNMP Trap
原文:https://www.ibm.com/developerworks/cn/linux/l-cn-snmp/index.html 简介 本文讲解 SNMP Trap,在介绍 Trap 概念之前, ...
- 详解Linux系统中的文件名和文件种类以及文件权限
Linux文件种类与副文件名 一直强调一个概念,那就是:任何装置在Linux底下都是文件, 不仅如此,连资料沟通的介面也有专属的文件在负责-所以,你会瞭解到,Linux的文件种类真的很多- 除了前面提 ...
- linux系统中文件的几种类型
Linux系统是以文件的形式来进行管理的.Linux文件类型常见的有:普通文件.目录.字符设备文件.块设备文件.符号链接文件等,如果想了解这方面知识的弟兄,就进来了解了解. Linux系统不同于win ...
- Linux系统中的Device Mapper学习
在linux系统中你使用一些命令时(例如nmon.iostat 如下截图所示),有可能会看到一些名字为dm-xx的设备,那么这些设备到底是什么设备呢,跟磁盘有什么关系呢?以前不了解的时候,我也很纳闷. ...
随机推荐
- JS创建一个数组1.求和 2.求平均值 3.最大值 4.最小值 5.数组逆序 6.数组去重 0.退出
rs = require("readline-sync"); let arr = []; console.log("请输入数组的长度:"); let arr_l ...
- vue provide和inject 父组件和子孙通信
父组件中通过provider来提供变量,然后在子组件中通过inject来注入变量.不论子组件有多深,只要调用了inject那么就可以注入provider中的数据.而不是局限于只能从当前父组件的prop ...
- angular1时间控件之时间比较大小,比如入住日期和离店日期,入住不能晚于离店时间
功能可能是大家有时遇到的,不过angluar1现在用的人很少了,希望帮上和我一样掉坑里的朋友吧,之前自己也在网上找过几乎没找到angluar1和我代码需求一样的控件,没办法自己造 了,看代码图 就差不 ...
- Redis和Memchaed缓存数据查询
使用TreeNMS数据库管理工具 支持Redis和Memchaed并存同时管理维护,支持集群管理,减少运维成本,提高工作效率 自带环境,免安装.免布署,下载解压即可使用.适用于windows,Linu ...
- Django form表单功能的引用(注册,复写form.clean方法 增加 验证密码功能)
1. 在app下 新建 forms.py 定义表单内容,类型models from django import forms class RegisterForm(forms.Form): userna ...
- centos7更改时区
1,linux系统更换时区 2,时间同步 1,linux系统更换时区 注:# timedatectl # 查看系统时间方面的各种状态 # timedatectl list-timezones # 列出 ...
- 2018-2019-2 20175126谢文航 实验一《Java开发环境的熟悉》实验报告
一.实验报告封面 课程:Java程序设计 班级:1751班 姓名:谢文航 学号:20175126 指导教师:娄嘉鹏 实验日期:2019年3月28日 实验时间:--- 实验序号:实验一 实验名称:Jav ...
- Integer 函数传参实现值交换
import java.lang.reflect.Field; public class MainClass { public static void main(String[] args) { In ...
- tiny4412 --Uboot移植(6) SD卡驱动,启动内核
开发环境:win10 64位 + VMware12 + Ubuntu14.04 32位 工具链:linaro提供的gcc-linaro-6.1.1-2016.08-x86_64_arm-linux-g ...
- java、php、.net关于web开发的区别
一提到web开发,目前在世界上流行性的三个帮派就是php,java和asp.net,这个世界上的百分之99的网站或者类似的应用都是由这三种语言的开发,这里请原谅我忽视某些小众语言如python之类.三 ...