引言
uC/OS-Ⅱ是一个源码开放的抢占式实时操作系统。它内核短小精悍、可裁减、执行时间确定。系统大部分代码采用C语言编写,与硬件有关的部分都集中在两个文件中,给出了规范的接口说明,移植相当方便,可应用于目前大多数型号的8位、16位、32位CPU。
uC/OS-Ⅱ提供的仅仅只是一个实时的调度及任务间通信的内核,没有集成网络协议。上网是当前嵌入式设备的广泛需求,本文讨论轻型TCP/IP协议栈的引入以及相关网络设备驱动程序,实现嵌入式系统的网络功能。
本文所用的硬件系统结构如图1所示。开发板基于TMS320LF2407A的含DSP核微处理器和LAN91C111以太网控制器。在成功移植了μCOS-Ⅱ的基础上进一步实现了以太网通讯功能。下面重点介绍TCP/IP协议栈的引入和LAN91C111驱动的编写。
图1 嵌入式以太网硬件系统结构图
TCP/IP网络协议栈的引入
在μCOS-Ⅱ上引入下TCP/IP协议栈,由于嵌入式系统的硬件资源有限,必须使用小型协议栈。这种协议栈很多,LwIP是其中之一。
关于LwIP简介
LwIP是瑞士计算机科学院(SCICS)的Adam Dunkels等开发的一套用于嵌入式系统的开放源码的轻型TCP/IP协议栈,但Lwip实现了较为完备的IP,ICMP, UDP, TCP协议,具有超时时间估计、快速恢复和重发、窗口调整等功能。IwIP在保持协议主要功能的基础上减少对RAM和ROM的占用,一般它只需要几十K的RAM和40K左右的ROM就可以运行,很适合同μCOS-Ⅱ相配合用在嵌入式系统中。LwIP在设计时就考虑到了将来的移植问题,它把所有与硬件、操作系统、编译器相关的部分独立出来,放在/src/arch目录下,因此LwIP在μCOS-Ⅱ上的实现就是修改这个目录下的文件,其它的文件一般不需要修改。下面分别予以说明:
协议栈的实现
·与CPU及编译器相关的include文件 /src/arch/include/arch目录下cc.h、cpu.h、perf.h中有一些与CPU或编译器相关的定义,如数据长度,字的高低位顺序等。这应该与用户实现μCOS-Ⅱ时定义的数据长度等参数一致。
·与操作系统相关部分 sys_arch.c中的内容是与操作系统相关的一些结构和函数,主要分四个部分: (1)sys_sem_t信号量LwIP中需用信号量通信,所以在sys_arch中应实现信号量结构体和处理函数:struct sys_sem_t{ sys_sem_new( )/创建一个信号量结构 sys_sem_free()/释放一个信号量结构sys_sem_signal( )/发送信号量 sys_arch_sem_wait( )/请求信号量}由于μCOS-Ⅱ已经实现了信号量OS_EVENT的各种操作,并且功能和LwlP上面几个函数的目的功能是完全一样的,所以只要把μCOS-Ⅱ的函数重新包装成上面的函数,就可以直接使用了。
(2 )sys_mbox_t消息
LwIP使用消息队列来缓冲、传递数据报文,因此要在sys_arch中实现消息队列结构sys_mbox_t,以及相应的操作函数。
sys_mbox_new()/创建一个消息队列 sys_mbox_free( ) /释放一个消息队列
sys_mbox_post( )/向消息队列发送消息
sys_arch_mbox_fetch( )/从消息队列中获取消息
μCOS-Ⅱ同样实现了消息队列结构及其操作,但是μCOS-Ⅱ没有对消息队列中的消息进行管理,因此不能直接使用,必须在μCOS-Ⅱ的基础上重新实现。
(3)sys_arch_timeout函数
LwIP中每个与外界网络连接的线程都有自己的timeout属性,即等待超时时间。这个属性表现为每个线程都对应一个sys_timeout结构体队列,包括这个线程的timeout时间长度,以及超时后应调用的timeout函数,该函数会做一些释放连接,回收资源的工作.timeout结构体已经由LwIP自己在sys.h中定义好了,而且对结构体队列的数据操作也由LwIP负责,我们所要实现的是如下函数:
struct sys_timeouts*sys_arch_timeouts(void)
这个函数的功能是返回目前正处于运行态的线程所对应的timeout队列指针。timeout队列属于线程的属性,它是OS相关的函数,只能由用户实现。
(4)sys_thread_new创建新线程
LwIP可以是单线程运行,也可以多线程运行。为提高效率并降低编程复杂度,就需要用户实现创建新线程的函数:
void sys_thread_new(void(*thread)(void*arg), void*arg);
在μCOS-Ⅱ中,没有线程(thread)的概念,只有任务(Task)。它已经提供了创建新任务的系统API调用OSTaskCreate,因此只要把OSTaskCreate封装一下,就可以实现sys_hread_new.
·lib_ arch中库函数的实现
LwIP协议栈中用到了8个外部函数,这些函数通常与用户使用的系统或编译器有关,因此留给用户自己实现,有关程序如下:
u16_t htons(u16_t n); /16位数据高低字节交换
u16_t ntohs(u16_t n);
int strlen(const char * str);/返回字符串长度
int strncmp(const char * strl,const char * str2,int len);/字符串比较
void bcopy(const void * src, void * dest, int len);/内存数据块之间的互相拷贝
void bzero(void *data, int n); /内存中指定长度的数据块清零
类似于操作系统在硬件上的移植,LwIP的移植也是根据实现的硬件以及操作系统对象,对相应的文件进行修改。整个通讯协议的引入可以很快实现。
LAN91C111驱动的实现
在上面为μCOS-Ⅱ引入了TCP/IP协议栈之后,为了实现以太网通信功能还必须完成相关网络设备驱动程序的添加。LwIP的网络驱动有一定的模板,其中src/netif/ethernetif.c文件即为驱动的模板,用户为自己的网络设备实现驱动时应参照这个模板,根据相应的网络芯片来实现。本系统选用的网络芯片是由SMSC公司生产的自适应10M/100M第三代快速以太网控制器芯片LAN91C111,集成了SMSC/CD协议的MAC(媒体层)和PHY(物理层)。由于其灵活性和集成度高,具有较高的性价比。
LAN91C111工作流程比较简单,驱动程序将要发送的数据包按指定格式写入芯片并启动发送命令,LAN91C111会自动把数据包转换成物理帧格式在物理信道上传输;反之芯片收到物理信号后自动将其还原成数据,并按指定格式存放在芯片RAM中以便主机程序取用。简言之就是LAN91C111完成数据包和电信号之间的相
互转换: 数据包 电信号。LAN91C111的编程主要包括:初始化、发送数据包、接收数据包三部分。
初始化上电后,LAN91C111内部的寄存器的值设置为缺省值,CPU根据需要设置它里面的Configuration, Base和Individual Address寄存器,以保证它正确工作。发送数据包流程
(1) DSP向控制器发送ALLOCATE MEMORY命令(设置MMUCOM寄存器,通常设置0x0020)。MMU为待发送包在控制器内部的packet buffer中分配存储空间。
(2) DSP查询中断状态寄存器中的ALLOC INT位,直到该位被置成1,也可以设置Interrupt Mask中的ALLOC INT位,然后等待硬件中断,这时MMU已经分配好存储空间。而且TX packet number放在Allocation Result寄存器中。
(3)将Allocation Result寄存器中的packet number拷贝到Packet Number:寄存器中,设置Pointer寄存器(设置为TX,WR,AUTOINC,即0x4000)。然后将包的数据从upper layer发送队列传送到控制器的数据寄存器中。要求依次写人Status Word, Byte Count, destination address,source address,packet size,packet data,control word。
(4) DSP向控制器发送"ENQUEUE PACKET NUMBER TO TX FIFO“命令(设置MMUCOM寄存器,通常设置Ox00C0),这个命令将Packet Number寄存器中的packetnumber拷贝到TX FIFO,说明发送的包已经放入队列中。同时设置Transmit control寄存器中的TXENA位,启动transmitter。到目前为止,DSP的设置工作完成,它可以IDLE,直到接收到一个控制器产生的发送中断。
(5)当控制器传送完包以后,memory中的第一个字(16bit)被CSMA/CD写入相应的Status Word,然后将TX FIFO中的packet number移到TX completion FIFO,当TX completion FIFO不为空时产生中断。
(6) DSP接收到中断后,开始执行中断处理程序,它读入中断状态寄存器,如果产生发送中断,则从FIFO ports寄存器读入发送的包的Packet Number,并将它写到Packet Number寄存器。然后从内存中读人状态字(包括设置Pointer寄存器为TX,RD,AUTOINC,即0x6000,然后从数据寄存器中读入包的状态字),它是EPH寄存器的镜像,根据状态字判断包发送是否成功。如果成功则DSP向控制器发布RELEASE命令(设置MMUCOM寄存器,设置为Ox00A0),控制器将释放发送包所使用的存储空间,同时设置TX INT Acknowledge寄存器,它将TX completion FIFO中的packet number清除。
(7)使用“每发送一个序列的包产生一个中断”方案:允许TX EMPTY INT和TX INT, AUTORELEASE="1",当发送完FIFO中的最后一个包后,产生TX EMPTY INT中断。当发生严重的发送错误时,产生TX INT中断,同时将发送失败的包的packet number保存到FIFO Ports寄存器,这样DSP就可以知道发送过程停止了。这种方案可以减少DSP的负担,而且存储空间的释放也更迅速。接收数据包流程
(1) DSP设置receive control寄存器中的RXEN位,允许接收包。
(2)含有正确地址的包被接收到,从MMU请求存储空间,并分派一个packet number,内部的DMA逻辑产生连续的地址,并将接收到的字写到memory中,如果超界,包被丢弃,存储空间被释放。当检测到包的结束,状态字被写到接收包的最前面,byte count写到第二个字。如果CRC校验正确,packet number被写到RX FIFO,由于RX FIFO非空,产生RCV INT中断;如果CRC校验不正确,存储空间被释放,而且不产生中断。
(3) DSP接收到中断后开始执行中断处理程序,它读入中断状态寄存器,如果产生接收中断(RCV INT位为1),则可以从FIFO ports寄存器得到接收的包的packet number,而且可以从数据寄存器将接收包传送到DSP的内存或外存中。当处理结束,DSP向处理器发布REMOVE AND RELEASE FROM TOP OF RX命令(即设置寄存器MMUCOM,即0x0060),释放使用的存储空间和packet number.
软件的调试与验证
调试环境包括我们做的TMS320LF2407A+LAN91C111板、PC机、仿真器、网线等。首先,新建工程,脱离操作系统和TCP/IP协议的环境下,单独调试通过LAN91C111的驱动程序,初始化,接收发送数据成功之后,另建工程集合μCOS-Ⅱ和LwIP结合驱动程序进行调试,在μCOS-Ⅱ中初始化LwlP,并创建TCP或UDP任务进行测试了。值得注意的是LwIP的初始化必须在μCOS-Ⅱ完全启动之后也就是在任务中进行,因为它的初始化用到了信号量等OS相关的操作。关键部份的代码和说明如下:
main(){OSlnit();OSTaskCreate(Iwip_init_task, Null, &Iwip-init-stk[TASK_STK_SIZE-1 ], 0);OSStart();}
主程序中创建了初始化LwIP任务Lwip_init_task(优先级0). Iwip_init_task任务中初始化硬件时钟和LwIP,还创建了tcpip_thread(优先级5)和tcpecho_thread(优先级6)两个任务。实际上tcpip_thread才是LwIP的主线程,多线程的Berkley API也是基于这个线程实现的,即上面的tcpecho_thread线程也要依靠tcpip_thread线程来与外界通信,这样做的好处是编程简单,结构清晰。
编译运行后,用ping IP地址命令可以得到ICMP reply响应。用telnet IP地址命令可以看到echo server的回显效果。说明ARP,ICMP,IP、下CP协议都已正确运行,调试通过。
结语
按课题的需求,这套系统用于电力保护系统的现场板卡的管理与和上下位机的通讯,现场采集的数据经处理后,通过数据线路连接到该板(本文所讨论的系统)。由该DSP板集中进行管理并实现和上位机的通讯。该系统目前效果令人满意,并且可以根据课题的需要,灵活地进行扩展。还可用于智能家电等领域,具有很好的发展前景。
声明:本网站所收集的部分公开资料来源于互联网,转载的目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。本站部分作品是由网友自主投稿和发布、编辑整理上传,对此类作品本站仅提供交流平台,不为其版权负责。如果您发现网站上所用视频、图片、文字如涉及作品版权问题,请第一时间告知,我们将根据您提供的证明材料确认版权并按国家标准支付稿酬或立即删除内容,以保证您的权益!联系电话:010-58612588 或 Email:editor@mmsonline.com.cn。
- 暂无反馈