1
0
mirror of https://github.com/paboyle/Grid.git synced 2024-11-10 15:55:37 +00:00
Grid/lib/algorithms/LinearOperator.h

132 lines
5.3 KiB
C
Raw Normal View History

2015-05-13 11:25:34 +01:00
#ifndef GRID_ALGORITHM_LINEAR_OP_H
#define GRID_ALGORITHM_LINEAR_OP_H
namespace Grid {
2015-05-16 23:35:08 +01:00
/////////////////////////////////////////////////////////////////////////////////////////////
// LinearOperators Take a something and return a something.
/////////////////////////////////////////////////////////////////////////////////////////////
//
// Hopefully linearity is satisfied and the AdjOp is indeed the Hermitian conjugateugate (transpose if real):
//SBase
2015-05-16 23:35:08 +01:00
// i) F(a x + b y) = aF(x) + b F(y).
// ii) <x|Op|y> = <y|AdjOp|x>^\ast
//
// Would be fun to have a test linearity & Herm Conj function!
/////////////////////////////////////////////////////////////////////////////////////////////
template<class Field> class LinearOperatorBase {
2015-05-13 11:25:34 +01:00
public:
2015-05-16 23:35:08 +01:00
virtual void Op (const Field &in, Field &out) = 0; // Abstract base
virtual void AdjOp (const Field &in, Field &out) = 0; // Abstract base
2015-05-13 11:25:34 +01:00
};
2015-05-16 23:35:08 +01:00
/////////////////////////////////////////////////////////////////////////////////////////////
// Hermitian operators are self adjoint and only require Op to be defined, so refine the base
/////////////////////////////////////////////////////////////////////////////////////////////
template<class Field> class HermitianOperatorBase : public LinearOperatorBase<Field> {
public:
2015-05-19 21:29:07 +01:00
virtual void OpAndNorm(const Field &in, Field &out,double &n1,double &n2)=0;
2015-05-16 23:35:08 +01:00
void AdjOp(const Field &in, Field &out) {
2015-05-17 00:19:03 +01:00
Op(in,out);
};
void Op(const Field &in, Field &out) {
double n1,n2;
OpAndNorm(in,out,n1,n2);
2015-05-16 23:35:08 +01:00
};
};
/////////////////////////////////////////////////////////////////////////////////////////////
// Whereas non hermitian takes a generic sparse matrix (e.g. lattice action)
// conforming to sparse matrix interface and builds the full checkerboard non-herm operator
// Op and AdjOp distinct.
// By sharing the class for Sparse Matrix across multiple operator wrappers, we can share code
// between RB and non-RB variants. Sparse matrix is like the fermion action def, and then
// the wrappers implement the specialisation of "Op" and "AdjOp" to the cases minimising
// replication of code.
/////////////////////////////////////////////////////////////////////////////////////////////
template<class Matrix,class Field>
2015-05-16 23:35:08 +01:00
class NonHermitianOperator : public LinearOperatorBase<Field> {
Matrix &_Mat;
2015-05-16 23:35:08 +01:00
public:
NonHermitianOperator(Matrix &Mat): _Mat(Mat){};
2015-05-16 23:35:08 +01:00
void Op (const Field &in, Field &out){
_Mat.M(in,out);
}
void AdjOp (const Field &in, Field &out){
_Mat.Mdag(in,out);
}
};
////////////////////////////////////////////////////////////////////////////////////
// Redblack Non hermitian wrapper
////////////////////////////////////////////////////////////////////////////////////
template<class Matrix,class Field>
class NonHermitianCheckerBoardedOperator : public LinearOperatorBase<Field> {
Matrix &_Mat;
2015-05-16 23:35:08 +01:00
public:
NonHermitianCheckerBoardedOperator(Matrix &Mat): _Mat(Mat){};
2015-05-16 23:35:08 +01:00
void Op (const Field &in, Field &out){
2015-05-17 00:19:03 +01:00
_Mat.Mpc(in,out);
2015-05-16 23:35:08 +01:00
}
void AdjOp (const Field &in, Field &out){ //
2015-05-17 00:19:03 +01:00
_Mat.MpcDag(in,out);
2015-05-16 23:35:08 +01:00
}
};
////////////////////////////////////////////////////////////////////////////////////
// Hermitian wrapper
////////////////////////////////////////////////////////////////////////////////////
template<class Matrix,class Field>
2015-05-16 23:35:08 +01:00
class HermitianOperator : public HermitianOperatorBase<Field> {
Matrix &_Mat;
2015-05-16 23:35:08 +01:00
public:
HermitianOperator(Matrix &Mat): _Mat(Mat) {};
void OpAndNorm(const Field &in, Field &out,double &n1,double &n2){
return _Mat.MdagM(in,out,n1,n2);
2015-05-16 23:35:08 +01:00
}
};
////////////////////////////////////////////////////////////////////////////////////
// Hermitian CheckerBoarded wrapper
2015-05-16 23:35:08 +01:00
////////////////////////////////////////////////////////////////////////////////////
template<class Matrix,class Field>
class HermitianCheckerBoardedOperator : public HermitianOperatorBase<Field> {
Matrix &_Mat;
2015-05-16 23:35:08 +01:00
public:
HermitianCheckerBoardedOperator(Matrix &Mat): _Mat(Mat) {};
void OpAndNorm(const Field &in, Field &out,RealD &n1,RealD &n2){
_Mat.MpcDagMpc(in,out,n1,n2);
2015-05-16 23:35:08 +01:00
}
};
/////////////////////////////////////////////////////////////
// Base classes for functions of operators
/////////////////////////////////////////////////////////////
template<class Field> class OperatorFunction {
public:
virtual void operator() (LinearOperatorBase<Field> &Linop, const Field &in, Field &out) = 0;
2015-05-16 23:35:08 +01:00
};
2015-05-25 13:42:36 +01:00
template<class Field> class HermitianOperatorFunction {
public:
virtual void operator() (HermitianOperatorBase<Field> &Linop, const Field &in, Field &out) = 0;
};
2015-05-16 23:35:08 +01:00
// FIXME : To think about
// Chroma functionality list defining LinearOperator
/*
virtual void operator() (T& chi, const T& psi, enum PlusMinus isign) const = 0;
virtual void operator() (T& chi, const T& psi, enum PlusMinus isign, Real epsilon) const
virtual const Subset& subset() const = 0;
virtual unsigned long nFlops() const { return 0; }
virtual void deriv(P& ds_u, const T& chi, const T& psi, enum PlusMinus isign) const
class UnprecLinearOperator : public DiffLinearOperator<T,P,Q>
const Subset& subset() const {return all;}
};
*/
2015-05-13 11:25:34 +01:00
}
#endif