向消息队列发送一个消息(后进先出LIFO),OSQPostFront()

来源:互联网 发布:access录入数据 编辑:程序博客网 时间:2024/05/17 19:16
OSQPostFront()函数和OSQPost()基本上是一样的,只是在插入新的消息到消息队列中时,使用.OSQOut作为指向下一个插入消息的单元的指针,而不是.OSQIn。程序清单 L6.24是它的源代码。值得注意的是,.OSQOut指针指向的是已经插入了消息指针的单元,所以再插入新的消息指针前,必须先将.OSQOut指针在消息队列中前移一个单元。如果.OSQOut指针指向的当前单元是队列中的第一个单元[L6.24(1)],这时再前移就会发生越界,需要特别地将该指针指向队列的末尾[L6.24(2)]。由于.OSQEnd指向的是消息队列中最后一个单元的下一个单元,因此.OSQOut必须被调整到指向队列的有效范围内[L6.24(3)]。因为QSQPend()函数取出的消息是由OSQPend()函数刚刚插入的,因此OSQPostFront()函数实现了一个LIFO队列。程序清单 L6.24 向消息队列发送一条消息(LIFO)INT8U OSQPostFront (OS_EVENT *pevent, void *msg){    OS_Q   *pq;    OS_ENTER_CRITICAL();    if (pevent->OSEventType != OS_EVENT_TYPE_Q) {        OS_EXIT_CRITICAL();        return (OS_ERR_EVENT_TYPE);    }    if (pevent->OSEventGrp) {        OSEventTaskRdy(pevent, msg, OS_STAT_Q);        OS_EXIT_CRITICAL();        OSSched();        return (OS_NO_ERR);    } else {        pq = pevent->OSEventPtr;        if (pq->OSQEntries >= pq->OSQSize) {            OS_EXIT_CRITICAL();            return (OS_Q_FULL);        } else {            if (pq->OSQOut == pq->OSQStart) {               (1)                pq->OSQOut = pq->OSQEnd;                    (2)            }            pq->OSQOut--;                                   (3)            *pq->OSQOut = msg;            pq->OSQEntries++;            OS_EXIT_CRITICAL();        }        return (OS_NO_ERR);    }}

0 0