新闻中心

EEPW首页 > 手机与无线通信 > 设计应用 > μC/OSII的CAN驱动程序设计

μC/OSII的CAN驱动程序设计

作者: 时间:2010-11-25 来源:网络 收藏

  编写的CAN中断服务程序应该越短越好,在不影响系统性能的情况下尽量将中断服务任务放到中断服务程序外执行,以便尽早退出FIQ中断模式,从而使节点能够响应新的中断,减少系统中的中断延时。其中,接收中断处理是最占用节点资源的,它不仅需要根据ICAN协议对报文进行解析,还需要执行报文指定的功能,所以必须放到中断服务程序外执行。解决的办法是,通过μC/OSII中的OSTaskCreate()函数建立一个报文处理任务,这个任务由一个请求消息队列函数OSQPend()和一个报文解析处理函数组成。报文处理函数如下:

voidCAN_RMSG_HANDLE(void* ptmr) {
  ptmr = ptmr;
  for( ; ; ) {
  OSQPend(CAN1_Q_RX,0,CAN_Q_ERROR);//根据ICAN协议解析报文实现报文指定功能
  }
}

  如果需要发送CAN报文,首先要查询是否有可用的发送缓冲区:若有则可用就直接发送,无须通过消息队列作为中介,从而提高程序运行效率;若都被锁定,则调用OSQPost()将报文发送到报文发送函数的消息队列MESSAGE_TX中,并执行TX_CNT++操作。

  ② 在繁忙的CAN网络中,节点可能会由于仲裁丢失而无法及时将数据传输,因此必须要对待发送的数据进行存储,等待节点获得总线使用权时再发送出去。LPC2368的CAN控制器有一个三态发送缓冲区,最多能够存储3个报文。若3个缓冲区都处于锁定状态(报文正在等待发送或正处于发送过程),而又有一个报文需要发送,则需要额外的缓冲区先将它存储起来,以待节点获得总线使用权时再发送。

  定义一个指针数组,把建立的消息数据缓冲区的首地址存入这个数组中,然后再调用OSQCreate()函数来创建一个用于存储发送报文的消息队列MESSAGE_TX,最后通过OSTaskCreate()函数建立一个负责发送报文的任务。该任务由一个请求消息队列函数OSQPend()和一个请求信号量函数OSSemPend()组成。报文发送函数如下:

void CAN_MESSAGE_SEND(void*ptmr ) {
  ptmr = ptmr;
  for( ; ; ) {
    S = OSQPend(MESSAGE_TX , 0 , Q_ERROR);
    OSSemPend(CAN_TX_OVER , 0, SEM_ERROR);
    OS_ENTER_CRITICAL( );//进入临界代码区
    SEND_TX_BUFFER( S );
    TX_CNT--;
    OS_EXIT_CRITICAL( );
  }
}

  其中,变量TX_CNT记录MESSAGE_TX中的报文数目。任务向MESSAGE_TX发送一个报文,TX_CNT就加1;报文发送函数成功发送一个报文,TX_CNT就减1。这样,中断服务程序就可以根据TX_CNT来判断是否有向CAN_TX_OVER发送信号量的必要,减少了不必要的冗余操作。

  除非在CAN节点任务中有比将处理好的CAN报文发送出去更重要的任务要做,一般来讲,报文发送任务在节点任务中应该具有最高的优先级,以保证CAN系统的

  ③ LPC2368的最高运行速率可达72 MHz,而CAN最高传输速率为1 Mb/s。一般情况下,即使连续接收到2个报文,CPU也完全有能力在接收完第、2个报文前将第1个报文处理完毕,所以只需要建立一个报文处理任务。

  还有些要完成较复杂任务的节点,譬如车载网络中的中央控制部件(BSI)。在全CAN车载网络中,它同时连接内部网、车身网和舒适网3个网络。作为汽车车载网络系统中枢,BSI任务繁重,对CAN报文的处理经常会被各种中断和内部任务打断,所以不能保证及时处理上一次接收的CAN报文。另外,由于消息队列是采取先进先出(FIFO)或者后进先出(LIFO)的方式来组织报文的,当消息队列中积攒多个还没处理的报文时,无法先取出优先级最高的报文进行处理。为了能够优先处理重要设备发送过来的报文,必须针对系统中每个与本节点有进行CAN关系的节点建立一个独立的报文处理任务。这个任务包含一个独立的消息队列,并且发送报文的节点优先级越高,该任务设置的优先级也应该越高。为此CAN1_RI_HANDLE()函数也应该做出相应的修改。修改之后的程序代码如下所示:

void CAN1_RI_HANDLE() {
  RI_DATA.FRAME = CAN1RFS;
  RI_DATA.ID = CAN1RID;
  RI_DATA.DataA = CAN1RDA;
  RI_DATA.DataB = CAN1RDB;//解析报文标识符RI_DATA.ID中的SrcMACID段,根据解析结果使用OSQPost( )将RI_DATA发送到对应节点任务的消息队列中
  CAN1_COMMAND_RRB();//释放接收缓冲区
}

  再结合CAN链路层的仲裁机制,就可以保证优先级别高的节点优先发送报文,并被接收节点优先处理。至此,CAN的整个脉络已经非常清晰,其总体流程略——编者注。

结语

  本文基于μC/OSII操作系统、针对要求较高的CAN系统编写的CAN简洁、高效,在不同的应用环境下只需添加相应的用户代码,就可以组成完整的CAN。但在提高高优先级节点的同时,在一定程度上也降低了低优先级节点的实时性,所以在工程应用中应根据实际需要兼顾高低优先级节点的实时性能。


上一页 1 2 3 下一页

评论


相关推荐

技术专区

关闭