File: cs_reachr_mex.c

package info (click to toggle)
suitesparse-metis 3.1.0-2
  • links: PTS, VCS
  • area: contrib
  • in suites: jessie, jessie-kfreebsd, wheezy
  • size: 36,560 kB
  • ctags: 7,484
  • sloc: ansic: 104,515; makefile: 5,984; fortran: 4,591; sh: 1,397; csh: 739; ruby: 603; perl: 219; sed: 164; awk: 18
file content (67 lines) | stat: -rw-r--r-- 1,708 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
#include "cs_mex.h"
/* find nonzero pattern of x=L\sparse(b).  L must be sparse, real, and lower
 * triangular.  b must be a real sparse vector. */

static
void dfsr (int j, const cs *L, int *top, int *xi, int *w)
{
    int p ;
    w [j] = 1 ;					/* mark node j */
    for (p = L->p [j] ; p < L->p [j+1] ; p++)	/* for each i in L(:,j) */
    {
	if (w [L->i [p]] != 1)			/* if i is unmarked */
	{
	    dfsr (L->i [p], L, top, xi, w) ;	/* start a dfs at i */
	}
    }
    xi [--(*top)] = j ;				/* push j onto the stack */
}

/* w [0..n-1] == 0 on input, <= 1 on output.  size n */
static
int reachr (const cs *L, const cs *B, int *xi, int *w)
{
    int p, n = L->n ;
    int top = n ;				/* stack is empty */
    for (p = B->p [0] ; p < B->p [1] ; p++)	/* for each i in pattern of b */
    {
	if (w [B->i [p]] != 1)			/* if i is unmarked */
	{
	    dfsr (B->i [p], L, &top, xi, w) ;	/* start a dfs at i */
	}
    }
    return (top) ;				/* return top of stack */
}

void mexFunction
(
    int nargout,
    mxArray *pargout [ ],
    int nargin,
    const mxArray *pargin [ ]
)
{
    cs Lmatrix, Bmatrix, *L, *B ;
    double *x ;
    int i, j, top, *xi ;

    if (nargout > 1 || nargin != 2)
    {
	mexErrMsgTxt ("Usage: x = cs_reachr(L,b)") ;
    }

    /* get inputs */
    L = cs_mex_get_sparse (&Lmatrix, 1, 1, pargin [0]) ;
    B = cs_mex_get_sparse (&Bmatrix, 0, 1, pargin [1]) ;
    cs_mex_check (0, L->n, 1, 0, 1, 1, pargin [1]) ;

    xi = cs_calloc (2*L->n, sizeof (int)) ;

    top = reachr (L, B, xi, xi + L->n) ;

    pargout [0] = mxCreateDoubleMatrix (L->n - top, 1, mxREAL) ;
    x = mxGetPr (pargout [0]) ;
    for (j = 0, i = top ; i < L->n ; i++, j++) x [j] = xi [i] ;

    cs_free (xi) ;
}