Linux原始套接字实现分析---转

本文深入分析Linux系统中原始套接字的实现,包括链路层和网络层的区别,重点阐述它们在接收和发送报文时的特点,以及在实际应用中的注意事项,如创建、绑定、选项设置和信息查看。此外,讨论了网络诊断工具如tcpdump、ping和traceroute如何利用原始套接字工作。
摘要由CSDN通过智能技术生成

http://blog.chinaunix.net/uid-27074062-id-3388166.html

本文从IPV4协议栈原始套接字的分类入手,详细介绍了链路层和网络层原始套接字的特点及其内核实现细节。并结合原始套接字的实际应用,说明各类型原始套接字的适应范围,以及在实际使用时需要注意的问题。

 

 

 

 

 

一、原始套接字概述

 

协议栈的原始套接字从实现上可以分为“链路层原始套接字”和“网络层原始套接字”两大类。本节主要描述各自的特点及其适用范围。

链路层原始套接字可以直接用于接收和发送链路层的MAC帧,在发送时需要由调用者自行构造和封装MAC首部。而网络层原始套接字可以直接用于接收和发送IP层的报文数据,在发送时需要自行构造IP报文头(取决是否设置IP_HDRINCL选项)。

 

1.1  链路层原始套接字

 

链路层原始套接字调用socket()函数创建。第一个参数指定协议族类型为PF_PACKET,第二个参数type可以设置为SOCK_RAW或SOCK_DGRAM,第三个参数是协议类型(该参数只对报文接收有意义)。协议类型protocol不同取值的意义具体见表1所示:

  1. socket(PF_PACKET, type, htons(protocol))

      

a)       参数type设置为SOCK_RAW时,套接字接收和发送的数据都是从MAC首部开始的。在发送时需要由调用者从MAC首部开始构造和封装报文数据。type设置为SOCK_RAW的情况应用是比较多的,因为某些项目会使用到自定义的二层报文类型。

  1. socket(PF_PACKET, SOCK_RAW, htons(protocol))

 

b)      参数type设置为SOCK_DGRAM时,套接字接收到的数据报文会将MAC首部去掉。同时在发送时也不需要再手动构造MAC首部,只需要从IP首部(或ARP首部,取决于封装的报文类型)开始构造即可,而MAC首部的填充由内核实现的。若对于MAC首部不关心的场景,可以使用这种类型,这种用法用得比较少。

  1. socket(PF_PACKET, SOCK_DGRAM, htons(protocol))

     

 

 

 

表1  protocol不同取值

 

 

 

 

 

 

protocol

作用

ETH_P_ALL

 0x0003

报收本机收到的所有二层报文

ETH_P_IP

0x0800

报收本机收到的所有IP报文

ETH_P_ARP

0x0806

报收本机收到的所有ARP报文

ETH_P_RARP

0x8035

报收本机收到的所有RARP报文

自定义协议

比如0x0810

报收本机收到的所有类型为0x0810的二层报文

不指定

0

不能用于接收,只用于发送

……

……

……

 

 

 

 

表1中protocol的取值中有两个值是比较特殊的。当protocol为ETH_P_ALL时,表示能够接收本机收到的所有二层报文(包括IP, ARP, 自定义二层报文等),同时这种类型套接字还能够将外发的报文再收回来。当protocol为0时,表示该套接字不能用于接收报文,只能用于发送。具体的实现细节在2.2节中会详细介绍。

 

 

 

 

 

 

1.2  网络层原始套接字

 

创建面向连接的TCP和创建面向无连接的UDP套接字,在接收和发送时只能操作数据部分,而不能对IP首部或TCP和UDP首部进行操作。如果想要操作IP首部或传输层协议首部,就需要调用如下socket()函数创建网络层原始套接字。第一个参数指定协议族的类型为PF_INET,第二个参数为SOCK_RAW,第三个参数protocol为协议类型(不同取值的意义见表2)。产品线有使用OSPF和RSVP等协议,需要使用这种类型的套接字。

  1. socktet(PF_INET, SOCK_RAW, protocol)
   

a)       接收报文

网络层原始套接字接收到的报文数据是从IP首部开始的,即接收到的数据包含了IP首部, TCP/UDP/ICMP等首部, 以及数据部分。

 

     

b)      发送报文

网络层原始套接字发送的报文数据,在默认情况下是从IP首部之后开始的,即需要由调用者自行构造和封装TCP/UDP等协议首部。

 

 

这种套接字也提供了发送时从IP首部开始构造数据的功能,通过setsockopt()给套接字设置上IP_HDRINCL选项,就需要在发送时自行构造IP首部。

 

   

  1. int val =
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值