/*
Copyright (C) 2006-2015 David Bateman
This file is part of Octave.
Octave is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation; either version 3 of the License, or (at your
option) any later version.
Octave is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with Octave; see the file COPYING. If not, see
.
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include
#include "MArray.h"
#include "MSparse.h"
#include "SparseQR.h"
#include "SparseCmplxQR.h"
#include "MatrixType.h"
#include "oct-sort.h"
#include "oct-locbuf.h"
#include "oct-inttypes.h"
template
static MSparse
dmsolve_extract (const MSparse &A, const octave_idx_type *Pinv,
const octave_idx_type *Q, octave_idx_type rst,
octave_idx_type rend, octave_idx_type cst,
octave_idx_type cend, octave_idx_type maxnz = -1,
bool lazy = false)
{
octave_idx_type nr = rend - rst;
octave_idx_type nc = cend - cst;
maxnz = (maxnz < 0 ? A.nnz () : maxnz);
octave_idx_type nz;
// Cast to uint64 to handle overflow in this multiplication
if (octave_uint64 (nr)*octave_uint64 (nc) < octave_uint64 (maxnz))
nz = nr*nc;
else
nz = maxnz;
MSparse B (nr, nc, (nz < maxnz ? nz : maxnz));
// Some sparse functions can support lazy indexing (where elements
// in the row are in no particular order), even though octave in
// general can't. For those functions that can using it is a big
// win here in terms of speed.
if (lazy)
{
nz = 0;
for (octave_idx_type j = cst ; j < cend ; j++)
{
octave_idx_type qq = (Q ? Q[j] : j);
B.xcidx (j - cst) = nz;
for (octave_idx_type p = A.cidx (qq) ; p < A.cidx (qq+1) ; p++)
{
octave_quit ();
octave_idx_type r = (Pinv ? Pinv[A.ridx (p)] : A.ridx (p));
if (r >= rst && r < rend)
{
B.xdata (nz) = A.data (p);
B.xridx (nz++) = r - rst ;
}
}
}
B.xcidx (cend - cst) = nz ;
}
else
{
OCTAVE_LOCAL_BUFFER (T, X, rend - rst);
octave_sort sort;
octave_idx_type *ri = B.xridx ();
nz = 0;
for (octave_idx_type j = cst ; j < cend ; j++)
{
octave_idx_type qq = (Q ? Q[j] : j);
B.xcidx (j - cst) = nz;
for (octave_idx_type p = A.cidx (qq) ; p < A.cidx (qq+1) ; p++)
{
octave_quit ();
octave_idx_type r = (Pinv ? Pinv[A.ridx (p)] : A.ridx (p));
if (r >= rst && r < rend)
{
X[r-rst] = A.data (p);
B.xridx (nz++) = r - rst ;
}
}
sort.sort (ri + B.xcidx (j - cst), nz - B.xcidx (j - cst));
for (octave_idx_type p = B.cidx (j - cst); p < nz; p++)
B.xdata (p) = X[B.xridx (p)];
}
B.xcidx (cend - cst) = nz ;
}
return B;
}
#if !defined (CXX_NEW_FRIEND_TEMPLATE_DECL)
static MSparse
dmsolve_extract (const MSparse &A, const octave_idx_type *Pinv,
const octave_idx_type *Q, octave_idx_type rst,
octave_idx_type rend, octave_idx_type cst,
octave_idx_type cend, octave_idx_type maxnz,
bool lazy);
static MSparse
dmsolve_extract (const MSparse &A, const octave_idx_type *Pinv,
const octave_idx_type *Q, octave_idx_type rst,
octave_idx_type rend, octave_idx_type cst,
octave_idx_type cend, octave_idx_type maxnz,
bool lazy);
#endif
template
static MArray
dmsolve_extract (const MArray &m, const octave_idx_type *,
const octave_idx_type *, octave_idx_type r1,
octave_idx_type r2, octave_idx_type c1,
octave_idx_type c2)
{
r2 -= 1;
c2 -= 1;
if (r1 > r2) { std::swap (r1, r2); }
if (c1 > c2) { std::swap (c1, c2); }
octave_idx_type new_r = r2 - r1 + 1;
octave_idx_type new_c = c2 - c1 + 1;
MArray result (dim_vector (new_r, new_c));
for (octave_idx_type j = 0; j < new_c; j++)
for (octave_idx_type i = 0; i < new_r; i++)
result.xelem (i, j) = m.elem (r1+i, c1+j);
return result;
}
#if !defined (CXX_NEW_FRIEND_TEMPLATE_DECL)
static MArray
dmsolve_extract (const MArray &m, const octave_idx_type *,
const octave_idx_type *, octave_idx_type r1,
octave_idx_type r2, octave_idx_type c1,
octave_idx_type c2)
static MArray
dmsolve_extract (const MArray &m, const octave_idx_type *,
const octave_idx_type *, octave_idx_type r1,
octave_idx_type r2, octave_idx_type c1,
octave_idx_type c2)
#endif
template
static void
dmsolve_insert (MArray &a, const MArray &b, const octave_idx_type *Q,
octave_idx_type r, octave_idx_type c)
{
T *ax = a.fortran_vec ();
const T *bx = b.fortran_vec ();
octave_idx_type anr = a.rows ();
octave_idx_type nr = b.rows ();
octave_idx_type nc = b.cols ();
for (octave_idx_type j = 0; j < nc; j++)
{
octave_idx_type aoff = (c + j) * anr;
octave_idx_type boff = j * nr;
for (octave_idx_type i = 0; i < nr; i++)
{
octave_quit ();
ax[Q[r + i] + aoff] = bx[i + boff];
}
}
}
#if !defined (CXX_NEW_FRIEND_TEMPLATE_DECL)
static void
dmsolve_insert (MArray &a, const MArray &b,
const octave_idx_type *Q, octave_idx_type r, octave_idx_type c);
static void
dmsolve_insert (MArray &a, const MArray &b,
const octave_idx_type *Q, octave_idx_type r, octave_idx_type c);
#endif
template
static void
dmsolve_insert (MSparse &a, const MSparse &b, const octave_idx_type *Q,
octave_idx_type r, octave_idx_type c)
{
octave_idx_type b_rows = b.rows ();
octave_idx_type b_cols = b.cols ();
octave_idx_type nr = a.rows ();
octave_idx_type nc = a.cols ();
OCTAVE_LOCAL_BUFFER (octave_idx_type, Qinv, nr);
for (octave_idx_type i = 0; i < nr; i++)
Qinv[Q[i]] = i;
// First count the number of elements in the final array
octave_idx_type nel = a.xcidx (c) + b.nnz ();
if (c + b_cols < nc)
nel += a.xcidx (nc) - a.xcidx (c + b_cols);
for (octave_idx_type i = c; i < c + b_cols; i++)
for (octave_idx_type j = a.xcidx (i); j < a.xcidx (i+1); j++)
if (Qinv[a.xridx (j)] < r || Qinv[a.xridx (j)] >= r + b_rows)
nel++;
OCTAVE_LOCAL_BUFFER (T, X, nr);
octave_sort sort;
MSparse tmp (a);
a = MSparse (nr, nc, nel);
octave_idx_type *ri = a.xridx ();
for (octave_idx_type i = 0; i < tmp.cidx (c); i++)
{
a.xdata (i) = tmp.xdata (i);
a.xridx (i) = tmp.xridx (i);
}
for (octave_idx_type i = 0; i < c + 1; i++)
a.xcidx (i) = tmp.xcidx (i);
octave_idx_type ii = a.xcidx (c);
for (octave_idx_type i = c; i < c + b_cols; i++)
{
octave_quit ();
for (octave_idx_type j = tmp.xcidx (i); j < tmp.xcidx (i+1); j++)
if (Qinv[tmp.xridx (j)] < r || Qinv[tmp.xridx (j)] >= r + b_rows)
{
X[tmp.xridx (j)] = tmp.xdata (j);
a.xridx (ii++) = tmp.xridx (j);
}
octave_quit ();
for (octave_idx_type j = b.cidx (i-c); j < b.cidx (i-c+1); j++)
{
X[Q[r + b.ridx (j)]] = b.data (j);
a.xridx (ii++) = Q[r + b.ridx (j)];
}
sort.sort (ri + a.xcidx (i), ii - a.xcidx (i));
for (octave_idx_type p = a.xcidx (i); p < ii; p++)
a.xdata (p) = X[a.xridx (p)];
a.xcidx (i+1) = ii;
}
for (octave_idx_type i = c + b_cols; i < nc; i++)
{
for (octave_idx_type j = tmp.xcidx (i); j < tmp.cidx (i+1); j++)
{
a.xdata (ii) = tmp.xdata (j);
a.xridx (ii++) = tmp.xridx (j);
}
a.xcidx (i+1) = ii;
}
}
#if !defined (CXX_NEW_FRIEND_TEMPLATE_DECL)
static void
dmsolve_insert (MSparse &a, const SparseMatrix &b,
const octave_idx_type *Q, octave_idx_type r, octave_idx_type c);
static void
dmsolve_insert (MSparse &a, const MSparse &b,
const octave_idx_type *Q, octave_idx_type r, octave_idx_type c);
#endif
template
static void
dmsolve_permute (MArray