51 #ifndef EIGEN_COLAMD_H
52 #define EIGEN_COLAMD_H
64 #define COLAMD_KNOBS 20
67 #define COLAMD_STATS 20
70 #define COLAMD_DENSE_ROW 0
73 #define COLAMD_DENSE_COL 1
76 #define COLAMD_DEFRAG_COUNT 2
79 #define COLAMD_STATUS 3
82 #define COLAMD_INFO1 4
83 #define COLAMD_INFO2 5
84 #define COLAMD_INFO3 6
88 #define COLAMD_OK_BUT_JUMBLED (1)
89 #define COLAMD_ERROR_A_not_present (-1)
90 #define COLAMD_ERROR_p_not_present (-2)
91 #define COLAMD_ERROR_nrow_negative (-3)
92 #define COLAMD_ERROR_ncol_negative (-4)
93 #define COLAMD_ERROR_nnz_negative (-5)
94 #define COLAMD_ERROR_p0_nonzero (-6)
95 #define COLAMD_ERROR_A_too_small (-7)
96 #define COLAMD_ERROR_col_length_negative (-8)
97 #define COLAMD_ERROR_row_index_out_of_bounds (-9)
98 #define COLAMD_ERROR_out_of_memory (-10)
99 #define COLAMD_ERROR_internal_error (-999)
105 #define COLAMD_MAX(a,b) (((a) > (b)) ? (a) : (b))
106 #define COLAMD_MIN(a,b) (((a) < (b)) ? (a) : (b))
108 #define ONES_COMPLEMENT(r) (-(r)-1)
112 #define COLAMD_EMPTY (-1)
119 #define DEAD_PRINCIPAL (-1)
120 #define DEAD_NON_PRINCIPAL (-2)
123 #define ROW_IS_DEAD(r) ROW_IS_MARKED_DEAD (Row[r].shared2.mark)
124 #define ROW_IS_MARKED_DEAD(row_mark) (row_mark < ALIVE)
125 #define ROW_IS_ALIVE(r) (Row [r].shared2.mark >= ALIVE)
126 #define COL_IS_DEAD(c) (Col [c].start < ALIVE)
127 #define COL_IS_ALIVE(c) (Col [c].start >= ALIVE)
128 #define COL_IS_DEAD_PRINCIPAL(c) (Col [c].start == DEAD_PRINCIPAL)
129 #define KILL_ROW(r) { Row [r].shared2.mark = DEAD ; }
130 #define KILL_PRINCIPAL_COL(c) { Col [c].start = DEAD_PRINCIPAL ; }
131 #define KILL_NON_PRINCIPAL_COL(c) { Col [c].start = DEAD_NON_PRINCIPAL ; }
138 template <
typename Index>
172 template <
typename Index>
208 template <
typename Index>
209 inline Index colamd_c(Index n_col)
210 {
return Index( ((n_col) + 1) *
sizeof (colamd_col<Index>) /
sizeof (Index) ) ; }
212 template <
typename Index>
213 inline Index colamd_r(Index n_row)
214 {
return Index(((n_row) + 1) *
sizeof (Colamd_Row<Index>) /
sizeof (Index)); }
217 template <
typename Index>
218 static Index init_rows_cols (Index n_row, Index n_col, Colamd_Row<Index> Row [], colamd_col<Index> col [], Index A [], Index p [], Index stats[COLAMD_STATS] );
220 template <
typename Index>
221 static void init_scoring (Index n_row, Index n_col, Colamd_Row<Index> Row [], colamd_col<Index> Col [], Index A [], Index head [],
double knobs[COLAMD_KNOBS], Index *p_n_row2, Index *p_n_col2, Index *p_max_deg);
223 template <
typename Index>
224 static Index find_ordering (Index n_row, Index n_col, Index Alen, Colamd_Row<Index> Row [], colamd_col<Index> Col [], Index A [], Index head [], Index n_col2, Index max_deg, Index pfree);
226 template <
typename Index>
227 static void order_children (Index n_col, colamd_col<Index> Col [], Index p []);
229 template <
typename Index>
230 static void detect_super_cols (colamd_col<Index> Col [], Index A [], Index head [], Index row_start, Index row_length ) ;
232 template <
typename Index>
233 static Index garbage_collection (Index n_row, Index n_col, Colamd_Row<Index> Row [], colamd_col<Index> Col [], Index A [], Index *pfree) ;
235 template <
typename Index>
236 static inline Index clear_mark (Index n_row, Colamd_Row<Index> Row [] ) ;
240 #define COLAMD_DEBUG0(params) ;
241 #define COLAMD_DEBUG1(params) ;
242 #define COLAMD_DEBUG2(params) ;
243 #define COLAMD_DEBUG3(params) ;
244 #define COLAMD_DEBUG4(params) ;
246 #define COLAMD_ASSERT(expression) ((void) 0)
263 template <
typename Index>
264 inline Index colamd_recommended ( Index nnz, Index n_row, Index n_col)
266 if ((nnz) < 0 || (n_row) < 0 || (n_col) < 0)
269 return (2 * (nnz) + colamd_c (n_col) + colamd_r (n_row) + (n_col) + ((nnz) / 5));
293 static inline void colamd_set_defaults(
double knobs[COLAMD_KNOBS])
303 for (i = 0 ; i < COLAMD_KNOBS ; i++)
307 knobs [COLAMD_DENSE_ROW] = 0.5 ;
308 knobs [COLAMD_DENSE_COL] = 0.5 ;
328 template <
typename Index>
329 static bool colamd(Index n_row, Index n_col, Index Alen, Index *A, Index *p,
double knobs[COLAMD_KNOBS], Index stats[COLAMD_STATS])
338 Colamd_Row<Index> *Row ;
339 colamd_col<Index> *Col ;
344 double default_knobs [COLAMD_KNOBS] ;
351 COLAMD_DEBUG0 ((
"colamd: stats not present\n")) ;
354 for (i = 0 ; i < COLAMD_STATS ; i++)
358 stats [COLAMD_STATUS] = COLAMD_OK ;
359 stats [COLAMD_INFO1] = -1 ;
360 stats [COLAMD_INFO2] = -1 ;
364 stats [COLAMD_STATUS] = COLAMD_ERROR_A_not_present ;
365 COLAMD_DEBUG0 ((
"colamd: A not present\n")) ;
371 stats [COLAMD_STATUS] = COLAMD_ERROR_p_not_present ;
372 COLAMD_DEBUG0 ((
"colamd: p not present\n")) ;
378 stats [COLAMD_STATUS] = COLAMD_ERROR_nrow_negative ;
379 stats [COLAMD_INFO1] = n_row ;
380 COLAMD_DEBUG0 ((
"colamd: nrow negative %d\n", n_row)) ;
386 stats [COLAMD_STATUS] = COLAMD_ERROR_ncol_negative ;
387 stats [COLAMD_INFO1] = n_col ;
388 COLAMD_DEBUG0 ((
"colamd: ncol negative %d\n", n_col)) ;
395 stats [COLAMD_STATUS] = COLAMD_ERROR_nnz_negative ;
396 stats [COLAMD_INFO1] = nnz ;
397 COLAMD_DEBUG0 ((
"colamd: number of entries negative %d\n", nnz)) ;
403 stats [COLAMD_STATUS] = COLAMD_ERROR_p0_nonzero ;
404 stats [COLAMD_INFO1] = p [0] ;
405 COLAMD_DEBUG0 ((
"colamd: p[0] not zero %d\n", p [0])) ;
413 colamd_set_defaults (default_knobs) ;
414 knobs = default_knobs ;
419 Col_size = colamd_c (n_col) ;
420 Row_size = colamd_r (n_row) ;
421 need = 2*nnz + n_col + Col_size + Row_size ;
426 stats [COLAMD_STATUS] = COLAMD_ERROR_A_too_small ;
427 stats [COLAMD_INFO1] = need ;
428 stats [COLAMD_INFO2] = Alen ;
429 COLAMD_DEBUG0 ((
"colamd: Need Alen >= %d, given only Alen = %d\n", need,Alen));
433 Alen -= Col_size + Row_size ;
434 Col = (colamd_col<Index> *) &A [Alen] ;
435 Row = (Colamd_Row<Index> *) &A [Alen + Col_size] ;
439 if (!Eigen::internal::init_rows_cols (n_row, n_col, Row, Col, A, p, stats))
442 COLAMD_DEBUG0 ((
"colamd: Matrix invalid\n")) ;
448 Eigen::internal::init_scoring (n_row, n_col, Row, Col, A, p, knobs,
449 &n_row2, &n_col2, &max_deg) ;
453 ngarbage = Eigen::internal::find_ordering (n_row, n_col, Alen, Row, Col, A, p,
454 n_col2, max_deg, 2*nnz) ;
458 Eigen::internal::order_children (n_col, Col, p) ;
462 stats [COLAMD_DENSE_ROW] = n_row - n_row2 ;
463 stats [COLAMD_DENSE_COL] = n_col - n_col2 ;
464 stats [COLAMD_DEFRAG_COUNT] = ngarbage ;
465 COLAMD_DEBUG0 ((
"colamd: done.\n")) ;
488 template <
typename Index>
489 static Index init_rows_cols
495 Colamd_Row<Index> Row [],
496 colamd_col<Index> Col [],
499 Index stats [COLAMD_STATS]
514 for (col = 0 ; col < n_col ; col++)
516 Col [col].start = p [col] ;
517 Col [col].length = p [col+1] - p [col] ;
519 if ((Col [col].length) < 0)
522 stats [COLAMD_STATUS] = COLAMD_ERROR_col_length_negative ;
523 stats [COLAMD_INFO1] = col ;
524 stats [COLAMD_INFO2] = Col [col].length ;
525 COLAMD_DEBUG0 ((
"colamd: col %d length %d < 0\n", col, Col [col].length)) ;
529 Col [col].shared1.thickness = 1 ;
530 Col [col].shared2.score = 0 ;
531 Col [col].shared3.prev = COLAMD_EMPTY ;
532 Col [col].shared4.degree_next = COLAMD_EMPTY ;
539 stats [COLAMD_INFO3] = 0 ;
541 for (row = 0 ; row < n_row ; row++)
543 Row [row].length = 0 ;
544 Row [row].shared2.mark = -1 ;
547 for (col = 0 ; col < n_col ; col++)
552 cp_end = &A [p [col+1]] ;
559 if (row < 0 || row >= n_row)
561 stats [COLAMD_STATUS] = COLAMD_ERROR_row_index_out_of_bounds ;
562 stats [COLAMD_INFO1] = col ;
563 stats [COLAMD_INFO2] = row ;
564 stats [COLAMD_INFO3] = n_row ;
565 COLAMD_DEBUG0 ((
"colamd: row %d col %d out of bounds\n", row, col)) ;
569 if (row <= last_row || Row [row].shared2.mark == col)
573 stats [COLAMD_STATUS] = COLAMD_OK_BUT_JUMBLED ;
574 stats [COLAMD_INFO1] = col ;
575 stats [COLAMD_INFO2] = row ;
576 (stats [COLAMD_INFO3]) ++ ;
577 COLAMD_DEBUG1 ((
"colamd: row %d col %d unsorted/duplicate\n",row,col));
580 if (Row [row].shared2.mark != col)
592 Row [row].shared2.mark = col ;
602 Row [0].start = p [n_col] ;
603 Row [0].shared1.p = Row [0].start ;
604 Row [0].shared2.mark = -1 ;
605 for (row = 1 ; row < n_row ; row++)
607 Row [row].start = Row [row-1].start + Row [row-1].length ;
608 Row [row].shared1.p = Row [row].start ;
609 Row [row].shared2.mark = -1 ;
614 if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED)
617 for (col = 0 ; col < n_col ; col++)
620 cp_end = &A [p [col+1]] ;
624 if (Row [row].shared2.mark != col)
626 A [(Row [row].shared1.p)++] = col ;
627 Row [row].shared2.mark = col ;
635 for (col = 0 ; col < n_col ; col++)
638 cp_end = &A [p [col+1]] ;
641 A [(Row [*cp++].shared1.p)++] = col ;
648 for (row = 0 ; row < n_row ; row++)
650 Row [row].shared2.mark = 0 ;
651 Row [row].shared1.degree = Row [row].length ;
656 if (stats [COLAMD_STATUS] == COLAMD_OK_BUT_JUMBLED)
658 COLAMD_DEBUG0 ((
"colamd: reconstructing column form, matrix jumbled\n")) ;
668 p [0] = Col [0].start ;
669 for (col = 1 ; col < n_col ; col++)
673 Col [col].start = Col [col-1].start + Col [col-1].length ;
674 p [col] = Col [col].start ;
679 for (row = 0 ; row < n_row ; row++)
681 rp = &A [Row [row].start] ;
682 rp_end = rp + Row [row].length ;
685 A [(p [*rp++])++] = row ;
704 template <
typename Index>
705 static void init_scoring
711 Colamd_Row<Index> Row [],
712 colamd_col<Index> Col [],
715 double knobs [COLAMD_KNOBS],
733 Index dense_row_count ;
734 Index dense_col_count ;
742 dense_row_count = COLAMD_MAX (0, COLAMD_MIN (knobs [COLAMD_DENSE_ROW] * n_col, n_col)) ;
743 dense_col_count = COLAMD_MAX (0, COLAMD_MIN (knobs [COLAMD_DENSE_COL] * n_row, n_row)) ;
744 COLAMD_DEBUG1 ((
"colamd: densecount: %d %d\n", dense_row_count, dense_col_count)) ;
753 for (c = n_col-1 ; c >= 0 ; c--)
755 deg = Col [c].length ;
759 Col [c].shared2.order = --n_col2 ;
760 KILL_PRINCIPAL_COL (c) ;
763 COLAMD_DEBUG1 ((
"colamd: null columns killed: %d\n", n_col - n_col2)) ;
768 for (c = n_col-1 ; c >= 0 ; c--)
775 deg = Col [c].length ;
776 if (deg > dense_col_count)
779 Col [c].shared2.order = --n_col2 ;
781 cp = &A [Col [c].start] ;
782 cp_end = cp + Col [c].length ;
785 Row [*cp++].shared1.degree-- ;
787 KILL_PRINCIPAL_COL (c) ;
790 COLAMD_DEBUG1 ((
"colamd: Dense and null columns killed: %d\n", n_col - n_col2)) ;
794 for (r = 0 ; r < n_row ; r++)
796 deg = Row [r].shared1.degree ;
797 COLAMD_ASSERT (deg >= 0 && deg <= n_col) ;
798 if (deg > dense_row_count || deg == 0)
807 max_deg = COLAMD_MAX (max_deg, deg) ;
810 COLAMD_DEBUG1 ((
"colamd: Dense and null rows killed: %d\n", n_row - n_row2)) ;
820 for (c = n_col-1 ; c >= 0 ; c--)
828 cp = &A [Col [c].start] ;
830 cp_end = cp + Col [c].length ;
836 if (ROW_IS_DEAD (row))
843 score += Row [row].shared1.degree - 1 ;
845 score = COLAMD_MIN (score, n_col) ;
848 col_length = (Index) (new_cp - &A [Col [c].start]) ;
853 COLAMD_DEBUG2 ((
"Newly null killed: %d\n", c)) ;
854 Col [c].shared2.order = --n_col2 ;
855 KILL_PRINCIPAL_COL (c) ;
860 COLAMD_ASSERT (score >= 0) ;
861 COLAMD_ASSERT (score <= n_col) ;
862 Col [c].length = col_length ;
863 Col [c].shared2.score = score ;
866 COLAMD_DEBUG1 ((
"colamd: Dense, null, and newly-null columns killed: %d\n",
878 for (c = 0 ; c <= n_col ; c++)
880 head [c] = COLAMD_EMPTY ;
885 for (c = n_col-1 ; c >= 0 ; c--)
888 if (COL_IS_ALIVE (c))
890 COLAMD_DEBUG4 ((
"place %d score %d minscore %d ncol %d\n",
891 c, Col [c].shared2.score, min_score, n_col)) ;
895 score = Col [c].shared2.score ;
897 COLAMD_ASSERT (min_score >= 0) ;
898 COLAMD_ASSERT (min_score <= n_col) ;
899 COLAMD_ASSERT (score >= 0) ;
900 COLAMD_ASSERT (score <= n_col) ;
901 COLAMD_ASSERT (head [score] >= COLAMD_EMPTY) ;
904 next_col = head [score] ;
905 Col [c].shared3.prev = COLAMD_EMPTY ;
906 Col [c].shared4.degree_next = next_col ;
910 if (next_col != COLAMD_EMPTY)
912 Col [next_col].shared3.prev = c ;
917 min_score = COLAMD_MIN (min_score, score) ;
928 *p_max_deg = max_deg ;
941 template <
typename Index>
942 static Index find_ordering
949 Colamd_Row<Index> Row [],
950 colamd_col<Index> Col [],
967 Index pivot_row_start ;
968 Index pivot_row_degree ;
969 Index pivot_row_length ;
970 Index pivot_col_score ;
971 Index needed_memory ;
983 Index set_difference ;
985 Index col_thickness ;
987 Index pivot_col_thickness ;
995 max_mark = INT_MAX - n_col ;
996 tag_mark = Eigen::internal::clear_mark (n_row, Row) ;
999 COLAMD_DEBUG1 ((
"colamd: Ordering, n_col2=%d\n", n_col2)) ;
1003 for (k = 0 ; k < n_col2 ; )
1009 COLAMD_ASSERT (min_score >= 0) ;
1010 COLAMD_ASSERT (min_score <= n_col) ;
1011 COLAMD_ASSERT (head [min_score] >= COLAMD_EMPTY) ;
1014 while (head [min_score] == COLAMD_EMPTY && min_score < n_col)
1018 pivot_col = head [min_score] ;
1019 COLAMD_ASSERT (pivot_col >= 0 && pivot_col <= n_col) ;
1020 next_col = Col [pivot_col].shared4.degree_next ;
1021 head [min_score] = next_col ;
1022 if (next_col != COLAMD_EMPTY)
1024 Col [next_col].shared3.prev = COLAMD_EMPTY ;
1027 COLAMD_ASSERT (COL_IS_ALIVE (pivot_col)) ;
1028 COLAMD_DEBUG3 ((
"Pivot col: %d\n", pivot_col)) ;
1031 pivot_col_score = Col [pivot_col].shared2.score ;
1034 Col [pivot_col].shared2.order = k ;
1037 pivot_col_thickness = Col [pivot_col].shared1.thickness ;
1038 k += pivot_col_thickness ;
1039 COLAMD_ASSERT (pivot_col_thickness > 0) ;
1043 needed_memory = COLAMD_MIN (pivot_col_score, n_col - k) ;
1044 if (pfree + needed_memory >= Alen)
1046 pfree = Eigen::internal::garbage_collection (n_row, n_col, Row, Col, A, &A [pfree]) ;
1049 COLAMD_ASSERT (pfree + needed_memory < Alen) ;
1051 tag_mark = Eigen::internal::clear_mark (n_row, Row) ;
1058 pivot_row_start = pfree ;
1061 pivot_row_degree = 0 ;
1065 Col [pivot_col].shared1.thickness = -pivot_col_thickness ;
1068 cp = &A [Col [pivot_col].start] ;
1069 cp_end = cp + Col [pivot_col].length ;
1074 COLAMD_DEBUG4 ((
"Pivot col pattern %d %d\n", ROW_IS_ALIVE (row), row)) ;
1076 if (ROW_IS_DEAD (row))
1080 rp = &A [Row [row].start] ;
1081 rp_end = rp + Row [row].length ;
1087 col_thickness = Col [col].shared1.thickness ;
1088 if (col_thickness > 0 && COL_IS_ALIVE (col))
1091 Col [col].shared1.thickness = -col_thickness ;
1092 COLAMD_ASSERT (pfree < Alen) ;
1095 pivot_row_degree += col_thickness ;
1101 Col [pivot_col].shared1.thickness = pivot_col_thickness ;
1102 max_deg = COLAMD_MAX (max_deg, pivot_row_degree) ;
1108 cp = &A [Col [pivot_col].start] ;
1109 cp_end = cp + Col [pivot_col].length ;
1114 COLAMD_DEBUG3 ((
"Kill row in pivot col: %d\n", row)) ;
1120 pivot_row_length = pfree - pivot_row_start ;
1121 if (pivot_row_length > 0)
1124 pivot_row = A [Col [pivot_col].start] ;
1125 COLAMD_DEBUG3 ((
"Pivotal row is %d\n", pivot_row)) ;
1130 pivot_row = COLAMD_EMPTY ;
1131 COLAMD_ASSERT (pivot_row_length == 0) ;
1133 COLAMD_ASSERT (Col [pivot_col].length > 0 || pivot_row_length == 0) ;
1156 COLAMD_DEBUG3 ((
"** Computing set differences phase. **\n")) ;
1160 COLAMD_DEBUG3 ((
"Pivot row: ")) ;
1162 rp = &A [pivot_row_start] ;
1163 rp_end = rp + pivot_row_length ;
1167 COLAMD_ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ;
1168 COLAMD_DEBUG3 ((
"Col: %d\n", col)) ;
1171 col_thickness = -Col [col].shared1.thickness ;
1172 COLAMD_ASSERT (col_thickness > 0) ;
1173 Col [col].shared1.thickness = col_thickness ;
1177 cur_score = Col [col].shared2.score ;
1178 prev_col = Col [col].shared3.prev ;
1179 next_col = Col [col].shared4.degree_next ;
1180 COLAMD_ASSERT (cur_score >= 0) ;
1181 COLAMD_ASSERT (cur_score <= n_col) ;
1182 COLAMD_ASSERT (cur_score >= COLAMD_EMPTY) ;
1183 if (prev_col == COLAMD_EMPTY)
1185 head [cur_score] = next_col ;
1189 Col [prev_col].shared4.degree_next = next_col ;
1191 if (next_col != COLAMD_EMPTY)
1193 Col [next_col].shared3.prev = prev_col ;
1198 cp = &A [Col [col].start] ;
1199 cp_end = cp + Col [col].length ;
1204 row_mark = Row [row].shared2.mark ;
1206 if (ROW_IS_MARKED_DEAD (row_mark))
1210 COLAMD_ASSERT (row != pivot_row) ;
1211 set_difference = row_mark - tag_mark ;
1213 if (set_difference < 0)
1215 COLAMD_ASSERT (Row [row].shared1.degree <= max_deg) ;
1216 set_difference = Row [row].shared1.degree ;
1219 set_difference -= col_thickness ;
1220 COLAMD_ASSERT (set_difference >= 0) ;
1222 if (set_difference == 0)
1224 COLAMD_DEBUG3 ((
"aggressive absorption. Row: %d\n", row)) ;
1230 Row [row].shared2.mark = set_difference + tag_mark ;
1238 COLAMD_DEBUG3 ((
"** Adding set differences phase. **\n")) ;
1241 rp = &A [pivot_row_start] ;
1242 rp_end = rp + pivot_row_length ;
1247 COLAMD_ASSERT (COL_IS_ALIVE (col) && col != pivot_col) ;
1250 cp = &A [Col [col].start] ;
1253 cp_end = cp + Col [col].length ;
1255 COLAMD_DEBUG4 ((
"Adding set diffs for Col: %d.\n", col)) ;
1261 COLAMD_ASSERT(row >= 0 && row < n_row) ;
1262 row_mark = Row [row].shared2.mark ;
1264 if (ROW_IS_MARKED_DEAD (row_mark))
1268 COLAMD_ASSERT (row_mark > tag_mark) ;
1274 cur_score += row_mark - tag_mark ;
1276 cur_score = COLAMD_MIN (cur_score, n_col) ;
1280 Col [col].length = (Index) (new_cp - &A [Col [col].start]) ;
1284 if (Col [col].length == 0)
1286 COLAMD_DEBUG4 ((
"further mass elimination. Col: %d\n", col)) ;
1288 KILL_PRINCIPAL_COL (col) ;
1289 pivot_row_degree -= Col [col].shared1.thickness ;
1290 COLAMD_ASSERT (pivot_row_degree >= 0) ;
1292 Col [col].shared2.order = k ;
1294 k += Col [col].shared1.thickness ;
1300 COLAMD_DEBUG4 ((
"Preparing supercol detection for Col: %d.\n", col)) ;
1303 Col [col].shared2.score = cur_score ;
1308 COLAMD_DEBUG4 ((
" Hash = %d, n_col = %d.\n", hash, n_col)) ;
1309 COLAMD_ASSERT (hash <= n_col) ;
1311 head_column = head [hash] ;
1312 if (head_column > COLAMD_EMPTY)
1316 first_col = Col [head_column].shared3.headhash ;
1317 Col [head_column].shared3.headhash = col ;
1322 first_col = - (head_column + 2) ;
1323 head [hash] = - (col + 2) ;
1325 Col [col].shared4.hash_next = first_col ;
1328 Col [col].shared3.hash = (Index) hash ;
1329 COLAMD_ASSERT (COL_IS_ALIVE (col)) ;
1337 COLAMD_DEBUG3 ((
"** Supercolumn detection phase. **\n")) ;
1339 Eigen::internal::detect_super_cols (Col, A, head, pivot_row_start, pivot_row_length) ;
1343 KILL_PRINCIPAL_COL (pivot_col) ;
1347 tag_mark += (max_deg + 1) ;
1348 if (tag_mark >= max_mark)
1350 COLAMD_DEBUG2 ((
"clearing tag_mark\n")) ;
1351 tag_mark = Eigen::internal::clear_mark (n_row, Row) ;
1356 COLAMD_DEBUG3 ((
"** Finalize scores phase. **\n")) ;
1359 rp = &A [pivot_row_start] ;
1362 rp_end = rp + pivot_row_length ;
1367 if (COL_IS_DEAD (col))
1373 A [Col [col].start + (Col [col].length++)] = pivot_row ;
1378 cur_score = Col [col].shared2.score + pivot_row_degree ;
1383 max_score = n_col - k - Col [col].shared1.thickness ;
1386 cur_score -= Col [col].shared1.thickness ;
1389 cur_score = COLAMD_MIN (cur_score, max_score) ;
1390 COLAMD_ASSERT (cur_score >= 0) ;
1393 Col [col].shared2.score = cur_score ;
1397 COLAMD_ASSERT (min_score >= 0) ;
1398 COLAMD_ASSERT (min_score <= n_col) ;
1399 COLAMD_ASSERT (cur_score >= 0) ;
1400 COLAMD_ASSERT (cur_score <= n_col) ;
1401 COLAMD_ASSERT (head [cur_score] >= COLAMD_EMPTY) ;
1402 next_col = head [cur_score] ;
1403 Col [col].shared4.degree_next = next_col ;
1404 Col [col].shared3.prev = COLAMD_EMPTY ;
1405 if (next_col != COLAMD_EMPTY)
1407 Col [next_col].shared3.prev = col ;
1409 head [cur_score] = col ;
1412 min_score = COLAMD_MIN (min_score, cur_score) ;
1418 if (pivot_row_degree > 0)
1422 Row [pivot_row].start = pivot_row_start ;
1423 Row [pivot_row].length = (Index) (new_rp - &A[pivot_row_start]) ;
1424 Row [pivot_row].shared1.degree = pivot_row_degree ;
1425 Row [pivot_row].shared2.mark = 0 ;
1452 template <
typename Index>
1453 static inline void order_children
1458 colamd_col<Index> Col [],
1471 for (i = 0 ; i < n_col ; i++)
1474 COLAMD_ASSERT (COL_IS_DEAD (i)) ;
1475 if (!COL_IS_DEAD_PRINCIPAL (i) && Col [i].shared2.order == COLAMD_EMPTY)
1481 parent = Col [parent].shared1.parent ;
1482 }
while (!COL_IS_DEAD_PRINCIPAL (parent)) ;
1488 order = Col [parent].shared2.order ;
1492 COLAMD_ASSERT (Col [c].shared2.order == COLAMD_EMPTY) ;
1495 Col [c].shared2.order = order++ ;
1497 Col [c].shared1.parent = parent ;
1500 c = Col [c].shared1.parent ;
1505 }
while (Col [c].shared2.order == COLAMD_EMPTY) ;
1508 Col [parent].shared2.order = order ;
1514 for (c = 0 ; c < n_col ; c++)
1516 p [Col [c].shared2.order] = c ;
1553 template <
typename Index>
1554 static void detect_super_cols
1558 colamd_col<Index> Col [],
1583 rp = &A [row_start] ;
1584 rp_end = rp + row_length ;
1588 if (COL_IS_DEAD (col))
1594 hash = Col [col].shared3.hash ;
1595 COLAMD_ASSERT (hash <= n_col) ;
1599 head_column = head [hash] ;
1600 if (head_column > COLAMD_EMPTY)
1602 first_col = Col [head_column].shared3.headhash ;
1606 first_col = - (head_column + 2) ;
1611 for (super_c = first_col ; super_c != COLAMD_EMPTY ;
1612 super_c = Col [super_c].shared4.hash_next)
1614 COLAMD_ASSERT (COL_IS_ALIVE (super_c)) ;
1615 COLAMD_ASSERT (Col [super_c].shared3.hash == hash) ;
1616 length = Col [super_c].length ;
1623 for (c = Col [super_c].shared4.hash_next ;
1624 c != COLAMD_EMPTY ; c = Col [c].shared4.hash_next)
1626 COLAMD_ASSERT (c != super_c) ;
1627 COLAMD_ASSERT (COL_IS_ALIVE (c)) ;
1628 COLAMD_ASSERT (Col [c].shared3.hash == hash) ;
1631 if (Col [c].length != length ||
1632 Col [c].shared2.score != Col [super_c].shared2.score)
1639 cp1 = &A [Col [super_c].start] ;
1640 cp2 = &A [Col [c].start] ;
1642 for (i = 0 ; i < length ; i++)
1645 COLAMD_ASSERT (ROW_IS_ALIVE (*cp1)) ;
1646 COLAMD_ASSERT (ROW_IS_ALIVE (*cp2)) ;
1649 if (*cp1++ != *cp2++)
1664 COLAMD_ASSERT (Col [c].shared2.score == Col [super_c].shared2.score) ;
1666 Col [super_c].shared1.thickness += Col [c].shared1.thickness ;
1667 Col [c].shared1.parent = super_c ;
1668 KILL_NON_PRINCIPAL_COL (c) ;
1670 Col [c].shared2.order = COLAMD_EMPTY ;
1672 Col [prev_c].shared4.hash_next = Col [c].shared4.hash_next ;
1678 if (head_column > COLAMD_EMPTY)
1681 Col [head_column].shared3.headhash = COLAMD_EMPTY ;
1686 head [hash] = COLAMD_EMPTY ;
1704 template <
typename Index>
1705 static Index garbage_collection
1711 Colamd_Row<Index> Row [],
1712 colamd_col<Index> Col [],
1729 for (c = 0 ; c < n_col ; c++)
1731 if (COL_IS_ALIVE (c))
1733 psrc = &A [Col [c].start] ;
1736 COLAMD_ASSERT (pdest <= psrc) ;
1737 Col [c].start = (Index) (pdest - &A [0]) ;
1738 length = Col [c].length ;
1739 for (j = 0 ; j < length ; j++)
1742 if (ROW_IS_ALIVE (r))
1747 Col [c].length = (Index) (pdest - &A [Col [c].start]) ;
1753 for (r = 0 ; r < n_row ; r++)
1755 if (ROW_IS_ALIVE (r))
1757 if (Row [r].length == 0)
1760 COLAMD_DEBUG3 ((
"Defrag row kill\n")) ;
1766 psrc = &A [Row [r].start] ;
1767 Row [r].shared2.first_column = *psrc ;
1768 COLAMD_ASSERT (ROW_IS_ALIVE (r)) ;
1770 *psrc = ONES_COMPLEMENT (r) ;
1779 while (psrc < pfree)
1786 r = ONES_COMPLEMENT (*psrc) ;
1787 COLAMD_ASSERT (r >= 0 && r < n_row) ;
1789 *psrc = Row [r].shared2.first_column ;
1790 COLAMD_ASSERT (ROW_IS_ALIVE (r)) ;
1793 COLAMD_ASSERT (pdest <= psrc) ;
1794 Row [r].start = (Index) (pdest - &A [0]) ;
1795 length = Row [r].length ;
1796 for (j = 0 ; j < length ; j++)
1799 if (COL_IS_ALIVE (c))
1804 Row [r].length = (Index) (pdest - &A [Row [r].start]) ;
1809 COLAMD_ASSERT (debug_rows == 0) ;
1813 return ((Index) (pdest - &A [0])) ;
1825 template <
typename Index>
1826 static inline Index clear_mark
1831 Colamd_Row<Index> Row []
1838 for (r = 0 ; r < n_row ; r++)
1840 if (ROW_IS_ALIVE (r))
1842 Row [r].shared2.mark = 0 ;