当前位置:文档之家› ucos操作系统在ARM上的移植

ucos操作系统在ARM上的移植

UC/OS-II 嵌入式系统在ARM 上的移植UC/OS-II 操作系统是一款完全公开的源代码,它非常精简,整个操作系统的代码只有几千行,是专门针对于嵌入式开发而产生的一款代码。

它有几个特点,分别是可移植性(Portable )、可固化(ROMable )、可裁剪(Scalable )、多任务、可确定性、任务栈、系统服务、中断管理、稳定性可靠性。

UC/OS-II 主要就是一个内核,由ANSIC 语言编写而成。

负责任务管理和任务调度,没有文件系统和界面系统。

它的代码是公开的,系统的实时性强、移植性好、可多任务。

UC/OS-II 作为基于优先级的抢占式多任务的实时操作系统,包含了实时内核、任务管理、时间管理、任务间通信同步和内存管理的功能。

它使得任务的独立性,不相互干涉,非常的准时和高效,且易于设计和扩展。

UO/OS-II 共有16个内核文件,11个与CPU 类型无关,就是说可以直接使用不需要修改。

还有3个内核文件与CPU 有关系,要根据需要作出相应的改动。

剩下的两个内核文件和具体的应用有关。

如图所示UC/OS-II 的16个内核文件的层次。

μC/OS -II 内核文件软件 硬件多任务操作的核心是系统调度器,利用TCB来管理任务调度功能。

它的主要功能是保存任务的当前态、优先级、等待事件、代码起始地址、初始堆栈指针等。

程序的设计关键就是确定划分多任务的问题,以及任务优先级和任务通信。

优先级的意思是每个任务都是无限循环的,有运行态度、就绪态、休眠态、挂起态和中断五种状态。

当有高一级优先级的任务就绪后,低优先级立即停止运行,转为挂起态或就绪态。

这就是可剥夺型的内核。

当中断一个高优先级任务,中断时挂起,中断结束后任务继续运行,并立即剥夺低优先级的任务。

对于这种可剥夺型内核,CPU的使用时可以确定的,可优化任务级响应。

在很多单片机或ARM板上很容易就可以移植UC/OS-II。

当然本次设计使用的TQ2440,也可以完美的移植它。

移植程序在网上都可以找得到,所以设计中就不做解释了。

本次设计实现的是串口协议和网口协议组合成的一个数据网关。

其主要的流程图如下所示:如图所示可以很清楚的看出内核操作系统对于整个系统的控制过程,下面我们要介绍一下UC/OS-II里面的一些程序。

用户应用程序任务定义代码如下:void MyTask( void *pdata) //开始定义用户任务{for(;;){...}}void main(){TargetInit(); //完成初始化目标开发板。

OSInit (); //完成初始化UCOS-II。

OSTimeSet(0); //完成初始化系统时基。

OSTaskCreate (MainTask,(void *)0, &MainTaskStk[MainTaskStkLengh - 1], MainTaskPrio); //开始创建系统初始任务。

OSStart (); //整个任务开始。

return 0;}void Task0(void *pdata) //主任务建立:taks0 和task1{#if OS_CRITICAL_METHOD == 3 /* 分配CPU状态寄存器*/ OS_CPU_SR cpu_sr;#endifOS_ENTER_CRITICAL();Timer0Init(); //初始化报时信号ISRInit(); //初始化中断优先级OS_EXIT_CRITICAL();OSPrintfInit(); //用户任务给串口DM9000_init();OSStatInit();while(1){OSPrintf("\nEnter Main Task\n");//测试Dm9000//打印DM9000寄存器OSTimeDly(OS_TICKS_PER_SEC);}}需要注意的是,μC/OS-II的应用程序要使用空闲任务OSTaskldle(),而它是不可删除的系统文件。

下面开始对应用程序进行移植了。

本次设计是在ARM开发板上实现串口、网口数据转换的的功能。

在TQ2440上有串口和网口。

通信过程简单的描述就是PC串口发送数据给ARM板,ARM接到串口数据后,从网口再传给PC。

相反也是同样的道理,由网口发送数据,有ARM 控制由串口发回数据。

首先是TCP/IP协议的移植。

在编写移植程序前,有必要解释一下基本的协议栈作用和意义。

`就常见的网络通信方式一般来说有两种:1、UART-RS232,此时只需要pc上有串口调试助手即可。

2、TCP/IP,这时候和普通pc与pc通信一样可以用socket套接字编程也可用别人写好的软件侦听。

而现在我们要实现的是串口控制单片机与PC机的通信,在这里用PC来代替以太网。

接下来还要介绍一下以太网接口。

以太网技术如今已经相当的成熟了,其相应的网络产品价格低廉、技术完善。

而数据总线如今越来越难以满足人们日益提升的需要,这时以太网控制网络技术得到了快速的发展,并形成了现场总线的新标准。

加上国内大部分局域网是以太网,给予以太网实现现场总线有了雄厚的物质基础。

以太网的接口就是以太网同信的基础,是通信介质通信的中间处理部件,实现报文的发送与接收功能,位于TCP/IP协议栈的数据链路层。

每一个以太网接口(有时候也叫网卡),在连通后就可以随时的发送和接收网络上的数据,执行EEE802.3标准。

TCP/IP对应的ISO结构如图所示:TCP/IP协议栈而单片机与计算机的TCP/IP协议的实现也有很大的不同,原本在计算机里编写的程序可以不考虑代码的大小和效率,但在嵌入式开发板上都要考虑到这些问题。

在操作系统、内存分配、指针、参数传递、协议支持以及硬件接口的设计方面有些不同。

首先就是操系统,嵌入式的特点之一就是简洁高效,有很强的专业功能。

相对而言计算机上的操作系统的侧重点就是兼容性,所以资源要求全面支持,所以很复杂。

其次就是内存上的分配,计算机像windows系统它的内存分配是动态的,而在单片机上却不能同样如此,应为RAM的容量所限,所以其中存放以太网的数据包是固态的。

由于ARM相对于单片机的能力而言有了很大的提升,所以可以突破单片机的一些约束。

如下图所示,TCP/IP协议栈中的内容,从上往下分别对应应用层、传输层、网络层和网络接口层。

TCP/IP协议栈在本次设计中采用Lwip协议栈来实现ARM与以太网的连接。

完成移植后,需要介绍一下以太网的初始化过程和数据收发过程。

LWIP的初始化要在UC/OS-II之后,在其它任务创建之前。

因为LWIP要对以太网协议栈初始化以及新线程的建立,LWIP初始化如图所示。

(3)以太网初始化流程LWIP程序可以实现很多功能,在本设计中没有实现他的全部功能。

而简单的PING通需要DM9000来实现。

以太网的接收是通过中断方式,如果有数据进入以太网中断函数。

其主要任务就是读取和分析数据包。

如果数据有效则在Tcpip_input()函数把数据发送出去,并在LWIP初始化创建的线程中就可以得到此消息。

然后通过ip_input()函数进入传输层后,再把把数据传到应用层。

具体流程如下图所示:以太网数据接收流程以太网数据发送流程为什么要选择DM9000网络驱动器呢?如今嵌入式中大量使用10/100M的以太网卡,实际上并不实用。

因为它不能既满足快速传输速率又满足成本控制。

这时,DM9000作为一种综合低成本的单一快速以太网控制芯片就有了很高的实用价值。

它具有通用的接口,设计简单,可满足不同系统的软件驱动。

DM9000程序的体系结构可以分为网络协议接口、设备接口层、功能层及媒介层。

下面是DM9000的部分驱动程序的编写。

定义DM9000:#include "config.h"#define DM_ADD (*((volatile unsigned short *) 0x20000300))#define DM_CMD (*((volatile unsigned short *) 0x20000304))#define Printf OSPrintfuint8 Buffer[1000]; //定义了一个1000字节的接收发送缓冲区uint8 host_mac_addr[6]= { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; //主机的MAC地址uint8 mac_addr[6] = { 0x0a, 0x1b, 0x2c, 0x3d, 0x4e, 0x5f }; //开发板{ 0a,1b,2c,3d,4e,5f },这实际上是一个软地址,在本设计中不需要特指。

uint8 ip_addr[4] = { 192, 168, 1, 6};uint8 host_ip_addr[4] = { 192, 168, 1, 100 };uint16 packet_len; //接收、发送数据包的长度,单位为字节uint8 arpsendbuf[60]={0xff,0xff,0xff,0xff,0xff,0xff, //以太网目标地址0x0a,0x1b,0x2c,0x3d,0x4e,0x5f, //以太网源地址0x08,0x06, //帧类型: ARP帧0x00,0x01, //硬件类型:以太网0x08,0x00, //协议类型:IP协议0x06, //硬件地址长度:6字节0x04, //协议地址长度:4字节0x00,0x00, //操作:ARP请求0x0a,0x1b,0x2c,0x3d,0x4e,0x5f, //发送端硬件地址192, 168, 1, 6, //发送端协议地址0x00,0x00,0x00,0x00,0x00,0x00, //接收端硬件地址192, 168, 1, 100 //接收端协议地址};#define DM9KS_ID 0x90000A46#define DM9KS_VID_L 0x28#define DM9KS_VID_H 0x29#define DM9KS_PID_L 0x2A#define DM9KS_PID_H 0x2B#define DM9KS_BASE_ADDR_ETH0 0x20000000//nGCS4#define DM9KS_Index (*((volatile unsigned short *)(DM9KS_BASE_ADDR_ETH0 + 0x300)))#define DM9KS_Data (*((volatile unsigned short *)(DM9KS_BASE_ADDR_ETH0 + 0x304)))接下是来对DM9000进行初始化的程序部分:void DM9000_init(void){uint32 i;//Test_DM9000AE();IOSetInit(); //设置中断EINIT7//初始化设置步骤: 1dm9000_reg_write(DM9000_GPCR, 0x01); //设置GPCR(1EH) bit[0]=1,使DM9000的GPIO3为输出。

相关主题