/* ========================================================================== */
/* === BTF package ========================================================== */
/* ========================================================================== */

/* MAXTRANS:  find a column permutation Q to give A*Q a zero-free diagonal
 * STRONGCOMP:  find a symmetric permutation P to put P*A*P' into block upper
 *	triangular form.
 * BTF_ORDER: do both of the above (maxtrans then strongcomp).
 */


/* ========================================================================== */
/* === MAXTRANS ============================================================= */
/* ========================================================================== */

/* MAXTRANS: finds a permutation of the columns of a matrix so that it has a
 * zero-free diagonal.  The input is an n-by-n sparse matrix in compressed
 * column form.  The array Ap of size n+1 gives the starting and ending
 * positions of the columns in the array Ai.  Ap[0] must be zero. The array Ai
 * contains the row indices of the nonzeros of the matrix A, and is of size
 * Ap[n].  The row indices of column j are located in Ai[Ap[j] ... Ap[j+1]-1].
 * Row indices must be in the range 0 to n-1.  Duplicate entries may be present
 * in any given column.  The input matrix  is not checked for validity (row
 * indices out of the range 0 to n-1 will lead to an undeterminate result -
 * possibly a core dump, for example).  Row indices in any given column need
 * not be in sorted order.  However, if they are sorted and the matrix already
 * has a zero-free diagonal, then the identity permutation is returned.
 *
 * The output of maxtrans is an array Match of size n.  If row i is matched with
 * column j, then A(i,j) is nonzero, and then Match[i] = j.  If the matrix is
 * structurally nonzero, all entries in the Match array are unique.  The Match
 * array can also be interpreted as a column permutation.  That is, column k of
 * the original matrix becomes column Match[k] of the permuted matrix.  In
 * MATLAB, this can be expressed as (for non-structurally singular matrices):
 *
 *	Match = maxtrans (A) ;
 *	B = A (:, Match) ;
 *
 * except of course here the A matrix and Match vector are all 0-based (rows
 * and columns in the range 0 to n-1), not 1-based (rows/cols in range 1 to n).
 * The MATLAB dmperm routine returns a row permutation.  See the maxtrans
 * mexFunction for more details.
 *
 * If row i is not matched to any column, then Match[i] is < -1.  To find a
 * complete permutation can be returned, let j = (-(Match[i])-2) if
 * Match[i] < -1. The column index j can be computed using the macro
 *
 *	j = MAXTRANS_UNFLIP (Match [i]) ;
 *
 * Then row i can be considered to be matched to column j, except that the
 * A(i,j) entry is zero if Match[i] < 0.  If any such row i exists, the matrix
 * is structurally singular.  The maxtrans routine returns the number of
 * nonzeros on the permuted matrix.
 *
 * In the MATLAB mexFunction interface to maxtrans, 1 is added to the Match
 * array to obtain a 1-based permutation.  Thus, in MATLAB:
 *
 *	Match = maxtrans (A) ;	% has entries in the range 1:n and -(1:n)
 *	p = abs (Match) ;	% the permutation (either full rank or singular)
 *	B = A (:, p) ;		% permuted matrix (either full rank or singular)
 *	find (Match < 0) ;	% gives a list of zero diagonal entries of B
 *	sum (Match > 0) ;	% same as "sprank (A)"
 *
 * This behaviour differs from p = dmperm (A) in MATLAB, which returns p(i)=0
 * if row i is unmatched.  Thus:
 *
 *	p = dmperm (A) ;	% has entries in the range 0:n
 *	p                       % the permutation (only if full rank)
 *	B = A (:, p) ;		% permuted matrix (only if full rank)
 *	find (p == 0) ;		% gives a list of zero diagonal entries of B
 *	sum (p > 0) ;		% definition of sprank (A)
 *
 * This algorithm is based on the paper "On Algorithms for obtaining a maximum
 * transversal" by Iain Duff, ACM Trans. Mathematical Software, vol 7, no. 1,
 * pp. 315-330, and "Algorithm 575: Permutations for a zero-free diagonal",
 * same issue, pp. 387-390.  Algorithm 575 is MC21A in the Harwell Subroutine
 * Library.  This code is not merely a translation of the Fortran code into C.
 * It is a completely new implementation of the basic underlying method (depth
 * first search over a subgraph with nodes corresponding to columns matched so
 * far, and cheap matching).  This code was written with minimal observation of
 * the MC21A/B code itself.  See comments below for a comparison between the
 * maxtrans and MC21A/B codes.
 *
 * This routine operates on a column-form matrix and produces a column
 * permutation.  MC21A uses a row-form matrix and produces a row permutation.
 * The difference is merely one of convention in the comments and interpretation
 * of the inputs and outputs.  If you want a row permutation, simply pass a
 * compressed-row sparse matrix to this routine and you will get a row
 * permutation (just like MC21A).  Similarly, you can pass a column-oriented
 * matrix to MC21A and it will happily return a column permutation.
 *
 * Copyright (c) 2004.  Tim Davis, May 15, 2004, University of Florida,
 * with support from Sandia National Laboratories.  All Rights Reserved.
 *
 * TODO: test with duplicate entries to make sure it still works OK
 * TODO: extend to rectangular matrices (like dmperm).
 * TODO: check inputs and return -1 if the input matrix is invalid.
 */

#ifndef _BTF_H
#define _BTF_H

/* make it easy for C++ programs to include BTF */
#ifdef __cplusplus
extern "C" {
#endif

int maxtrans	    /* returns # of columns matched */
(
    /* input, not modified: */
    int nrow,	    /* A is nrow-by-ncol in compressed column form */
    int ncol,
    int Ap [ ],	    /* size ncol+1 */
    int Ai [ ],	    /* size nz = Ap [ncol] */

    /* output, not defined on input */
    int Match [ ],  /* size nrow.  Match [i] = j if column j matched to row i
		     * (see above for the singular-matrix case) */

    /* workspace, not defined on input or output */
    int Work [ ]    /* size 5*ncol */
) ;


/* ========================================================================== */
/* === MAXTRANS marking of singular columns ================================= */
/* ========================================================================== */


/* MAXTRANS_FLIP is a "negation about -1", and is used to mark an integer j
 * that is normally non-negative.  MAXTRANS_FLIP (-1) is -1.  MAXTRANS_FLIP of
 * a number > -1 is negative, and MAXTRANS_FLIP of a number < -1 is positive.
 * MAXTRANS_FLIP (MAXTRANS_FLIP (j)) = j for all integers j.  UNFLIP (j) acts
 * like an "absolute value" operation, and is always >= -1.  You can test
 * whether or not an integer j is "flipped" with the MAXTRANS_ISFLIPPED (j)
 * macro.
 */

#define MAXTRANS_FLIP(j) (-(j)-2)
#define MAXTRANS_ISFLIPPED(j) ((j) < -1)
#define MAXTRANS_UNFLIP(j) ((MAXTRANS_ISFLIPPED (j)) ? MAXTRANS_FLIP (j) : (j))

/* ========================================================================== */
/* === STRONGCOMP =========================================================== */
/* ========================================================================== */


int strongcomp	    /* return # of strongly connected components */
(
    /* input, not modified: */
    int n,	    /* A is n-by-n in compressed column form */
    int Ap [ ],	    /* size n+1 */
    int Ai [ ],	    /* size nz = Ap [n] */

    /* optional input, modified (if present) on output: */
    int Q [ ],	    /* size n, input column permutation */

    /* output, not defined on input */
    int P [ ],	    /* size n.  P [k] = j if row and column j are kth row/col
		     * in permuted matrix. */

    int R [ ],	    /* size n+1.  block b is in rows/cols R[b] ... R[b+1]-1 */

    /* workspace, not defined on input or output */
    int Work [ ]    /* size 4n */
) ;


/* ========================================================================== */
/* === BTF_ORDER ============================================================ */
/* ========================================================================== */


int btf_order	    /* returns number of blocks found */
(
    /* input, not modified: */
    int n,	    /* A is n-by-n in compressed column form */
    int Ap [ ],	    /* size n+1 */
    int Ai [ ],	    /* size nz = Ap [n] */

    /* output, not defined on input */
    int Q [ ],	    /* size n, column permutation */
    int P [ ],	    /* size n, row permutation */
    int R [ ],	    /* size n+1.  block b is in rows/cols R[b] ... R[b+1]-1 */
    int *nfound,    /* # nonzeros on diagonal of P*A*Q */

    /* workspace, not defined on input or output */
    int Work [ ]    /* size 5n */
) ;

/* ========================================================================== */
/* === BTF version ========================================================== */
/* ========================================================================== */

/* All versions of BTF include these definitions.
 * As an example, to test if the version you are using is 1.2 or later:
 *
 *	if (BTF_VERSION >= BTF_VERSION_CODE (1,2)) ...
 *
 * This also works during compile-time:
 *
 *	#if (BTF >= BTF_VERSION_CODE (1,2))
 *	    printf ("This is version 1.2 or later\n") ;
 *	#else
 *	    printf ("This is an early version\n") ;
 *	#endif
 */

#define BTF_DATE "Dec 12, 2006"
#define BTF_VERSION_CODE(main,sub) ((main) * 1000 + (sub))
#define BTF_MAIN_VERSION 0
#define BTF_SUB_VERSION 11
#define BTF_VERSION BTF_VERSION_CODE(BTF_MAIN_VERSION,BTF_SUB_VERSION)

#ifdef __cplusplus
}
#endif


#endif


syntax highlighted by Code2HTML, v. 0.9.1