home *** CD-ROM | disk | FTP | other *** search
- /*Program 4_2g -- The Final Solution to the IRS Sort
- by Stephen R. Davis, 1987
-
- This is merely a combination of the various parts of the solution
- already developed. This program would normally be linked together
- the General Sort procedure -- we have included directly into the
- same source here for clarity.
- */
-
- #include <stdio.h>
- #include <ctype.h>
- #include <alloc.h>
- #include <process.h>
-
- #define NULL (struct IRSdata *)0
- #define IRSsignature 0x1234
-
- /*prototype declarations --*/
- /*structure creation*/
- struct IRSdata *create (void);
- void getfield (char *, char *, unsigned);
- void output (struct IRSdata *);
- /*linked list routines*/
- int insert (struct IRSdata *, struct IRSdata *, struct IRSdata *);
- int remove (struct IRSdata *);
- int check (struct IRSdata *, char *);
- struct IRSdata *alloc (void);
- void init (void);
- /*swap routines*/
- int swap (struct IRSdata *, struct IRSdata *);
- int compare (struct IRSdata *, struct IRSdata *);
- struct IRSdata *sequence (struct IRSdata *);
- int sort (int (*)(void *, void *), int (*)(void *, void *),
- void *(*)(void *));
-
- /*structure declaration for IRS data*/
- struct IRSdata{
- struct IRSdata *previous,*next;
- unsigned fingerprint;
- char lastname [11];
- char firstname [11];
- char sex;
- struct {
- char street [16];
- char city [11];
- char state [3];
- } address;
- char ssnum [10];
- int taxrate;
- };
- struct IRSdata *MARKER;
- char buffer [256];
-
- /*Create - allocate an IRSdata entry and fill in the data from 'stdin'*/
- struct IRSdata * create ()
- {
- char answer [2];
- struct IRSdata *ptr;
-
- getfield ("Another entry? ", answer, 1);
- if (tolower (answer [0]) == 'n')
- return NULL;
-
- if (ptr = alloc ()) {
- getfield ("Enter last name: ", ptr -> lastname, 10);
- getfield (" first name: ", ptr -> firstname, 10);
- getfield (" street addr: ", ptr -> address.street, 15);
- getfield (" city address: ", ptr -> address.city, 10);
- getfield (" state (2 ltr):", ptr -> address.state, 2);
- getfield (" soc sec #: ", ptr -> ssnum, 9);
- printf (" tax bracket: ");
-
- gets (buffer);
- sscanf (buffer, "%d", &(ptr -> taxrate));
- } else {
- printf ("Sorry. No more room for data.\n");
- }
-
- return (ptr);
- }
-
- /*GetField - pose a question, then get an answer. Save up to
- 'size' characters*/
- void getfield (question, answer, size)
- char *question,*answer;
- unsigned size;
- {
- unsigned i;
-
- printf (question);
- while (!gets (buffer));
- for (i = 0; size; size--)
- *answer++ = buffer [i++];
- *answer = '\0';
- }
-
- /*Output - output a subset of IRSdata structure to 'stdout'*/
- void output (ptr)
- struct IRSdata *ptr;
- {
- if (ptr)
- printf ("%s %s, %s, taxrate = %u\n", ptr -> firstname,
- ptr -> lastname, ptr -> ssnum, ptr -> taxrate);
- }
-
- /*Insert - insert a structure in between two doubly linked entries.
- Return a 0 if successful, and a nonzero if not*/
- int insert (before, after, current)
- struct IRSdata *before, *after, *current;
- {
- if (before -> next != after) return -1;
- if (before != after -> previous) return -1;
-
- if (check (before, "arg 'before' to insert()")) return -1;
- if (check (after, "arg 'after' to insert()")) return -1;
- if (check (current, "arg 'current' to insert()")) return -1;
-
- before -> next = current;
- current -> previous = before;
-
- after -> previous = current;
- current -> next = after;
- return 0;
- }
-
- /*Remove - remove an entry from a doubly linked list*/
- int remove (entry)
- struct IRSdata *entry;
- {
- struct IRSdata *before, *after;
-
- if (check (entry, "arg 'entry' to remove()")) return -1;
-
- before = entry -> previous;
- after = entry -> next;
-
- before -> next = after;
- after -> previous = before;
-
- entry -> previous = entry -> next = NULL;
- return 0;
- }
-
- /*Init - initialize the linked list to empty*/
- void init (void)
- {
- struct IRSdata *alloc ();
-
- MARKER = alloc ();
- MARKER -> previous = MARKER -> next = MARKER;
- }
-
- /*Check - check the integrity of an IRS pointer. If OK, return a
- 0, else print message and return a -1.*/
- int check (ptr, msg)
- struct IRSdata *ptr;
- char *msg;
- {
- if (ptr -> fingerprint == IRSsignature)
- return 0;
- printf ("Error:\n Pointer failure on %s\n", msg);
- return -1;
- }
-
- /*Alloc - allocate a structure and "sign it" with the IRS
- signature*/
- struct IRSdata *alloc(void)
- {
- struct IRSdata *ptr;
-
- if (ptr = (struct IRSdata *)malloc (sizeof (struct IRSdata)))
- ptr -> fingerprint = IRSsignature;
- return ptr;
- }
-
- /*Sort - impliment bubble sort*/
- int sort (compare, swap, sequence)
- int (*compare)(void *, void *), (*swap)(void *, void *);
- void *(*sequence)(void *);
- {
- int flag;
- void *p1, *p2;
-
- do {
- flag = 0;
- p2 = (*sequence)(0); /*starting w/ first entry...*/
- while (p1 = p2, p2 = (*sequence) (p2)) /*...sequence thru*/
- {
- if ((*compare)(p1, p2) > 0) { /*if p1 > p2...*/
- if ((*swap)(p1, p2)) return -1;/*...swap p1 & p2*/
- flag = 1;
- }
- }
- } while (flag); /*stop when all are in order*/
- return 0;
- }
-
- /*Swap - swap the position of the two entries passed. Return
- a 0 if successful and a -1 if not.*/
- int swap (first, second)
- struct IRSdata *first, *second;
- {
-
- if (remove (second))
- return -1;
- return (insert (first -> previous, first, second));
- }
-
- /*Compare - compare two IRS data structures. Return as follows:
- 1 -> a.taxrate > b.taxrate
- 0 -> a.taxrate = b.taxrate
- -1 -> a.taxrate < b.taxrate*/
- int compare (a, b)
- struct IRSdata *a, *b;
- {
- if (a -> taxrate > b -> taxrate)
- return 1;
- if (a -> taxrate < b -> taxrate)
- return -1;
- return 0;
- }
-
- /*Sequence - given an entry, return the address of the next
- entry. Return a 0 on end of list.*/
- struct IRSdata *sequence (entry)
- struct IRSdata *entry;
- {
- struct IRSdata *value;
-
- if (entry == 0) /*given 0...*/
- entry = MARKER; /*...start with beginning*/
-
- if (check (entry, "arg 'entry' to sequence()"))
- return NULL;
-
- if ((value = entry -> next) == MARKER) /*if end of chain...*/
- return NULL; /*...return a NULL*/
- else
- return value;
- }
-
- /*Main - now invoke the above routines*/
- main ()
- {
- struct IRSdata *ptr, *ptrold;
-
- /*initialize linked list to empty*/
- init ();
-
- /*now read in entries and add them to the chain*/
- ptrold = MARKER;
- while (ptr = create ()) {
- if (insert (ptrold, MARKER, ptr))
- abort ();
- ptrold = ptr;
- }
-
- /*display the entries in input order*/
- printf ("\n\n\nHere are the entries as entered:\n\n");
- ptr = MARKER;
- while (ptr = sequence (ptr))
- output (ptr);
-
- /*now sort the list*/
- if (sort (compare, swap, sequence))
- printf ("failure during sort!\n");
-
- /*and reoutput the list*/
- printf ("\n\n\nHere are the entries sorted by taxrate:\n\n");
- ptr = MARKER;
- while (ptr = sequence (ptr))
- output (ptr);
-
- printf ("\nCompleted successfully\n");
- }