ncsa-mosaic/libnet/list.c

272 lines
5.2 KiB
C
Raw Normal View History

2010-03-08 04:55:21 -06:00
/*
* Copyright (C) 1992, Board of Trustees of the University of Illinois.
*
* Permission is granted to copy and distribute source with out fee.
* Commercialization of this product requires prior licensing
* from the National Center for Supercomputing Applications of the
* University of Illinois. Commercialization includes the integration of this
* code in part or whole into a product for resale. Free distribution of
* unmodified source and use of NCSA software is not considered
2010-03-08 04:55:21 -06:00
* commercialization.
*
*/
/*
* list.c: This module contains list manipulation routines that cunstruct
* and maintain a linked list of data items. The record at the head of each
* list contains pointers to the head, tail, and current list position.
* the list itsself is doubly linked with both next and previous pointers.
*/
#include <stdio.h>
#include "listP.h"
#ifndef MALLOC
#define MALLOC malloc
#define FREE free
#endif
#define NIL 0
static void ListPrintErr(s)
char *s;
{
fprintf(stderr,"%s",s);
}
/*
* This function returns the data located at the head of the linked list,
* or NIL if the list is empty. As a side effect current is also set to
* the head of the list.
*/
char *ListHead(theList)
List theList;
{
if (!theList)
return(NIL);
theList->current = theList->head;
if (theList->head)
return(theList->head->value);
else
return(NIL);
}
/*
* This function returns the data located at the tail of the linked list,
* or NIL if the list is empty. As a side effect current is also set to
* the tail of the list.
*/
char *ListTail(theList)
List theList;
{
if (!theList)
return(NIL);
theList->current = theList->tail;
if (theList->tail)
return(theList->tail->value);
else
return(NIL);
}
/*
* This function returns the data located at the current position in the
* linked list, or NIL if the list is empty.
*/
char *ListCurrent(theList)
List theList;
{
if (!theList)
return(NIL);
if (theList->current)
return(theList->current->value);
else
return(NIL);
}
/*
* This function returns the data located at the next element of the linked
* list after the current position, or NIL if the list is empty, or you
* are at its end.
* As a side effect current is also set to the next entry in the list.
*/
char *ListNext(theList)
List theList;
{
if (!theList)
return(NIL);
if (theList->current) {
theList->current = theList->current->next;
return(ListCurrent(theList));
}
else
return(NIL);
}
/*
* This function returns the data located at the previous element of the linked
* list before the current position, or NIL if the list is empty.
* As a side effect current is also set to the previous entry in the list.
*/
char *ListPrev(theList)
List theList;
{
if (!theList)
return(NIL);
if (theList->current) {
theList->current = theList->current->prev;
return(ListCurrent(theList));
}
else
return(NIL);
}
/*
* Cretae a list head and initialize it to NIL.
*/
List ListCreate()
{
List retVal;
retVal = (List) MALLOC(sizeof(struct LISTSTRUCT));
retVal->head = NIL;
retVal->tail = NIL;
retVal->current = NIL;
return(retVal);
}
/*
* Destroy a list head, and free all associated memory.
*/
void ListDestroy(theList)
List theList;
{
struct LISTINSTANCE *l;
struct LISTINSTANCE *m;
if (!theList)
return;
l = theList->head;
while(l) {
m = l;
l = l->next;
FREE(m);
}
FREE(theList);
2010-03-08 04:55:21 -06:00
}
/*
* Add an entry to the end of the linked list. Current is changed to point to
* the added element.
*/
int ListAddEntry(theList,v)
/* return 0 on failure */
List theList;
char *v; /* data to be added */
{
struct LISTINSTANCE *l;
if (!(l =(struct LISTINSTANCE *) MALLOC(sizeof(struct LISTINSTANCE)))){
ListPrintErr("Out of Memory\n");
return(0);
}
l->value = v;
l->next = NIL;
l->prev = NIL;
if (theList->head == NIL)
theList->tail = theList->head = l;
else {
theList->tail->next = l;
l->prev = theList->tail;
theList->tail = l;
}
theList->current = l;
2010-03-08 04:55:21 -06:00
return(1);
}
/*
* Search the list for an entry with a matching value field, and return
* a pointer to that list element. Current is changed to point to the
* element returned.
*/
static struct LISTINSTANCE *SearchListByValue(theList,v)
List theList;
char *v;
{
struct LISTINSTANCE *l;
l = theList->head;
while (l != NIL) {
if (l->value == v) {
theList->current = l;
return(l);
}
else {
l = l->next;
}
}
theList->current = l;
return(NIL);
2010-03-08 04:55:21 -06:00
}
/*
* Find the list entry with a matching value field, and delete it
* from the list. Set current to point to the element after the deleted
* element in the list.
*/
int ListDeleteEntry(theList,v)
/* removes the first occurance of v from the list */
/* return 0 if value not in list else 1 */
List theList;
char *v;
{
struct LISTINSTANCE *l;
char *retV;
if (!(l = SearchListByValue(theList,v)))
return(0);
if (l->prev)
l->prev->next = l->next;
else
theList->head = l->next;
if (l->next)
l->next->prev = l->prev;
else
theList->tail = l->prev;
theList->current = l->next;
retV = l->value;
FREE(l);
return(1);
}
int ListMakeEntryCurrent(theList,entry)
/* return 0 on failure */
List theList;
char *entry;
{
struct LISTINSTANCE *l;
if (theList) {
if (!(l = SearchListByValue(theList,entry)))
return(0);
theList->current = l;
return(1);
}
return(0);
}