/* This software was developed by Bruce Hendrickson and Robert Leland   *
 * at Sandia National Laboratories under US Department of Energy        *
 * contract DE-AC04-76DP00789 and is copyrighted by Sandia Corporation. */

#include	<stdio.h>
#include	"params.h"
#include	"structs.h"
#include	"defs.h"

/* Idea:
   'buckets[i][j]' is a set of buckets to sort moves from i to j.
   listspace[i] is space for lists in buckets[i][j].
   Loop through all nonequal pairs [i][j], taking the first element
   in each list.  Compare them all to find the largest allowed move.
   Make that move, and save it in movelist.
*/


void      bucketsorts(graph, nvtxs, buckets, listspace, dvals, sets, term_wgts,
             maxdval, nsets, parity, hops, bspace, list_length, npass, using_ewgts)
struct vtx_data **graph;	/* graph data structure */
int       nvtxs;		/* number of vertices */
struct bilist ****buckets;	/* array of lists for bucket sort */
struct bilist **listspace;	/* list data structure for each vertex */
int     **dvals;		/* d-values for each vertex for removing */
short    *sets;			/* processor each vertex is assigned to */
float    *term_wgts[];		/* weights for terminal propogation */
int       maxdval;		/* maximum possible dvalue for a vertex */
int       nsets;		/* number of sets being divided into */
int       parity;		/* work in forward or backward direction? */
short     (*hops)[MAXSETS];	/* hop cost between sets */
int      *bspace;		/* indices for randomly ordering vtxs */
int       list_length;		/* number of values in bspace to work with */
int       npass;		/* which pass through KL is this? */
int       using_ewgts;		/* are edge weights being used? */
{
    extern int KL_RANDOM;	/* use randomness in KL? */
    extern int KL_UNDO_LIST;	/* only sort vertices who have moved. */
    extern double CUT_TO_HOP_COST;	/* if term_prop, cut/hop importance */
    struct bilist **bptr;	/* loops through set of buckets */
    struct bilist *lptr;	/* pointer to an element in listspace */
    float    *ewptr;		/* loops through edge weights */
    int      *bsptr;		/* loops through bspace */
    int      *edges;		/* edge list for a vertex */
    int       myset;		/* set that current vertex belongs to */
    int       newset;		/* set current vertex could move to */
    int       set;		/* set that neighboring vertex belongs to */
    int       weight;		/* edge weight for a particular edge */
    int       vtx;		/* vertex in graph */
    float     tval;		/* terminal propagation value */
    int       val;		/* terminal propogation rounded value */
    double    cut_cost;		/* relative cut/hop importance */
    double    hop_cost;		/* relative hop/cut importance */
    int       myhop;		/* hops associated with current vertex */
    int       i, j, l;		/* loop counters */
    void      randomize(), add2bilist();

    /* For each vertex, compute d-values for each possible transition. */
    /* Then store them in each appropriate bucket. */

    if (npass == 1 || !KL_UNDO_LIST || list_length == nvtxs) {
	/* Empty all the buckets. */
	/* Last clause catches case where lists weren't undone. */
	bptr = buckets[0][1];
	for (i = nsets * (nsets - 1) * (2 * maxdval + 1); i; i--)
	    *bptr++ = NULL;
    }

    /* Randomize the order of the vertices */

    if ((KL_UNDO_LIST && list_length == nvtxs) ||
	    (!KL_UNDO_LIST && !KL_RANDOM)) {
	/* Don't need to reoder if about to randomize. */
	bsptr = bspace;
	list_length = nvtxs;
	if (parity)
	    for (i = 1; i <= nvtxs; i++)
		*bsptr++ = i;
	else
	    for (i = nvtxs; i; i--)
		*bsptr++ = i;
    }
    if (KL_RANDOM)
	randomize(bspace - 1, list_length);

    /* Now compute d-vals by seeing which sets neighbors belong to. */
    cut_cost = hop_cost = 1;
    if (term_wgts[1] != NULL) {
	if (CUT_TO_HOP_COST > 1) {
	    cut_cost = CUT_TO_HOP_COST;
	}
	else {
	    hop_cost = 1.0 / CUT_TO_HOP_COST;
	}
    }
    weight = cut_cost + .5;
    bsptr = bspace;
    for (i = 0; i < list_length; i++) {	/* Loop through vertices. */
	vtx = *bsptr++;
	myset = sets[vtx];

	/* Initialize all the preference values. */
	if (term_wgts[1] != NULL) {
	    /* Using terminal propogation. */
	    if (myset == 0) {	/* No terminal value. */
		for (newset = 1; newset < nsets; newset++) {
		    tval = (term_wgts[newset])[vtx];
		    if (tval < 0) {
			val = - tval * hop_cost + .5;
			val = -val;
		    }
		    else {
			val = tval * hop_cost + .5;
		    }
		    dvals[vtx][newset - 1] = val;
		}
	    }
	    else {
		tval = -(term_wgts[myset])[vtx];
		if (tval < 0) {
		    val = - tval * hop_cost + .5;
		    val = -val;
		}
		else {
		    val = tval * hop_cost + .5;
		}
		dvals[vtx][0] = val;
		l = 1;
		for (newset = 1; newset < nsets; newset++) {
		    if (newset != myset) {
			tval = (term_wgts[newset])[vtx] - (term_wgts[myset])[vtx];
		        if (tval < 0) {
			    val = - tval * hop_cost + .5;
			    val = -val;
		        }
		        else {
			    val = tval * hop_cost + .5;
		        }
			dvals[vtx][l] = val;
			l++;
		    }
		}
	    }
	}
	else {
	    for (j = 0; j < nsets - 1; j++)
		dvals[vtx][j] = 0;
	}

	/* First count the neighbors in each set. */
	edges = graph[vtx]->edges;
	if (using_ewgts)
	    ewptr = graph[vtx]->ewgts;
	for (j = graph[vtx]->nedges - 1; j; j--) {
	    set = sets[*(++edges)];
	    if (set < 0)
		set = -set - 1;
	    if (using_ewgts)
		weight = *(++ewptr) * cut_cost + .5;
	    myhop = hops[myset][set];

	    l = 0;
	    for (newset = 0; newset < nsets; newset++) {
		if (newset != myset) {
		    dvals[vtx][l] += weight * (myhop - hops[newset][set]);
		    l++;
		}
	    }
	}

	/* Now add to appropriate buckets. */
	l = 0;
	for (newset = 0; newset < nsets; newset++) {
	    if (newset != myset) {
		lptr = listspace[l];
		add2bilist(&lptr[vtx], &buckets[myset][newset][dvals[vtx][l] + maxdval]);
		++l;
	    }
	}
    }
}


syntax highlighted by Code2HTML, v. 0.9.1