/src/match.c

https://github.com/nclack/whisk · C · 543 lines · 400 code · 80 blank · 63 comment · 88 complexity · 811ffd9fc7606035599baf903d204369 MD5 · raw file

  1. /*
  2. * match.c
  3. *
  4. * Adapted from Markus Buehren's implimentation of the Munkres (a.k.a. Hungarian) algorithm.
  5. *
  6. * Adapted by Nathan Clack on 3/5/08.
  7. * Copyright 2008 HHMI. All rights reserved.
  8. *
  9. */
  10. #include <stdio.h>
  11. #include <stdlib.h>
  12. #include <math.h>
  13. #include <time.h>
  14. #include <string.h>
  15. #include <float.h>
  16. #include "common.h"
  17. #include "match.h"
  18. #undef COLUMN_ORDER
  19. #define CHECK_FOR_INF
  20. #undef ONE_INDEXING
  21. #undef MATLAB
  22. #ifndef INFINITY
  23. #define INFINITY FLT_MAX
  24. #endif
  25. #ifndef isfinite
  26. #define isfinite( x ) (fabs(x)<FLT_MAX)
  27. #endif
  28. #ifndef isinf
  29. #define isinf(x) (!isfinite(x))
  30. #endif
  31. SHARED_EXPORT void assignmentoptimal(double *assignment, double *cost, double *distMatrix, int nOfRows, int nOfColumns);
  32. void buildassignmentvector(double *assignment, char *starMatrix, int nOfRows, int nOfColumns);
  33. void computeassignmentcost(double *assignment, double *cost, double *distMatrix, int nOfRows);
  34. void step2a(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim);
  35. void step2b(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim);
  36. void step3 (double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim);
  37. void step4 (double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim, int row, int col);
  38. void step5 (double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim);
  39. SHARED_EXPORT Assignment match( double* distMatrixIn, int nOfRows, int nOfColumns )
  40. { Assignment result;
  41. result.n = nOfRows;
  42. result.assignment = (double*) malloc( sizeof(double)*nOfRows );
  43. assignmentoptimal( result.assignment, &result.cost, distMatrixIn, nOfRows, nOfColumns );
  44. return result;
  45. }
  46. SHARED_EXPORT void assignmentoptimal(double *assignment, double *cost, double *distMatrixIn, int nOfRows, int nOfColumns)
  47. {
  48. double *distMatrix, *distMatrixTemp, *distMatrixEnd, *columnEnd, value, minValue;
  49. char *coveredColumns, *coveredRows, *starMatrix, *newStarMatrix, *primeMatrix;
  50. int nOfElements, minDim, row, col;
  51. #ifdef CHECK_FOR_INF
  52. char infiniteValueFound;
  53. double maxFiniteValue, infValue;
  54. #endif
  55. /* initialization */
  56. *cost = 0;
  57. for(row=0; row<nOfRows; row++)
  58. #ifdef ONE_INDEXING
  59. assignment[row] = 0.0;
  60. #else
  61. assignment[row] = -1.0;
  62. #endif
  63. /* generate working copy of distance Matrix */
  64. /* check if all matrix elements are positive */
  65. nOfElements = nOfRows * nOfColumns;
  66. distMatrix = (double *)malloc(nOfElements * sizeof(double));
  67. distMatrixEnd = distMatrix + nOfElements;
  68. for(row=0; row<nOfElements; row++)
  69. {
  70. value = distMatrixIn[row];
  71. if(isfinite(value) && (value < 0))
  72. fprintf(stderr, "All matrix elements have to be non-negative.\n");
  73. #ifdef COLUMN_ORDER
  74. /* translate an input column-order matrix into row-order form */
  75. distMatrix[ ((int) row/nOfColumns) + nOfRows * (row%nOfColumns) ] = value;
  76. #else
  77. distMatrix[row] = value;
  78. #endif
  79. }
  80. #ifdef CHECK_FOR_INF
  81. /* check for infinite values */
  82. maxFiniteValue = -1;
  83. infiniteValueFound = 0;
  84. distMatrixTemp = distMatrix;
  85. while(distMatrixTemp < distMatrixEnd)
  86. {
  87. value = *distMatrixTemp++;
  88. if(isfinite(value))
  89. {
  90. if(value > maxFiniteValue)
  91. maxFiniteValue = value;
  92. }
  93. else
  94. infiniteValueFound = 1;
  95. }
  96. if(infiniteValueFound)
  97. {
  98. if(maxFiniteValue == -1) /* all elements are infinite */
  99. return;
  100. /* set all infinite elements to big finite value */
  101. if(maxFiniteValue > 0)
  102. infValue = 10 * maxFiniteValue * nOfElements;
  103. else
  104. infValue = 10;
  105. distMatrixTemp = distMatrix;
  106. while(distMatrixTemp < distMatrixEnd)
  107. if(isinf(*distMatrixTemp++))
  108. *(distMatrixTemp-1) = infValue;
  109. }
  110. #endif
  111. /* memory allocation */
  112. coveredColumns = (char *)malloc(nOfColumns* sizeof(char));
  113. memset( coveredColumns, 0, nOfColumns * sizeof(char));
  114. coveredRows = (char *)malloc(nOfRows * sizeof(char));
  115. memset( coveredRows , 0, nOfRows * sizeof(char));
  116. starMatrix = (char *)malloc(nOfElements* sizeof(char));
  117. memset( starMatrix , 0, nOfElements * sizeof(char));
  118. primeMatrix = (char *)malloc(nOfElements* sizeof(char));
  119. memset( primeMatrix , 0, nOfElements * sizeof(char));
  120. newStarMatrix = (char *)malloc(nOfElements* sizeof(char)); /* used in step4 */
  121. memset( newStarMatrix , 0, nOfElements * sizeof(char));
  122. /* preliminary steps */
  123. if(nOfRows <= nOfColumns)
  124. {
  125. minDim = nOfRows;
  126. for(row=0; row<nOfRows; row++)
  127. {
  128. /* find the smallest element in the row */
  129. distMatrixTemp = distMatrix + row;
  130. minValue = *distMatrixTemp;
  131. distMatrixTemp += nOfRows;
  132. while(distMatrixTemp < distMatrixEnd)
  133. {
  134. value = *distMatrixTemp;
  135. if(value < minValue)
  136. minValue = value;
  137. distMatrixTemp += nOfRows;
  138. }
  139. /* subtract the smallest element from each element of the row */
  140. distMatrixTemp = distMatrix + row;
  141. while(distMatrixTemp < distMatrixEnd)
  142. {
  143. *distMatrixTemp -= minValue;
  144. distMatrixTemp += nOfRows;
  145. }
  146. }
  147. /* Steps 1 and 2a */
  148. for(row=0; row<nOfRows; row++)
  149. for(col=0; col<nOfColumns; col++)
  150. if(distMatrix[row + nOfRows*col] == 0)
  151. if(!coveredColumns[col])
  152. {
  153. starMatrix[row + nOfRows*col] = 1;
  154. coveredColumns[col] = 1;
  155. break;
  156. }
  157. }
  158. else /* if(nOfRows > nOfColumns) */
  159. {
  160. minDim = nOfColumns;
  161. for(col=0; col<nOfColumns; col++)
  162. {
  163. /* find the smallest element in the column */
  164. distMatrixTemp = distMatrix + nOfRows*col;
  165. columnEnd = distMatrixTemp + nOfRows;
  166. minValue = *distMatrixTemp++;
  167. while(distMatrixTemp < columnEnd)
  168. {
  169. value = *distMatrixTemp++;
  170. if(value < minValue)
  171. minValue = value;
  172. }
  173. /* subtract the smallest element from each element of the column */
  174. distMatrixTemp = distMatrix + nOfRows*col;
  175. while(distMatrixTemp < columnEnd)
  176. *distMatrixTemp++ -= minValue;
  177. }
  178. /* Steps 1 and 2a */
  179. for(col=0; col<nOfColumns; col++)
  180. for(row=0; row<nOfRows; row++)
  181. if(distMatrix[row + nOfRows*col] == 0)
  182. if(!coveredRows[row])
  183. {
  184. starMatrix[row + nOfRows*col] = 1;
  185. coveredColumns[col] = 1;
  186. coveredRows[row] = 1;
  187. break;
  188. }
  189. for(row=0; row<nOfRows; row++)
  190. coveredRows[row] = 0;
  191. }
  192. /* move to step 2b */
  193. step2b(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  194. /* compute cost and remove invalid assignments */
  195. computeassignmentcost(assignment, cost, distMatrixIn, nOfRows);
  196. /* free allocated memory */
  197. free(distMatrix);
  198. free(coveredColumns);
  199. free(coveredRows);
  200. free(starMatrix);
  201. free(primeMatrix);
  202. free(newStarMatrix);
  203. return;
  204. }
  205. /********************************************************/
  206. void buildassignmentvector(double *assignment, char *starMatrix, int nOfRows, int nOfColumns)
  207. {
  208. int row, col;
  209. for(row=0; row<nOfRows; row++)
  210. for(col=0; col<nOfColumns; col++)
  211. if(starMatrix[row + nOfRows*col])
  212. {
  213. #ifdef ONE_INDEXING
  214. assignment[row] = col + 1; /* MATLAB-Indexing */
  215. #else
  216. assignment[row] = col;
  217. #endif
  218. break;
  219. }
  220. }
  221. /********************************************************/
  222. void computeassignmentcost(double *assignment, double *cost, double *distMatrix, int nOfRows)
  223. {
  224. int row, col;
  225. #ifdef CHECK_FOR_INF
  226. double value;
  227. #endif
  228. for(row=0; row<nOfRows; row++)
  229. {
  230. #ifdef ONE_INDEXING
  231. col = assignment[row]-1; /* MATLAB-Indexing */
  232. #else
  233. col = assignment[row];
  234. #endif
  235. if(col >= 0)
  236. {
  237. #ifdef CHECK_FOR_INF
  238. value = distMatrix[row + nOfRows*col];
  239. if(isfinite(value))
  240. *cost += value;
  241. else
  242. #ifdef ONE_INDEXING
  243. assignment[row] = 0.0;
  244. #else
  245. assignment[row] = -1.0;
  246. #endif
  247. #else
  248. *cost += distMatrix[row + nOfRows*col];
  249. #endif
  250. }
  251. }
  252. }
  253. /********************************************************/
  254. void step2a(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim)
  255. {
  256. char *starMatrixTemp, *columnEnd;
  257. int col;
  258. /* cover every column containing a starred zero */
  259. for(col=0; col<nOfColumns; col++)
  260. {
  261. starMatrixTemp = starMatrix + nOfRows*col;
  262. columnEnd = starMatrixTemp + nOfRows;
  263. while(starMatrixTemp < columnEnd){
  264. if(*starMatrixTemp++)
  265. {
  266. coveredColumns[col] = 1;
  267. break;
  268. }
  269. }
  270. }
  271. /* move to step 3 */
  272. step2b(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  273. }
  274. /********************************************************/
  275. void step2b(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim)
  276. {
  277. int col, nOfCoveredColumns;
  278. /* count covered columns */
  279. nOfCoveredColumns = 0;
  280. for(col=0; col<nOfColumns; col++)
  281. if(coveredColumns[col])
  282. nOfCoveredColumns++;
  283. if(nOfCoveredColumns == minDim)
  284. {
  285. /* algorithm finished */
  286. buildassignmentvector(assignment, starMatrix, nOfRows, nOfColumns);
  287. }
  288. else
  289. {
  290. /* move to step 3 */
  291. step3(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  292. }
  293. }
  294. /********************************************************/
  295. void step3(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim)
  296. {
  297. char zerosFound;
  298. int row, col, starCol;
  299. zerosFound = 1;
  300. while(zerosFound)
  301. {
  302. zerosFound = 0;
  303. for(col=0; col<nOfColumns; col++)
  304. if(!coveredColumns[col])
  305. for(row=0; row<nOfRows; row++)
  306. if((!coveredRows[row]) && (distMatrix[row + nOfRows*col] == 0))
  307. {
  308. /* prime zero */
  309. primeMatrix[row + nOfRows*col] = 1;
  310. /* find starred zero in current row */
  311. for(starCol=0; starCol<nOfColumns; starCol++)
  312. if(starMatrix[row + nOfRows*starCol])
  313. break;
  314. if(starCol == nOfColumns) /* no starred zero found */
  315. {
  316. /* move to step 4 */
  317. step4(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim, row, col);
  318. return;
  319. }
  320. else
  321. {
  322. coveredRows[row] = 1;
  323. coveredColumns[starCol] = 0;
  324. zerosFound = 1;
  325. break;
  326. }
  327. }
  328. }
  329. /* move to step 5 */
  330. step5(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  331. }
  332. /********************************************************/
  333. void step4(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim, int row, int col)
  334. {
  335. int n, starRow, starCol, primeRow, primeCol;
  336. int nOfElements = nOfRows*nOfColumns;
  337. /* generate temporary copy of starMatrix */
  338. for(n=0; n<nOfElements; n++)
  339. newStarMatrix[n] = starMatrix[n];
  340. /* star current zero */
  341. newStarMatrix[row + nOfRows*col] = 1;
  342. /* find starred zero in current column */
  343. starCol = col;
  344. for(starRow=0; starRow<nOfRows; starRow++)
  345. if(starMatrix[starRow + nOfRows*starCol])
  346. break;
  347. while(starRow<nOfRows)
  348. {
  349. /* unstar the starred zero */
  350. newStarMatrix[starRow + nOfRows*starCol] = 0;
  351. /* find primed zero in current row */
  352. primeRow = starRow;
  353. for(primeCol=0; primeCol<nOfColumns; primeCol++)
  354. if(primeMatrix[primeRow + nOfRows*primeCol])
  355. break;
  356. /* star the primed zero */
  357. newStarMatrix[primeRow + nOfRows*primeCol] = 1;
  358. /* find starred zero in current column */
  359. starCol = primeCol;
  360. for(starRow=0; starRow<nOfRows; starRow++)
  361. if(starMatrix[starRow + nOfRows*starCol])
  362. break;
  363. }
  364. /* use temporary copy as new starMatrix */
  365. /* delete all primes, uncover all rows */
  366. for(n=0; n<nOfElements; n++)
  367. {
  368. primeMatrix[n] = 0;
  369. starMatrix[n] = newStarMatrix[n];
  370. }
  371. for(n=0; n<nOfRows; n++)
  372. coveredRows[n] = 0;
  373. /* move to step 2a */
  374. step2a(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  375. }
  376. /********************************************************/
  377. void step5(double *assignment, double *distMatrix, char *starMatrix, char *newStarMatrix, char *primeMatrix, char *coveredColumns, char *coveredRows, int nOfRows, int nOfColumns, int minDim)
  378. {
  379. double h, value;
  380. int row, col;
  381. /* find smallest uncovered element h */
  382. h = INFINITY;
  383. for(row=0; row<nOfRows; row++)
  384. if(!coveredRows[row])
  385. for(col=0; col<nOfColumns; col++)
  386. if(!coveredColumns[col])
  387. {
  388. value = distMatrix[row + nOfRows*col];
  389. if(value < h)
  390. h = value;
  391. }
  392. /* add h to each covered row */
  393. for(row=0; row<nOfRows; row++)
  394. if(coveredRows[row])
  395. for(col=0; col<nOfColumns; col++)
  396. distMatrix[row + nOfRows*col] += h;
  397. /* subtract h from each uncovered column */
  398. for(col=0; col<nOfColumns; col++)
  399. if(!coveredColumns[col])
  400. for(row=0; row<nOfRows; row++)
  401. distMatrix[row + nOfRows*col] -= h;
  402. /* move to step 3 */
  403. step3(assignment, distMatrix, starMatrix, newStarMatrix, primeMatrix, coveredColumns, coveredRows, nOfRows, nOfColumns, minDim);
  404. }
  405. /****************************************
  406. **
  407. ** TEST CASE
  408. **
  409. ****************************************/
  410. #ifdef TEST_MATCH
  411. #define M 10
  412. #define N 1
  413. int B[5*5] = { 7, 2, 1, 9, 4,
  414. 9, 6, 9, 5, 5,
  415. 3, 8, 3, 1, 8,
  416. 7, 9, 4, 2, 2,
  417. 8, 4, 7, 4, 8 };
  418. void pmat( double* array, int m, int n )
  419. { int i,j;
  420. for( i=0; i<m; ++i )
  421. { for( j=0; j<n; ++j )
  422. printf("%10.3g, ", array[i+j*m]);
  423. printf("\n");
  424. }
  425. }
  426. void pimat( int* array, int m, int n )
  427. { int i,j;
  428. for( i=0; i<m; ++i )
  429. { for( j=0; j<n; ++j )
  430. printf("%7d, ", array[i+j*m]);
  431. printf("\n");
  432. }
  433. }
  434. void pcmat( char* array, int m, int n )
  435. { int i,j;
  436. for( i=0; i<m; ++i )
  437. { for( j=0; j<n; ++j )
  438. printf("%7d, ", (int)array[i+j*m]);
  439. printf("\n");
  440. }
  441. }
  442. void f()
  443. { int i,j;
  444. double A[M*N];
  445. Assignment result;
  446. for( i=0; i<M*N; i++) A[i] = fabs( ((int)rand())/1024./1024./1024. );
  447. //printf("*************\n");
  448. //pmat( A, M,N );
  449. result = match( A, M, N );
  450. printf("----\n");
  451. pmat( A, M,N );
  452. for( i=0; i<result.n; i++ )
  453. printf( "%d --> %g\n", i, result.assignment[i] );
  454. printf("Cost: %g\n", result.cost);
  455. }
  456. int main(int argc, char* argv[] )
  457. { int i;
  458. srand( time(NULL) );
  459. for(i=0;i<1; i++)
  460. f();
  461. return 0;
  462. }
  463. #endif