Loading
新闻中心库存中心价格中心PDF中心图片中心
ICGLE 电子技术支持中心,为电气人员和相关企业提供全面的信息(IC技术\IC库存\IC图片\PDF资料等)服务
总线设计PCB设计单片机DSPARM
JAVA驱动设计C/C++汇编J2EE网络协议中间件技术嵌入式系统无线通信通信网络模拟技术接口电路显示光电传感与控制EDA/PLD
气流体控制电子产品电机及工具通信设备仪器仪表电线电缆建筑电气低压电器高压电器电源工控自动化广电设备医疗器械More..
电源设计仪器仪表技术专递电路图片电子专栏储存技术汽车电子测量测试音响技术家用电器
成功方案市场分析行业标准应用前沿芯片应用综合专区
使用 ADS 移植mC/OS-II的实例分析
来源:转载   作者:清华大学智能技术与系统国家重点实验室 李
字体大小:[大][中][小]


  本文介绍了使用 ARM 公司提供的 ADS 开发工具,进行移植mC/OS-II 的工作。结合基于 StrongARM 评估板的硬件结构,对移植工作中的若干要点做了详细分析。最后,给出了移植体会和程序技巧分析。
 
选择开发工具

  在嵌入式系统设计中,开发工具的选取是一个重要的考虑因素,通常这是与开发项目的需求和应用背景相关。一般嵌入式开发工具包含用于目标系统的交叉编译器、连接器、调试器以及辅助处理用的二进制文件分析工具等。

  目前可以用来编译链接产生 ARM 处理器执行代码的开发工具主要有如下几类:

1. ARM 公司提供的 ARM Developer Suite 集成开发环境

  主要工具有 armasm、armcc、armlink、fromelf 等。

2. GNU 组织提供的 tool chain for arm

  主要工具有 arm-elf-gcc、arm-elf-gdb、arm-elf-objcopy 等

3. Microsoft公司提供的 eMbedded Visual Tools

  主要工具有 clarm、clthumb、c2_arm、link、lib等

  这里我们选用 ARM 公司提供的 ADS 下的工具集来编译我们的程序和链接目标代码并最终生成可执行的二进制映像。这里介绍一下主要会用到的一些工具:

armasm.exe : 汇编文件编译器
armcc.exe : C 文件编译器
armlink.exe : 目标文件连接器
fromelf.exe : 用于将 axf 或者 elf 格式转换成其他格式的文件,例如二进制映像。
armprof.exe : 对调试过程中生成的 profiling 记录文件做分析用的工具软件

启动代码

  由于板子的 0x0 地址处是 32M 的Flash ROM,因此在板子加电后,会从 Flash 中顺序执行启动代码。为了能使得mC/OS-II 运行,启动代码需要完成如下工作:

1. 设置 异常向量表,即在 0x0-0x1c 位置放置7条跳转指令(其中 0x14 为空)

2.分别实现每种异常的处理程序,其中包括 Reset_Handler、Undefined_Handler、SWI_Handler、Prefetch_Handler、Abort_Handler、IRQ_Handler、FIQ_Handler。

3. 程序从 Reset_Handler 进入后,需要首先进行相关硬件的初始化操作,例如 初始化SDRAM、CPU speed、Interrupt Controller、UART、timer 等。

4. 建立每种异常状态下的系统堆栈,为了简单起见可以只在 svc 态 和 irq 态下建立堆栈:setup_svc_stack,setup_irq_stack。

5. 强制 ARM 处理器状态转换为 svc 管理态。

6. 跳转到mC/OS-II 代码的 main 入口,实际上是编译链接后产生的 __main 入口。

时钟与中断处理

时钟控制逻辑

  在图1中,有4种和系统时钟相关寄存器,它们的含义如下:

● OSCR: 一个自动递增计数的 32 位计数器。
● OSMR3-0: 4 个 32 位的匹配寄存器,当 OSCR 的值匹配时产生中断。
● OSSR: 状态寄存器,当 OSCR 和 OSMR 匹配时,会对 OSSR 做标志。
● OIER: 使能寄存器,表示当匹配发生时,允许在 OSSR 设置一个标识位。

  OSCR 在自动累加的过程中,与OSMR里面设定的那些匹配寄存器进行匹配,发现有匹配的事件时,就会对 OSSR 中的相应位置设一个标志位“1”,表示OSCR与对应的OSMR 发生了匹配。当然这个匹配发生的前提是发生匹配的那个OSMR在OIER中的相应位被使能,否则OSMR中的设置将不起作用。

系统时钟初始化流程

  mC/OS-II 中创建的第一个任务将负责启动时钟节拍,时钟的初始化设置流程如下:

1) 设置 OSMR0 = x ,表示 初始化 OSMR0,即当计数器为x时发生匹配
2) 设置 OSSR = 0xf ,表示 清除所有已经发生的匹配,写“1”清除
3) 设置 OIER = OIER_EO ,表示 使能 OSMR0 来产生匹配
4) 设置 OSCR = 0 ,表示 初始化计数器的开始值 为 0

系统时钟中断复位

1) 清除 OSSR 中的相应位,即向发生匹配的OSMR的那个对应位写“1”

2) 设置 OSCR = 0 ,表示 继续初始化计数器的值为 0

中断控制器相关的寄存器

● ICPR: 中断标示寄存器,表示了当前系统正处于激活状态的中断源。
● ICMR: 中断屏蔽寄存器,用来屏蔽相应位的中断。
● ICLR: 中断级别设置寄存器,设定报告中断的级别是 IRQ 或者是 FIQ 。
● ICIP: IRQ 级别的中断源寄存器,用来标识 IRQ 中断发生的源设备。
● ICFP: FIQ 级别的中断源寄存器,用来标识 FIQ 中断发生的源设备。

中断控制器初始化流程

1) 设置 ICMR 屏蔽位为不屏蔽时钟中断 OSMR0 (相应位写“1”)

2) 设置 ICLR 为都报告为 IRQ 级别(所有位写“0”)

移植工作总结

难点分析

  移植mC/OS-II 到 StrongARM 的芯片上,基本上和移植到 ARM7 的芯片例如S3C4510,AT91x等工作类似,因为所有的ARM处理器都共享ARM通用的基础体系结构,这使得移植工作变得相对简单,其中绝大部分工作都集中在 os_cpu_a.S 文件的移植,这个文件的实现集中体现了所要移植到处理器的体系结构和mC/OS-II 的移植原理;在这个文件里,最困难的工作主要是在 OSIntCtxSw 和 OSTickISR 这两个函数的实现上。因为它们的实现是和移植者的移植思路以及相关硬件定时器、中断寄存器的设置有关。在实际的移植工作中,这两个地方也是比较容易出错的地方。

  OSIntCtxSw 最重要的作用就是它完成了在中断ISR中直接进行任务切换,从而提高了实时响应的速度。它发生的时机是在 ISR 执行到 OSIntExit 时,如果发现有高优先级的任务因为等待的 time tick 到来获得了执行的条件,这样就可以马上被调度执行,而不用返回被中断的那个任务之后再进行任务切换,因为那样的话就不够实时了。

  实现 OSIntCtxSw 的方法大致也有两种情况:一种是通过调整 sp 堆栈指针的方法,根据所用的编译器对于函数嵌套的处理,通过精确计算出所需要调整的 sp 位置来使得进入中断时所作的保存现场的工作可以被重用。这种方法的好处是直接在函数嵌套内部发生任务切换,使得高优先级的任务能够最快的被调度执行。但是这个办法需要和具体的编译器以及编译参数的设置相关,需要较多技巧。
另一种是设置需要切换标志位的方法,在 OSIntCtxSw 里面不发生切换,而是设置一个需要切换的标志,等函数嵌套从进入OSIntExit => OS_ENTER_CRITICAL() => OSIntCtxSw() => OS_EXIT_CRITICAL() => OSIntExit退出后,再根据标志位来判断是否需要进行中断级的任务切换。这种方法的好处是不需要考虑编译器的因素,也不用做计算,但是从实时响应上不是最快,不过这种方法实现起来比较简单。

  在中断态下进行任务切换,需要特别说明的一个问题是如何获得被中断任务的 lr_svc 。因为进入中断态后,lr 变成了lr_irq ,原来任务的 lr_svc 无法在中断态下获得,这样要得到lr_svc,就必须在中断 ISR 里面进行一次cpu mode强制转换,即对CPSR赋值为0x000000d3,只有返回到svc态之后才能得到 原来任务的lr,这个对于任务切换很重要。还有一个需要留意的问题是在强制CPSR变成svc态之后,SPSR 也会相应地变成 SPSR_irq,这样就需要在强制转变之前保存 SPSR ,也就是被中断任务中断前的 CPSR 。
移植中使用的编程技巧

  ADS 编译器在编译 C 语言的程序时,如果程序中使用了 main 函数,则编译器将自动添加如下代码,完成初始化堆栈和C库等工作,工作流程如下:

1> 将执行文件中的 RO 段和 RW 段从 load address 复制到 execution address
2> 初始化 ZI 区域,用 0 来初始化变量
3> 跳转到 __rt_entry 执行如下 4 个调用
3.1> 调用 __rt_statckheap_init ,建立程序的堆和栈
3.2> 调用 __rt_lib_init ,初始化程序用到的 C 库,并为 main 传递参数
3.3> 调用 main ,即用户程序的入口
3.4> 调用 exit

  因为系统复位后,在启动代码中已经设置了系统堆栈,同时也不需要使用C库,因此可以从 __rt_entry 处直接跳转到mC/OS-II 的代码中,即直接执行 main 函数,可以用新的 __rt_entry 来作为链接的目标入口。

IMPORT main
EXPORT __rt_entry
__rt_entry
b main
这样在启动代码的最后,加入一条跳转语句:
bl __main

  __main 入口是用户程序执行的真正入口,我们利用 ARMCC 编译 C 里面的 main 入口以求得到 1> 和 2> 的代码,使得可以支持全局变量。否则的话,必须自己来实现全局变量的初始化或者把这些初始化操作放到函数内部来实现。

  另外一个非常有用的编程技巧是通过串口实现自己的 printf 输出。 如果使用ARMCC编译器的 semihosting 的话,会把 printf 通过 target 的 swi 0x123456 输出。如果已经实现的 serial_putchar 之类的函数,那么可以用它来实现 fputc 接口,也就是低级的输出函数,这样就可以使用 printf 来输出了,详细的做法在 ADS 安装目录下面的文档里可以找到,这里就不再赘述。
Upload by 小刘(2007-3-2)
IC
IC
推荐技术文章:
·μC/OSII任务创建和销毁的用户接....
·一个uCOS ARM系统的启动过程分....
·基于ARM7 μC/OSII的数据采....
·嵌入式操作系统UC/OSII的内核实....
·u-boot移植日记1..
·u-boot移植日记2..
·u-boot移植日记3..
·u-boot移植日记4..
·如何在uC/GUI中实现窗体半透明效....
·μC/OS-Ⅱ在ATmega128上....
·嵌入式实时操作系统μC/OS-II及....
·μC/OS-II任务栈处理的改进设计..
·基于μC/OS-II的VG2以太网和....
·μC/OS-Ⅱ在ARM7上移植方法的....
·μC/OS-ll中任务调度算法的改进..
·基于uC/OS-II的低速率语音编码....
·μC/OSII嵌入式操作系统在机电控....
·基于μC/OS-Ⅱ的变频器变结构控制....
·实时操作系统μC/OS-II在MCF....
·嵌入式实时操作系统μC/OS- II....
·嵌入式系统μC/OS-II在LPC2..
·uC/ OS - II 在ARM系统..
·嵌入式系统中入侵检测的设计与实现
·基于AT91R40008的嵌入式GP..
·uC/OS-II在ARM上的移植体会
·uC/OS II针对TMS320C3..
·μC/OS的应用和扩展
·三种嵌入式操作系统的分析与比较
·嵌入式操作系统Nucleus下触摸屏..
·嵌入式实时操作系统Nucleus P..
·嵌入式实时操作系统ECOS在S3C2..
·如何构建eCos嵌入式系统
·使用图形化的开发环境——LabVie..
·网络只发不收怎么解决
·量身订制的DSP元件设计策略
·达芬奇技术助力,ISS加速实现车辆事..
·支持蓝牙和无线USB的单芯片解决方案
·TI、联合信源携手共推业界首款支持A..
·ARM公司推出三维图形系统解决方案
·集成电路应用与抗辐射加固(上)
IC
IC

©2007 版权归ICGLE所有   页面执行时间:62.500毫秒