mathjs
Version:
Math.js is an extensive math library for JavaScript and Node.js. It features a flexible expression parser with support for symbolic computation, comes with a large set of built-in functions and constants, and offers an integrated solution to work with dif
92 lines (86 loc) • 2.63 kB
JavaScript
function factory (type, config, load) {
const csCumsum = load(require('./csCumsum'))
const conj = load(require('../../complex/conj'))
const SparseMatrix = type.SparseMatrix
/**
* Computes the symmetric permutation of matrix A accessing only
* the upper triangular part of A.
*
* C = P * A * P'
*
* @param {Matrix} a The A matrix
* @param {Array} pinv The inverse of permutation vector
* @param {boolean} values Process matrix values (true)
*
* @return {Matrix} The C matrix, C = P * A * P'
*
* Reference: http://faculty.cse.tamu.edu/davis/publications.html
*/
const csSymperm = function (a, pinv, values) {
// A matrix arrays
const avalues = a._values
const aindex = a._index
const aptr = a._ptr
const asize = a._size
// columns
const n = asize[1]
// C matrix arrays
const cvalues = values && avalues ? [] : null
const cindex = [] // (nz)
const cptr = [] // (n + 1)
// variables
let i, i2, j, j2, p, p0, p1
// create workspace vector
const w = [] // (n)
// count entries in each column of C
for (j = 0; j < n; j++) {
// column j of A is column j2 of C
j2 = pinv ? pinv[j] : j
// loop values in column j
for (p0 = aptr[j], p1 = aptr[j + 1], p = p0; p < p1; p++) {
// row
i = aindex[p]
// skip lower triangular part of A
if (i > j) { continue }
// row i of A is row i2 of C
i2 = pinv ? pinv[i] : i
// column count of C
w[Math.max(i2, j2)]++
}
}
// compute column pointers of C
csCumsum(cptr, w, n)
// loop columns
for (j = 0; j < n; j++) {
// column j of A is column j2 of C
j2 = pinv ? pinv[j] : j
// loop values in column j
for (p0 = aptr[j], p1 = aptr[j + 1], p = p0; p < p1; p++) {
// row
i = aindex[p]
// skip lower triangular part of A
if (i > j) { continue }
// row i of A is row i2 of C
i2 = pinv ? pinv[i] : i
// C index for column j2
const q = w[Math.max(i2, j2)]++
// update C index for entry q
cindex[q] = Math.min(i2, j2)
// check we need to process values
if (cvalues) { cvalues[q] = (i2 <= j2) ? avalues[p] : conj(avalues[p]) }
}
}
// return C matrix
return new SparseMatrix({
values: cvalues,
index: cindex,
ptr: cptr,
size: [n, n]
})
}
return csSymperm
}
exports.name = 'csSymperm'
exports.path = 'algebra.sparse'
exports.factory = factory