// ----------------------------------------------------------------------- // // Original Matlab code by John Burkardt, Florida State University // Triangle.NET code by Christian Woltering, http://triangle.codeplex.com/ // // ----------------------------------------------------------------------- namespace TriangleNet.Tools { using System; /// /// Applies the Cuthill and McKee renumbering algorithm to reduce the bandwidth of /// the adjacency matrix associated with the mesh. /// public class CuthillMcKee { // The adjacency matrix of the mesh. AdjacencyMatrix matrix; /// /// Gets the permutation vector for the Reverse Cuthill-McKee numbering. /// /// The mesh. /// Permutation vector. public int[] Renumber(Mesh mesh) { // Algorithm needs linear numbering of the nodes. mesh.Renumber(NodeNumbering.Linear); return Renumber(new AdjacencyMatrix(mesh)); } /// /// Gets the permutation vector for the Reverse Cuthill-McKee numbering. /// /// The mesh. /// Permutation vector. public int[] Renumber(AdjacencyMatrix matrix) { this.matrix = matrix; int bandwidth1 = matrix.Bandwidth(); var pcol = matrix.ColumnPointers; // Adjust column pointers (1-based indexing). Shift(pcol, true); // TODO: Make RCM work with 0-based matrix. // Compute the RCM permutation. int[] perm = GenerateRcm(); int[] perm_inv = PermInverse(perm); int bandwidth2 = PermBandwidth(perm, perm_inv); if (Log.Verbose) { Log.Instance.Info(String.Format("Reverse Cuthill-McKee (Bandwidth: {0} > {1})", bandwidth1, bandwidth2)); } // Adjust column pointers (0-based indexing). Shift(pcol, false); return perm_inv; } #region RCM /// /// Finds the reverse Cuthill-Mckee ordering for a general graph. /// /// The RCM ordering. /// /// For each connected component in the graph, the routine obtains /// an ordering by calling RCM. /// int[] GenerateRcm() { // Number of nodes in the mesh. int n = matrix.N; int[] perm = new int[n]; int i, num, root; int iccsze = 0; int level_num = 0; /// Index vector for a level structure. The level structure is stored in the /// currently unused spaces in the permutation vector PERM. int[] level_row = new int[n + 1]; /// Marks variables that have been numbered. int[] mask = new int[n]; for (i = 0; i < n; i++) { mask[i] = 1; } num = 1; for (i = 0; i < n; i++) { // For each masked connected component... if (mask[i] != 0) { root = i; // Find a pseudo-peripheral node ROOT. The level structure found by // ROOT_FIND is stored starting at PERM(NUM). FindRoot(ref root, mask, ref level_num, level_row, perm, num - 1); // RCM orders the component using ROOT as the starting node. Rcm(root, mask, perm, num - 1, ref iccsze); num += iccsze; // We can stop once every node is in one of the connected components. if (n < num) { return perm; } } } return perm; } /// /// RCM renumbers a connected component by the reverse Cuthill McKee algorithm. /// /// the node that defines the connected component. It is used as the starting /// point for the RCM ordering. /// Input/output, int MASK(NODE_NUM), a mask for the nodes. Only those nodes with /// nonzero input mask values are considered by the routine. The nodes numbered by RCM will have /// their mask values set to zero. /// Output, int PERM(NODE_NUM), the RCM ordering. /// Output, int ICCSZE, the size of the connected component that has been numbered. /// the number of nodes. /// /// The connected component is specified by a node ROOT and a mask. /// The numbering starts at the root node. /// /// An outline of the algorithm is as follows: /// /// X(1) = ROOT. /// /// for ( I = 1 to N-1) /// Find all unlabeled neighbors of X(I), /// assign them the next available labels, in order of increasing degree. /// /// When done, reverse the ordering. /// void Rcm(int root, int[] mask, int[] perm, int offset, ref int iccsze) { int[] pcol = matrix.ColumnPointers; int[] irow = matrix.RowIndices; int fnbr; int i, j, k, l; int jstop, jstrt; int lbegin, lnbr, lperm, lvlend; int nbr, node; // Number of nodes in the mesh. int n = matrix.N; /// Workspace, int DEG[NODE_NUM], a temporary vector used to hold /// the degree of the nodes in the section graph specified by mask and root. int[] deg = new int[n]; // Find the degrees of the nodes in the component specified by MASK and ROOT. Degree(root, mask, deg, ref iccsze, perm, offset); mask[root] = 0; if (iccsze <= 1) { return; } lvlend = 0; lnbr = 1; // LBEGIN and LVLEND point to the beginning and // the end of the current level respectively. while (lvlend < lnbr) { lbegin = lvlend + 1; lvlend = lnbr; for (i = lbegin; i <= lvlend; i++) { // For each node in the current level... node = perm[offset + i - 1]; jstrt = pcol[node]; jstop = pcol[node + 1] - 1; // Find the unnumbered neighbors of NODE. // FNBR and LNBR point to the first and last neighbors // of the current node in PERM. fnbr = lnbr + 1; for (j = jstrt; j <= jstop; j++) { nbr = irow[j - 1]; if (mask[nbr] != 0) { lnbr += 1; mask[nbr] = 0; perm[offset + lnbr - 1] = nbr; } } // Node has neighbors if (lnbr > fnbr) { // Sort the neighbors of NODE in increasing order by degree. // Linear insertion is used. k = fnbr; while (k < lnbr) { l = k; k = k + 1; nbr = perm[offset + k - 1]; while (fnbr < l) { lperm = perm[offset + l - 1]; if (deg[lperm - 1] <= deg[nbr - 1]) { break; } perm[offset + l] = lperm; l = l - 1; } perm[offset + l] = nbr; } } } } // We now have the Cuthill-McKee ordering. Reverse it. ReverseVector(perm, offset, iccsze); return; } /// /// Finds a pseudo-peripheral node. /// /// On input, ROOT is a node in the the component of the graph for /// which a pseudo-peripheral node is sought. On output, ROOT is the pseudo-peripheral /// node obtained. /// MASK[NODE_NUM], specifies a section subgraph. Nodes for which MASK /// is zero are ignored by FNROOT. /// Output, int LEVEL_NUM, is the number of levels in the level /// structure rooted at the node ROOT. /// Output, int LEVEL_ROW(NODE_NUM+1), the level structure array pair /// containing the level structure found. /// Output, int LEVEL(NODE_NUM), the level structure array pair /// containing the level structure found. /// the number of nodes. /// /// The diameter of a graph is the maximum distance (number of edges) /// between any two nodes of the graph. /// /// The eccentricity of a node is the maximum distance between that /// node and any other node of the graph. /// /// A peripheral node is a node whose eccentricity equals the /// diameter of the graph. /// /// A pseudo-peripheral node is an approximation to a peripheral node; /// it may be a peripheral node, but all we know is that we tried our /// best. /// /// The routine is given a graph, and seeks pseudo-peripheral nodes, /// using a modified version of the scheme of Gibbs, Poole and /// Stockmeyer. It determines such a node for the section subgraph /// specified by MASK and ROOT. /// /// The routine also determines the level structure associated with /// the given pseudo-peripheral node; that is, how far each node /// is from the pseudo-peripheral node. The level structure is /// returned as a list of nodes LS, and pointers to the beginning /// of the list of nodes that are at a distance of 0, 1, 2, ..., /// NODE_NUM-1 from the pseudo-peripheral node. /// /// Reference: /// Alan George, Joseph Liu, /// Computer Solution of Large Sparse Positive Definite Systems, /// Prentice Hall, 1981. /// /// Norman Gibbs, William Poole, Paul Stockmeyer, /// An Algorithm for Reducing the Bandwidth and Profile of a Sparse Matrix, /// SIAM Journal on Numerical Analysis, /// Volume 13, pages 236-250, 1976. /// /// Norman Gibbs, /// Algorithm 509: A Hybrid Profile Reduction Algorithm, /// ACM Transactions on Mathematical Software, /// Volume 2, pages 378-387, 1976. /// void FindRoot(ref int root, int[] mask, ref int level_num, int[] level_row, int[] level, int offset) { int[] pcol = matrix.ColumnPointers; int[] irow = matrix.RowIndices; int iccsze; int j, jstrt; int k, kstop, kstrt; int mindeg; int nghbor, ndeg; int node; int level_num2 = 0; // Determine the level structure rooted at ROOT. GetLevelSet(ref root, mask, ref level_num, level_row, level, offset); // Count the number of nodes in this level structure. iccsze = level_row[level_num] - 1; // Extreme cases: // A complete graph has a level set of only a single level. // Every node is equally good (or bad). // or // A "line graph" 0--0--0--0--0 has every node in its only level. // By chance, we've stumbled on the ideal root. if (level_num == 1 || level_num == iccsze) { return; } // Pick any node from the last level that has minimum degree // as the starting point to generate a new level set. for (; ; ) { mindeg = iccsze; jstrt = level_row[level_num - 1]; root = level[offset + jstrt - 1]; if (jstrt < iccsze) { for (j = jstrt; j <= iccsze; j++) { node = level[offset + j - 1]; ndeg = 0; kstrt = pcol[node - 1]; kstop = pcol[node] - 1; for (k = kstrt; k <= kstop; k++) { nghbor = irow[k - 1]; if (mask[nghbor] > 0) { ndeg += 1; } } if (ndeg < mindeg) { root = node; mindeg = ndeg; } } } // Generate the rooted level structure associated with this node. GetLevelSet(ref root, mask, ref level_num2, level_row, level, offset); // If the number of levels did not increase, accept the new ROOT. if (level_num2 <= level_num) { break; } level_num = level_num2; // In the unlikely case that ROOT is one endpoint of a line graph, // we can exit now. if (iccsze <= level_num) { break; } } return; } /// /// Generates the connected level structure rooted at a given node. /// /// the node at which the level structure is to be rooted. /// MASK[NODE_NUM]. On input, only nodes with nonzero MASK are to be processed. /// On output, those nodes which were included in the level set have MASK set to 1. /// Output, int LEVEL_NUM, the number of levels in the level structure. ROOT is /// in level 1. The neighbors of ROOT are in level 2, and so on. /// Output, int LEVEL_ROW[NODE_NUM+1], the rooted level structure. /// Output, int LEVEL[NODE_NUM], the rooted level structure. /// the number of nodes. /// /// Only nodes for which MASK is nonzero will be considered. /// /// The root node chosen by the user is assigned level 1, and masked. /// All (unmasked) nodes reachable from a node in level 1 are /// assigned level 2 and masked. The process continues until there /// are no unmasked nodes adjacent to any node in the current level. /// The number of levels may vary between 2 and NODE_NUM. /// /// Reference: /// Alan George, Joseph Liu, /// Computer Solution of Large Sparse Positive Definite Systems, /// Prentice Hall, 1981. /// void GetLevelSet(ref int root, int[] mask, ref int level_num, int[] level_row, int[] level, int offset) { int[] pcol = matrix.ColumnPointers; int[] irow = matrix.RowIndices; int i, iccsze; int j, jstop, jstrt; int lbegin, lvlend, lvsize; int nbr; int node; mask[root] = 0; level[offset] = root; level_num = 0; lvlend = 0; iccsze = 1; // LBEGIN is the pointer to the beginning of the current level, and // LVLEND points to the end of this level. for (; ; ) { lbegin = lvlend + 1; lvlend = iccsze; level_num += 1; level_row[level_num - 1] = lbegin; // Generate the next level by finding all the masked neighbors of nodes // in the current level. for (i = lbegin; i <= lvlend; i++) { node = level[offset + i - 1]; jstrt = pcol[node]; jstop = pcol[node + 1] - 1; for (j = jstrt; j <= jstop; j++) { nbr = irow[j - 1]; if (mask[nbr] != 0) { iccsze += 1; level[offset + iccsze - 1] = nbr; mask[nbr] = 0; } } } // Compute the current level width (the number of nodes encountered.) // If it is positive, generate the next level. lvsize = iccsze - lvlend; if (lvsize <= 0) { break; } } level_row[level_num] = lvlend + 1; // Reset MASK to 1 for the nodes in the level structure. for (i = 0; i < iccsze; i++) { mask[level[offset + i]] = 1; } return; } /// /// Computes the degrees of the nodes in the connected component. /// /// the node that defines the connected component. /// MASK[NODE_NUM], is nonzero for those nodes which are to be considered. /// Output, int DEG[NODE_NUM], contains, for each node in the connected component, its degree. /// Output, int ICCSIZE, the number of nodes in the connected component. /// Output, int LS[NODE_NUM], stores in entries 1 through ICCSIZE the nodes in the /// connected component, starting with ROOT, and proceeding by levels. /// the number of nodes. /// /// The connected component is specified by MASK and ROOT. /// Nodes for which MASK is zero are ignored. /// /// Reference: /// Alan George, Joseph Liu, /// Computer Solution of Large Sparse Positive Definite Systems, /// Prentice Hall, 1981. /// void Degree(int root, int[] mask, int[] deg, ref int iccsze, int[] ls, int offset) { int[] pcol = matrix.ColumnPointers; int[] irow = matrix.RowIndices; int i, ideg; int j, jstop, jstrt; int lbegin, lvlend; int lvsize = 1; int nbr, node; // The sign of ADJ_ROW(I) is used to indicate if node I has been considered. ls[offset] = root; pcol[root] = -pcol[root]; lvlend = 0; iccsze = 1; // If the current level width is nonzero, generate another level. while (lvsize > 0) { // LBEGIN is the pointer to the beginning of the current level, and // LVLEND points to the end of this level. lbegin = lvlend + 1; lvlend = iccsze; // Find the degrees of nodes in the current level, // and at the same time, generate the next level. for (i = lbegin; i <= lvlend; i++) { node = ls[offset + i - 1]; jstrt = -pcol[node]; jstop = Math.Abs(pcol[node + 1]) - 1; ideg = 0; for (j = jstrt; j <= jstop; j++) { nbr = irow[j - 1]; if (mask[nbr] != 0) // EDIT: [nbr - 1] { ideg = ideg + 1; if (0 <= pcol[nbr]) // EDIT: [nbr - 1] { pcol[nbr] = -pcol[nbr]; // EDIT: [nbr - 1] iccsze = iccsze + 1; ls[offset + iccsze - 1] = nbr; } } } deg[node] = ideg; } // Compute the current level width. lvsize = iccsze - lvlend; } // Reset ADJ_ROW to its correct sign and return. for (i = 0; i < iccsze; i++) { node = ls[offset + i]; pcol[node] = -pcol[node]; } return; } #endregion #region Tools /// /// Computes the bandwidth of a permuted adjacency matrix. /// /// The permutation. /// The inverse permutation. /// Bandwidth of the permuted adjacency matrix. /// /// The matrix is defined by the adjacency information and a permutation. /// The routine also computes the bandwidth and the size of the envelope. /// int PermBandwidth(int[] perm, int[] perm_inv) { int[] pcol = matrix.ColumnPointers; int[] irow = matrix.RowIndices; int col, i, j; int band_lo = 0; int band_hi = 0; int n = matrix.N; for (i = 0; i < n; i++) { for (j = pcol[perm[i]]; j < pcol[perm[i] + 1]; j++) { col = perm_inv[irow[j - 1]]; band_lo = Math.Max(band_lo, i - col); band_hi = Math.Max(band_hi, col - i); } } return band_lo + 1 + band_hi; } /// /// Produces the inverse of a given permutation. /// /// Number of items permuted. /// PERM[N], a permutation. /// The inverse permutation. int[] PermInverse(int[] perm) { int n = matrix.N; int[] perm_inv = new int[n]; for (int i = 0; i < n; i++) { perm_inv[perm[i]] = i; } return perm_inv; } /// /// Reverses the elements of an integer vector. /// /// number of entries in the array. /// the array to be reversed. /// /// Input: /// N = 5, /// A = ( 11, 12, 13, 14, 15 ). /// /// Output: /// A = ( 15, 14, 13, 12, 11 ). /// void ReverseVector(int[] a, int offset, int size) { int i; int j; for (i = 0; i < size / 2; i++) { j = a[offset + i]; a[offset + i] = a[offset + size - 1 - i]; a[offset + size - 1 - i] = j; } return; } void Shift(int[] a, bool up) { int length = a.Length; if (up) { for (int i = 0; i < length; a[i]++, i++) ; } else { for (int i = 0; i < length; a[i]--, i++) ; } } #endregion } }