Windows2003-3790/base/mvdm/wow16/ddeml/dmgq.c
2020-09-30 16:53:55 +02:00

211 lines
5.3 KiB
C

/****************************** Module Header ******************************\
* Module Name: DMGQ.C
*
* DDE Manager queue control functions.
*
* Created: 9/1/89 Sanford Staab
* Modified:5/31/90 Rich Gartland, Aldus (Windows 3.0 port)
*
* This is a general queue manager - yes another one!
* Queues are each allocated within their own segment and have a
* QST structure associated with that heap. Each queue item
* is allocated within the heap segment. The offset of the items
* address combined with an instance count is used as the item ID.
* This is both unique and allows for instant location of an item.
* New items are added to the head of the queue which is a doubly linked
* list. The next links point to more recent entries, the prev pointers
* to older entries. The next of the head is the tail. The prev of the
* tail is the head. All pointers are far.
* Queue Data may be of any structure type that begins identical to
* a QUEUEITEM structure. Functions that require an cbItem perameter
* should be given the size of the specialized queue item structure.
*
* Copyright (c) 1988, 1989 Microsoft Corporation
\***************************************************************************/
#include "ddemlp.h"
/***************************** Private Function ****************************\
*
* Creates a Queue for items of cbItem.
* Returns NULL on error.
*
*
* History:
* Created 9/1/89 Sanfords
\***************************************************************************/
PQST CreateQ(cbItem)
WORD cbItem;
{
QST cq;
PQST pQ;
cq.cItems = 0;
cq.instLast = 0;
cq.cbItem = cbItem;
cq.pqiHead = NULL;
if (!(cq.hheap = DmgCreateHeap(sizeof(QST) + cbItem << 3)))
return(NULL);
if (!(pQ = (PQST)FarAllocMem(cq.hheap, sizeof(QST)))) {
DmgDestroyHeap(cq.hheap);
return(0);
}
*pQ = cq;
return(pQ);
}
/***************************** Private Function ****************************\
*
*
* History:
* Created 9/1/89 Sanfords
\***************************************************************************/
BOOL DestroyQ(pQ)
PQST pQ;
{
if (pQ)
DmgDestroyHeap(pQ->hheap);
return(TRUE);
}
/***************************** Private Function ****************************\
*
* returns a long pointer to the queue item data created. The new item
* is added to the head of the queue. The queue's cbItem specified at
* creation is used for allocation.
*
*
* History:
* Created 9/1/89 Sanfords
\***************************************************************************/
PQUEUEITEM Addqi(pQ)
PQST pQ;
{
PQUEUEITEM pqi;
if ((pqi = (PQUEUEITEM)FarAllocMem(pQ->hheap, pQ->cbItem)) == NULL) {
return(NULL);
}
SEMENTER();
if (pQ->cItems == 0) {
pQ->pqiHead = pqi->prev = pqi->next = pqi;
} else {
pqi->prev = pQ->pqiHead;
pqi->next = pQ->pqiHead->next;
pQ->pqiHead->next->prev = pqi;
pQ->pqiHead->next = pqi;
pQ->pqiHead = pqi;
}
SEMLEAVE();
pQ->cItems++;
pqi->inst = ++pQ->instLast;
return(pqi);
}
/***************************** Private Function ****************************\
*
* The id given is an external LONG id, not an item instance number.
* If id is QID_NEWEST, the head item is deleted.
* If id is QID_OLDEST, the tail item is deleted.
*
*
* History:
* Created 9/1/89 Sanfords
\***************************************************************************/
void Deleteqi(pQ, id)
PQST pQ;
DWORD id;
{
PQUEUEITEM pqi;
SEMENTER();
pqi = Findqi(pQ, id);
if (pqi == NULL) {
SEMLEAVE();
return;
}
pqi->prev->next = pqi->next;
pqi->next->prev = pqi->prev;
if (pqi == pQ->pqiHead)
pQ->pqiHead = pqi->prev;
if (!(--pQ->cItems))
pQ->pqiHead = NULL;
FarFreeMem((LPSTR)pqi);
SEMLEAVE();
}
/***************************** Private Function ****************************\
*
* The id given is an external LONG id, not an item instance number.
*
* if id == QID_NEWEST, returns the head queue data item.
* if id == QID_OLDEST == 0L, returns the tail queue data item.
* if the id is not found or the queue is empty, NULL is returned.
* if found, pqi is returned.
*
*
* History:
* Created 9/1/89 Sanfords
\***************************************************************************/
PQUEUEITEM Findqi(pQ, id)
PQST pQ;
DWORD id;
{
PQUEUEITEM pqi;
SEMCHECKIN();
if (pQ == NULL || pQ->pqiHead == NULL)
return(NULL);
if (id == QID_OLDEST)
return(pQ->pqiHead->next);
if (id == QID_NEWEST)
return(pQ->pqiHead);
if (id) {
pqi = PFROMID(pQ, id);
if (pqi->inst == HIWORD(id)) {
return(pqi);
}
return(NULL);
}
}
/*
* useful for traversing queues and deleting particular stuff in them.
*/
PQUEUEITEM FindNextQi(
PQST pQ,
PQUEUEITEM pqi,
BOOL fDelete)
{
PQUEUEITEM pqiNext;
if (pqi == NULL) {
return(pQ->cItems ? pQ->pqiHead : NULL);
}
pqiNext = pqi->next;
if (fDelete) {
Deleteqi(pQ, MAKEID(pqi));
}
return(pqiNext != pQ->pqiHead && pQ->cItems ? pqiNext : NULL);
}