2003-05-11 14:42:36 +02:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* liblx.h: definitions necessary for the liblx package
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* $Revision: 1.2 $
|
|
|
|
* $Date: 2001/06/25 19:39:00 $
|
|
|
|
* $Author: hakenes $
|
|
|
|
*
|
|
|
|
* (C) 1992-2001 Rolf Hakenes <hakenes@hippomi.de>, under the GNU GPL.
|
|
|
|
*
|
|
|
|
* liblx is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* liblx is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You may have received a copy of the GNU General Public License
|
|
|
|
* along with liblx; see the file COPYING. If not, write to the
|
|
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
*/
|
2001-08-15 13:56:11 +02:00
|
|
|
|
|
|
|
#ifndef LIBLX_H
|
|
|
|
#define LIBLX_H
|
|
|
|
|
|
|
|
#ifndef NULL
|
|
|
|
#define NULL 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* list support structures
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
struct NODE
|
|
|
|
{
|
|
|
|
struct NODE *Succ;
|
|
|
|
struct NODE *Pred;
|
|
|
|
char *Name;
|
|
|
|
unsigned short HashKey;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LIST
|
|
|
|
{
|
|
|
|
struct NODE *Head;
|
|
|
|
struct NODE *Tail;
|
|
|
|
char *Name;
|
|
|
|
unsigned long Size;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* memory managment structures
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
struct MEM_ENTRY
|
|
|
|
{
|
|
|
|
struct MEM_ENTRY *Succ;
|
|
|
|
struct MEM_ENTRY *Pred;
|
|
|
|
unsigned long Size;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct MEM_CHUNK
|
|
|
|
{
|
|
|
|
struct MEM_CHUNK *Succ;
|
|
|
|
struct MEM_CHUNK *Pred;
|
|
|
|
unsigned long Size;
|
|
|
|
struct MEM_ENTRY *FirstFreeMemEntry;
|
|
|
|
struct MEM_ENTRY *FirstUsedMemEntry;
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* list functions (package xList)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
unsigned short xHashKey (char *);
|
|
|
|
struct LIST *xNewList (char *);
|
|
|
|
struct NODE *xNewNode (char *, unsigned long);
|
|
|
|
struct NODE *xFindName (struct LIST *, char *);
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* memory management
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
void xMemAllo (unsigned long, unsigned char **);
|
|
|
|
void xMemFre (unsigned char *);
|
|
|
|
void xMemFreeAll (struct MEM_CHUNK **);
|
|
|
|
void xMemMerge (struct MEM_CHUNK **);
|
|
|
|
struct MEM_CHUNK **xGetRemember (void);
|
|
|
|
void xSetRemember (struct MEM_CHUNK **);
|
|
|
|
void xPrintMemList (struct MEM_CHUNK **);
|
|
|
|
unsigned long xGetMemSize (struct MEM_CHUNK **);
|
|
|
|
extern unsigned long xAllocatedMemory;
|
|
|
|
char *xSetText (char *);
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#define MEM_CHUNK_SIZE 65536
|
|
|
|
|
|
|
|
#define xMemAlloc(size, ptr) \
|
|
|
|
xMemAllo (((unsigned long)((size))), ((unsigned char **)((ptr))))
|
|
|
|
#define xMemFree(ptr) xMemFre (((unsigned char *)((ptr))))
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* list support macros
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xCreateNode (NodeStruct,Name) allocates a correctly sized and |
|
|
|
|
| typed node struct. |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xCreateNode(NodeStruct,Name) \
|
|
|
|
(NodeStruct) = (void *) xNewNode(Name, sizeof(*(NodeStruct)))
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xSize (List) scans for the ->Size field of a list struct |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xSize(List) ((List) ? ((struct LIST *)(List))->Size : 0)
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xName (NodeStruct) scans for the ->Node.Name of a node struct |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xName(NodeStruct) (((struct NODE *)(NodeStruct))->Name)
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xSucc (NodeStruct) scans for the ->Node.Succ of a node struct |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xSucc(NodeStruct) (((struct NODE *)(NodeStruct))->Succ)
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xPred (NodeStruct) scans for the ->Node.Pred of a node struct |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xPred(NodeStruct) (((struct NODE *)(NodeStruct))->Pred)
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xForeach(List,NodeStruct) builds a loop to process each list |
|
|
|
|
| element. |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xForeach(List,NodeStruct) \
|
|
|
|
if (List) for ((NodeStruct) = (void *) ((struct LIST *)(List))->Head; \
|
|
|
|
(NodeStruct); (NodeStruct) = (void *) xSucc (NodeStruct))
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xForeachReverse(List,NodeStruct) builds a loop to process each |
|
|
|
|
| element in reverse order. |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xForeachReverse(List,NodeStruct) \
|
|
|
|
if (List) for ((NodeStruct) = (void *) ((struct LIST *)(List))->Tail; \
|
|
|
|
NodeStruct; (NodeStruct) = (void *) xPred (NodeStruct))
|
|
|
|
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------*
|
|
|
|
| |
|
|
|
|
| xRemove(List,NodeStruct) unchains a node struct out of a list. |
|
|
|
|
| |
|
|
|
|
*---------------------------------------------------------------------*/
|
|
|
|
#define xRemove(List,NodeStruct) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
struct LIST *TmpList; \
|
|
|
|
\
|
|
|
|
TmpNode = ((struct NODE *)(NodeStruct)); \
|
|
|
|
TmpList = ((struct LIST *)(List)); \
|
|
|
|
\
|
|
|
|
if (TmpNode->Pred) \
|
|
|
|
(TmpNode->Pred)->Succ = TmpNode->Succ; \
|
|
|
|
else TmpList->Head = TmpNode->Succ; \
|
|
|
|
if (TmpNode->Succ) \
|
|
|
|
(TmpNode->Succ)->Pred = TmpNode->Pred; \
|
|
|
|
else TmpList->Tail = TmpNode->Pred; \
|
|
|
|
TmpList->Size --; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* function : xAddHead *
|
|
|
|
* *
|
|
|
|
* arguments : List - pointer to a LIST structure *
|
|
|
|
* *
|
|
|
|
* Node - pointer to a NODE structure *
|
|
|
|
* *
|
|
|
|
*-----------------------------------------------------------------------*
|
|
|
|
* *
|
|
|
|
* xAddHead() inserts 'Node' at the head of 'List'. *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
#define xAddHead(List, NodeStruct) \
|
|
|
|
do { \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
struct LIST *TmpList; \
|
|
|
|
\
|
|
|
|
TmpNode = ((struct NODE *)(NodeStruct)); \
|
|
|
|
TmpList = ((struct LIST *)(List)); \
|
|
|
|
\
|
|
|
|
if (TmpList->Head) { \
|
|
|
|
TmpNode->Pred = NULL; \
|
|
|
|
TmpNode->Succ = TmpList->Head; \
|
|
|
|
(TmpList->Head)->Pred = TmpNode; \
|
|
|
|
TmpList->Head = TmpNode; } \
|
|
|
|
else { \
|
|
|
|
TmpList->Head = TmpNode; \
|
|
|
|
TmpList->Tail = TmpNode; \
|
|
|
|
TmpNode->Pred = NULL; \
|
|
|
|
TmpNode->Succ = NULL; } \
|
|
|
|
TmpList->Size++; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* function : xAddTail *
|
|
|
|
* *
|
|
|
|
* arguments : List - pointer to a LIST structure *
|
|
|
|
* *
|
|
|
|
* Node - pointer to a NODE structure *
|
|
|
|
* *
|
|
|
|
*-----------------------------------------------------------------------*
|
|
|
|
* *
|
|
|
|
* xAddTail() inserts 'Node' at the tail of 'List'. *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
#define xAddTail(List, NodeStruct) \
|
|
|
|
do { \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
struct LIST *TmpList; \
|
|
|
|
\
|
|
|
|
TmpNode = ((struct NODE *)(NodeStruct)); \
|
|
|
|
TmpList = ((struct LIST *)(List)); \
|
|
|
|
\
|
|
|
|
if (TmpList->Head) { \
|
|
|
|
TmpNode->Succ = NULL; \
|
|
|
|
TmpNode->Pred = TmpList->Tail; \
|
|
|
|
(TmpList->Tail)->Succ = TmpNode; \
|
|
|
|
TmpList->Tail = TmpNode; } \
|
|
|
|
else { \
|
|
|
|
TmpList->Head = TmpNode; \
|
|
|
|
TmpList->Tail = TmpNode; \
|
|
|
|
TmpNode->Pred = NULL; \
|
|
|
|
TmpNode->Succ = NULL; } \
|
|
|
|
TmpList->Size++; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* function : xRemHead *
|
|
|
|
* *
|
|
|
|
* arguments : List - pointer to a LIST structure *
|
|
|
|
* *
|
|
|
|
*-----------------------------------------------------------------------*
|
|
|
|
* *
|
|
|
|
* xRemHead() removes a Node from head of 'List'. *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
#define xRemHead(List) \
|
|
|
|
do { \
|
|
|
|
struct LIST *TmpList; \
|
|
|
|
\
|
|
|
|
TmpList = ((struct LIST *)(List)); \
|
|
|
|
\
|
|
|
|
if (TmpList->Head) \
|
|
|
|
{ \
|
|
|
|
TmpList->Head = (TmpList->Head)->Succ; \
|
|
|
|
if (TmpList->Head) (TmpList->Head)->Pred = NULL; \
|
|
|
|
else TmpList->Tail = NULL; \
|
|
|
|
TmpList->Size--; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* function : xRemTail *
|
|
|
|
* *
|
|
|
|
* arguments : List - pointer to a LIST structure *
|
|
|
|
* *
|
|
|
|
*-----------------------------------------------------------------------*
|
|
|
|
* *
|
|
|
|
* xRemTail() removes a Node from the tail of 'List'. *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
#define xRemTail(List) \
|
|
|
|
do { \
|
|
|
|
struct LIST *TmpList; \
|
|
|
|
\
|
|
|
|
TmpList = ((struct LIST *)(List)); \
|
|
|
|
\
|
|
|
|
if (TmpList->Tail) \
|
|
|
|
{ \
|
|
|
|
TmpList->Tail = (TmpList->Tail)->Pred; \
|
|
|
|
if (TmpList->Tail) (TmpList->Tail)->Succ = NULL; \
|
|
|
|
else TmpList->Head = NULL; \
|
|
|
|
TmpList->Size--; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************
|
|
|
|
* *
|
|
|
|
* function : xConCat *
|
|
|
|
* *
|
|
|
|
* arguments : DestinationList - pointer to the destination *
|
|
|
|
* LIST structure *
|
|
|
|
* *
|
|
|
|
* SourceList - pointer to the source LIST structure *
|
|
|
|
* *
|
|
|
|
*-----------------------------------------------------------------------*
|
|
|
|
* *
|
|
|
|
* xConCat() concats 'SourceList' with 'DestinationList' and clears *
|
|
|
|
* 'SourceList'. *
|
|
|
|
* *
|
|
|
|
*************************************************************************/
|
|
|
|
#define xConCat(DestinationList, SourceList) \
|
|
|
|
do { \
|
|
|
|
struct LIST *SrcList; \
|
|
|
|
struct LIST *DstList; \
|
|
|
|
\
|
|
|
|
SrcList = ((struct LIST *)(SourceList)); \
|
|
|
|
DstList = ((struct LIST *)(DestinationList)); \
|
|
|
|
\
|
|
|
|
if (DstList && SrcList) \
|
|
|
|
{ \
|
|
|
|
if (DstList->Head) { \
|
|
|
|
if (SrcList->Head) { \
|
|
|
|
(DstList->Tail)->Succ = SrcList->Head; \
|
|
|
|
(SrcList->Head)->Pred = DstList->Tail; \
|
|
|
|
DstList->Tail = SrcList->Tail; \
|
|
|
|
DstList->Size += SrcList->Size; \
|
|
|
|
SrcList->Size = 0; \
|
|
|
|
SrcList->Head = NULL; \
|
|
|
|
SrcList->Tail = NULL; } } \
|
|
|
|
else { \
|
|
|
|
DstList->Head = SrcList->Head; \
|
|
|
|
DstList->Tail = SrcList->Tail; \
|
|
|
|
DstList->Size += SrcList->Size; \
|
|
|
|
SrcList->Size = 0; \
|
|
|
|
SrcList->Head = NULL; \
|
|
|
|
SrcList->Tail = NULL; } \
|
|
|
|
} \
|
|
|
|
else if (SrcList) ((struct LIST *)(DestinationList)) = SrcList; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define xJoinList(SourceList, DestinationList, NodeStruct) \
|
|
|
|
do { \
|
|
|
|
struct NODE *KeyNode; \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
struct LIST *SrcList; \
|
|
|
|
struct LIST *DstList; \
|
|
|
|
\
|
|
|
|
KeyNode = ((struct NODE *)(NodeStruct)); \
|
|
|
|
SrcList = ((struct LIST *)(SourceList)); \
|
|
|
|
DstList = ((struct LIST *)(DestinationList)); \
|
|
|
|
\
|
|
|
|
if (SrcList->Head) \
|
|
|
|
{ \
|
|
|
|
TmpNode = KeyNode->Succ; \
|
|
|
|
KeyNode->Succ = SrcList->Head; \
|
|
|
|
SrcList->Tail->Succ = TmpNode; \
|
|
|
|
SrcList->Head->Pred = KeyNode; \
|
|
|
|
if (!TmpNode) DstList->Tail = SrcList->Tail; \
|
|
|
|
else TmpNode->Pred = SrcList->Tail; \
|
|
|
|
DstList->Size += SrcList->Size; \
|
|
|
|
SrcList->Size = 0; \
|
|
|
|
SrcList->Head = NULL; \
|
|
|
|
SrcList->Tail = NULL; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define xJoin(SourceNode, DestinationList, NodeStruct) \
|
|
|
|
do { \
|
|
|
|
struct NODE *KeyNode; \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
struct NODE *SrcNode; \
|
|
|
|
struct LIST *DstList; \
|
|
|
|
\
|
|
|
|
KeyNode = ((struct NODE *)(NodeStruct)); \
|
|
|
|
SrcNode = ((struct NODE *)(SourceNode)); \
|
|
|
|
DstList = ((struct LIST *)(DestinationList)); \
|
|
|
|
\
|
|
|
|
if (SrcNode) \
|
|
|
|
{ \
|
|
|
|
TmpNode = KeyNode->Succ; \
|
|
|
|
KeyNode->Succ = SrcNode; \
|
|
|
|
SrcNode->Succ = TmpNode; \
|
|
|
|
SrcNode->Pred = KeyNode; \
|
|
|
|
if (!TmpNode) DstList->Tail = SrcNode; \
|
|
|
|
else TmpNode->Pred = SrcNode; \
|
|
|
|
DstList->Size += 1; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define xClearList(SrcList) \
|
|
|
|
do { \
|
|
|
|
(SrcList)->Size = 0; \
|
|
|
|
(SrcList)->Head = NULL; \
|
|
|
|
(SrcList)->Tail = NULL; \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define xSetName(nodestruct, name) \
|
|
|
|
do { \
|
|
|
|
struct NODE *TmpNode; \
|
|
|
|
\
|
|
|
|
TmpNode = (struct NODE *) (nodestruct); \
|
|
|
|
\
|
|
|
|
TmpNode->Name = xSetText (name); \
|
|
|
|
TmpNode->HashKey = xHashKey (name); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#endif
|