COIN-OR::LEMON - Graph Library

source: glpk-cmake/src/amd/amd_order.c @ 1:c445c931472f

Last change on this file since 1:c445c931472f was 1:c445c931472f, checked in by Alpar Juttner <alpar@…>, 14 years ago

Import glpk-4.45

  • Generated files and doc/notes are removed
File size: 6.3 KB
Line 
1/* ========================================================================= */
2/* === AMD_order =========================================================== */
3/* ========================================================================= */
4
5/* ------------------------------------------------------------------------- */
6/* AMD, Copyright (c) Timothy A. Davis,                                      */
7/* Patrick R. Amestoy, and Iain S. Duff.  See ../README.txt for License.     */
8/* email: davis at cise.ufl.edu    CISE Department, Univ. of Florida.        */
9/* web: http://www.cise.ufl.edu/research/sparse/amd                          */
10/* ------------------------------------------------------------------------- */
11
12/* User-callable AMD minimum degree ordering routine.  See amd.h for
13 * documentation.
14 */
15
16#include "amd_internal.h"
17
18/* ========================================================================= */
19/* === AMD_order =========================================================== */
20/* ========================================================================= */
21
22GLOBAL Int AMD_order
23(
24    Int n,
25    const Int Ap [ ],
26    const Int Ai [ ],
27    Int P [ ],
28    double Control [ ],
29    double Info [ ]
30)
31{
32    Int *Len, *S, nz, i, *Pinv, info, status, *Rp, *Ri, *Cp, *Ci, ok ;
33    size_t nzaat, slen ;
34    double mem = 0 ;
35
36#ifndef NDEBUG
37    AMD_debug_init ("amd") ;
38#endif
39
40    /* clear the Info array, if it exists */
41    info = Info != (double *) NULL ;
42    if (info)
43    {
44        for (i = 0 ; i < AMD_INFO ; i++)
45        {
46            Info [i] = EMPTY ;
47        }
48        Info [AMD_N] = n ;
49        Info [AMD_STATUS] = AMD_OK ;
50    }
51
52    /* make sure inputs exist and n is >= 0 */
53    if (Ai == (Int *) NULL || Ap == (Int *) NULL || P == (Int *) NULL || n < 0)
54    {
55        if (info) Info [AMD_STATUS] = AMD_INVALID ;
56        return (AMD_INVALID) ;      /* arguments are invalid */
57    }
58
59    if (n == 0)
60    {
61        return (AMD_OK) ;           /* n is 0 so there's nothing to do */
62    }
63
64    nz = Ap [n] ;
65    if (info)
66    {
67        Info [AMD_NZ] = nz ;
68    }
69    if (nz < 0)
70    {
71        if (info) Info [AMD_STATUS] = AMD_INVALID ;
72        return (AMD_INVALID) ;
73    }
74
75    /* check if n or nz will cause size_t overflow */
76    if (((size_t) n) >= SIZE_T_MAX / sizeof (Int)
77     || ((size_t) nz) >= SIZE_T_MAX / sizeof (Int))
78    {
79        if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ;
80        return (AMD_OUT_OF_MEMORY) ;        /* problem too large */
81    }
82
83    /* check the input matrix:  AMD_OK, AMD_INVALID, or AMD_OK_BUT_JUMBLED */
84    status = AMD_valid (n, n, Ap, Ai) ;
85
86    if (status == AMD_INVALID)
87    {
88        if (info) Info [AMD_STATUS] = AMD_INVALID ;
89        return (AMD_INVALID) ;      /* matrix is invalid */
90    }
91
92    /* allocate two size-n integer workspaces */
93    Len = amd_malloc (n * sizeof (Int)) ;
94    Pinv = amd_malloc (n * sizeof (Int)) ;
95    mem += n ;
96    mem += n ;
97    if (!Len || !Pinv)
98    {
99        /* :: out of memory :: */
100        amd_free (Len) ;
101        amd_free (Pinv) ;
102        if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ;
103        return (AMD_OUT_OF_MEMORY) ;
104    }
105
106    if (status == AMD_OK_BUT_JUMBLED)
107    {
108        /* sort the input matrix and remove duplicate entries */
109        AMD_DEBUG1 (("Matrix is jumbled\n")) ;
110        Rp = amd_malloc ((n+1) * sizeof (Int)) ;
111        Ri = amd_malloc (MAX (nz,1) * sizeof (Int)) ;
112        mem += (n+1) ;
113        mem += MAX (nz,1) ;
114        if (!Rp || !Ri)
115        {
116            /* :: out of memory :: */
117            amd_free (Rp) ;
118            amd_free (Ri) ;
119            amd_free (Len) ;
120            amd_free (Pinv) ;
121            if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ;
122            return (AMD_OUT_OF_MEMORY) ;
123        }
124        /* use Len and Pinv as workspace to create R = A' */
125        AMD_preprocess (n, Ap, Ai, Rp, Ri, Len, Pinv) ;
126        Cp = Rp ;
127        Ci = Ri ;
128    }
129    else
130    {
131        /* order the input matrix as-is.  No need to compute R = A' first */
132        Rp = NULL ;
133        Ri = NULL ;
134        Cp = (Int *) Ap ;
135        Ci = (Int *) Ai ;
136    }
137
138    /* --------------------------------------------------------------------- */
139    /* determine the symmetry and count off-diagonal nonzeros in A+A' */
140    /* --------------------------------------------------------------------- */
141
142    nzaat = AMD_aat (n, Cp, Ci, Len, P, Info) ;
143    AMD_DEBUG1 (("nzaat: %g\n", (double) nzaat)) ;
144    ASSERT ((MAX (nz-n, 0) <= nzaat) && (nzaat <= 2 * (size_t) nz)) ;
145
146    /* --------------------------------------------------------------------- */
147    /* allocate workspace for matrix, elbow room, and 6 size-n vectors */
148    /* --------------------------------------------------------------------- */
149
150    S = NULL ;
151    slen = nzaat ;                      /* space for matrix */
152    ok = ((slen + nzaat/5) >= slen) ;   /* check for size_t overflow */
153    slen += nzaat/5 ;                   /* add elbow room */
154    for (i = 0 ; ok && i < 7 ; i++)
155    {
156        ok = ((slen + n) > slen) ;      /* check for size_t overflow */
157        slen += n ;                     /* size-n elbow room, 6 size-n work */
158    }
159    mem += slen ;
160    ok = ok && (slen < SIZE_T_MAX / sizeof (Int)) ; /* check for overflow */
161    ok = ok && (slen < Int_MAX) ;       /* S[i] for Int i must be OK */
162    if (ok)
163    {
164        S = amd_malloc (slen * sizeof (Int)) ;
165    }
166    AMD_DEBUG1 (("slen %g\n", (double) slen)) ;
167    if (!S)
168    {
169        /* :: out of memory :: (or problem too large) */
170        amd_free (Rp) ;
171        amd_free (Ri) ;
172        amd_free (Len) ;
173        amd_free (Pinv) ;
174        if (info) Info [AMD_STATUS] = AMD_OUT_OF_MEMORY ;
175        return (AMD_OUT_OF_MEMORY) ;
176    }
177    if (info)
178    {
179        /* memory usage, in bytes. */
180        Info [AMD_MEMORY] = mem * sizeof (Int) ;
181    }
182
183    /* --------------------------------------------------------------------- */
184    /* order the matrix */
185    /* --------------------------------------------------------------------- */
186
187    AMD_1 (n, Cp, Ci, P, Pinv, Len, slen, S, Control, Info) ;
188
189    /* --------------------------------------------------------------------- */
190    /* free the workspace */
191    /* --------------------------------------------------------------------- */
192
193    amd_free (Rp) ;
194    amd_free (Ri) ;
195    amd_free (Len) ;
196    amd_free (Pinv) ;
197    amd_free (S) ;
198    if (info) Info [AMD_STATUS] = status ;
199    return (status) ;       /* successful ordering */
200}
Note: See TracBrowser for help on using the repository browser.