/* 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 by geometrically near neighbors. */
int maxmatch5(graph, nvtxs, mflag, igeom, coords)
struct vtx_data **graph; /* array of vtx data for graph */
int nvtxs; /* number of vertices in graph */
int *mflag; /* flag indicating vtx selected or not */
int igeom; /* geometric dimensionality */
float **coords; /* coordinates of each vertex */
{
extern double DOUBLE_MAX; /* largest floating point value */
double dist; /* distance to free neighbor */
double min_dist; /* smallest distance to free neighbor */
int *jptr; /* loops through integer arrays */
int vtx; /* vertex to process next */
int neighbor; /* neighbor of a vertex */
int nmerged; /* number of edges in matching */
int jsave; /* best edge so far */
int i, j, k; /* loop counters */
double drandom();
/* Initialize mflag array. */
jptr = mflag;
for (i = 1; i <= nvtxs; i++) {
*(++jptr) = 0;
}
nmerged = 0;
/* Select random starting point in list of vertices. */
vtx = 1 + drandom() * nvtxs;
if (igeom == 1) {
for (i = nvtxs; i; i--) { /* Choose geometrically nearest neighbor */
if (mflag[vtx] == 0) { /* Not already matched. */
/* Select nearest free edge. */
jsave = 0;
min_dist = DOUBLE_MAX;
for (j = 1; j < graph[vtx]->nedges; j++) {
neighbor = graph[vtx]->edges[j];
if (mflag[neighbor] == 0) {
dist = (coords[0][vtx] - coords[0][neighbor]) *
(coords[0][vtx] - coords[0][neighbor]);
if (dist < min_dist) {
jsave = j;
min_dist = dist;
}
}
}
if (jsave > 0) {
neighbor = graph[vtx]->edges[jsave];
mflag[vtx] = neighbor;
mflag[neighbor] = vtx;
nmerged++;
}
}
if (++vtx > nvtxs)
vtx = 1;
}
}
else if (igeom == 2) {
for (i = nvtxs; i; i--) { /* Choose geometrically nearest neighbor */
if (mflag[vtx] == 0) { /* Not already matched. */
/* Select nearest free edge. */
jsave = 0;
min_dist = DOUBLE_MAX;
for (j = 1; j < graph[vtx]->nedges; j++) {
neighbor = graph[vtx]->edges[j];
if (mflag[neighbor] == 0) {
dist = (coords[0][vtx] - coords[0][neighbor]) *
(coords[0][vtx] - coords[0][neighbor]);
if (dist < min_dist) {
dist += (coords[1][vtx] - coords[1][neighbor]) *
(coords[1][vtx] - coords[1][neighbor]);
if (dist < min_dist) {
jsave = j;
min_dist = dist;
}
}
}
}
if (jsave > 0) {
neighbor = graph[vtx]->edges[jsave];
mflag[vtx] = neighbor;
mflag[neighbor] = vtx;
nmerged++;
}
}
if (++vtx > nvtxs)
vtx = 1;
}
}
else if (igeom >= 2) {
for (i = nvtxs; i; i--) { /* Choose geometrically nearest neighbor */
if (mflag[vtx] == 0) { /* Not already matched. */
/* Select nearest free edge. */
jsave = 0;
min_dist = DOUBLE_MAX;
for (j = 1; j < graph[vtx]->nedges; j++) {
neighbor = graph[vtx]->edges[j];
if (mflag[neighbor] == 0) {
dist = (coords[0][vtx] - coords[0][neighbor]) *
(coords[0][vtx] - coords[0][neighbor]);
if (dist < min_dist) {
dist += (coords[1][vtx] - coords[1][neighbor]) *
(coords[1][vtx] - coords[1][neighbor]);
if (dist < min_dist) {
dist += (coords[2][vtx] - coords[2][neighbor]) *
(coords[2][vtx] - coords[2][neighbor]);
if (dist < min_dist) {
jsave = j;
min_dist = dist;
}
}
}
}
}
if (jsave > 0) {
neighbor = graph[vtx]->edges[jsave];
mflag[vtx] = neighbor;
mflag[neighbor] = vtx;
nmerged++;
}
}
if (++vtx > nvtxs)
vtx = 1;
}
}
return (nmerged);
}
syntax highlighted by Code2HTML, v. 0.9.1