215 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			215 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* $Id$ */
 | 
						|
/*
 | 
						|
 * (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | 
						|
 * See the copyright notice in the ACK home directory, in the file "Copyright".
 | 
						|
 */
 | 
						|
/*  L O N G   S E T S
 | 
						|
 *
 | 
						|
 *  L S E T . C
 | 
						|
 */
 | 
						|
 | 
						|
 | 
						|
#include <stdlib.h>
 | 
						|
#include "types.h"
 | 
						|
#include "lset.h"
 | 
						|
#include "alloc.h"
 | 
						|
#include "debug.h"
 | 
						|
 | 
						|
 | 
						|
/* A 'long' set is represented as a linear list of 'elemholder'
 | 
						|
 * records. Every such record contains a pointer to an element
 | 
						|
 * of the set and to the next elemholder. An empty set is
 | 
						|
 * represented as a null pointer.
 | 
						|
 * An element of a long set must be of some pointer type or,
 | 
						|
 * in any case, must have the size of a pointer. Note that
 | 
						|
 * the strict typing rules are not obeyed here.
 | 
						|
 * This package implements the usual operations on sets.
 | 
						|
 * The name of every operation is preceeded by a 'L' to
 | 
						|
 * distinguish it from the operation on 'compact' (bitvector)
 | 
						|
 * sets with a similar name.
 | 
						|
 */
 | 
						|
 | 
						|
 | 
						|
lset Lempty_set()
 | 
						|
{
 | 
						|
	return ((lset) 0);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Lis_elem(x,s)
 | 
						|
	register Lelem_t x;
 | 
						|
	register lset    s;
 | 
						|
{
 | 
						|
 | 
						|
	/* Search the list to see if x is an element of s */
 | 
						|
	while (s != (elem_p) 0) {
 | 
						|
		if (s->e_elem == x) {
 | 
						|
			return TRUE;
 | 
						|
		}
 | 
						|
		s = s->e_next;
 | 
						|
	}
 | 
						|
	return FALSE;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Ladd(x,s_p)
 | 
						|
	Lelem_t x;
 | 
						|
	lset    *s_p;
 | 
						|
{
 | 
						|
	/* add x to a set. Note that the set is given as in-out
 | 
						|
	 * parameter, because it may be changed.
 | 
						|
	 */
 | 
						|
 | 
						|
	elem_p t;
 | 
						|
 | 
						|
	if (!Lis_elem(x,*s_p)) {
 | 
						|
		t = newelem();  /* allocate a new elemholder */
 | 
						|
		t->e_elem = x;
 | 
						|
		t->e_next = *s_p;  /* insert it at the head of the list */
 | 
						|
		*s_p = t;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Lremove(x,s_p)
 | 
						|
	Lelem_t x;
 | 
						|
	lset    *s_p;
 | 
						|
{
 | 
						|
	/* Remove x from a set. If x was not an element of
 | 
						|
	 * the set, nothing happens.
 | 
						|
	 */
 | 
						|
 | 
						|
	register elem_p *epp, ep;
 | 
						|
	lset s;
 | 
						|
 | 
						|
	s = *s_p;
 | 
						|
	epp = &s;
 | 
						|
	while ((ep = *epp) != (elem_p) 0) {
 | 
						|
		if (ep->e_elem == x) {
 | 
						|
			*epp = ep->e_next;
 | 
						|
			oldelem(ep);
 | 
						|
			break;
 | 
						|
		} else {
 | 
						|
			epp = &ep->e_next;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	*s_p = s;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/* The operations first, next and elem can be used to iterate
 | 
						|
 * over a set. For example:
 | 
						|
 *	for (i = Lfirst(s); i != (Lindex) 0; i = Lnext(i,s) {
 | 
						|
 *		x = Lelem(i);
 | 
						|
 *		use x
 | 
						|
 *	}
 | 
						|
 * which is like:
 | 
						|
 *      'for all elements x of s do'
 | 
						|
 *		use x
 | 
						|
 */
 | 
						|
 | 
						|
 | 
						|
Lindex Lfirst(s)
 | 
						|
	lset s;
 | 
						|
{
 | 
						|
	return ((Lindex) s);
 | 
						|
	/* Note that an index for long sets is just
 | 
						|
	 * a pointer to an elemholder.
 | 
						|
	 */
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*ARGSUSED1*/
 | 
						|
Lindex Lnext(i,s)
 | 
						|
	Lindex i;
 | 
						|
	lset   s;
 | 
						|
{
 | 
						|
	assert(i != (Lindex) 0);
 | 
						|
	return (i->e_next);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Lelem_t Lelem(i)
 | 
						|
	Lindex i;
 | 
						|
{
 | 
						|
	return (i->e_elem);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
Ljoin(s1,s2_p)
 | 
						|
	lset s1,*s2_p;
 | 
						|
{
 | 
						|
	/* Join two sets, assign the result to the second set
 | 
						|
	 * and delete the first set (i.e. the value of the
 | 
						|
	 * first set becomes undefined).
 | 
						|
	 */
 | 
						|
 | 
						|
	 register elem_p *epp, ep;
 | 
						|
	 lset s2;
 | 
						|
 | 
						|
	 /* First all elements of s1 that are also an element of s2
 | 
						|
	  * are removed from the s1 list. The two resulting lists
 | 
						|
	  * (for s1 and s2) are linked (s1 first).
 | 
						|
	  * Note the usage of epp, which points to a pointer that
 | 
						|
	  * points to the next elemholder record of the list.
 | 
						|
	  */
 | 
						|
 | 
						|
	s2 = *s2_p;
 | 
						|
	epp = &s1;
 | 
						|
	while ((ep = *epp) != (elem_p) 0) {
 | 
						|
		if (Lis_elem(ep->e_elem,s2)) {
 | 
						|
			/* remove an element */
 | 
						|
			*epp = ep->e_next;
 | 
						|
			oldelem(ep);
 | 
						|
		} else {
 | 
						|
			epp = &ep->e_next;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	*epp = s2; /* last record of s1 (or s1 itself) now points
 | 
						|
		    * to first record of s2.
 | 
						|
		    */
 | 
						|
	*s2_p = s1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
Ldeleteset(s)
 | 
						|
	lset s;
 | 
						|
{
 | 
						|
	register elem_p ep, next;
 | 
						|
 | 
						|
	for (ep = s; ep != (elem_p) 0; ep = next) {
 | 
						|
		next = ep->e_next;
 | 
						|
		oldelem(ep);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
bool Lis_subset(s1,s2)
 | 
						|
	lset s1,s2;
 | 
						|
{
 | 
						|
	/* See if s1 is a subset of s2 */
 | 
						|
 | 
						|
	register Lindex i;
 | 
						|
 | 
						|
	for (i = Lfirst(s1); i != (Lindex) 0; i = Lnext(i,s1)) {
 | 
						|
		if (!Lis_elem(Lelem(i),s2)) return FALSE;
 | 
						|
	}
 | 
						|
	return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
short Lnrelems(s)
 | 
						|
	lset s;
 | 
						|
{
 | 
						|
	/* Compute the number of elements of a set */
 | 
						|
 | 
						|
	register elem_p ep;
 | 
						|
	register short  cnt;
 | 
						|
 | 
						|
	cnt = 0;
 | 
						|
	for (ep = s; ep != (elem_p) 0; ep = ep->e_next) {
 | 
						|
		cnt++;
 | 
						|
	}
 | 
						|
	return cnt;
 | 
						|
}
 |