1. 原始套接字能力:

(1) 进程可以读写ICMP,IGMP等分组,如ping程序;

(2) 进程可以读写内核不处理协议字段的ipv4数据报;如OSPF等;

(3) 进程可以使用IP_HDRINCL套接字选项自行构造ipv4首部;

2. 原始套接字的创建:

int sockfd;
sockfd = socket(AF_INET, SOCK_RAW, protocol);

开启ip头构造选项:

const int on = ;
if (setsockopt(sockfd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < ){
error...
}

当选项开启时,我们需要构造完整的IP首部,下列情况除外:

(1) IP总是计算并存储IP首部校验和;

(2) 如果我们将IP标识字段设置为0,内核将设置该字段;

(3) 如果源ip地址是INADDR_ANY,IP将把它设置为外出接口的IP地址;

(4) 如何设置IP选项取决于实现。有些实现取出我们预先使用IP_OPTIONS套接字选项设置的任何IP选项,把它们添加到我们的构造首部中,而其他实现则要求我们亲自在首部指定任何期望的IP选项;

(5) IP首部中的有些字段必须以主机字节序填写,有些字段必须以网络字节序填写,具体取决于实现。linux上所有字节都采用网络字节序;

3. 原始套接字的输出:

(1) 普通输出通过调用sendto或者sendmsg并指定目的IP地址完成,如果套接字已经连接,那么也可以使用write,writev,send;

(2) 如果IP_HDRINCL套接字选项未开启,那么由进程让内核发送的数据的起始地址是IP首部之后的第一个字节,因为内核将构造IP首部并把它置于来自进程的数据之前。内把所构造的ipv4首部的协议字段设置成来自socket调用的第三个参数;

(3) 如果IP_HDRINCL套接字选项已开启,那么由进程让内核发送的数据的起始地址是IP首部的第一个字节。进程调用输出函数写出的数据量必须包括IP首部大小。整个IP首部由进程构造。不过(a)ipv4标识字段可以设置为0,从而告知内核设置该值;(b) ipv4首部校验和字段总是由内核计算并存储 (c) ipv4选项字段是可选的;

(4) 内核会对超出外出接口MTU的原始分组执行分片;

另:ip首部之后所含有的其他任何首部的校验和需要用户进程计算;

4. 原始套接字的输入:

(1) 接收到的udp/tcp分组决不传递到任何原始套接字;进程只能通过数据链路层读取这些分组;但是可以发哦;

(2) 大多数icmp分组在内核处理完其中的icmp消息后传递到原始套接字;源自Berkeley的实现把不是回射请求,时间戳请求或者地址掩码请求(这三类均有内核处理)的所有接收到的icmp分组传递给原始套接字;

(3) 所有igmp分组在内核完成处理其中的igmp消息后传递给原始套接字;

(4) 内核不认识协议字段的所有ip数据报传递到原始套接字;内核对这些分组进行ip头最小验证:ip版本,ip校验和,首部长度,目的ip地址;

(5) 如果某个数据报以片段的形式到达,那么在它的所有片段均到达并重组出该数据报之前,不会传递任何片段分组到原始套接字;

(6) 如果创建原始套接字制定了非0协议参数,那么接收到的数据报协议字段必须匹配该值,否则该数据报不会传递到该套接字;

(7) 如果原始套接字已由bind调用绑定了某个本地ip地址,那么接收到的数据报目的IP地址必须匹配这个绑定地址,否则该数据报不会传递到这个套接字;

(8) 如果这个原始套接字由connect调用制定了某个外地ip地址,那么接收到的数据报的源ip地址必须匹配该连接,否则数据报不会传递到该套接字;

注意:如果一个原始套接字是以0值协议参数创建,并且既未调用bind,也未调用connect,那么该套接字将接收可由内核传递到原始套接字的每个原始数据报的一个副本;

无论何时往一个原始IPv4套接字传递一个接收的数据报,传到该套接字所在进程都包括ip首部在内的完整数据报;ipv6不包含首部;

5. 数据链路层访问:

接收所有帧:

fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));

接收ipv4帧:

fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_IP));

注意:如果设置接口为混杂模式,则可接收所有经过数据流,而不论其目的地址是否是它;

原始套接字&&数据链路层访问的更多相关文章

  1. Linux Socket 原始套接字编程

    对于linux网络编程来说,可以简单的分为标准套接字编程和原始套接字编程,标准套接字主要就是应用层数据的传输,原始套接字则是可以获得不止是应用层的其他层不同协议的数据.与标准套接字相区别的主要是要开发 ...

  2. Linux系统编程(37)—— socket编程之原始套接字

    原始套接字的特点 原始套接字(SOCK_RAW)可以用来自行组装IP数据包,然后将数据包发送到其他终端.也就是说原始套接字是基于IP数据包的编程(SOCK_PACKET是基于数据链路层的编程).另外, ...

  3. UNIX网络编程——原始套接字SOCK_RAW

    实际上,我们常用的网络编程都是在应用层的报文的收发操作,也就是大多数程序员接触到的流式套接字(SOCK_STREAM)和数据包式套接字(SOCK_DGRAM).而这些数据包都是由系统提供的协议栈实现, ...

  4. Go中原始套接字的深度实践

    1. 介绍 2. 传输层socket 2.1 ICMP 2.2 TCP 2.3 传输层协议 3. 网络层socket 3.1 使用Go库 3.2 系统调用 3.3 网络层协议 4. 总结 4.1 参考 ...

  5. raw_socket(原始套接字)以及普通socket使用终极总结

      一.传输层socket(四层socket,普通socket) 可参考本人以下博客: Windows Socket编程之UDP实现大文件的传输:http://blog.csdn.net/luchen ...

  6. Linux网络编程——原始套接字能干什么?

    通常情况下程序员接所接触到的套接字(Socket)为两类: (1)流式套接字(SOCK_STREAM):一种面向连接的 Socket,针对于面向连接的TCP 服务应用: (2)数据报式套接字(SOCK ...

  7. Linux基础(11)原始套接字

    一边接收函数返回一边判断返回值时一定要把接收的优先级加()提高再去判断 例 if((sockfd = socket()) < 0) 问题: 如何实现SYN扫描器扫描端口 , 比如AB两个设备要进 ...

  8. Linux原始套接字抓取底层报文

    1.原始套接字使用场景 我们平常所用到的网络编程都是在应用层收发数据,每个程序只能收到发给自己的数据,即每个程序只能收到来自该程序绑定的端口的数据.收到的数据往往只包括应用层数据,原有的头部信息在传递 ...

  9. Linux网络编程:原始套接字简介

    Linux网络编程:原始套接字编程 一.原始套接字用途 通常情况下程序员接所接触到的套接字(Socket)为两类: 流式套接字(SOCK_STREAM):一种面向连接的Socket,针对于面向连接的T ...

随机推荐

  1. Linux中实现在系统启动时自动加载模块

    下面是以前学习Linux时写的,后来仔细研究rc.sysinit后发现,只需要修改下列地方就可以了,不必这么麻烦的: rc.sysinit中有这样的一段代码: # Load other user-de ...

  2. hdu5575 Discover Water Tank

    题意: 给出个水箱,水箱两侧有无限高的隔板,水箱内有整数高度的隔板将水箱分成n-1份,现在给出m个限制,每个限制表示某个位置的某个高度有水或没水,问最多能同时满足多少个限制.n,m<=2*10^ ...

  3. 【hdu4734】F(x) 数位dp

    题目描述 对于一个非负整数 $x=​​\overline{a_na_{n-1}...a_2a_1}$ ,设 $F(x)=a_n·2^{n-1}+a_{n-1}·2^{n-2}+...+a_2·2^1+ ...

  4. prototype的本质

    在<关于思维方式的思绪>那篇文章里提到了, 原型的本质就是一种委托关系. 即我这里没有,就到我的原型里去看看,一旦找到就当成我的用. 本文详细说一下这个事情. 比如某女买东西,钱都是她老公 ...

  5. 2017中国大学生程序设计竞赛-哈尔滨站 H - A Simple Stone Game

    A Simple Stone Game Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Ot ...

  6. hdu ACM Steps Section 1 花式A+B 输入输出格式

    acm与oi很大的一个不同就是在输入格式上.oi往往是单组数据,而acm往往是多组数据,而且题目对数据格式往往各有要求,这8道a+b(吐槽..)涉及到了大量的常用的输入输出格式.https://wen ...

  7. BZOJ1934:[SHOI2007]善意的投票 & BZOJ2768:[JLOI2010]冠军调查——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=1934 https://www.lydsy.com/JudgeOnline/problem.php? ...

  8. C++中swap函数

    本文是我用到swap函数时,对其产生好奇,所以结合网上有关博文写下的.个人水平有限,若有错误的地方,欢迎留言指出.谢谢! 一.通用的函数交换模板 template<class T> voi ...

  9. React受控组件和非受控组件

    受控组件和非受控组件主要是用来解决表单组件状态谁来控制的问题.因为用户的输入会反应在界面上,相当于视图的状态发生了变化,而react是通过虚拟DOM比对修改视图的,这里就要决定谁来控制表单组件的状态. ...

  10. Codechef MARCH14 GERALD07加强版

    强制在线不代表不能预处理! 考虑暴力怎么干? 开始n个联通块.now=n 不断加入边,如果连接两个联通块,--now 否则不动. 后者的前提是和[l,id-1]的边构成环 所以,我们考虑每个[l,r] ...