引言
随着信息技术的不断发展,以及人们对日常生活舒适度、方便度要求的提高,信息家电、智能仪表等产品越来越频繁的出现在我们的生活当中;人们也越来越热衷于把家电、仪表等设备连接到Internet 中,从而可以方便、及时的对它们进行远程察看、远程控制。把这些设备接入Internet ,就需要考虑TCP/IP 网络协议的实现。
MSP430 系列单片机是由TI 公司开发的16 位单片机,其突出特点是超低功耗,非常适合于各种功率要求比较低的场合,该系列已经应用在智能仪表、医疗设备和保安系统等方面。本文给出了在MSP430F149 硬件平台上移植TCP/IP 协议的方案,实现了终端设备到Internet 的接入。
嵌入式TCP/IP 协议栈uIP
TCP/IP 是一个协议族,它是一个四层网络协议模型,分别包含应用层、传输层、网络层、网络接口层。应用层定义清晰的会话过程,平常所用的协议如HTTP、FTP、SMTP、Telnet 等都属于应用层。传输层提供端对端的通信,该层协议有传输控制协议(TCP) 和用户数据协议(UDP) 。网络层负责数据打包和逻辑寻址,这一层的协议有IP、ICMP、ARP 等协议。网络接口层负责在源和目的节点间的线路上进行无差错的传送数据,并且具有流量控制等功能。
在嵌入式系统中,应用TCP/IP 协议是主要为了完成数据采集和数据传输,不需要实现网页浏览、文件传输等功能,同时,MSP430 芯片也没有足够的空间资源实现所有的TCP/IP协议,所以在本文的方案中,采用了UIP TCP/IP 栈。它是瑞士计算机科学院的Adam Dunkels 等开发的一种免费公开源代码的小型TCP/IP 协议栈,它专门为8 位和16 位MCU 编写。uIP 代码的大小和RAM的需求比其它一般的TCP/IP 栈要小得多。
UIP实现了TCP/IP 协议组的四个基本协议:ARP(地址解析协议) ,IP(网际协议) ,ICMP(因特网信息控制协议) 和TCP(传输控制协议) 。链路层协议例如PPP 等可以由UIP 下面的设备驱动实现;应用层协议例如HTTP、FTP、SMTP、Telnet 等可以由uIP 之上的应用程序实现。
ARP 协议
ARP 协议把目标IP 地址解析为相应的以太网MAC 地址。当一个IP 包要在以太网上发出时,先查询ARP 表,找出包要发送去的MAC 地址。如果在表里找不到对应的IP 地址,就会广播ARP 请求包,以获取给出IP 地址所对应的MAC地址。目的主机收到请求包后发出一个ARP 回应包,给出自己的MAC 地址和IP 地址。
当ARP 表中没有对应的地址条目时,就会发送ARP 请求包时,同时该请求包会覆盖掉发出请求的IP 包,以节省储存器。ARP 表每十秒更新一次。
IP 协议
UIP 的IP 协议主要负责验证输入包的IP 头的正确性,以及在ICMP 和TCP 之间复用数据包。IP 层没有实现数据包的分段和重组,从而代码得到极大的简化。
ICMP 协议
ICMP 中echo 和echo reply 信息常常用在ping 程序里,以检查目的主机能否连通。在uIP 只实现echo 。在处理收到的echo 信息时,只需要把ICMP 类型字段从“echo”类型改变到“echo reply”类型,调整ICMP 校验和,并互掉IP 数据包头里的目的地址和源地址,把包发回到发送方。
TCP
为了减少储存器的使用,在UIP 里,TCP 不再实现发送和接收数据的窗口调整;不会缓存刚刚收到的TCP 段,而是立即由应用程序处理,应用程序可以自己缓冲数据;在输出数据时,在每个连接只能有一个正在传输的TCP 段。
以上四个协议实现的过程中,极大的简化了代码和处理过程,节省了存储空间和缓存空间。
UIP 协议栈的接口
UIP 协议通过一系列接口函数与底层系统和上层应用通信,它内部的协议集合对外部系统来说是透明的,从而增强了该协议的通用性和独立性,可以非常方便地移植到不同系统和应用平台。
图1 描述了UIP、底层系统和应用程序三者之间的调用关系。其中UIP 提供了三个函数给底层系统:UIP_init ( ) ,UIP_input ( ) ,UIP_periodic ( ) 。应用程序向UIP 提供一个调用函数UIP_ APPCALL( ) ,在网络事件或计时事件发生时进行调用;同时,UIP 也要向应用程序提供一些与协议栈的接口函数,应用程序根据接口函数提供的信息或者状态,执行相应的操作。
图1 UIP 协议栈接口
UIP 应用接口
UIP 使用基于事件的程序模式,应用程序由C 语言函数实现。当收发数据、新连接建立或者数据需要重新传输时,UIP 都会调用应用程序。同时,应用程序还要周期查询是否有新的数据收发。因为应用程序只提供了一个回调函数,所以应用程序还要把不同的网络服务映射到不同的端口和连接。
UIP 在接受到底层传来的数据包后,如果需要送上层应用程序处理,就调用UIP_APPCALL( ) 。同时,UIP 设置结构体UIP_conn 指针指向当前连接。UIP_conn 记录一条TCP 连接的所有相关信息,它是维持uIP 运行的关键结构,定义如下:
struct uip_conn {
u8_t tcpstateflags ; PPTCP 的状态和标志
u16_t lport , rport ; PP当地和远端端口
u16_t ripaddr[2] ; PP远端的IP 地址
u8_t rcv- nxt [4] ; PP下一个要接收的序列号
u8_t snd- nxt [4] ; PP上一个已发送的序列号
u8_t ack- nxt [4] ; PP对端下一个应答序列号
u8_t timer ; PP重传时间
u8_t nrtx ; PP计算特殊段的重发数量
u8_t mss ; PP连接中最大分段的大小
u8_t appstate[UIP_APPSTATE_SIZE] ;
} ;
UIP提供给应用程序的接口函数如: uip_listen ( ) 、uip_connect ( ) 、uip_send( ) 、uip_datalen( ) 、uip_close ( ) 、uip_abort ( ) 、uip_stop ( ) 、uip_stopped( ) 、uip_restart ( ) 等,实现了TCP/IP 协议栈的基本功能。