关闭

【Linux网络编程】原始套接字实例:发送 UDP 数据包

标签: Linux网络编程UDP包的封装UDP协议实现UDP
704人阅读 评论(0) 收藏 举报
分类:

以太网报文格式:


详细的说明,请看《MAC 头部报文分析》


IP 报文格式:



详细的说明,请看《IP 数据报格式详解》


UDP 报文格式:



详细的说明,请看《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;
}


这里是在 ubuntu 下通过原始套接字组一个 udp 数据包,给 PC 机的网络调试助手发送信息:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <net/if.h>				//struct ifreq
#include <sys/ioctl.h>			//ioctl、SIOCGIFADDR
#include <sys/socket.h>
#include <netinet/ether.h>		//ETH_P_ALL
#include <netpacket/packet.h>	//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-D8
		0xc8, 0x9c, 0xdc, 0xb7, 0x0f, 0x19, //src_mac: c8:9c:dc:b7:0f:19
		0x08, 0x00,                         //类型:0x0800 IP协议
		//--------------组IP---------20------
		0x45, 0x00, 0x00, 0x00,             //版本号:4, 首部长度:20字节, TOS:0, --总长度--:
		0x00, 0x00, 0x00, 0x00,				//16位标识、3位标志、13位片偏移都设置0
		0x80, 17,   0x00, 0x00,				//TTL:128、协议:UDP(17)、16位首部校验和
		10,  221,   20,  11,				//src_ip: 10.221.20.11
		10,  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.11
		10,  221,   20,  10,				//dst_ip: 10.221.20.10
		0x00, 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;
}


运行结果如下:



0
0
查看评论

Linux 网络编程——原始套接字实例:发送 UDP 数据包

以太网报文格式:详细的说明,请看《MAC 头部报文分析》。IP 报文格式:详细的说明,请看《IP 数据报格式详解》。UDP 报文格式:详细的说明,请看《UDP 数据报格式详解》。校验和函数:/******************************************************...
  • tennysonsky
  • tennysonsky
  • 2015-04-08 16:57
  • 70408

Linux网络编程:原始套接字编程及实例分析

Linux网络编程:原始套接字编程及实例分析 一、原始套接字能干什么? 二、原始套接字编程 三、原始套接字实例:MAC头部报文分析
  • q623702748
  • q623702748
  • 2016-07-29 11:25
  • 1039

Linux下使用RAW SOCKET原始套接字构造UDP原始数据帧广播到局域网,在局域网的另一台计算机上显示UDP发送的信息

因为使用IEC61850需要直接访问以太网数据链路层
  • zgrjkflmkyc
  • zgrjkflmkyc
  • 2014-09-12 15:48
  • 4579

Linux C raw_socket (原始套接字)

我们平常所用到的网络编程都是在应用层收发数据,每个程序只能收到发给自己的数据,即每个程序只能收到来自该程序绑定的端口的数据。收到的数据往往只包括应用层数据。某些情况下我们需要执行更底层的操作,比如监听所有本机收发的数据、修改报头等。 通过原始套接字,我们可以抓取所有发送到本机的IP包(包括IP头和...
  • zgrjkflmkyc
  • zgrjkflmkyc
  • 2014-09-11 15:46
  • 8740

黑马程序员--Java基础学习之网络编程(TCP、UDP、Socket、模拟发送和接收数据)

网络传输过程: 1,找到对方ip 2,数据发送到指定端口(每个应用程序对应一个端口) 3,定义通信规则,即协议   TCP/IP UDP 1,面向无连接,将数据、源、目的封装(像邮递东西) 2,数据报大小限制在64k内 3,不可靠协议 4,速度快 TCP 1,面向连接,形成传输数据的通道...
  • u011746071
  • u011746071
  • 2013-09-28 10:42
  • 2135

linux网络编程(如何编写一个UDP通信程序)

UDP数据通讯原理      UDP数据通讯分服务端(软件)和客户端端:      服务端(软件)(服务器)先运行,服务端,不需要事先知道客户端IP和port      客户端(软件)(客户端机器)后运行...
  • Linux_dream_2015
  • Linux_dream_2015
  • 2015-06-14 11:07
  • 1415

原始套接字,接收所有数据的设置方法,及代码。

在看一本安全编程书的时候读到原始套接字地方,自己也尝试写了下来,将遇到的一个问题发出来。 // 原始套接字数据的接收.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include "stdio.h" #incl...
  • Stay_Deep
  • Stay_Deep
  • 2015-03-20 09:13
  • 490

《网络编程》原始套接字 ---ping程序实现

基于字节流套接字(SOCK_STREAM)和数据报套接字(SOCK_DGRAM)不可以访问传输层协议,只是对应用层的报文进行操作,传输层的数据报格式都是由系统提供的协议栈实现,用户只需要填充相应的应用层报文,由系统完成底层报文首部的填充并发送。原始套接字(SOCK_RAW)可以访问位于基层的传输层协...
  • chenhanzhun
  • chenhanzhun
  • 2014-12-23 18:08
  • 2860

学习--原始套接字

  • eroswang
  • eroswang
  • 2008-03-05 14:02
  • 3341

linux原始套接字-发送ARP报文

linux原始套接字,可以直接发送和接收链路层和网络层的报文,对我们理解TCP/IP协议栈有很多帮助。 也可写出很多有趣的程序。 下面的例子是向192.168.1.60的电脑,发送伪造的ARP报文,使其更新ARP表,导致无法PING通192.168.1.201。 使用命令arp -d 删除ar...
  • maxzero
  • maxzero
  • 2016-09-27 11:58
  • 1352
    个人资料
    • 访问:213902次
    • 积分:4461
    • 等级:
    • 排名:第7967名
    • 原创:221篇
    • 转载:69篇
    • 译文:3篇
    • 评论:22条
    博客专栏
    最新评论