做了这么多年的程序员后,总想资源回收一下,写一点点什么,却又发现无从写起。

SLIP—串行线路上传输数据报的非标准协议

简介

TCP/IP 协议族可以在许多网络介质上运行,如: IEEE 802.3 (以太网)和802.5(令牌环)局域网,X.25线路,卫星连接和串行线路。除了串行线路外,其它的介质上都有包格式的标准。

SLIP(Serial Line Internet Protocol,串行线路网际协议),该协议是Windows远程访问的一种旧工业标准,主要在Unix远程访问服务器中使用,现今仍然用于连接某些ISP。因为SLIP协议是面向低速串行线路的,可以用于专用线路,也可以用于拨号线路,Modem的传输速率在1200bps到19200bps。

SLIP ,即串行线路IP,实际上是一个标准,它通常用于运行TCP/IP协议点对点连接之中。

它并非Internet标准。

历史

SLIP起始于八十年代初3COM UNET TCP/IP实现,它仅是一个包协议:SLIP定义了一系列字符将IP包在串行线路上变成帧格式,仅此而已。它不提供寻址,包类型标识,差错控制或压缩机制。因为此协议十分简单,所以非常容易实现。

在1984年左右,Rick Adams为Berkeley Unix和Sun Microsystems工作站实现了SLIP并推广到世界。它很快被用于在主机和路由器之间的串行线路连接。SLIP通常用于专线连接,有时也用于拔号连接,其速度经常在1200bps和19.2Kbps之间。对于主机和路由器之间的连接是十分有用的。

实用性

SLIP在大部分基于Berkeley UNIX的系统上可用,在Berkeley 4.3BSD 中也包括SLIP。SLIP在Ultrix,Sun UNIX和大部分由Berkeley演变而来的UNIX上可用。一些终端集中器和IBM PC也支持它。

协议

SLIP协议定义两个特殊字符:ENDESCEND是八进制300(十进制192)ESC是八进制(十进制219),这与ASCII码中的ESC字符不冲突;为了讨论的方便,这里所说的ESC均是SLIP的ESC字符。

若要发送一个包,SLIP主机只需要以包的形式发送数据即可。

如果数据与END字符相同,则发送ESC和八进制的334(十进制220)代替。

如果和ESC相同,则以ESC和八进制335(十进制221)代替。

当包数据发送结束,则发送一个END字符。

Phil Karn提出一个改进的算法,可以在包头和饱包尾都使用END。这将消除由于线路噪声带来的错误。在一般情况下,接收方只用观察两个END,这将产生错误的IP包。如果SLIP实现不放弃0长度包,那IP实现会这样做的。如果因为噪声,此包将被抛弃,而不影响下面的包。因为没有标准的SLIP说明,因此没有真正定义的最大SLIP包大小。我们最好接受由Berkeley UNIX SLIP drivers定义的大小:1006字节,包括IP和传输协议头(不包括帧字符。因此,新的SLIP实现应该准备接收1006字节的数据报,而且不应该发送大于1006字节的数据报。

不足之处

有一些用户希望SLIP提供但它没有提供的功能,公平地说,SLIP仅仅是很久前,问题并不那么重要时设计的普通协议。下面是显而易见的SLIP的不足之处:

  • 寻址功能

    SLIP连接的双方都出于路由的目的需要知道对方的IP地址。并且,当使用SLIP作为主机拔号到路由器的目的时,寻址机制会是动态的,路由器需要通知拔号主机主机的IP地址。而现在,SLIP却没有提供通过SLIP连接传送地址信息的机制。

  • 类型标识

    SLIP没有类型域,因此,在SLIP连接上仅能运行一种协议,所有在配置了TCP/IP和DECnet的主机之间不可能使用SLIP。而SLIP是串行线路IP,如果以串行线路连接多协议的计算机,这些计算机应该具有以一种以上协议通信的能力。

  • 差错检测与校正

    线路噪声可能使包在传送过程中损坏,因为线路速率比较低,因此,重新发送的代价是昂贵的。在SLIP层,差错控制并不是必须的,因为IP应用程序可以检测到损坏的包(IP头和UDP,TCP校验码是足够的),但是一些应用程序如NFS通常忽略错误而单纯依靠网络介质来检测损坏的包。因为重新传送的代价很大,因此SLIP提供差错检测与校正是更有效的方法。

  • 压缩

    因为拔号线路速率比较慢,包的压缩将大大提高包的吞吐量。通常,在单独一个TCP连接的包序列中的IP和TCP头中几乎没有多少变化,所以普通的压缩算法就可以仅发送改变的包头部分而不是整个包头。已经在这方面做了一些工作,上面的问题中的全部或一部分正在研究之中。

SLIP驱动程序

下面的C语言函数可以发送并接收SLIP包。他们依靠两个函数完成功能:send_char()和recv_char(),它们分别在串行线路上发送和接收一个字节。


  1. /* SLIP特殊字符 */
  2. #define END 0300 /*标明包结束*/
  3. #define ESC 0333 /*标明字节填充*/
  4. #define ESC_END 0334 /*ESC ESC_END用于包中数据和和END相同时的转意字符*/
  5. #define ESC_ESC 0335 /*ESC ESC_ESC用于包中数据和和ESC相同时的转意字符*/
  6. /* SEND_PACKET:发送长度为LEN的的包,起始位置在P*/
  7. void send_packet(char *p, int len)
  8. {
  9. /*发送一个END字符*/
  10. send_char(END);
  11. /*发送包内的数据*/
  12. while (len--)
  13. {
  14. switch (*p)
  15. {
  16. /*如果需要转意,则进行相应的处理*/
  17. case END:
  18. send_char(ESC);
  19. send_char(ESC_END);
  20. break;
  21. case ESC:
  22. send_char(ESC);
  23. send_char(ESC_ESC);
  24. break;
  25. /*如果不需要转意,则直接发送*/
  26. default:
  27. send_char(*p);
  28. }
  29. p++;
  30. }
  31. /*通知接收方发送结束*/
  32. send_char(END);
  33. }
  34. /* RECV_PACKET:接收包数据,存储于P位置,如果接收到的数据大于LEN,则被截断,函数返回接收到的字节数*/
  35. int recv_packet(char *p, int len)
  36. {
  37. char c;
  38. int received = 0;
  39. while (1)
  40. {
  41. /*接收字符*/
  42. c = recv_char();
  43. switch (c)
  44. {
  45. /*如果接收到END,包数据结束,如果包内没有数据,直接抛弃*/
  46. case END:
  47. if (received)
  48. return received;
  49. else
  50. break;
  51. /*下面的代码用于处理转意字符*/
  52. case ESC:
  53. c = recv_char();
  54. switch (c)
  55. {
  56. case ESC_END:
  57. c = END;
  58. break;
  59. case ESC_ESC:
  60. c = ESC;
  61. break;
  62. }
  63. default:
  64. if (received < len)
  65. {
  66. p[received++] = c;
  67. }
  68. }
  69. }
  70. }

SLIP—串行线路上传输数据报的非标准协议的更多相关文章

  1. 《ServerSuperIO Designer IDE使用教程》-1.标准Modbus和非标准协议的使用、测试以及驱动开发。附:v4.2发布

    ServerSuperIO Designer IDE v4.2版本更新内容: 增加ServerSuperIO.Host运行程序,可以使用IDE进行测试,Host为运行环境. 针对设备驱动增加导入监测点 ...

  2. Python并发编程系列之常用概念剖析:并行 串行 并发 同步 异步 阻塞 非阻塞 进程 线程 协程

    1 引言 并发.并行.串行.同步.异步.阻塞.非阻塞.进程.线程.协程是并发编程中的常见概念,相似却也有却不尽相同,令人头痛,这一篇博文中我们来区分一下这些概念. 2 并发与并行 在解释并发与并行之前 ...

  3. filebeat向kafka中传输数据报WARN Failed to connect to broker DOMSDev07:9092: dial tcp: lookup DOMSDev07: getaddrinfow: No such host is known.解决方法

    打开filebeat客户端所在机器C:\Windows\System32\drivers\etc目录,找到hosts文件 以记事本形式打开,在底部追加 “IP 主机名” 即可

  4. 第11节-BLE协议HCI层的硬件接口

    本篇博客由韦东山视频整理所得 如何控制链路层让其发出广播包.数据包?通过HCI层向它发出命令,也可以通过ATT层.L2CAP层向LL层发出数据. 学习资料: 蓝牙协议core_v5.0.pdf < ...

  5. 第07节-开源蓝牙协议BTStack框架代码阅读(下)

    上篇博客中已经对BTStack框架进行了较为详细的说明,本篇博客将进一步总结一下(由韦大仙笔记所得). 可以从5个方面来理解BTStack的框架: 1.硬件操作:hci_transport_t BTS ...

  6. 链路层 - SLIP,PPP,

    最常使用的封装格式是RFC 894定义的格式.图2 - 1显示了两种不同形式的封装格式.图中每个方框下面的数字是它们的字节长度. 两种帧格式都采用48 bit(6字节)的目的地址和源地址( 8 0 2 ...

  7. RFC(请求注解)--各种协议-标准

    转自:http://blog.sina.com.cn/s/blog_65d6476a0101cj8n.html RFC(Request For Comments)-意即“请求注解”,包含了关于Inte ...

  8. ★RFC标准库_目录链接

    RFC(Request For Comments)是一个国际标准化的数据库,记录了从计算机到互联网的海量标准协议.它是一个免费公开的IT标准文件分享平台,其内容也在不断增长,与时俱进.它与ISO等组织 ...

  9. ★RFC标准库_目录链接

    RFC(Request For Comments)是一个国际标准化的数据库,记录了从计算机到互联网的海量标准协议.它是一个免费公开的IT标准文件分享平台,其内容也在不断增长,与时俱进.它与ISO等组织 ...

随机推荐

  1. 1、概率vs统计

  2. U盘安装RedHat linux 5.3

    U盘安装RedHat linux 5.3 1.下载rhel-5.3-server-i386-dvd.iso文件: 2.下载绿色版UltraISO软件: 3.将rhel-5.3-server-i386- ...

  3. [GO]简单的并发服务器

    package main import ( "net" "fmt" "strings" ) func HandleConn(conn net ...

  4. (转) c/c++调用libcurl库发送http请求的两种基本用法

    libcurl主要提供了两种发送http请求的方式,分别是Easy interface方式和multi interface方式,前者是采用阻塞的方式发送单条数据,后者采用组合的方式可以一次性发送多条数 ...

  5. mybatis SqlMapConfig.xml

    一.SqlMapConfig.xml 1.属性properties 在入门时,以抽取出连接数据库的属性得到properties文件. a.可以通过resource和url来获得属性. b.proper ...

  6. 团体程序设计天梯赛L1-024 后天 2017-03-22 17:59 68人阅读 评论(0) 收藏

    L1-024. 后天 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 如果今天是星期三,后天就是星期五:如果今天是星期六,后天就 ...

  7. linux 挂载和使用文件系统

    从分区,到创建文件系统,再到把磁盘或分区挂载到一个目录后才能够使用. Windows或Mac系统会自动进行挂载,一旦创建好文件系统后会自动挂载到系统上,Windows我们称之为C\D盘,而Linux需 ...

  8. vector容器(一)

    一. Vector简要描述 vector是C++标准模版库STL提出的一种顺序存储结构,之所以称之为“容器”,是因为vector是一个模板类,它允许我们重复利用已有的实现构造自己的特定类型下的数据结构 ...

  9. linux下PHP5.5的安装【oci8,pdo-oci,memcache,Zend OPCache扩展】

    最近一段时间学习了一下PHP,用CI做了一个小项目,为了开发方便,本地windows下使用了集成环境XAMPP,不过当把项目部署到linux上时,确实遇到了很多问题,下面把我在linux上安装php的 ...

  10. 在 Mac OSX 上安装 nginx

    今天在使用 brew 安装 nginx 时,提示错误,安装不上去: brew install nginx, 提示:/usr/local is not writable. 这个是需要修改 /usr/lo ...