/* 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 "defs.h"
#include "structs.h"


/* Find a maximal matching in a graph using Luby's algorithm.  Assign a */
/* random value to each edge and include an edge in the matching if it  */
/* has a higher value than all neighboring edges that aren't disallowed. */
/* (Use float instead of double values to save space.) */
/* THIS ROUTINE IS SLOWER THAN MAXMATCH3, AND SO PROBABLY OBSOLETE. */

int       maxmatch4(graph, nvtxs, nedges, mflag, using_ewgts)
/* Find a maximal matching in the graph, using variant of Luby's algorithm. */
/* Assign a random value to each edge, and include an edge in */
/* the matching if its value larger than all neighboring edges. */
struct vtx_data **graph;	/* array of vtx data for graph */
int       nvtxs;		/* number of vertices in graph */
int       nedges;		/* number of edges in graph */
int      *mflag;		/* flag indicating vtx selected or not */
{
    extern int HEAVY_MATCH;	/* try for heavy edges in matching? */
    int      *iptr;		/* loops through integer arrays */
    float    *edgevals;		/* random values for all edges */
    float    *evptr;		/* loops through edgevals */
    double    maxval;		/* largest edge value for a vertex */
    int       neighbor;		/* neighbor of a vertex */
    int       nmerged;		/* number of edges in matching */
    int       change;		/* any new edges in matching? */
    int      *start;		/* start of edgevals list for each vertex */
    int       i, j, k;		/* loop counters */
    double   *smalloc();
    int       sfree();
    double    drandom();

    /* Allocate and initialize space. */
    evptr = edgevals = (float *) smalloc((unsigned) 2 * nedges * sizeof(float));

    start = (int *) smalloc((unsigned) (nvtxs + 2) * sizeof(int));
    start[1] = 0;
    for (i = 1; i <= nvtxs; i++)
	start[i + 1] = start[i] + graph[i]->nedges - 1;

    /* Assign a random value to each edge. */
    if (!using_ewgts || !HEAVY_MATCH) {	/* All edges are equal. */
	for (i = 1; i <= nvtxs; i++) {
	    for (j = 1; j < graph[i]->nedges; j++) {
		neighbor = graph[i]->edges[j];
		if (neighbor > i) {
		    *evptr = (float) drandom();
		}
		else {		/* Look up already-generated value. */
		    for (k = 1; graph[neighbor]->edges[k] != i; k++);
		    *evptr = edgevals[start[neighbor] + k - 1];
		}
		evptr++;
	    }
	}
    }
    else {			/* Prefer heavy weight edges. */
	for (i = 1; i <= nvtxs; i++) {
	    for (j = 1; j < graph[i]->nedges; j++) {
		neighbor = graph[i]->edges[j];
		if (neighbor > i) {
		    *evptr = (float) graph[i]->ewgts[j] * drandom();
		}
		else {		/* Look up already-generated value. */
		    for (k = 1; graph[neighbor]->edges[k] != i; k++);
		    *evptr = edgevals[start[neighbor] + k - 1];
		}
		evptr++;
	    }
	}
    }

    for (iptr = mflag, i = nvtxs; i; i--)
	*(++iptr) = -(nvtxs + 1);
    nmerged = 0;
    change = TRUE;
    while (change) {
	change = FALSE;

	for (i = 1; i <= nvtxs; i++) {	/* Find largest valued edge of each vtx */
	    if (mflag[i] < 0) {
		maxval = 0.0;
		k = -1;
		evptr = &(edgevals[start[i]]);
		for (j = 1; j < graph[i]->nedges; j++) {
		    if (*evptr > maxval && mflag[graph[i]->edges[j]] < 0) {
			maxval = *evptr;
			k = j;
		    }
		    evptr++;
		}
		if (k == -1)
		    mflag[i] = 0;	/* No neighbors are alive. */
		else {
		    mflag[i] = -graph[i]->edges[k];
		}
	    }
	}

	/* If vtxs agree on largest valued edge, add to independent set. */
	for (i = 1; i <= nvtxs; i++) {
	    if (-mflag[i] > i) {
		if (-mflag[-mflag[i]] == i) {	/* Add edge to independent set. */
		    nmerged++;
		    mflag[i] = -mflag[i];
		    mflag[mflag[i]] = i;
		    change = TRUE;
		}
	    }
	}
    }

    /* Maximal independent set is indicated by corresponding pairs */
    /* of positive values in the mflag array. */
    for (i = 1; i <= nvtxs; i++)
	if (mflag[i] < 0)
	    mflag[i] = 0;

    sfree((char *) start);
    sfree((char *) edgevals);
    return (nmerged);
}


syntax highlighted by Code2HTML, v. 0.9.1