avr_chen 发表于 2009-9-17 17:04:55

ucos-ii高手请进

最近在学习ucosii,有一个问题希望哪位高手指点一下就是消息邮箱的挂起函数,有一段不知道为什么要这样。源程序如下:
void *OSMboxPend (OS_EVENT *pevent, INT16U timeout, INT8U *err)
{
    void*msg;

    OS_ENTER_CRITICAL();
    if (pevent->OSEventType != OS_EVENT_TYPE_MBOX) {                      (1)
      OS_EXIT_CRITICAL();
      *err = OS_ERR_EVENT_TYPE;
      return ((void *)0);
    }
    msg = pevent->OSEventPtr;
    if (msg != (void *)0) {                                               (2)
      pevent->OSEventPtr = (void *)0;                                   (3)
      OS_EXIT_CRITICAL();
      *err = OS_NO_ERR;
    } else if (OSIntNesting > 0) {                                        (4)
      OS_EXIT_CRITICAL();
      *err = OS_ERR_PEND_ISR;
    } else {
      OSTCBCur->OSTCBStat |= OS_STAT_MBOX;                            (5)
      OSTCBCur->OSTCBDly   = timeout;
      OSEventTaskWait(pevent);
      OS_EXIT_CRITICAL();
      OSSched();
      OS_ENTER_CRITICAL();
      if ((msg = OSTCBCur->OSTCBMsg) != (void *)0) {                    (6)
            OSTCBCur->OSTCBMsg      = (void *)0;
            OSTCBCur->OSTCBStat   = OS_STAT_RDY;
            OSTCBCur->OSTCBEventPtr = (OS_EVENT *)0;
            OS_EXIT_CRITICAL();
            *err                  = OS_NO_ERR;
      } else if (OSTCBCur->OSTCBStat & OS_STAT_MBOX) {                (7)
            OSEventTO(pevent);                                            (8)
            OS_EXIT_CRITICAL();
            msg                     = (void *)0;                          (9)
            *err                  = OS_TIMEOUT;
      } else {                                                      (?????)
            msg                     = pevent->OSEventPtr;                (10)
            pevent->OSEventPtr      = (void *)0;                            (11)
            OSTCBCur->OSTCBEventPtr = (OS_EVENT *)0;                   (12)
            OS_EXIT_CRITICAL();
            *err                  = OS_NO_ERR;
      }
    }
    return (msg);
}



问题是:(10)标号上方这个else(即标有问号的那一行),程序在什么情况下会出现这种可能性呢???即任务控制块中没有消息指针,而任务等待消息的状态标志又被清掉了,这是一种什么情况呢???恳请哪位高手指点一下,呵呵,不胜感激。

ralfak 发表于 2009-9-22 16:02:39

这个问题很简单,(6)之前任务重新调度,
6之后任务重新就绪,
1是队列就绪,2是超时,第三种情况我给出一种答案
#if OS_Q_PEND_ABORT_EN > 0
INT8UOSQPendAbort (OS_EVENT *pevent, INT8U opt, INT8U *perr)
{
    INT8U      nbr_tasks;
#if OS_CRITICAL_METHOD == 3                              /* Allocate storage for CPU status register */
    OS_CPU_SRcpu_sr = 0;
#endif



#if OS_ARG_CHK_EN > 0
    if (perr == (INT8U *)0) {                              /* Validate 'perr'                        */
      return (0);
    }
    if (pevent == (OS_EVENT *)0) {                         /* Validate 'pevent'                        */
      *perr = OS_ERR_PEVENT_NULL;
      return (0);
    }
#endif
    if (pevent->OSEventType != OS_EVENT_TYPE_Q) {          /* Validate event block type                */
      *perr = OS_ERR_EVENT_TYPE;
      return (0);
    }
    OS_ENTER_CRITICAL();
    if (pevent->OSEventGrp != 0) {                         /* See if any task waiting on queue?      */
      nbr_tasks = 0;
      switch (opt) {
            case OS_PEND_OPT_BROADCAST:                  /* Do we need to abort ALL waiting tasks?   */
               while (pevent->OSEventGrp != 0) {         /* Yes, ready ALL tasks waiting on queue    */
                     (void)OS_EventTaskRdy(pevent, (void *)0, OS_STAT_Q, OS_STAT_PEND_ABORT);
                     nbr_tasks++;
               }
               break;
               
            case OS_PEND_OPT_NONE:
            default:                                       /* No,ready HPT       waiting on queue    */
               (void)OS_EventTaskRdy(pevent, (void *)0, OS_STAT_Q, OS_STAT_PEND_ABORT);
               nbr_tasks++;
               break;
      }
      OS_EXIT_CRITICAL();
      OS_Sched();                                        /* Find HPT ready to run                  */
      *perr = OS_ERR_PEND_ABORT;
      return (nbr_tasks);
    }
    OS_EXIT_CRITICAL();
    *perr = OS_ERR_NONE;
    return (0);                                          /* No tasks waiting on queue                */
}

ralfak 发表于 2009-9-22 16:03:47

具体情况需要具体分析,每个版本的代码不一样,你这种问内核的事情需要内核版本

avr_chen 发表于 2009-9-22 17:17:47

谢谢 回复
我是初学者看得是卲贝贝的书
你给的答案我不是很懂,这个函数我没有见过
我只看了书里面介绍的函数呵呵
是不是有这样的可能性就是其他任务发送消息的时候发送过程还没有全部完成即退出临界状态,而后又被中断,而这个挂起的任务重新取得了CPU的控制权。所以如此。
谢谢您的回复。

ralfak 发表于 2009-9-26 14:58:16

任务发送消息的时候是有临界保护的,这是系统内核保证的,如果你不放心,可以仔细的阅读代码
我给出的答案是在别的任务中将执行请求取消函数
这样当前的请求任务就可以就绪,结果就是既没有超时也没有请求到消息。
我的这个版本应该是2.83以上的

maomao2126 发表于 2009-10-11 12:46:22

mark,后面看到这部分的时候注意一下,呵呵

sugarxyc 发表于 2009-11-14 08:19:12

dddddddddddd
页: [1]
查看完整版本: ucos-ii高手请进