ZT:实时操作系统µC/OS II下TCP/IP协议栈的实现

简介: 实时操作系统µC/OS II下TCP/IP协议栈的实现 摘要: 结合ez80和ARM7两种系统上的具体实现,说明了如何在嵌入式实时操作系统µC/OSII上移植实现LwIP这套TCP/IP协议栈,使µC/OS II成为支持网络的RTOS。

实时操作系统µC/OS IITCP/IP协议栈的实现

摘要: 结合ez80ARM7两种系统上的具体实现,说明了如何在嵌入式实时操作系统µC/OSII上移植实现LwIP这套TCP/IP协议栈,使µC/OS II成为支持网络的RTOS

 

关键词 µC/OS IITCP/IPLwIP,网络设备驱动

引言

随着嵌入式系统与网络的日益结合,在嵌入式实时操作系统中引入TCP/IP协议栈,以支持嵌入式设备接入网络,成为嵌入式领域重要的研究方向。µC/OS II是近年来发展迅速的一个开放源码实时操作系统,但它只是一个实时的任务调度及通信内核,缺少对外围设备和接口的支持,如没有文件系统、网络协议、图形界面。笔者在多个嵌入式项目的开发过程中,以开源TCP/IP协议栈LwIP为基础,给µC/OS II加上了网络支持。下面就以µC/OS II LwIP分别在8MCU ez8032MCU ARM7TDMI上的实现为例进行说明。

需要说明的是,笔者使用的ez80系统是Zilog公司的ez80190开发板,自带网络芯片。而ARM7系统是使用笔者参与开发的Skyeye,一个基于GDBARM7TDMI指令级软件仿真器。Skyeye小组最近为Skyeye加上了软件模拟的Ne2k兼容网络芯片,可以运行带网络支持的µcLinuxµC/OS II。以下的全部相关程序和代码都可以在Skyeye网站(hpclab.cs.tsinghua.edu.cn/~skyeye/)下载。

1  基于 µC/OS II 的网络平台概述

嵌入式操作系统µC/OS II是一个公开源代码的占先式多任务的微内核RTOS,其性能和安全性可以与商业产品竞争。µC/OS II的特点可以概括为以下几个方面:公开源代码,代码结构清晰、明了,注释详尽,组织有条理,可移植性好。可裁剪,可固化。内核属于抢占式,最多可以管理60个任务。µC/OS II1992年的第一版(µC/OS)以来已经有好几百个应用,是一个经实践证明好用且稳定可靠的内核。目前国内对µC/OS II的研究和应用都很多。

       TCP/IPInternet的基本协议,以其实用性、高效性已经成为事实上的工业标准。嵌入式设备要与Internet网络直接交换信息,就必须支持TCP/IP协议。目前嵌入式设备上TCP/IP方案有很多种,但面向低端应用的开源嵌入式网络平台还很少见。

       µC/OS II是一个富有开放色彩的RTOS,只要买一本书就可获得源代码,对学校和教育的使用完全免费,商业应用的费用相对也很低。但是它目前的一些第三方TCP/IP支持都是完全商业化的,用户需要付费才能获得,很少给出源代码,这影响了µC/OS II的研究和推广。通过把开放源代码的TCP/IP协议栈LwIP移植到µC/OS II上来,就获得了一套可免费研究、学习的嵌入式网络软件平台。系统示意图如图1

LwIP协议栈

Ez80SkyeyeMCU

Ne2k网络芯片

µC/OS II

网络设备驱动

应用程序(网络或非网络应用)


1 µC/OS II LwIP 系统示意图 2  开源 TCP/IP 协议栈 LwIP 简介

LwIP是瑞士计算机科学院(Swedish Institute of Computer Science)的Adam Dunkels等开发的一套用于嵌入式系统的开放源代码TCP/IP协议栈。LwIP的含义是Light Weight(轻型)IP协议。LwIP可以移植到操作系统上,也可以在无操作系统的情况下独立运行。LwIP TCP/IP实现的重点是在保持TCP协议主要功能的基础上减少对RAM的占用,一般它只需要几十KRAM40K左右的ROM就可以运行,这使LwIP协议栈适合在低端嵌入式系统中使用。

LwIP的特性如下:

(1)              支持多网络接口下的IP转发

(2)              支持ICMP协议

(3)              包括实验性扩展的的UDP(用户数据报协议)

(4)              包括阻塞控制,RTT估算和快速恢复和快速转发的TCP(传输控制协议)

(5)              提供专门的内部回调接口(Raw API)用于提高应用程序性能

(6)              可选择的Berkeley接口API(多线程情况下)

我们目前使用的是LwIP的最新稳定版V0.5.3。有关LwIP的详细内容,可以参考其代码和网站上的文档。

4  LwIPµC/OS II下的实现 4.1 概述

       LwIP协议栈在设计时就考虑到了将来的移植问题,因此把所有与硬件、OS、编译器相关的部份独立出来,放在/src/arch目录下。因此LwIPµC/OS II上的实现就是修改这个目录下的文件,其它的文件一般不应该修改。下面分几部份分别说明相应文件的实现原理和过程。具体的代码限于篇幅没有给出,Skyeye网站上有完整的代码和说明。

4.2 CPU 或编译器相关的 include 文件

       /src/arch/include/arch目录下cc.hcpu.hperf.h中有一些与CPU或编译器相关的定义,如数据长度,字的高低位顺序等。这应该与用户实现µC/OS II时定义的数据长度等参数是一致的。

#define BYTE_ORDER LITTLE_ENDIAN  //ARM7默认为小端存储系统

//数据类型长度的定义

typedef unsigned char   u8_t;

typedef signed char     s8_t;

typedef unsigned short  u16_t;

typedef signed short    s16_t;

typedef unsigned int    u32_t;

typedef signed int      s32_t;

此外还有一点:一般情况下C语言的结构体struct4字节对齐的,但是在处理数据包的时候,LwIP使用的是通过结构体中不同数据的长度来读取相应的数据的,所以,一定要在定义struct的时候使用_packed关键字,让编译器放弃struct的字节对齐。LwIP也考虑到了这个问题,所以,在它的结构体定义中有几个PACKED_FIELD_xxx宏,默认的时候这几个宏都是空的,可以在移植的时候添加不同的编译器所对应的_packed关键字。比如在SkyeyeARM7)上对应gcc编译器的定义:

#define PACK_STRUCT_FIELD(x) x __attribute__((packed))

#define PACK_STRUCT_STRUCT __attribute__((packed))

#define PACK_STRUCT_BEGIN

#define PACK_STRUCT_END

4.3 sys_arch 操作系统相关部份

       sys_arch.[ch]中的内容是与OS相关的一些结构和函数,主要可以分为四个部份:

(1)              sys_sem_t 信号量

LwIP中需要使用信号量通信,所以在sys_arch中应实现信号量结构体和处理函数:

struct  sys_sem_t

       sys_sem_new()                //创建一个信号量结构

       sys_ sem _free()                     //释放一个信号量结构

       sys_ sem _signal()            //发送信号量

       sys_ arch_sem _wait()      //请求信号量

由于µC/OSII已经实现了信号量OS_EVENT的各种操作,并且功能和LwIP上面几个函数的目的功能是完全一样的,所以只要把µC/OSII的函数重新包装成上面的函数,就可以直接使用了。

(2)              sys_mbox_t 消息

LwIP使用消息队列来缓冲、传递数据报文,因此要在sys_arch中实现消息队列结构sys_mbox_t,以及相应的操作函数:

sys_mbox_new()                //创建一个消息队列

sys_mbox_free()                //释放一个消息队列

sys_mbox_post()                //向消息队列发送消息

sys_arch_mbox_fetch()       //从消息队列中获取消息

µC/OSII同样实现了消息队列结构OSQ及其操作,但是µC/OSII没有对消息队列中的消息进行管理,因此不能直接使用,必须在µC/OSII的基础上重新实现。为了实现对消息的管理,我们定义了以下结构:

typedef struct {

           OS_EVENT*   pQ;

           void* pvQEntries[MAX_QUEUE_ENTRIES];

} sys_mbox_t

在以上结构中,包括OS_EVENT类型的队列指针(pQ)和队列内的消息(pvQEntries)两部分,对队列本身的管理利用µC/OSII自己的OSQ操作完成,然后使用µC/OSII中的内存管理模块实现对消息的创建、使用、删除回收,两部分综合起来形成了LwIP的消息队列功能。

(3)              sys_arch_timeout 函数

LwIP中每个与外界网络连接的线程都有自己的timeout属性,即等待超时时间。这个属性表现为每个线程都对应一个sys_timeout结构体队列,包括这个线程的timeout时间长度,以及超时后应调用的timeout函数,该函数会做一些释放连接,回收资源的工作。如果一个线程对应的sys_timeout为空(NULL),说明该线程对连接做永久的等待。

timeout结构体已经由LwIP自己在sys.h中定义好了,而且对结构体队列的数据操作也由LwIP负责,我们所要实现的是如下函数:

struct sys_timeouts * sys_arch_timeouts(void)

这个函数的功能是返回目前正处于运行态的线程所对应的timeout队列指针。timeout队列属于线程的属性,因此是OS相关的函数,只能由用户实现。

(4)              sys_thread_new 创建新线程

LwIP可以是单线程运行,即只有一个tcpip线程(tcpip_thread),负责处理所有的tcp/ucp连接,各种网络程序都通过tcpip线程与网络交互。但LwIP也可以多线程运行,以提高效率,降低编程复杂度。这时就需要用户实现创建新线程的函数:

void sys_thread_new(void (* thread)(void *arg), void *arg);

µC/OS II中,没有线程(thread)的概念,只有任务(Task)。它已经提供了创建新任务的系统API调用OSTaskCreate,因此只要把OSTaskCreate封装一下,就可以实现sys_thread_new。需要注意的是LwIP中的thread并没有µC/OS II中优先级的概念,实现时要由用户事先为LwIP中创建的线程分配好优先级。

4.4 lib_arch 中库函数的实现

       LwIP协议栈中用到了8个外部函数,这些函数通常与用户使用的系统或编译器有关,因此留给用户自己实现。如下:

u16_t htons(u16_t n);     //16位数据高低字节交换

u16_t ntohs(u16_t n);

u32_t htonl(u32_t n);       //32位数据大小头对调

u32_t ntohl(u32_t n);

int strlen(const char *str);      //返回字符串长度

int strncmp(const char *str1, const char *str2, int len);  //字符串比较

void bcopy(const void *src, void *dest, int len);  //内存数据块之间的互相拷贝

void bzero(void *data, int n);              //内存中指定长度的数据块清零

       前四个函数通常由用户自己实现。SkyeyeARM7)中,由于使用了gcc编译器,gcclib库里已经有了后四个函数。而ez80的编译器函数库中缺少bcopybzero两个,需要自己编写。用户在其它CPU上实现时应根据自己的编译器来决定。

      

4.5 网络设备驱动程序

      ez80开发板自带的网络芯片为RealTek8019as芯片,这是ISA 10BASET的以太网芯片,与Ne2k兼容。而我们在AT91模拟器Skyeye中所仿真的网络芯片也是Ne2k,所以目前实现的网络设备驱动是针对Ne2k的,其它类型的网络芯片驱动可以在LwIP的网站上找到。LwIP的网络驱动有一定的模型,/src/netif/ethernetif.c 文件即为驱动的模板,用户为自己的网络设备实现驱动时应参照此模板。

LwIP中可以有多个网络接口,每个网络接口都对应了一个struct netif,这个netif包含了相应网络接口的属性、收发函数。LwIP调用netif的方法netif->input()netif->output()进行以太网packet的收、发等操作。在驱动中主要做的,就是实现网络接口的收、发、初始化以及中断处理函数。驱动程序工作在IP协议模型的网络接口层,它提供给上层(IP层)的接口函数如下:

//网卡初始化函数

void ethernetif_init(struct netif *netif)

//网卡接收函数,从网络接口接收以太网数据包并把其中的IP报文向IP层发送

//在中断方式下由网卡ISR调用

void ethernetif_input(struct netif *netif)

//网卡发送函数,给IP层传过来的IP报文加上以太网包头并通过网络接口发送

err_t ethernetif_output(struct netif *netif, struct pbuf *p, struct ip_addr *ipaddr)

//网卡中断处理函数ISR

void ethernetif_isr(void);

以上的函数都可以分为协议栈本身的处理和对网络接口硬件的操作两部份,但硬件操作是对上层屏蔽的,具体参见RTL8019asDM9008Ne2k网络芯片的数据手册。驱动程序可以到SkyeyeLwIP的网站下载。

5 应用实例的建立和测试

       做完上面的移植修改工作以后,就可以在µC/OSII中初始化LwIP,并创建TCPUDP任务进行测试了。这部份完全是C语言的实现,因此这部份在ez80ARM7上基本都是一样的。值得注意的是LwIP的初始化必须在µC/OSII完全启动之后也就是在任务中进行,因为它的初始化用到了信号量等OS相关的操作。关键部份的代码和说明如下:

       main(){

OSInit();

OSTaskCreate(lwip_init_task, &LineNo11, &lwip_init_stk[TASK_STK_SIZE-1], 0);      

OSTaskCreate(usr_task,&LineNo12,&usr_stk[TASK_STK_SIZE-1],1);

          OSStart();

}

主程序中创建了lwip_init_task初始化LwIP任务(优先级0)和usr_task用户任务(优先级1)。lwip_init_task任务中除了初始化硬件时钟和LwIP之外,还创建了tcpip_thread(优先级5)和tcpecho_thread(优先级6)。实际上tcpip_thread才是LwIP的主线程,多线程的Berkley API也是基于这个线程实现的,即上面的tcpecho_thread线程也要依靠tcpip_thread线程来与外界通信,这样做的好处是编程简单,结构清晰。

实用Berkley API实现的tcpecho_thread是一个TCP echo服务器,监听7号端口,程序框架如下:

void tcpecho_thread(void *arg){

conn = netconn_new(NETCONN_TCP);  //创建新的连接标识

netconn_bind(conn, NULL, 7);         //绑定到7号端口

netconn_listen(conn);                      //开始监听端口

  while(1){

       newconn = netconn_accept(conn);        //接收外部到来的连接

       buf = netconn_recv(newconn)              //获取数据

       …….                                                //处理数据

       netconn_write(newconn, data, len, NETCONN_COPY); //发送数据

       netconn_delete(newconn);                   //释放本次连接

}

       }

       编译运行后,用ping ip地址命令可以得到ICMP reply响应。用telnet ip地址 7(登录7号端口)命令可以看到echo server的回显效果。说明ARPICMPIPTCP协议都已正确运行。

 

 

 

目录
相关文章
|
6天前
|
网络协议 网络架构 数据格式
TCP/IP基础:工作原理、协议栈与网络层
TCP/IP(传输控制协议/互联网协议)是互联网通信的基础协议,支持数据传输和网络连接。本文详细阐述了其工作原理、协议栈构成及网络层功能。TCP/IP采用客户端/服务器模型,通过四个层次——应用层、传输层、网络层和数据链路层,确保数据可靠传输。网络层负责IP寻址、路由选择、分片重组及数据包传输,是TCP/IP的核心部分。理解TCP/IP有助于深入掌握互联网底层机制。
28 2
|
23天前
|
人工智能 弹性计算 自然语言处理
AI奇思妙想之旅 —— 操作系统智能助手OS Copilot
AI奇思妙想之旅 —— 操作系统智能助手OS Copilot
175 1
|
1月前
|
安全 Linux 网络安全
龙蜥Anolis OS:国产操作系统的逆袭之路,它将如何引领中国IT业翻天覆地的变化?揭秘未来数字世界的心脏!
【8月更文挑战第13天】在信息化时代,操作系统作为软硬件桥梁至关重要。国产操作系统如龙蜥Anolis OS,凭借其开源、灵活与安全特性,成为了探索未来发展的重要样本。基于Linux内核的Anolis OS不仅技术先进,生态完善,还针对国内用户习惯进行了优化,并通过如SELinux等安全技术确保系统稳固。其开源属性促进了社区的活跃发展与技术创新。随着政策支持和产业链成熟,Anolis OS正引领国产操作系统走向更广阔的应用领域,加速软硬件生态系统的成型,展现出无限发展潜力。
112 2
|
15天前
|
安全 Linux 网络安全
龙蜥Anolis OS:国产操作系统的逆袭之路,它将如何引领中国IT业翻天覆地的变化?揭秘未来数字世界的心脏!
【9月更文挑战第4天】在信息化时代,操作系统作为计算机系统的核心,连接着上层软件与底层硬件。随着全球化及地缘政治的影响,国产操作系统愈发重要。龙蜥Anolis OS作为佼佼者,基于Linux内核,兼具开源、灵活与安全特性,针对国内用户优化,支持多种编码标准和汉字输入法,提升中文用户体验。其采用角色访问控制、SELinux等技术,保障系统安全。Anolis OS还拥有活跃的开源社区,促进功能完善与创新。随着国家政策扶持和产业链协同,Anolis OS正引领国产操作系统迈向更广阔的应用领域,推动软硬件生态系统的成熟,成为全球多元化计算生态的重要组成部分。
45 0
|
15天前
|
存储 自然语言处理 搜索推荐
探索OS Copilot——那款让你瞬间变身超级用户、编程如呼吸般自然、文件管理如同魔法般的神奇操作系统辅助神器!
【9月更文挑战第4天】“OS Copilot”是一款高效的操作系统辅助软件,通过智能化手段简化电脑使用,涵盖智能命令建议、代码片段生成、文件管理及任务自动化等强大功能。其简洁的界面与友好的用户体验使其成为提升生产力的理想选择,无论是专业人士还是普通用户都能从中受益。从安装到实际应用都非常流畅,能显著提升工作效率,是优化数字生活的得力助手。
25 0
|
2月前
|
弹性计算 运维
阿里云操作系统智能助手OS Copilot实验测评报告
**OS Copilot 产品体验与功能反馈摘要** 运维人员发现OS Copilot易上手,文档清晰,助其高效排查故障(8/10分)。愿意推荐并参与开源开发。亮点在于知识问答,能快速筛选答案。相较于竞品,优点是新手友好、文档清晰,但功能扩展性待增强。期望增加系统错误排查与解决方案,并集成ECS等,以优化系统安装流程。
阿里云操作系统智能助手OS Copilot实验测评报告
|
1月前
|
存储 自然语言处理 搜索推荐
【颠覆你的数字生活!】探索OS Copilot——那款让你瞬间变身超级用户、编程如呼吸般自然、文件管理如同魔法般的神奇操作系统辅助神器!
【8月更文挑战第8天】OS Copilot是一款新兴的操作系统辅助软件,通过智能化手段简化电脑使用,从办公到开发全面赋能。安装简易,启动即有引导教程。其智能命令建议功能,可在命令行输入时提供后续选项及其说明,特别适合Linux用户。内置代码片段生成器,根据需求或代码框架自动生成代码,大幅提升开发效率。文件管理助手支持批量操作且可预览结果,降低误操作风险。任务自动化功能便于设置重复性工作流程,如定时备份。搜索功能强大,支持自然语言查询。尽管尚有改进空间,OS Copilot已是提升生产力的得力助手。
94 5
|
2月前
|
弹性计算 Kubernetes 开发者
操作系统OS Copilot 产品体验评测
OS Copilot体验摘要: 开发者评价OS Copilot在软件开发和系统维护中提供帮助。新人易上手,界面直观,但高级功能说明不足。工具在编程时给出智能建议,提升效率,专长于操作系统任务。相比同类产品,如GitHub Copilot,OS Copilot在OS相关建议上更专业,但特定场景准确性待提高。期望增加更多操作系统支持及自动错误排查功能。适合与ACK智能助手等产品联动,提供云环境全面支持。
200 1
操作系统OS Copilot 产品体验评测
|
2月前
|
弹性计算 运维 开发者
阿里云操作系统智能助手OS Copilot产品测评报告
阿里云的OS Copilot是面向开发者和运维的智能助手,提升云上应用的构建和管理效率,支持Linux、Windows等操作系统。它具有智能化、自动化特点,帮助进行系统管理和故障排查。用户反馈上手简单,文档清晰,但在高级功能和特定场景支持上还有待加强。用户对其辅助命令执行功能赞赏有加,期望能支持更多操作系统和提供系统级错误排查工具。与ECS、Workbench等集成可提升管理与开发效率。适合云基础设施新手和前端开发者,但在复杂任务中可能需要改进。
362 5
阿里云操作系统智能助手OS Copilot产品测评报告
|
1月前
|
定位技术 开发者 Anolis
【开发者评测】操作系统OS Copilot获奖名单
操作系统OS Copilot获奖名单新鲜出炉!
166 2