QP/C
qa_lifo.c
Go to the documentation of this file.
00001 /*****************************************************************************
00002 * Product: QF/C
00003 * Last Updated for Version: 4.4.00
00004 * Date of the Last Update:  Jan 14, 2012
00005 *
00006 *                    Q u a n t u m     L e a P s
00007 *                    ---------------------------
00008 *                    innovating embedded systems
00009 *
00010 * Copyright (C) 2002-2012 Quantum Leaps, LLC. All rights reserved.
00011 *
00012 * This software may be distributed and modified under the terms of the GNU
00013 * General Public License version 2 (GPL) as published by the Free Software
00014 * Foundation and appearing in the file GPL.TXT included in the packaging of
00015 * this file. Please note that GPL Section 2[b] requires that all works based
00016 * on this software must also be made publicly available under the terms of
00017 * the GPL ("Copyleft").
00018 *
00019 * Alternatively, this software may be distributed and modified under the
00020 * terms of Quantum Leaps commercial licenses, which expressly supersede
00021 * the GPL and are specifically designed for licensees interested in
00022 * retaining the proprietary status of their code.
00023 *
00024 * Contact information:
00025 * Quantum Leaps Web site:  http://www.quantum-leaps.com
00026 * e-mail:                  info@quantum-leaps.com
00027 *****************************************************************************/
00028 #include "qf_pkg.h"
00029 #include "qassert.h"
00030 
00031 Q_DEFINE_THIS_MODULE("qa_lifo")
00032 
00033 
00042 /*..........................................................................*/
00043 void QActive_postLIFO(QActive *me, QEvent const *e) {
00044     QF_CRIT_STAT_
00045     QF_CRIT_ENTRY_();
00046 
00047     QS_BEGIN_NOCRIT_(QS_QF_ACTIVE_POST_LIFO, QS_aoObj_, me)
00048         QS_TIME_();                                            /* timestamp */
00049         QS_SIG_(e->sig);                        /* the signal of this event */
00050         QS_OBJ_(me);                                  /* this active object */
00051         QS_U8_(QF_EVT_POOL_ID_(e));             /* the pool Id of the event */
00052         QS_U8_(QF_EVT_REF_CTR_(e));           /* the ref count of the event */
00053         QS_EQC_(me->eQueue.nFree);                /* number of free entries */
00054         QS_EQC_(me->eQueue.nMin);             /* min number of free entries */
00055     QS_END_NOCRIT_()
00056 
00057     if (QF_EVT_POOL_ID_(e) != (uint8_t)0) {          /* is it a pool event? */
00058         QF_EVT_REF_CTR_INC_(e);          /* increment the reference counter */
00059     }
00060 
00061     if (me->eQueue.frontEvt == (QEvent *)0) {               /* empty queue? */
00062         me->eQueue.frontEvt = e;                  /* deliver event directly */
00063         QACTIVE_EQUEUE_SIGNAL_(me);               /* signal the event queue */
00064     }
00065     else {            /* queue is not empty, leave event in the ring-buffer */
00066             /* the queue must be able to accept the event (cannot overflow) */
00067         Q_ASSERT(me->eQueue.nFree != (QEQueueCtr)0);
00068 
00069         ++me->eQueue.tail;
00070         if (me->eQueue.tail == me->eQueue.end) {  /* need to wrap the tail? */
00071             me->eQueue.tail = (QEQueueCtr)0;                 /* wrap around */
00072         }
00073 
00074         QF_PTR_AT_(me->eQueue.ring, me->eQueue.tail) = me->eQueue.frontEvt;
00075         me->eQueue.frontEvt = e;                      /* put event to front */
00076 
00077         --me->eQueue.nFree;                 /* update number of free events */
00078         if (me->eQueue.nMin > me->eQueue.nFree) {
00079             me->eQueue.nMin = me->eQueue.nFree;    /* update minimum so far */
00080         }
00081     }
00082     QF_CRIT_EXIT_();
00083 }
00084