nodelist.c   [plain text]


/*
    This software may only be used by you under license from AT&T Corp.
    ("AT&T").  A copy of AT&T's Source Code Agreement is available at
    AT&T's Internet website having the URL:
    <http://www.research.att.com/sw/tools/graphviz/license/source.html>
    If you received this software without first entering into a license
    with AT&T, you have an infringing copy of this software and cannot use
    it without violating AT&T's intellectual property rights.
*/
#pragma prototyped

#include	<assert.h>
#include	"nodelist.h"
#include	"circular.h"

static nodelistitem_t* 
init_nodelistitem(Agnode_t* n)
{
	nodelistitem_t* p = NEW(nodelistitem_t);
	p->curr = n;
	return p;
}

nodelist_t*
mkNodelist()
{
	nodelist_t* list = NEW(nodelist_t);
	return list;
}

void
freeNodelist(nodelist_t* list)
{
	nodelistitem_t* temp;
	nodelistitem_t* next;

	if(!list)
		return;

	for (temp = list->first; temp; temp = next) {
		next = temp->next;
		free(temp);
	}
	free(list);
}

/* appendNodelist:
 * Add node after one.
 * If one == NULL, add n to end.
 */
void
appendNodelist(nodelist_t* list, nodelistitem_t* one, Agnode_t* n)
{
	nodelistitem_t* np = init_nodelistitem(n);

	list->sz++;
	if (!one) one = list->last;
	if (one == list->last) {
		if (one) one->next = np;
		else list->first = np;
		np->prev = one;
		np->next = NULL;
		list->last = np;
	}
	else {
		nodelistitem_t* temp = one->next;
		one->next = np;
		np->prev = one;
		temp->prev = np;
		np->next = temp;
	}
}

#ifdef OLD
/* addNodelist:
 * Adds node to end of list if not already present.
 */
void
addNodelist(nodelist_t* list, Agnode_t* n)
{
	nodelistitem_t* temp;
	nodelistitem_t* item = 0;

	for(temp = list->first; temp; temp = temp->next) {
		if(n == temp->curr) {
			item = temp;
			break;
		}
	}

	if(item) return;

	item = init_nodelistitem(n);
	if (list->last) {
		list->last->next = item;
		item->prev = list->last;
	}
	else
		list->first = item;
	list->last = item;
	list->sz++;
}

void
removeNodelist(nodelist_t* list, Agnode_t* n)
{
	nodelistitem_t* temp;

	for (temp = list->first; temp; temp = temp->next) {
		if(n == temp->curr) {
			list->sz--;
			if(temp->prev == NULL) { /* the first node */
				list->first = temp->next;
			}
			else {
				temp->prev->next = temp->next;
			}
			if(temp == list->last) { /* the last node */
				list->last = temp->prev;				
			} 
			else {
				temp->next->prev = temp->prev;
			}
			free(temp);
			return;
		}
	}
}
#endif

/* reverseNodelist;
 * Destructively reverse a list.
 */
nodelist_t*
reverseNodelist(nodelist_t* list)
{
	nodelistitem_t*	temp;
	nodelistitem_t*	p;
	
	for(p = list->first; p; p = p->prev) {
		temp = p->next;
		p->next = p->prev;
		p->prev = temp;
	}
	temp = list->last;
	list->last = list->first;
	list->first = temp;
	return list;
}

/* realignNodelist:
 * Make np new front of list, with current last hooked to
 * current first. I.e., make list circular, then cut between
 * np and np->prev.
 */
void
realignNodelist(nodelist_t* list, nodelistitem_t* np)
{
	nodelistitem_t* temp;
	nodelistitem_t* prev;
	
	if(np == list->first)
		return;

	temp = list->first;
	prev = np->prev;

	list->first = np;
	np->prev = NULL;
	list->last->next = temp;
	temp->prev = list->last;
	list->last = prev;
	prev->next = NULL;
}

/* cloneNodelist:
 * Create a copy of list.
 */
nodelist_t*
cloneNodelist(nodelist_t* list)
{
	nodelist_t* newlist = mkNodelist();
	nodelistitem_t*	temp;
	nodelistitem_t*	prev = 0;

	for(temp = list->first; temp; temp = temp->next) {
		appendNodelist(newlist, prev, temp->curr);
		prev = newlist->last;
	}
	return newlist;
}

/* insertNodelist:
 * Remove cn. Then, insert cn before neighbor if pos == 0 and 
 * after neighbor otherwise.
 */
void
insertNodelist(nodelist_t* list, Agnode_t* cn, Agnode_t* neighbor, int pos)
{
	nodelistitem_t* temp;
	nodelistitem_t* prev;
	nodelistitem_t* next;
	nodelistitem_t* actual = 0;
	
	for(temp = list->first; temp; temp = temp->next) {
		if(temp->curr == cn) {
			actual = temp;
			prev = actual->prev;
			next = actual->next;
			if(prev)     /* not first */
				prev->next = next;
			else
				list->first = next;

			if(next)     /* not last */
				next->prev = prev;
			else
				list->last = prev;
			break;
		}
	}
	assert (actual);

	prev = NULL;
	for(temp = list->first; temp; temp = temp->next) {
		if(temp->curr == neighbor) {
			if(pos == 0) {
				if(temp == list->first) {
					list->first = actual;
					actual->next = temp;
					actual->prev = NULL;
					temp->prev = actual;
					return;
				}				
				prev->next = actual;
				actual->prev = prev;
				actual->next = temp;
				temp->prev = actual;
				return;
			} else {
				if(temp == list->last) {
					list->last = actual;
					actual->next = NULL;
					actual->prev = temp;
					temp->next = actual;
					return;
				}
				actual->prev = temp;
				actual->next = temp->next;
				temp->next->prev = actual;
				temp->next = actual;
				return;
			}
			break;
		}
		prev = temp;
	}
}

int
sizeNodelist(nodelist_t* list)
{
	return list->sz;
#ifdef OLD
	int i = 0;
	nodelistitem_t* temp = NULL;

	temp = list->first;
	while(temp) {
		i++;
		temp = temp->next;
	}
	return i;
#endif
}

#ifdef OLD
/* node_exists:
 * Return true if node is in list.
 */
int
node_exists(nodelist_t* list, Agnode_t* n)
{
	nodelistitem_t* temp;

	for(temp = list->first; temp; temp = temp->next) {
		if(temp->curr == n) {
			return 1;
		}
	}
	return 0;
}

/* nodename_exists:
 * Return true if node with given name is in list.
 * Assumes n == np->name for some node np;
 */
int
nodename_exists(nodelist_t* list, char* n)
{
	nodelistitem_t* temp;

	temp = list->first;

	for(temp = list->first; temp; temp = temp->next) {
		if(temp->curr->name == n) {
			return 1;
		}
	}
	return 0;
}
#endif

/* node_position:
 * Returns index of node n in list, starting at 0.
 * Returns -1 if not in list.
 */
int
node_position(nodelist_t* list, Agnode_t* n)
{
	return POSITION(n); 
#ifdef OLD
	nodelistitem_t* temp;
	int           i = 0;
	char*         name = n->name;

	for (temp = list->first; temp; temp = temp->next) {
		if(temp->curr->name == name) {
			return i;
		}
		i++;
	}
	return -1;	
#endif
}

/* concatNodelist:
 * attach l2 to l1.
 */
static void
concatNodelist(nodelist_t* l1, nodelist_t* l2)
{
	if (l2->first) {
		if (l2->first) {
			l1->last->next = l2->first;
			l2->first->prev = l1->last;
			l1->last = l2->last;
			l1->sz += l2->sz;
		}
    	else {
	  		*l1 = *l2;
		}
	}
}

/* reverse_append;
 * Create l1 @ (rev l2)
 * Destroys and frees l2.
 */
void 
reverseAppend(nodelist_t* l1, nodelist_t* l2)
{
	l2 = reverseNodelist (l2);
	concatNodelist (l1, l2);
	free (l2);
}

#ifdef DEBUG
void
printNodelist(nodelist_t* list)
{
	nodelistitem_t* temp = NULL;

	temp = list->first;
	while(temp != NULL) {
		fprintf(stderr, "%s ", temp->curr->name);
		temp = temp->next;
	}
	fputs("\n", stderr);
}
#endif