μC/OS-II的多任务信息流与CAN总线驱动 (2)

2010-07-26 16:42:56   作者:张海峰 段登平 陈文芗   来源:单片机及嵌入式系统应用   

关键字:μC/OS-Ⅱ RTOS嵌入式系统 设备驱动 中断处理程序(ISR) 进程调度

  2 μC/OS-II的中断处理

  μC/OS-II中,中断服务程序一般用汇编语言来写。以下是中断服务程序的示意代码。

  用户中断服务程序:

  保存全部CPU寄存器;

  调用OSIntEnter或OSIntNesting直接加1;

  执行用户代码做中断服务;

  调用OSIntExit;

  恢复所有CPU寄存器;

  执行中断返回指令;

  这里μC/OS-II提供了两个ISR与内核的接口函数:OSIntEnter和OSIntExit。OSIntEnter通知μC/OS-II内核,中断服务程序开始运行了。实际上,此函数做的工作是把一个全局变量OSIntNesting加1。在x86等有累加指令的CPU中,可以用指令代替OSIntEnter:

  INC BYTE PTR OSIntNesting

  此中断嵌套计数器可以确保所有中断处理完成后再作任务调度。另一个接口函数OSIntExit则通知内核,中断服务已结束。根据相应情况,返回被中断点(可能是一个任务或者被嵌套的中断服务程序)或由内核作任务调度。

  用户编写的ISR必须被安装到某一位置,以便中断发生后,CPU根据相应的中断号运行准确的服务程序。许多实时操作系统都提供了安装、卸载中断服务程序的API接口函数,有些成熟的RTOS甚至对中断控制器的管理都有相应的API函数。但 μC/OS-II内核没有提供类似的接口函数,需要用户在对应的CPU移植中自己实现。这些接口函数与具体的硬件环境有关,接下来PC体系下的中断处理对此有详细的说明。

  3 PC体系下的中断

  X86系列的处理器可支持256个中断,并用向量表的方法来关联每个中断和相应ISR的位置。在实模式下,中断向量表(IVT)存于内存的低端1K。每个向量表条目占4字节,保存一个ISR的段地址和偏移信息。PC系统使用两个级联的可编程中断控制器82C59A。一个82C59A能连接8个硬件中断,编号为IRQ0~IRQ7。 PC总共可管理15个外部中断源,PC的中断控制器如图4所示。(关于82C59A的详细使用可参见有关资料。)

  在μC/OS下,CAN总线I/O端口中断向量设置伪代码:

  void CanInitHW(UI segment,BYTE Irq0,BYTE Irq1){
  保存原有的中断向量
  保存掩码寄存器的值
  使82C59A的掩码寄存器(0x21)各位置1,关闭中断输入
  关闭CPU中断
  设置新的中断向量
  正在服务的中断禁止再次响应服务(假定当前服务中断是IRQ5)
  开CPU中断
  清除82C59A的掩码寄存器(0X21、0XA1)各位,开启中断输入
  }

  4 信号量与缓冲队列支持下的CAN总线驱动

  前面介绍了μC/OS-II内核下多任务调度的关键技术、中断与PC体系下中断的一般方法。又以82C59A的中断5(IRQ5)、0x0D中断向量为例,介绍了中断服务子程序的重新分配和响应SJA1000控制器收发的中断服务子程序。

  下面介绍信号量配合下的环形缓冲队列与中断处理程序之间的关系问题,这也是设备驱动部分的核心内容。

  ERTOS的驱动程序与其它操作系统有所不同。比如Windows、Unix、Solaris、Linux等操作系统弱化了设备的概念,用户进程对设备的使用可以通过文件系统来完成。然而,在μC /OS-II上开发CAN总线驱动程序没有那么严格,只要满足设备在连续的CPU时间上使用时不发生时间重叠就可以了。

  串行设备或者其它字符型设备都存在外设处理速度和CPU速度不匹配的问题,所以需要建立相应的缓冲区。向CAN口发送数据时,只要把数据写到缓冲区,然后由SJA1000控制器逐个取出往外发。从CAN口接收数据时,往往等收到若干个字节后才需要CPU进行处理,所以这些预收的数据可以先存于缓冲区。缓冲区可以设置收到若干个字节后再中断CPU,这样避免了因为CPU的频繁中断而降低系统的实时性。

μC/OS-II的多任务信息流与CAN总线驱动

  在对缓冲区读写的过程中,经常会遇到想发送数据时,发送缓冲已满;想去读时,接收缓冲却是空的。对于用户程序端,可以采用查询工作方式,即放弃无法读写的操作,然后再频繁地去尝试这个操作直到成功,这样程序效率显然降低。如果引入读、写两个信号量分别对缓冲区两端的操作进行同步,问题将迎刃而解。用户任务想写但缓冲区满时,在信号量上睡眠,让CPU运行别的任务,待ISR从缓冲区读走数据后唤醒此睡眠的任务;类似地,用户任务想读但缓冲区空时,也可以在信号量上睡眠,待外部设备有数据来了再唤醒。由于μC/OS-II的信号量提供了超时等待机制,CAN口当然也具有超时读写能力。

  带缓冲和信号量的CAN口接收和发送部分见本刊网络补充版(http://www.dpj.com.cn)。

  接口函数总结如下。

void CanInitHW(UI segment,BYTE irq0,BYTE IRQ1)
/*设置SJA1000控制器端口中断向量*/
int canReleaseHW() /* 清除SJA1000控制器端口中断向量*/
int canSendMsg( CANBYTE port, MSG_STRUCT msg)
/* 向定制SJA1000控制器端口发送数据*/
int canReceiveMsg( CANBYTE port, MSG_STRUCT msg_ptr)
/*从定制SJA1000控制器端口接收数据
int canConfig( CANBYTE port, CAN_STRUCT can)
/*初始化和配置SJA1000控制器 */
int canNormalRun( CANBYTE port )
/*设置SJA1000正常(Normal)运行模式 */
int canReset( CANBYTE port )
/* SJA1000控制器端口重新设置,缓冲区置位0xff*/
CANBYTE can0r( CANBYTE addr)
/*读取SJA1000控制器端口0的定制寄存器的值 */
CANBYTE can1r( CANBYTE addr)
/*读取SJA1000控制器端口1的定制寄存器的值 */

  接收和发送数据缓冲区数据结构定义:

typedef struct {
INT16U RingBufRxCtr; /* 接收缓冲中字符数目 */
OS_EVENT RingBufRxSem; /* 接收信号量 */
INT8U RingBufRxInPtr; /* 接收缓冲中下一字符的写入位置 */
INT8U RingBufRxOutPtr; /* 接收缓冲中下一待读出字符的位置 */
INT8U RingBufRx[CAN_RX_BUF_SIZE]; /* 接收环形缓冲区*/
INT16U RingBufTxCtr;
/* 发送缓冲中字符数目 */
OS_EVENT *RingBufTxSem; /* 发送信号量 */
INT8U *RingBufTxInPtr;
/* 发送缓冲中下一字符的写入位置 */
INT8U *RingBufTxOutPtr;
/* 发送缓冲中下一待读出字符的位置 */
INT8U RingBufTx[CAN_TX_BUF_SIZE]; /* 发送环形缓冲区*/
} CAN_RING_BUF;

  结 语

  本文是在嵌入式计算机技术领域的应用背景下提出的,整个工程开发结束以后,系统正常运作时间超过27天。希望本文的提出对开发嵌入式操作系统的技术人员能有所帮助,同时也希望同一领域的开发人员共同探讨、共同发展。

[1] [2]
分享到:
相关阅读
WindowsCE.Net下基于流驱动的AD驱动设计与实现 2012-03-24
基于VxBus设备驱动程序架构的设备驱动开发 2010-11-17
基于嵌入式应用的单芯片USB方案 2010-10-20
基于Linux平台的FPGA驱动开发 2010-09-05
44-如何编写嵌入式Linux的设备驱动(三) 2010-01-07
43-如何编写嵌入式Linux的设备驱动(二) 2010-01-07
42-如何编写嵌入式Linux的设备驱动(一) 2010-01-07
基于嵌入式Linux的矩阵键盘驱动程序开发 2009-04-30
编辑:金海
本文引用地址: http://www.eeworld.com.cn/qrs/2010/0726/article_2874.html
[发表评论]
[加入收藏]
[告诉好友]
[打印本页]
[关闭窗口]
[返回顶部]
[RSS订阅]
小广播
最热点击
专栏
向农,EEWORLD副总编。被英特尔董事长贝瑞特称为“中国可与之对话的两名记者之一”

【详细】

总编随笔
汤宏琳,人皆称为“汤汤”,电子工程世界高级编辑。随着EEWORLD一起成长。

【详细】

汤汤手记
今年,是中国集成电路产业丰收的一年,相比较往年都有大幅提升。

【详细】

凯哥博客
论坛精华
精选博文