os_q.c
上传用户:yyyd609
上传日期:2022-07-18
资源大小:183k
文件大小:36k
源码类别:

微处理器开发

开发平台:

C/C++

  1. /*
  2. *********************************************************************************************************
  3. *                                                uC/OS-II
  4. *                                          The Real-Time Kernel
  5. *                                        MESSAGE QUEUE MANAGEMENT
  6. *
  7. *                          (c) Copyright 1992-2003, Jean J. Labrosse, Weston, FL
  8. *                                           All Rights Reserved
  9. *
  10. * File    : OS_Q.C
  11. * By      : Jean J. Labrosse
  12. * Version : V2.76
  13. *********************************************************************************************************
  14. */
  15. #ifndef  OS_MASTER_FILE
  16. #include <ucos_ii.h>
  17. #endif
  18. #if (OS_Q_EN > 0) && (OS_MAX_QS > 0)
  19. /*
  20. *********************************************************************************************************
  21. *                                      ACCEPT MESSAGE FROM QUEUE
  22. *
  23. * Description: This function checks the queue to see if a message is available.  Unlike OSQPend(),
  24. *              OSQAccept() does not suspend the calling task if a message is not available.
  25. *
  26. * Arguments  : pevent        is a pointer to the event control block
  27. *
  28. *              err           is a pointer to where an error message will be deposited.  Possible error
  29. *                            messages are:
  30. *
  31. *                            OS_NO_ERR           The call was successful and your task received a
  32. *                                                message.
  33. *                            OS_ERR_EVENT_TYPE   You didn't pass a pointer to a queue
  34. *                            OS_ERR_PEVENT_NULL  If 'pevent' is a NULL pointer
  35. *                            OS_Q_EMPTY          The queue did not contain any messages
  36. *
  37. * Returns    : != (void *)0  is the message in the queue if one is available.  The message is removed
  38. *                            from the so the next time OSQAccept() is called, the queue will contain
  39. *                            one less entry.
  40. *              == (void *)0  if you received a NULL pointer message
  41. *                            if the queue is empty or,
  42. *                            if 'pevent' is a NULL pointer or,
  43. *                            if you passed an invalid event type
  44. *
  45. * Note(s)    : As of V2.60, you can now pass NULL pointers through queues.  Because of this, the argument
  46. *              'err' has been added to the API to tell you about the outcome of the call.
  47. *********************************************************************************************************
  48. */
  49. #if OS_Q_ACCEPT_EN > 0
  50. void  *OSQAccept (OS_EVENT *pevent, INT8U *err)
  51. {
  52.     void      *msg;
  53.     OS_Q      *pq;
  54. #if OS_CRITICAL_METHOD == 3                      /* Allocate storage for CPU status register           */
  55.     OS_CPU_SR  cpu_sr;
  56.     cpu_sr = 0;                                  /* Prevent compiler warning                           */
  57. #endif    
  58. #if OS_ARG_CHK_EN > 0
  59.     if (pevent == (OS_EVENT *)0) {               /* Validate 'pevent'                                  */
  60.         *err = OS_ERR_PEVENT_NULL;
  61.         return ((void *)0);
  62.     }
  63. #endif
  64.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {/* Validate event block type                          */
  65.         *err = OS_ERR_EVENT_TYPE;
  66.         return ((void *)0);
  67.     }
  68.     OS_ENTER_CRITICAL();
  69.     pq = (OS_Q *)pevent->OSEventPtr;             /* Point at queue control block                       */
  70.     if (pq->OSQEntries > 0) {                    /* See if any messages in the queue                   */
  71.         msg = *pq->OSQOut++;                     /* Yes, extract oldest message from the queue         */
  72.         pq->OSQEntries--;                        /* Update the number of entries in the queue          */
  73.         if (pq->OSQOut == pq->OSQEnd) {          /* Wrap OUT pointer if we are at the end of the queue */
  74.             pq->OSQOut = pq->OSQStart;
  75.         }
  76.         *err = OS_NO_ERR;
  77.     } else {
  78.         *err = OS_Q_EMPTY;
  79.         msg  = (void *)0;                        /* Queue is empty                                     */
  80.     }
  81.     OS_EXIT_CRITICAL();
  82.     return (msg);                                /* Return message received (or NULL)                  */
  83. }
  84. #endif
  85. /*$PAGE*/
  86. /*
  87. *********************************************************************************************************
  88. *                                        CREATE A MESSAGE QUEUE
  89. *
  90. * Description: This function creates a message queue if free event control blocks are available.
  91. *
  92. * Arguments  : start         is a pointer to the base address of the message queue storage area.  The
  93. *                            storage area MUST be declared as an array of pointers to 'void' as follows
  94. *
  95. *                            void *MessageStorage[size]
  96. *
  97. *              size          is the number of elements in the storage area
  98. *
  99. * Returns    : != (OS_EVENT *)0  is a pointer to the event control clock (OS_EVENT) associated with the
  100. *                                created queue
  101. *              == (OS_EVENT *)0  if no event control blocks were available or an error was detected
  102. *********************************************************************************************************
  103. */
  104. OS_EVENT  *OSQCreate (void **start, INT16U size)
  105. {
  106.     OS_EVENT  *pevent;
  107.     OS_Q      *pq;
  108. #if OS_CRITICAL_METHOD == 3                      /* Allocate storage for CPU status register           */
  109.     OS_CPU_SR  cpu_sr;
  110.     cpu_sr = 0;                                  /* Prevent compiler warning                           */
  111. #endif    
  112.     if (OSIntNesting > 0) {                      /* See if called from ISR ...                         */
  113.         return ((OS_EVENT *)0);                  /* ... can't CREATE from an ISR                       */
  114.     }
  115.     OS_ENTER_CRITICAL();
  116.     pevent = OSEventFreeList;                    /* Get next free event control block                  */
  117.     if (OSEventFreeList != (OS_EVENT *)0) {      /* See if pool of free ECB pool was empty             */
  118.         OSEventFreeList = (OS_EVENT *)OSEventFreeList->OSEventPtr;
  119.     }
  120.     OS_EXIT_CRITICAL();
  121.     if (pevent != (OS_EVENT *)0) {               /* See if we have an event control block              */
  122.         OS_ENTER_CRITICAL();
  123.         pq = OSQFreeList;                        /* Get a free queue control block                     */
  124.         if (pq != (OS_Q *)0) {                   /* Were we able to get a queue control block ?        */
  125.             OSQFreeList            = OSQFreeList->OSQPtr; /* Yes, Adjust free list pointer to next free*/
  126.             OS_EXIT_CRITICAL();
  127.             pq->OSQStart           = start;               /*      Initialize the queue                 */
  128.             pq->OSQEnd             = &start[size];
  129.             pq->OSQIn              = start;
  130.             pq->OSQOut             = start;
  131.             pq->OSQSize            = size;
  132.             pq->OSQEntries         = 0;
  133.             pevent->OSEventType    = OS_EVENT_TYPE_Q;
  134.             pevent->OSEventCnt     = 0;
  135.             pevent->OSEventPtr     = pq;
  136. #if OS_EVENT_NAME_SIZE > 1
  137.             pevent->OSEventName[0] = '?';                  /* Unknown name                             */
  138.             pevent->OSEventName[1] = OS_ASCII_NUL;
  139. #endif
  140.             OS_EventWaitListInit(pevent);                 /*      Initalize the wait list              */
  141.         } else {
  142.             pevent->OSEventPtr = (void *)OSEventFreeList; /* No,  Return event control block on error  */
  143.             OSEventFreeList    = pevent;
  144.             OS_EXIT_CRITICAL();
  145.             pevent = (OS_EVENT *)0;
  146.         }
  147.     }
  148.     return (pevent);
  149. }
  150. /*$PAGE*/
  151. /*
  152. *********************************************************************************************************
  153. *                                        DELETE A MESSAGE QUEUE
  154. *
  155. * Description: This function deletes a message queue and readies all tasks pending on the queue.
  156. *
  157. * Arguments  : pevent        is a pointer to the event control block associated with the desired
  158. *                            queue.
  159. *
  160. *              opt           determines delete options as follows:
  161. *                            opt == OS_DEL_NO_PEND   Delete the queue ONLY if no task pending
  162. *                            opt == OS_DEL_ALWAYS    Deletes the queue even if tasks are waiting.
  163. *                                                    In this case, all the tasks pending will be readied.
  164. *
  165. *              err           is a pointer to an error code that can contain one of the following values:
  166. *                            OS_NO_ERR               The call was successful and the queue was deleted
  167. *                            OS_ERR_DEL_ISR          If you tried to delete the queue from an ISR
  168. *                            OS_ERR_INVALID_OPT      An invalid option was specified
  169. *                            OS_ERR_TASK_WAITING     One or more tasks were waiting on the queue
  170. *                            OS_ERR_EVENT_TYPE       If you didn't pass a pointer to a queue
  171. *                            OS_ERR_PEVENT_NULL      If 'pevent' is a NULL pointer.
  172. *
  173. * Returns    : pevent        upon error
  174. *              (OS_EVENT *)0 if the queue was successfully deleted.
  175. *
  176. * Note(s)    : 1) This function must be used with care.  Tasks that would normally expect the presence of
  177. *                 the queue MUST check the return code of OSQPend().
  178. *              2) OSQAccept() callers will not know that the intended queue has been deleted unless
  179. *                 they check 'pevent' to see that it's a NULL pointer.
  180. *              3) This call can potentially disable interrupts for a long time.  The interrupt disable
  181. *                 time is directly proportional to the number of tasks waiting on the queue.
  182. *              4) Because ALL tasks pending on the queue will be readied, you MUST be careful in
  183. *                 applications where the queue is used for mutual exclusion because the resource(s)
  184. *                 will no longer be guarded by the queue.
  185. *              5) If the storage for the message queue was allocated dynamically (i.e. using a malloc()
  186. *                 type call) then your application MUST release the memory storage by call the counterpart
  187. *                 call of the dynamic allocation scheme used.  If the queue storage was created statically
  188. *                 then, the storage can be reused.
  189. *********************************************************************************************************
  190. */
  191. #if OS_Q_DEL_EN > 0
  192. OS_EVENT  *OSQDel (OS_EVENT *pevent, INT8U opt, INT8U *err)
  193. {
  194.     BOOLEAN    tasks_waiting;
  195.     OS_Q      *pq;
  196. #if OS_CRITICAL_METHOD == 3                                /* Allocate storage for CPU status register */
  197.     OS_CPU_SR  cpu_sr;
  198.     cpu_sr = 0;                                            /* Prevent compiler warning                 */
  199. #endif    
  200.     if (OSIntNesting > 0) {                                /* See if called from ISR ...               */
  201.         *err = OS_ERR_DEL_ISR;                             /* ... can't DELETE from an ISR             */
  202.         return (pevent);
  203.     }
  204. #if OS_ARG_CHK_EN > 0
  205.     if (pevent == (OS_EVENT *)0) {                         /* Validate 'pevent'                        */
  206.         *err = OS_ERR_PEVENT_NULL;
  207.         return (pevent);
  208.     }
  209. #endif
  210.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {          /* Validate event block type                */
  211.         *err = OS_ERR_EVENT_TYPE;
  212.         return (pevent);
  213.     }
  214.     OS_ENTER_CRITICAL();
  215.     if (pevent->OSEventGrp != 0x00) {                      /* See if any tasks waiting on queue        */
  216.         tasks_waiting = TRUE;                              /* Yes                                      */
  217.     } else {
  218.         tasks_waiting = FALSE;                             /* No                                       */
  219.     }
  220.     switch (opt) {
  221.         case OS_DEL_NO_PEND:                               /* Delete queue only if no task waiting     */
  222.              if (tasks_waiting == FALSE) {
  223. #if OS_EVENT_NAME_SIZE > 1
  224.                  pevent->OSEventName[0] = '?';             /* Unknown name                             */
  225.                  pevent->OSEventName[1] = OS_ASCII_NUL;
  226. #endif
  227.                  pq                     = (OS_Q *)pevent->OSEventPtr;  /* Return OS_Q to free list     */
  228.                  pq->OSQPtr             = OSQFreeList;
  229.                  OSQFreeList            = pq;
  230.                  pevent->OSEventType    = OS_EVENT_TYPE_UNUSED;
  231.                  pevent->OSEventPtr     = OSEventFreeList; /* Return Event Control Block to free list  */
  232.                  pevent->OSEventCnt     = 0;
  233.                  OSEventFreeList        = pevent;          /* Get next free event control block        */
  234.                  OS_EXIT_CRITICAL();
  235.                  *err                   = OS_NO_ERR;
  236.                  return ((OS_EVENT *)0);                   /* Queue has been deleted                   */
  237.              } else {
  238.                  OS_EXIT_CRITICAL();
  239.                  *err                   = OS_ERR_TASK_WAITING;
  240.                  return (pevent);
  241.              }
  242.         case OS_DEL_ALWAYS:                                /* Always delete the queue                  */
  243.              while (pevent->OSEventGrp != 0x00) {          /* Ready ALL tasks waiting for queue        */
  244.                  (void)OS_EventTaskRdy(pevent, (void *)0, OS_STAT_Q);
  245.              }
  246. #if OS_EVENT_NAME_SIZE > 1
  247.              pevent->OSEventName[0] = '?';                 /* Unknown name                             */
  248.              pevent->OSEventName[1] = OS_ASCII_NUL;
  249. #endif
  250.              pq                     = (OS_Q *)pevent->OSEventPtr;   /* Return OS_Q to free list        */
  251.              pq->OSQPtr             = OSQFreeList;
  252.              OSQFreeList            = pq;
  253.              pevent->OSEventType    = OS_EVENT_TYPE_UNUSED;
  254.              pevent->OSEventPtr     = OSEventFreeList;     /* Return Event Control Block to free list  */
  255.              pevent->OSEventCnt     = 0;
  256.              OSEventFreeList        = pevent;              /* Get next free event control block        */
  257.              OS_EXIT_CRITICAL();
  258.              if (tasks_waiting == TRUE) {                  /* Reschedule only if task(s) were waiting  */
  259.                  OS_Sched();                               /* Find highest priority task ready to run  */
  260.              }
  261.              *err = OS_NO_ERR;
  262.              return ((OS_EVENT *)0);                       /* Queue has been deleted                   */
  263.         default:
  264.              OS_EXIT_CRITICAL();
  265.              *err = OS_ERR_INVALID_OPT;
  266.              return (pevent);
  267.     }
  268. }
  269. #endif
  270. /*$PAGE*/
  271. /*
  272. *********************************************************************************************************
  273. *                                             FLUSH QUEUE
  274. *
  275. * Description : This function is used to flush the contents of the message queue.
  276. *
  277. * Arguments   : none
  278. *
  279. * Returns     : OS_NO_ERR           upon success
  280. *               OS_ERR_EVENT_TYPE   If you didn't pass a pointer to a queue
  281. *               OS_ERR_PEVENT_NULL  If 'pevent' is a NULL pointer
  282. *
  283. * WARNING     : You should use this function with great care because, when to flush the queue, you LOOSE
  284. *               the references to what the queue entries are pointing to and thus, you could cause
  285. *               'memory leaks'.  In other words, the data you are pointing to that's being referenced
  286. *               by the queue entries should, most likely, need to be de-allocated (i.e. freed).
  287. *********************************************************************************************************
  288. */
  289. #if OS_Q_FLUSH_EN > 0
  290. INT8U  OSQFlush (OS_EVENT *pevent)
  291. {
  292.     OS_Q      *pq;
  293. #if OS_CRITICAL_METHOD == 3                           /* Allocate storage for CPU status register      */
  294.     OS_CPU_SR  cpu_sr;
  295.     cpu_sr = 0;                                       /* Prevent compiler warning                      */
  296. #endif    
  297. #if OS_ARG_CHK_EN > 0
  298.     if (pevent == (OS_EVENT *)0) {                    /* Validate 'pevent'                             */
  299.         return (OS_ERR_PEVENT_NULL);
  300.     }
  301.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {     /* Validate event block type                     */
  302.         return (OS_ERR_EVENT_TYPE);
  303.     }
  304. #endif
  305.     OS_ENTER_CRITICAL();
  306.     pq             = (OS_Q *)pevent->OSEventPtr;      /* Point to queue storage structure              */
  307.     pq->OSQIn      = pq->OSQStart;
  308.     pq->OSQOut     = pq->OSQStart;
  309.     pq->OSQEntries = 0;
  310.     OS_EXIT_CRITICAL();
  311.     return (OS_NO_ERR);
  312. }
  313. #endif
  314. /*$PAGE*/
  315. /*
  316. *********************************************************************************************************
  317. *                                     PEND ON A QUEUE FOR A MESSAGE
  318. *
  319. * Description: This function waits for a message to be sent to a queue
  320. *
  321. * Arguments  : pevent        is a pointer to the event control block associated with the desired queue
  322. *
  323. *              timeout       is an optional timeout period (in clock ticks).  If non-zero, your task will
  324. *                            wait for a message to arrive at the queue up to the amount of time
  325. *                            specified by this argument.  If you specify 0, however, your task will wait
  326. *                            forever at the specified queue or, until a message arrives.
  327. *
  328. *              err           is a pointer to where an error message will be deposited.  Possible error
  329. *                            messages are:
  330. *
  331. *                            OS_NO_ERR           The call was successful and your task received a
  332. *                                                message.
  333. *                            OS_TIMEOUT          A message was not received within the specified timeout
  334. *                            OS_ERR_EVENT_TYPE   You didn't pass a pointer to a queue
  335. *                            OS_ERR_PEVENT_NULL  If 'pevent' is a NULL pointer
  336. *                            OS_ERR_PEND_ISR     If you called this function from an ISR and the result
  337. *                                                would lead to a suspension.
  338. *
  339. * Returns    : != (void *)0  is a pointer to the message received
  340. *              == (void *)0  if you received a NULL pointer message or,
  341. *                            if no message was received or,
  342. *                            if 'pevent' is a NULL pointer or,
  343. *                            if you didn't pass a pointer to a queue.
  344. *
  345. * Note(s)    : As of V2.60, this function allows you to receive NULL pointer messages.
  346. *********************************************************************************************************
  347. */
  348. void  *OSQPend (OS_EVENT *pevent, INT16U timeout, INT8U *err)
  349. {
  350.     void      *msg;
  351.     OS_Q      *pq;
  352. #if OS_CRITICAL_METHOD == 3                      /* Allocate storage for CPU status register           */
  353.     OS_CPU_SR  cpu_sr;
  354.     cpu_sr = 0;                                  /* Prevent compiler warning                           */
  355. #endif    
  356.     if (OSIntNesting > 0) {                      /* See if called from ISR ...                         */
  357.         *err = OS_ERR_PEND_ISR;                  /* ... can't PEND from an ISR                         */
  358.         return ((void *)0);
  359.     }
  360. #if OS_ARG_CHK_EN > 0
  361.     if (pevent == (OS_EVENT *)0) {               /* Validate 'pevent'                                  */
  362.         *err = OS_ERR_PEVENT_NULL;
  363.         return ((void *)0);
  364.     }
  365.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {/* Validate event block type                          */
  366.         *err = OS_ERR_EVENT_TYPE;
  367.         return ((void *)0);
  368.     }
  369. #endif
  370.     OS_ENTER_CRITICAL();
  371.     pq = (OS_Q *)pevent->OSEventPtr;             /* Point at queue control block                       */
  372.     if (pq->OSQEntries > 0) {                    /* See if any messages in the queue                   */
  373.         msg = *pq->OSQOut++;                     /* Yes, extract oldest message from the queue         */
  374.         pq->OSQEntries--;                        /* Update the number of entries in the queue          */
  375.         if (pq->OSQOut == pq->OSQEnd) {          /* Wrap OUT pointer if we are at the end of the queue */
  376.             pq->OSQOut = pq->OSQStart;
  377.         }
  378.         OS_EXIT_CRITICAL();
  379.         *err = OS_NO_ERR;
  380.         return (msg);                            /* Return message received                            */
  381.     }
  382.     OSTCBCur->OSTCBStat   |= OS_STAT_Q;          /* Task will have to pend for a message to be posted  */
  383.     OSTCBCur->OSTCBPendTO  = FALSE;
  384.     OSTCBCur->OSTCBDly     = timeout;            /* Load timeout into TCB                              */
  385.     OS_EventTaskWait(pevent);                    /* Suspend task until event or timeout occurs         */
  386.     OS_EXIT_CRITICAL();
  387.     OS_Sched();                                  /* Find next highest priority task ready to run       */
  388.     OS_ENTER_CRITICAL();
  389.     if (OSTCBCur->OSTCBPendTO == TRUE) {         /* Was task readied because of a timeout?             */
  390.         OS_EventTO(pevent);                      /* Yes                                                */
  391.         OS_EXIT_CRITICAL();
  392.         *err = OS_TIMEOUT;                       /*     Indicate a timeout occured                     */
  393.         return ((void *)0);                      /*     No message received                            */
  394.     }
  395.     msg                     = OSTCBCur->OSTCBMsg;/* No, Extract message from TCB (Put there by QPost)  */
  396.     OSTCBCur->OSTCBMsg      = (void *)0;
  397.     OSTCBCur->OSTCBStat     = OS_STAT_RDY;
  398.     OSTCBCur->OSTCBEventPtr = (OS_EVENT *)0;     /*     No longer waiting for event                    */
  399.     OS_EXIT_CRITICAL();
  400.     *err                    = OS_NO_ERR;
  401.     return (msg);                                /*     Return message received                        */
  402. }
  403. /*$PAGE*/
  404. /*
  405. *********************************************************************************************************
  406. *                                        POST MESSAGE TO A QUEUE
  407. *
  408. * Description: This function sends a message to a queue
  409. *
  410. * Arguments  : pevent        is a pointer to the event control block associated with the desired queue
  411. *
  412. *              msg           is a pointer to the message to send.
  413. *
  414. * Returns    : OS_NO_ERR             The call was successful and the message was sent
  415. *              OS_Q_FULL             If the queue cannot accept any more messages because it is full.
  416. *              OS_ERR_EVENT_TYPE     If you didn't pass a pointer to a queue.
  417. *              OS_ERR_PEVENT_NULL    If 'pevent' is a NULL pointer
  418. *
  419. * Note(s)    : As of V2.60, this function allows you to send NULL pointer messages.
  420. *********************************************************************************************************
  421. */
  422. #if OS_Q_POST_EN > 0
  423. INT8U  OSQPost (OS_EVENT *pevent, void *msg)
  424. {
  425.     OS_Q      *pq;
  426. #if OS_CRITICAL_METHOD == 3                            /* Allocate storage for CPU status register     */
  427.     OS_CPU_SR  cpu_sr;
  428.     cpu_sr = 0;                                        /* Prevent compiler warning                     */
  429. #endif    
  430. #if OS_ARG_CHK_EN > 0
  431.     if (pevent == (OS_EVENT *)0) {                     /* Validate 'pevent'                            */
  432.         return (OS_ERR_PEVENT_NULL);
  433.     }
  434. #endif
  435.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {      /* Validate event block type                    */
  436.         return (OS_ERR_EVENT_TYPE);
  437.     }
  438.     OS_ENTER_CRITICAL();
  439.     if (pevent->OSEventGrp != 0x00) {                  /* See if any task pending on queue             */
  440.         (void)OS_EventTaskRdy(pevent, msg, OS_STAT_Q); /* Ready highest priority task waiting on event */
  441.         OS_EXIT_CRITICAL();
  442.         OS_Sched();                                    /* Find highest priority task ready to run      */
  443.         return (OS_NO_ERR);
  444.     }
  445.     pq = (OS_Q *)pevent->OSEventPtr;                   /* Point to queue control block                 */
  446.     if (pq->OSQEntries >= pq->OSQSize) {               /* Make sure queue is not full                  */
  447.         OS_EXIT_CRITICAL();
  448.         return (OS_Q_FULL);
  449.     }
  450.     *pq->OSQIn++ = msg;                                /* Insert message into queue                    */
  451.     pq->OSQEntries++;                                  /* Update the nbr of entries in the queue       */
  452.     if (pq->OSQIn == pq->OSQEnd) {                     /* Wrap IN ptr if we are at end of queue        */
  453.         pq->OSQIn = pq->OSQStart;
  454.     }
  455.     OS_EXIT_CRITICAL();
  456.     return (OS_NO_ERR);
  457. }
  458. #endif
  459. /*$PAGE*/
  460. /*
  461. *********************************************************************************************************
  462. *                                   POST MESSAGE TO THE FRONT OF A QUEUE
  463. *
  464. * Description: This function sends a message to a queue but unlike OSQPost(), the message is posted at
  465. *              the front instead of the end of the queue.  Using OSQPostFront() allows you to send
  466. *              'priority' messages.
  467. *
  468. * Arguments  : pevent        is a pointer to the event control block associated with the desired queue
  469. *
  470. *              msg           is a pointer to the message to send.
  471. *
  472. * Returns    : OS_NO_ERR             The call was successful and the message was sent
  473. *              OS_Q_FULL             If the queue cannot accept any more messages because it is full.
  474. *              OS_ERR_EVENT_TYPE     If you didn't pass a pointer to a queue.
  475. *              OS_ERR_PEVENT_NULL    If 'pevent' is a NULL pointer
  476. *
  477. * Note(s)    : As of V2.60, this function allows you to send NULL pointer messages.
  478. *********************************************************************************************************
  479. */
  480. #if OS_Q_POST_FRONT_EN > 0
  481. INT8U  OSQPostFront (OS_EVENT *pevent, void *msg)
  482. {
  483.     OS_Q      *pq;
  484. #if OS_CRITICAL_METHOD == 3                           /* Allocate storage for CPU status register      */
  485.     OS_CPU_SR  cpu_sr;
  486.     cpu_sr = 0;                                       /* Prevent compiler warning                      */
  487. #endif    
  488. #if OS_ARG_CHK_EN > 0
  489.     if (pevent == (OS_EVENT *)0) {                    /* Validate 'pevent'                             */
  490.         return (OS_ERR_PEVENT_NULL);
  491.     }
  492. #endif
  493.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {     /* Validate event block type                     */
  494.         return (OS_ERR_EVENT_TYPE);
  495.     }
  496.     OS_ENTER_CRITICAL();
  497.     if (pevent->OSEventGrp != 0x00) {                 /* See if any task pending on queue              */
  498.         (void)OS_EventTaskRdy(pevent, msg, OS_STAT_Q);/* Ready highest priority task waiting on event  */
  499.         OS_EXIT_CRITICAL();
  500.         OS_Sched();                                   /* Find highest priority task ready to run       */
  501.         return (OS_NO_ERR);
  502.     }
  503.     pq = (OS_Q *)pevent->OSEventPtr;                  /* Point to queue control block                  */
  504.     if (pq->OSQEntries >= pq->OSQSize) {              /* Make sure queue is not full                   */
  505.         OS_EXIT_CRITICAL();
  506.         return (OS_Q_FULL);
  507.     }
  508.     if (pq->OSQOut == pq->OSQStart) {                 /* Wrap OUT ptr if we are at the 1st queue entry */
  509.         pq->OSQOut = pq->OSQEnd;
  510.     }
  511.     pq->OSQOut--;
  512.     *pq->OSQOut = msg;                                /* Insert message into queue                     */
  513.     pq->OSQEntries++;                                 /* Update the nbr of entries in the queue        */
  514.     OS_EXIT_CRITICAL();
  515.     return (OS_NO_ERR);
  516. }
  517. #endif
  518. /*$PAGE*/
  519. /*
  520. *********************************************************************************************************
  521. *                                        POST MESSAGE TO A QUEUE
  522. *
  523. * Description: This function sends a message to a queue.  This call has been added to reduce code size
  524. *              since it can replace both OSQPost() and OSQPostFront().  Also, this function adds the
  525. *              capability to broadcast a message to ALL tasks waiting on the message queue.
  526. *
  527. * Arguments  : pevent        is a pointer to the event control block associated with the desired queue
  528. *
  529. *              msg           is a pointer to the message to send.
  530. *
  531. *              opt           determines the type of POST performed:
  532. *                            OS_POST_OPT_NONE         POST to a single waiting task
  533. *                                                     (Identical to OSQPost())
  534. *                            OS_POST_OPT_BROADCAST    POST to ALL tasks that are waiting on the queue
  535. *                            OS_POST_OPT_FRONT        POST as LIFO (Simulates OSQPostFront())
  536. *
  537. *                            Below is a list of ALL the possible combination of these flags:
  538. *
  539. *                                 1) OS_POST_OPT_NONE
  540. *                                    identical to OSQPost()
  541. *
  542. *                                 2) OS_POST_OPT_FRONT
  543. *                                    identical to OSQPostFront()
  544. *
  545. *                                 3) OS_POST_OPT_BROADCAST
  546. *                                    identical to OSQPost() but will broadcast 'msg' to ALL waiting tasks
  547. *
  548. *                                 4) OS_POST_OPT_FRONT + OS_POST_OPT_BROADCAST  is identical to
  549. *                                    OSQPostFront() except that will broadcast 'msg' to ALL waiting tasks
  550. *
  551. * Returns    : OS_NO_ERR             The call was successful and the message was sent
  552. *              OS_Q_FULL             If the queue cannot accept any more messages because it is full.
  553. *              OS_ERR_EVENT_TYPE     If you didn't pass a pointer to a queue.
  554. *              OS_ERR_PEVENT_NULL    If 'pevent' is a NULL pointer
  555. *
  556. * Warning    : Interrupts can be disabled for a long time if you do a 'broadcast'.  In fact, the
  557. *              interrupt disable time is proportional to the number of tasks waiting on the queue.
  558. *********************************************************************************************************
  559. */
  560. #if OS_Q_POST_OPT_EN > 0
  561. INT8U  OSQPostOpt (OS_EVENT *pevent, void *msg, INT8U opt)
  562. {
  563.     OS_Q      *pq;
  564. #if OS_CRITICAL_METHOD == 3                           /* Allocate storage for CPU status register      */
  565.     OS_CPU_SR  cpu_sr;
  566.                      
  567.     cpu_sr = 0;                                       /* Prevent compiler warning                      */
  568. #endif    
  569. #if OS_ARG_CHK_EN > 0
  570.     if (pevent == (OS_EVENT *)0) {                    /* Validate 'pevent'                             */
  571.         return (OS_ERR_PEVENT_NULL);
  572.     }
  573. #endif
  574.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {     /* Validate event block type                     */
  575.         return (OS_ERR_EVENT_TYPE);
  576.     }
  577.     OS_ENTER_CRITICAL();
  578.     if (pevent->OSEventGrp != 0x00) {                 /* See if any task pending on queue              */
  579.         if ((opt & OS_POST_OPT_BROADCAST) != 0x00) {  /* Do we need to post msg to ALL waiting tasks ? */
  580.             while (pevent->OSEventGrp != 0x00) {      /* Yes, Post to ALL tasks waiting on queue       */
  581.                 (void)OS_EventTaskRdy(pevent, msg, OS_STAT_Q);
  582.             }
  583.         } else {
  584.             (void)OS_EventTaskRdy(pevent, msg, OS_STAT_Q);  /* No,  Post to HPT waiting on queue       */
  585.         }
  586.         OS_EXIT_CRITICAL();
  587.         OS_Sched();                                         /* Find highest priority task ready to run */
  588.         return (OS_NO_ERR);
  589.     }
  590.     pq = (OS_Q *)pevent->OSEventPtr;                  /* Point to queue control block                  */
  591.     if (pq->OSQEntries >= pq->OSQSize) {              /* Make sure queue is not full                   */
  592.         OS_EXIT_CRITICAL();
  593.         return (OS_Q_FULL);
  594.     }
  595.     if ((opt & OS_POST_OPT_FRONT) != 0x00) {          /* Do we post to the FRONT of the queue?         */
  596.         if (pq->OSQOut == pq->OSQStart) {             /* Yes, Post as LIFO, Wrap OUT pointer if we ... */
  597.             pq->OSQOut = pq->OSQEnd;                  /*      ... are at the 1st queue entry           */
  598.         }
  599.         pq->OSQOut--;
  600.         *pq->OSQOut = msg;                            /*      Insert message into queue                */
  601.     } else {                                          /* No,  Post as FIFO                             */
  602.         *pq->OSQIn++ = msg;                           /*      Insert message into queue                */
  603.         if (pq->OSQIn == pq->OSQEnd) {                /*      Wrap IN ptr if we are at end of queue    */
  604.             pq->OSQIn = pq->OSQStart;
  605.         }
  606.     }
  607.     pq->OSQEntries++;                                 /* Update the nbr of entries in the queue        */
  608.     OS_EXIT_CRITICAL();
  609.     return (OS_NO_ERR);
  610. }
  611. #endif
  612. /*$PAGE*/
  613. /*
  614. *********************************************************************************************************
  615. *                                        QUERY A MESSAGE QUEUE
  616. *
  617. * Description: This function obtains information about a message queue.
  618. *
  619. * Arguments  : pevent        is a pointer to the event control block associated with the desired queue
  620. *
  621. *              p_q_data      is a pointer to a structure that will contain information about the message
  622. *                            queue.
  623. *
  624. * Returns    : OS_NO_ERR           The call was successful and the message was sent
  625. *              OS_ERR_EVENT_TYPE   If you are attempting to obtain data from a non queue.
  626. *              OS_ERR_PEVENT_NULL  If 'pevent' is a NULL pointer
  627. *********************************************************************************************************
  628. */
  629. #if OS_Q_QUERY_EN > 0
  630. INT8U  OSQQuery (OS_EVENT *pevent, OS_Q_DATA *p_q_data)
  631. {
  632.     OS_Q      *pq;
  633.     INT8U     *psrc;
  634.     INT8U     *pdest;
  635. #if OS_CRITICAL_METHOD == 3                      /* Allocate storage for CPU status register           */
  636.     OS_CPU_SR  cpu_sr;
  637.     cpu_sr = 0;                                            /* Prevent compiler warning                 */
  638. #endif    
  639. #if OS_ARG_CHK_EN > 0
  640.     if (pevent == (OS_EVENT *)0) {                         /* Validate 'pevent'                        */
  641.         return (OS_ERR_PEVENT_NULL);
  642.     }
  643. #endif
  644.     if (pevent->OSEventType != OS_EVENT_TYPE_Q) {          /* Validate event block type                */
  645.         return (OS_ERR_EVENT_TYPE);
  646.     }
  647.     OS_ENTER_CRITICAL();
  648.     p_q_data->OSEventGrp = pevent->OSEventGrp;             /* Copy message queue wait list             */
  649.     psrc                 = &pevent->OSEventTbl[0];
  650.     pdest                = &p_q_data->OSEventTbl[0];
  651. #if OS_EVENT_TBL_SIZE > 0
  652.     *pdest++ = *psrc++;
  653. #endif
  654. #if OS_EVENT_TBL_SIZE > 1
  655.     *pdest++ = *psrc++;
  656. #endif
  657. #if OS_EVENT_TBL_SIZE > 2
  658.     *pdest++ = *psrc++;
  659. #endif
  660. #if OS_EVENT_TBL_SIZE > 3
  661.     *pdest++ = *psrc++;
  662. #endif
  663. #if OS_EVENT_TBL_SIZE > 4
  664.     *pdest++ = *psrc++;
  665. #endif
  666. #if OS_EVENT_TBL_SIZE > 5
  667.     *pdest++ = *psrc++;
  668. #endif
  669. #if OS_EVENT_TBL_SIZE > 6
  670.     *pdest++ = *psrc++;
  671. #endif
  672. #if OS_EVENT_TBL_SIZE > 7
  673.     *pdest   = *psrc;
  674. #endif
  675.     pq = (OS_Q *)pevent->OSEventPtr;
  676.     if (pq->OSQEntries > 0) {
  677.         p_q_data->OSMsg = *pq->OSQOut;                     /* Get next message to return if available  */
  678.     } else {
  679.         p_q_data->OSMsg = (void *)0;
  680.     }
  681.     p_q_data->OSNMsgs = pq->OSQEntries;
  682.     p_q_data->OSQSize = pq->OSQSize;
  683.     OS_EXIT_CRITICAL();
  684.     return (OS_NO_ERR);
  685. }
  686. #endif                                                     /* OS_Q_QUERY_EN                            */
  687. /*$PAGE*/
  688. /*
  689. *********************************************************************************************************
  690. *                                      QUEUE MODULE INITIALIZATION
  691. *
  692. * Description : This function is called by uC/OS-II to initialize the message queue module.  Your
  693. *               application MUST NOT call this function.
  694. *
  695. * Arguments   :  none
  696. *
  697. * Returns     : none
  698. *
  699. * Note(s)    : This function is INTERNAL to uC/OS-II and your application should not call it.
  700. *********************************************************************************************************
  701. */
  702. void  OS_QInit (void)
  703. {
  704. #if OS_MAX_QS == 1
  705.     OSQFreeList         = &OSQTbl[0];                /* Only ONE queue!                                */
  706.     OSQFreeList->OSQPtr = (OS_Q *)0;
  707. #endif
  708. #if OS_MAX_QS >= 2
  709.     INT16U  i;
  710.     OS_Q   *pq1;
  711.     OS_Q   *pq2;
  712.     OS_MemClr((INT8U *)&OSQTbl[0], sizeof(OSQTbl));  /* Clear the queue table                          */
  713.     pq1 = &OSQTbl[0];
  714.     pq2 = &OSQTbl[1];
  715.     for (i = 0; i < (OS_MAX_QS - 1); i++) {          /* Init. list of free QUEUE control blocks        */
  716.         pq1->OSQPtr = pq2;
  717.         pq1++;
  718.         pq2++;
  719.     }
  720.     pq1->OSQPtr = (OS_Q *)0;
  721.     OSQFreeList = &OSQTbl[0];
  722. #endif
  723. }
  724. #endif                                               /* OS_Q_EN                                        */