From ca0393dabc83020a617fe6b24305d7d7364b9402 Mon Sep 17 00:00:00 2001
From: Jesper Andersson <jesper@fmrib.ox.ac.uk>
Date: Mon, 7 Apr 2008 17:30:21 +0000
Subject: [PATCH] BFSparseMatrix now templated to allow double or float
 representation

---
 bfmatrix.cpp | 382 ++++++++++++++-------------------------------------
 bfmatrix.h   | 250 ++++++++++++++++++++++++++++++---
 2 files changed, 334 insertions(+), 298 deletions(-)

diff --git a/bfmatrix.cpp b/bfmatrix.cpp
index 9f59d9e..9305b73 100644
--- a/bfmatrix.cpp
+++ b/bfmatrix.cpp
@@ -17,216 +17,6 @@
 
 namespace MISCMATHS {
 
-//
-// Member functions for SparseBFMatrix
-//
-
-//
-// Transpose
-//
-
-boost::shared_ptr<BFMatrix> SparseBFMatrix::Transpose()
-const
-{
-  boost::shared_ptr<SparseBFMatrix>   tm(new SparseBFMatrix(mp->t()));
-  return(tm);
-}
-
-//
-// Concatenation of two matrices returning a third
-//
-void SparseBFMatrix::HorConcat(const BFMatrix& B, BFMatrix& AB) const
-{
-  if (B.Nrows() && Nrows() != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat: Matrices must have same # of rows");}
-
-  try {
-    SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);      // Means that output is sparse
-    lAB = *this;
-    lAB.HorConcat2MyRight(B);
-  }
-  catch (std::bad_cast) {
-    try {
-      FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);        // Means that output is full
-      lAB = FullBFMatrix(this->AsMatrix());
-      lAB.HorConcat2MyRight(B);
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::HorConcat: dynamic cast error"); 
-    }
-  }
-}
-
-void SparseBFMatrix::HorConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
-{
-  if (B.Nrows() && int(Nrows()) != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat: Matrices must have same # of rows");}
-
-  try {
-    SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);   // Means that output is sparse
-    lAB = *this;
-    lAB.HorConcat2MyRight(B);
-  }
-  catch (std::bad_cast) {
-    try {
-      FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);     // Means that output is full
-      lAB = FullBFMatrix(this->AsMatrix());
-      lAB.HorConcat2MyRight(B);
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::HorConcat: dynamic cast error"); 
-    }
-  }
-}
-
-void SparseBFMatrix::VertConcat(const BFMatrix& B, BFMatrix& AB) const
-{
-  if (B.Ncols() && Ncols() != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcat: Matrices must have same # of columns");}
-
-  try {
-    SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);      // Means that output is sparse
-    lAB = *this;
-    lAB.VertConcatBelowMe(B);
-  }
-  catch (std::bad_cast) {
-    try {
-      FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);        // Means that output is full
-      lAB = FullBFMatrix(this->AsMatrix());
-      lAB.VertConcatBelowMe(B);
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::VertConcat: dynamic cast error"); 
-    }
-  }
-}
-
-void SparseBFMatrix::VertConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
-{
-  if (B.Ncols() && int(Ncols()) != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcat: Matrices must have same # of columns");}
-
-  try {
-    SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);      // Means that output is sparse
-    lAB = *this;
-    lAB.VertConcatBelowMe(B);
-  }
-  catch (std::bad_cast) {
-    try {
-      FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);        // Means that output is full
-      lAB = FullBFMatrix(this->AsMatrix());
-      lAB.VertConcatBelowMe(B);
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::VertConcat: dynamic cast error"); 
-    }
-  }
-}
-
-//
-// Concatenate another matrix to *this
-//
-void SparseBFMatrix::HorConcat2MyRight(const BFMatrix& B)
-{
-  if (!B.Nrows()) return;
-
-  if (Nrows() != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: Matrices must have same # of rows");}
-
-  try {
-    const SparseBFMatrix& lB = dynamic_cast<const SparseBFMatrix&>(B);   // Means that we want to concatenate a sparse matrix
-    *mp |= *(lB.mp);
-  }
-  catch (std::bad_cast) {
-    try {
-      const FullBFMatrix& lB = dynamic_cast<const FullBFMatrix&>(B);     // Means that we want to concatenate a full
-      this->HorConcat2MyRight(lB.AsMatrix());
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: dynamic cast error");
-    } 
-  }
-}
-
-void SparseBFMatrix::HorConcat2MyRight(const NEWMAT::Matrix& B)
-{
-  if (!B.Nrows()) return;
-
-  if (int(Nrows()) != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: Matrices must have same # of rows");}
-  *mp |= B;
-}
-
-void SparseBFMatrix::VertConcatBelowMe(const BFMatrix& B)
-{
-  if (!B.Ncols()) return;
-
-  if (Ncols() != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcatBelowMe: Matrices must have same # of columns");}
-
-  try {
-    const SparseBFMatrix& lB = dynamic_cast<const SparseBFMatrix&>(B);   // Means that we want to concatenate a sparse matrix
-    *mp &= *(lB.mp);
-  }
-  catch (std::bad_cast) {
-    try {
-      const FullBFMatrix& lB = dynamic_cast<const FullBFMatrix&>(B);     // Means that we want to concatenate a full
-      this->VertConcatBelowMe(lB.AsMatrix());
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: dynamic cast error");
-    } 
-  }
-}
-
-void SparseBFMatrix::VertConcatBelowMe(const NEWMAT::Matrix& B)
-{
-  if (!B.Ncols()) return;
-
-  if (int(Ncols()) != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcatBelowMe: Matrices must have same # of columns");}
-  *mp &= B;
-}
-
-// Multiply by vector
-NEWMAT::ReturnMatrix SparseBFMatrix::MulByVec(const NEWMAT::ColumnVector& invec) const
-{
-  if (invec.Nrows() != int(Ncols())) {throw BFMatrixException("Matrix-vector size mismatch");}
-  NEWMAT::ColumnVector   outvec = *mp * invec;
-  outvec.Release();
-  return(outvec);
-}
-
-// Add another matrix to this one
-void SparseBFMatrix::AddToMe(const BFMatrix& M, double s)
-{
-  if (Ncols() != M.Ncols() || Nrows() != M.Nrows()) {
-    throw BFMatrixException("SparseBFMatrix::AddToMe: Matrix size mismatch");
-  }
-
-  try {
-    const SparseBFMatrix& lM = dynamic_cast<const SparseBFMatrix&>(M);  // Add sparse matrix to this sparse matrix
-    if (s == 1.0) *mp += *(lM.mp);
-    else *mp += s * *(lM.mp);
-  }
-  catch (std::bad_cast) {
-    try {
-      const FullBFMatrix& lM = dynamic_cast<const FullBFMatrix&>(M);      // Add full matrix to this sparse matrix
-      if (s == 1.0) *mp += SpMat<double>(lM.ReadAsMatrix());
-      else *mp += s * SpMat<double>(lM.ReadAsMatrix());
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("SparseBFMatrix::AddToMe: dynamic cast error"); 
-    }
-  }
-}
-
-// Given A*x=b, solve for x
-NEWMAT::ReturnMatrix SparseBFMatrix::SolveForx(const NEWMAT::ColumnVector& b,
-					       MISCMATHS::MatrixType       type,
-					       double                      tol,
-                                               int                         miter) const
-{
-  if (b.Nrows() != int(Nrows())) {
-    throw BFMatrixException("SparseBFMatrix::SolveForx: Matrix-vector size mismatch");
-  }
-  NEWMAT::ColumnVector  x = mp->SolveForx(b,type,tol,miter);
-  x.Release();
-  return(x);
-}
-
 //
 // Member functions for FullBFMatrix
 //
@@ -252,19 +42,24 @@ void FullBFMatrix::HorConcat(const BFMatrix& B, BFMatrix& AB) const
 {
   if (B.Nrows() && Nrows() != B.Nrows()) {throw BFMatrixException("FullBFMatrix::HorConcat: Matrices must have same # of rows");}
 
-  try {
-    FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);  // This means output is a full matrix
-    lAB = *this;
-    lAB.HorConcat2MyRight(B);
+  FullBFMatrix *pAB = dynamic_cast<FullBFMatrix *>(&AB);  
+  if (pAB) { // This means output is a full matrix
+    *pAB = *this;
+    pAB->HorConcat2MyRight(B);
   }
-  catch (std::bad_cast) {
-    try {
-      SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);
-      lAB = SparseBFMatrix(this->AsMatrix());
-      lAB.HorConcat2MyRight(B);
+  else {
+    SparseBFMatrix<double> *psdAB = dynamic_cast<SparseBFMatrix<double> *>(&AB);
+    if (psdAB) { 
+      *psdAB = SparseBFMatrix<double>(this->AsMatrix());
+      psdAB->HorConcat2MyRight(B);
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::HorConcat: dynamic cast error"); 
+    else {
+      SparseBFMatrix<float> *psfAB = dynamic_cast<SparseBFMatrix<float> *>(&AB);
+      if (psfAB) { 
+        *psfAB = SparseBFMatrix<float>(this->AsMatrix());
+        psfAB->HorConcat2MyRight(B);
+      }
+      else throw BFMatrixException("FullBFMatrix::HorConcat: dynamic cast error"); 
     }
   }
 }
@@ -273,19 +68,24 @@ void FullBFMatrix::HorConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
 {
   if (B.Nrows() && int(Nrows()) != B.Nrows()) {throw BFMatrixException("FullBFMatrix::HorConcat: Matrices must have same # of rows");}
 
-  try {
-    FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);  // This means output is a full matrix
-    lAB = *this;
-    lAB.HorConcat2MyRight(B);
+  FullBFMatrix *pAB = dynamic_cast<FullBFMatrix *>(&AB);  
+  if (pAB) { // This means output is a full matrix
+    *pAB = *this;
+    pAB->HorConcat2MyRight(B);
   }
-  catch (std::bad_cast) {
-    try {
-      SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);
-      lAB = SparseBFMatrix(this->AsMatrix());
-      lAB.HorConcat2MyRight(B);
+  else {
+    SparseBFMatrix<double> *psdAB = dynamic_cast<SparseBFMatrix<double> *>(&AB);
+    if (psdAB) {
+      *psdAB = SparseBFMatrix<double>(this->AsMatrix());
+      psdAB->HorConcat2MyRight(B);
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::HorConcat: dynamic cast error"); 
+    else {
+      SparseBFMatrix<float> *psfAB = dynamic_cast<SparseBFMatrix<float> *>(&AB);
+      if (psfAB) {
+        *psfAB = SparseBFMatrix<float>(this->AsMatrix());
+        psfAB->HorConcat2MyRight(B);
+      }
+      else throw BFMatrixException("FullBFMatrix::HorConcat: dynamic cast error"); 
     }
   }
 }
@@ -294,19 +94,24 @@ void FullBFMatrix::VertConcat(const BFMatrix& B, BFMatrix& AB) const
 {
   if (B.Ncols() && Ncols() != B.Ncols()) {throw BFMatrixException("FullBFMatrix::VertConcat: Matrices must have same # of columns");}
 
-  try {
-    FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);  // This means output is a full matrix
-    lAB = *this;
-    lAB.VertConcatBelowMe(B);
+  FullBFMatrix *pAB = dynamic_cast<FullBFMatrix *>(&AB);  
+  if (pAB) { // This means output is a full matrix
+    *pAB = *this;
+    pAB->VertConcatBelowMe(B);
   }
-  catch (std::bad_cast) {
-    try {
-      SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);
-      lAB = SparseBFMatrix(this->AsMatrix());
-      lAB.VertConcatBelowMe(B);
+  else {
+    SparseBFMatrix<double> *psdAB = dynamic_cast<SparseBFMatrix<double> *>(&AB);
+    if (psdAB) {
+      *psdAB = SparseBFMatrix<double>(this->AsMatrix());
+      psdAB->VertConcatBelowMe(B);
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::VertConcat: dynamic cast error"); 
+    else {
+      SparseBFMatrix<float> *psfAB = dynamic_cast<SparseBFMatrix<float> *>(&AB);
+      if (psfAB) {
+        *psfAB = SparseBFMatrix<float>(this->AsMatrix());
+        psfAB->VertConcatBelowMe(B);
+      }
+      else throw BFMatrixException("FullBFMatrix::VertConcat: dynamic cast error"); 
     }
   }
 }
@@ -315,19 +120,24 @@ void FullBFMatrix::VertConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
 {
   if (B.Ncols() && int(Ncols()) != B.Ncols()) {throw BFMatrixException("FullBFMatrix::VertConcat: Matrices must have same # of columns");}
 
-  try {
-    FullBFMatrix& lAB = dynamic_cast<FullBFMatrix&>(AB);  // This means output is a full matrix
-    lAB = *this;
-    lAB.VertConcatBelowMe(B);
+  FullBFMatrix *pAB = dynamic_cast<FullBFMatrix *>(&AB);  
+  if (pAB) { // This means output is a full matrix
+    *pAB = *this;
+    pAB->VertConcatBelowMe(B);
   }
-  catch (std::bad_cast) {
-    try {
-      SparseBFMatrix& lAB = dynamic_cast<SparseBFMatrix&>(AB);
-      lAB = SparseBFMatrix(this->AsMatrix());
-      lAB.VertConcatBelowMe(B);
+  else {
+    SparseBFMatrix<double> *psdAB = dynamic_cast<SparseBFMatrix<double> *>(&AB);
+    if (psdAB) {
+      *psdAB = SparseBFMatrix<double>(this->AsMatrix());
+      psdAB->VertConcatBelowMe(B);
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::VertConcat: dynamic cast error"); 
+    else {
+      SparseBFMatrix<float> *psfAB = dynamic_cast<SparseBFMatrix<float> *>(&AB);
+      if (psfAB) {
+        *psfAB = SparseBFMatrix<float>(this->AsMatrix());
+        psfAB->VertConcatBelowMe(B);
+      }
+      else throw BFMatrixException("FullBFMatrix::VertConcat: dynamic cast error"); 
     }
   }
 }
@@ -341,17 +151,21 @@ void FullBFMatrix::HorConcat2MyRight(const BFMatrix& B)
 
   if (Nrows() != B.Nrows()) {throw BFMatrixException("FullBFMatrix::HorConcat2MyRight: Matrices must have same # of rows");}
 
-  try {
-    const FullBFMatrix& lB = dynamic_cast<const FullBFMatrix&>(B);
-    *mp |= *(lB.mp);
+  const FullBFMatrix *pB = dynamic_cast<const FullBFMatrix *>(&B);
+  if (pB) { // If B was full
+    *mp |= *(pB->mp);
   }
-  catch (std::bad_cast) {
-    try {
-      const SparseBFMatrix& lB = dynamic_cast<const SparseBFMatrix&>(B);
-      this->HorConcat2MyRight(lB.AsMatrix());
+  else {
+    const SparseBFMatrix<double> *psdB = dynamic_cast<const SparseBFMatrix<double> *>(&B);
+    if (psdB) {
+      this->HorConcat2MyRight(psdB->AsMatrix());
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::HorConcat2MyRight: dynamic cast error"); 
+    else { 
+      const SparseBFMatrix<float> *psfB = dynamic_cast<const SparseBFMatrix<float> *>(&B);
+      if (psfB) {
+        this->HorConcat2MyRight(psfB->AsMatrix());
+      }
+      else throw BFMatrixException("FullBFMatrix::HorConcat2MyRight: dynamic cast error"); 
     }
   }
 }
@@ -370,17 +184,21 @@ void FullBFMatrix::VertConcatBelowMe(const BFMatrix& B)
 
   if (Ncols() != B.Ncols()) {throw BFMatrixException("FullBFMatrix::VertConcatBelowMe: Matrices must have same # of columns");}
 
-  try {
-    const FullBFMatrix& lB = dynamic_cast<const FullBFMatrix&>(B);
-    *mp &= *(lB.mp);
+  const FullBFMatrix *pB = dynamic_cast<const FullBFMatrix *>(&B);
+  if (pB) { // Means B is full
+    *mp &= *(pB->mp);
   }
-  catch (std::bad_cast) {
-    try {
-      const SparseBFMatrix& lB = dynamic_cast<const SparseBFMatrix&>(B);
-      this->VertConcatBelowMe(lB.AsMatrix());
+  else {
+    const SparseBFMatrix<double> *psdB = dynamic_cast<const SparseBFMatrix<double> *>(&B);
+    if (psdB) {
+      this->VertConcatBelowMe(psdB->AsMatrix());
     }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::HorConcatBelowMe: dynamic cast error"); 
+    else {
+      const SparseBFMatrix<float> *psfB = dynamic_cast<const SparseBFMatrix<float> *>(&B);
+      if (psfB) {
+        this->VertConcatBelowMe(psfB->AsMatrix());
+      }
+      else throw BFMatrixException("FullBFMatrix::HorConcatBelowMe: dynamic cast error"); 
     }
   }
 }
@@ -417,17 +235,17 @@ void FullBFMatrix::AddToMe(const BFMatrix& m, double s)
     throw BFMatrixException("FullBFMatrix::AddToMe: Matrix size mismatch");
   }
 
-  try {
-    const FullBFMatrix& lm = dynamic_cast<const FullBFMatrix&>(m);
-    *mp += s*(*lm.mp);
+  const FullBFMatrix *pm = dynamic_cast<const FullBFMatrix *>(&m);
+  if (pm) { // If m is full matrix
+    *mp += s*(*(pm->mp));
   }
-  catch (std::bad_cast) {
-    try {
-      const SparseBFMatrix& lm = dynamic_cast<const SparseBFMatrix&>(m);
-      *mp += s*lm.AsMatrix();
-    }
-    catch (std::bad_cast) {
-      throw BFMatrixException("FullBFMatrix::AddToMe: dynamic cast error");
+  else {
+    const SparseBFMatrix<double> *psdm = dynamic_cast<const SparseBFMatrix<double> *>(&m);
+    if (psdm) *mp += s*psdm->AsMatrix();
+    else {
+      const SparseBFMatrix<float> *psfm = dynamic_cast<const SparseBFMatrix<float> *>(&m);
+      if (psfm) *mp += s*psfm->AsMatrix();
+      else throw BFMatrixException("FullBFMatrix::AddToMe: dynamic cast error");
     }
   }
 }
diff --git a/bfmatrix.h b/bfmatrix.h
index a6032a7..604f92c 100644
--- a/bfmatrix.h
+++ b/bfmatrix.h
@@ -49,6 +49,8 @@ public:
   ~BFMatrixException() throw() {}
 };
 
+enum BFMatrixPrecisionType {BFMatrixDoublePrecision, BFMatrixFloatPrecision};
+
 class BFMatrix
 {
 protected:
@@ -70,7 +72,8 @@ public:
   virtual void Print(const std::string fname=std::string("")) const = 0;
 
   // Setting, deleting or resizing the whole sparse matrix.
-  virtual void SetMatrix(const MISCMATHS::SpMat<double>& M) = 0;
+  // virtual void SetMatrix(const MISCMATHS::SpMat<double>& M) = 0;
+  // virtual void SetMatrix(const MISCMATHS::SpMat<float>& M) = 0;
   virtual void SetMatrix(const NEWMAT::Matrix& M) = 0;
   virtual void Clear() = 0;
   virtual void Resize(unsigned int m, unsigned int n) = 0;
@@ -114,26 +117,27 @@ public:
                                          int                         miter=200) const = 0;  
 };
 
+template<class T>
 class SparseBFMatrix : public BFMatrix
 {
 private:
-  boost::shared_ptr<MISCMATHS::SpMat<double> >    mp;
-  
+  boost::shared_ptr<MISCMATHS::SpMat<T> >    mp;
+
 public:
   // Constructors, destructor and assignment
   SparseBFMatrix() 
-  : mp(boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>())) {}
+  : mp(boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>())) {}
   SparseBFMatrix(unsigned int m, unsigned int n) 
-  : mp(boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(m,n))) {}
+  : mp(boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(m,n))) {}
   SparseBFMatrix(unsigned int m, unsigned int n, const unsigned int *irp, const unsigned int *jcp, const double *sp)
-  : mp(boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(m,n,irp,jcp,sp))) {}
-  SparseBFMatrix(const MISCMATHS::SpMat<double>& M) 
-  : mp(boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(M))) {}
+  : mp(boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(m,n,irp,jcp,sp))) {}
+  SparseBFMatrix(const MISCMATHS::SpMat<T>& M) 
+  : mp(boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(M))) {}
   SparseBFMatrix(const NEWMAT::Matrix& M) 
-  : mp(boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(M))) {}
+  : mp(boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(M))) {}
   virtual ~SparseBFMatrix() {}
-  virtual const SparseBFMatrix& operator=(const SparseBFMatrix& M) {
-    mp = boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(*(M.mp))); return(*this);
+  virtual const SparseBFMatrix& operator=(const SparseBFMatrix<T>& M) {
+    mp = boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(*(M.mp))); return(*this);
   }
 
   // Access as NEWMAT::Matrix
@@ -147,11 +151,12 @@ public:
   virtual void Print(const std::string fname=std::string("")) const {mp->Print(fname);}
 
   // Setting, deleting or resizing the whole sparse matrix.
-  virtual void SetMatrix(const MISCMATHS::SpMat<double>& M) {mp = boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(M));}
-  virtual void SetMatrix(const NEWMAT::Matrix& M) {mp = boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(M));}
-  virtual void SetMatrixPtr(boost::shared_ptr<MISCMATHS::SpMat<double> >& mptr) {mp = mptr;}
-  virtual void Clear() {mp = boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>());}
-  virtual void Resize(unsigned int m, unsigned int n) {mp = boost::shared_ptr<MISCMATHS::SpMat<double> >(new MISCMATHS::SpMat<double>(m,n));}
+  virtual void SetMatrix(const MISCMATHS::SpMat<T>& M) {mp = boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(M));}
+  // virtual void SetMatrix(const MISCMATHS::SpMat<float>& M) {mp = boost::shared_ptr<MISCMATHS::SpMat<float> >(new MISCMATHS::SpMat<float>(M));}
+  virtual void SetMatrix(const NEWMAT::Matrix& M) {mp = boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(M));}
+  virtual void SetMatrixPtr(boost::shared_ptr<MISCMATHS::SpMat<T> >& mptr) {mp = mptr;}
+  virtual void Clear() {mp = boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>());}
+  virtual void Resize(unsigned int m, unsigned int n) {mp = boost::shared_ptr<MISCMATHS::SpMat<T> >(new MISCMATHS::SpMat<T>(m,n));}
 
   // Accessing values
   virtual double Peek(unsigned int r, unsigned int c) const {return(mp->Peek(r,c));}
@@ -221,6 +226,7 @@ public:
 
   // Setting, deleting or resizing the whole matrix.
   virtual void SetMatrix(const MISCMATHS::SpMat<double>& M) {mp = boost::shared_ptr<NEWMAT::Matrix>(new NEWMAT::Matrix(M.AsNEWMAT()));} 
+  virtual void SetMatrix(const MISCMATHS::SpMat<float>& M) {mp = boost::shared_ptr<NEWMAT::Matrix>(new NEWMAT::Matrix(M.AsNEWMAT()));} 
   virtual void SetMatrix(const NEWMAT::Matrix& M) {mp = boost::shared_ptr<NEWMAT::Matrix>(new NEWMAT::Matrix(M));}
   virtual void SetMatrixPtr(boost::shared_ptr<NEWMAT::Matrix>& mptr) {mp = mptr;}
   virtual void Clear() {mp->ReSize(0,0);}
@@ -266,6 +272,218 @@ public:
     
 };
 
+//
+// Here comes member functions for SparseBFMatrix. Since it is templated
+// these need to go here rather than in bfmatrix.cpp.
+//
+
+//
+// Member functions for SparseBFMatrix
+//
+
+//
+// Transpose
+//
+
+template<class T>
+boost::shared_ptr<BFMatrix> SparseBFMatrix<T>::Transpose() const
+{
+  boost::shared_ptr<SparseBFMatrix<T> >   tm(new SparseBFMatrix<T>(mp->t()));
+  return(tm);
+}
+
+//
+// Concatenation of two matrices returning a third
+//
+template<class T>
+void SparseBFMatrix<T>::HorConcat(const BFMatrix& B, BFMatrix& AB) const
+{
+  if (B.Nrows() && Nrows() != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat: Matrices must have same # of rows");}
+
+  SparseBFMatrix<T> *pAB = dynamic_cast<SparseBFMatrix<T> *>(&AB);
+  if (pAB) { // Means that output is sparse of type T
+    *pAB = *this;
+    pAB->HorConcat2MyRight(B);
+  }
+  else {
+    FullBFMatrix *fpAB = dynamic_cast<FullBFMatrix *>(&AB);        
+    if (fpAB) { // Means that output is full 
+      *fpAB = FullBFMatrix(this->AsMatrix());
+      fpAB->HorConcat2MyRight(B);
+    }
+    else throw BFMatrixException("SparseBFMatrix::HorConcat: dynamic cast error"); 
+  }
+}
+
+template<class T>
+void SparseBFMatrix<T>::HorConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
+{
+  if (B.Nrows() && int(Nrows()) != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat: Matrices must have same # of rows");}
+
+  SparseBFMatrix<T> *pAB = dynamic_cast<SparseBFMatrix<T> *>(&AB);   
+  if (pAB) { // Means that output is sparse
+    *pAB = *this;
+    pAB->HorConcat2MyRight(B);
+  }
+  else {
+    FullBFMatrix *fpAB = dynamic_cast<FullBFMatrix *>(&AB);     
+    if (fpAB) {// Means that output is full
+      *fpAB = FullBFMatrix(this->AsMatrix());
+      fpAB->HorConcat2MyRight(B);
+    }
+    else throw BFMatrixException("SparseBFMatrix::HorConcat: dynamic cast error"); 
+  }
+}
+
+template<class T>
+void SparseBFMatrix<T>::VertConcat(const BFMatrix& B, BFMatrix& AB) const
+{
+  if (B.Ncols() && Ncols() != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcat: Matrices must have same # of columns");}
+
+  SparseBFMatrix<T> *pAB = dynamic_cast<SparseBFMatrix<T> *>(&AB);      
+  if (pAB) { // Means that output is sparse
+    *pAB = *this;
+    pAB->VertConcatBelowMe(B);
+  }
+  else {
+    FullBFMatrix *fpAB = dynamic_cast<FullBFMatrix *>(&AB);        
+    if (fpAB) { // Means that output is full
+      *fpAB = FullBFMatrix(this->AsMatrix());
+      fpAB->VertConcatBelowMe(B);
+    }
+    else throw BFMatrixException("SparseBFMatrix::VertConcat: dynamic cast error"); 
+  }
+}
+
+template<class T>
+void SparseBFMatrix<T>::VertConcat(const NEWMAT::Matrix& B, BFMatrix& AB) const
+{
+  if (B.Ncols() && int(Ncols()) != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcat: Matrices must have same # of columns");}
+
+  SparseBFMatrix<T> *pAB = dynamic_cast<SparseBFMatrix<T> *>(&AB);      
+  if (pAB) { // Means that output is sparse
+    *pAB = *this;
+    pAB->VertConcatBelowMe(B);
+  }
+  else {
+    FullBFMatrix *fpAB = dynamic_cast<FullBFMatrix *>(&AB);        
+    if (fpAB) { // Means that output is full
+      *fpAB = FullBFMatrix(this->AsMatrix());
+      fpAB->VertConcatBelowMe(B);
+    }
+    else throw BFMatrixException("SparseBFMatrix::VertConcat: dynamic cast error"); 
+  }
+}
+
+//
+// Concatenate another matrix to *this
+//
+template<class T>
+void SparseBFMatrix<T>::HorConcat2MyRight(const BFMatrix& B)
+{
+  if (!B.Nrows()) return;
+
+  if (Nrows() != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: Matrices must have same # of rows");}
+
+  const SparseBFMatrix<T> *pB = dynamic_cast<const SparseBFMatrix<T> *>(&B);   
+  if (pB) { // Means that we want to concatenate a sparse matrix
+    *mp |= *(pB->mp);
+  }
+  else {
+    const FullBFMatrix *fpB = dynamic_cast<const FullBFMatrix *>(&B);     
+    if (fpB) { // Means that we want to concatenate a full
+      this->HorConcat2MyRight(fpB->AsMatrix());
+    }
+    else throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: dynamic cast error");
+  }
+}
+
+template<class T>
+void SparseBFMatrix<T>::HorConcat2MyRight(const NEWMAT::Matrix& B)
+{
+  if (!B.Nrows()) return;
+
+  if (int(Nrows()) != B.Nrows()) {throw BFMatrixException("SparseBFMatrix::HorConcat2MyRight: Matrices must have same # of rows");}
+  *mp |= B;
+}
+
+template<class T>
+void SparseBFMatrix<T>::VertConcatBelowMe(const BFMatrix& B)
+{
+  if (!B.Ncols()) return;
+
+  if (Ncols() != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcatBelowMe: Matrices must have same # of columns");}
+
+  const SparseBFMatrix<T> *pB = dynamic_cast<const SparseBFMatrix<T> *>(&B);   
+  if (pB) { // Means that we want to concatenate a sparse matrix
+    *mp &= *(pB->mp);
+  }
+  else {
+    const FullBFMatrix *fpB = dynamic_cast<const FullBFMatrix *>(&B);     
+    if (fpB) { // Means that we want to concatenate a full
+      this->VertConcatBelowMe(fpB->AsMatrix());
+    }
+    else throw BFMatrixException("SparseBFMatrix::VertConcatBelowMe: dynamic cast error");
+  }
+}
+
+template<class T>
+void SparseBFMatrix<T>::VertConcatBelowMe(const NEWMAT::Matrix& B)
+{
+  if (!B.Ncols()) return;
+
+  if (int(Ncols()) != B.Ncols()) {throw BFMatrixException("SparseBFMatrix::VertConcatBelowMe: Matrices must have same # of columns");}
+  *mp &= B;
+}
+
+// Multiply by vector
+template<class T>
+NEWMAT::ReturnMatrix SparseBFMatrix<T>::MulByVec(const NEWMAT::ColumnVector& invec) const
+{
+  if (invec.Nrows() != int(Ncols())) {throw BFMatrixException("Matrix-vector size mismatch");}
+  NEWMAT::ColumnVector   outvec = *mp * invec;
+  outvec.Release();
+  return(outvec);
+}
+
+// Add another matrix to this one
+template<class T>
+void SparseBFMatrix<T>::AddToMe(const BFMatrix& M, double s)
+{
+  if (Ncols() != M.Ncols() || Nrows() != M.Nrows()) {
+    throw BFMatrixException("SparseBFMatrix::AddToMe: Matrix size mismatch");
+  }
+
+  const SparseBFMatrix<T> *pM = dynamic_cast<const SparseBFMatrix<T> *>(&M);  
+  if (pM) { // Add sparse matrix to this sparse matrix
+    if (s == 1.0) *mp += *(pM->mp);
+    else *mp += s * *(pM->mp);
+  }
+  else {
+    const FullBFMatrix *fpM = dynamic_cast<const FullBFMatrix *>(&M);      
+    if (fpM) { // Add full matrix to this sparse matrix
+      if (s == 1.0) *mp += SpMat<T>(fpM->ReadAsMatrix());
+      else *mp += s * SpMat<T>(fpM->ReadAsMatrix());
+    }
+    else throw BFMatrixException("SparseBFMatrix::AddToMe: dynamic cast error"); 
+  }
+}
+
+// Given A*x=b, solve for x
+template<class T>
+NEWMAT::ReturnMatrix SparseBFMatrix<T>::SolveForx(const NEWMAT::ColumnVector& b,
+					          MISCMATHS::MatrixType       type,
+					          double                      tol,
+                                                  int                         miter) const
+{
+  if (b.Nrows() != int(Nrows())) {
+    throw BFMatrixException("SparseBFMatrix::SolveForx: Matrix-vector size mismatch");
+  }
+  NEWMAT::ColumnVector  x = mp->SolveForx(b,type,tol,miter);
+  x.Release();
+  return(x);
+}
+
 } // End namespace MISCMATHS
 
 #endif // End #ifndef BFMatrix_h
-- 
GitLab