2011-12-12 15:29:51 +00:00
|
|
|
#ifndef HORUS_FGBPSOLVER_H
|
|
|
|
#define HORUS_FGBPSOLVER_H
|
|
|
|
|
|
|
|
#include <set>
|
|
|
|
#include <vector>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include "Solver.h"
|
|
|
|
#include "Factor.h"
|
|
|
|
#include "FactorGraph.h"
|
2012-03-22 11:33:24 +00:00
|
|
|
#include "Util.h"
|
2011-12-12 15:29:51 +00:00
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
|
|
|
|
class SpLink
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SpLink (FgFacNode* fn, FgVarNode* vn)
|
|
|
|
{
|
|
|
|
fac_ = fn;
|
|
|
|
var_ = vn;
|
2012-03-31 23:27:37 +01:00
|
|
|
v1_.resize (vn->nrStates(), LogAware::tl (1.0 / vn->nrStates()));
|
|
|
|
v2_.resize (vn->nrStates(), LogAware::tl (1.0 / vn->nrStates()));
|
2011-12-12 15:29:51 +00:00
|
|
|
currMsg_ = &v1_;
|
|
|
|
nextMsg_ = &v2_;
|
|
|
|
msgSended_ = false;
|
|
|
|
residual_ = 0.0;
|
|
|
|
}
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
virtual ~SpLink (void) { };
|
2011-12-12 15:29:51 +00:00
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
FgFacNode* getFactor (void) const { return fac_; }
|
|
|
|
|
|
|
|
FgVarNode* getVariable (void) const { return var_; }
|
|
|
|
|
|
|
|
const Params& getMessage (void) const { return *currMsg_; }
|
|
|
|
|
|
|
|
Params& getNextMessage (void) { return *nextMsg_; }
|
|
|
|
|
|
|
|
bool messageWasSended (void) const { return msgSended_; }
|
|
|
|
|
|
|
|
double getResidual (void) const { return residual_; }
|
|
|
|
|
|
|
|
void clearResidual (void) { residual_ = 0.0; }
|
|
|
|
|
|
|
|
void updateResidual (void)
|
2011-12-12 15:29:51 +00:00
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
residual_ = LogAware::getMaxNorm (v1_,v2_);
|
2011-12-12 15:29:51 +00:00
|
|
|
}
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
virtual void updateMessage (void)
|
2011-12-12 15:29:51 +00:00
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
swap (currMsg_, nextMsg_);
|
|
|
|
msgSended_ = true;
|
2011-12-12 15:29:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string toString (void) const
|
|
|
|
{
|
|
|
|
stringstream ss;
|
|
|
|
ss << fac_->getLabel();
|
|
|
|
ss << " -- " ;
|
|
|
|
ss << var_->label();
|
|
|
|
return ss.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2012-03-31 23:27:37 +01:00
|
|
|
FgFacNode* fac_;
|
|
|
|
FgVarNode* var_;
|
|
|
|
Params v1_;
|
|
|
|
Params v2_;
|
|
|
|
Params* currMsg_;
|
|
|
|
Params* nextMsg_;
|
|
|
|
bool msgSended_;
|
|
|
|
double residual_;
|
2011-12-12 15:29:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef vector<SpLink*> SpLinkSet;
|
|
|
|
|
|
|
|
|
|
|
|
class SPNodeInfo
|
|
|
|
{
|
|
|
|
public:
|
2012-03-31 23:27:37 +01:00
|
|
|
void addSpLink (SpLink* link) { links_.push_back (link); }
|
|
|
|
const SpLinkSet& getLinks (void) { return links_; }
|
2011-12-12 15:29:51 +00:00
|
|
|
private:
|
2012-03-31 23:27:37 +01:00
|
|
|
SpLinkSet links_;
|
2011-12-12 15:29:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class FgBpSolver : public Solver
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FgBpSolver (const FactorGraph&);
|
2012-03-31 23:27:37 +01:00
|
|
|
|
2011-12-12 15:29:51 +00:00
|
|
|
virtual ~FgBpSolver (void);
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
void runSolver (void);
|
|
|
|
|
|
|
|
virtual Params getPosterioriOf (VarId);
|
|
|
|
|
|
|
|
virtual Params getJointDistributionOf (const VarIds&);
|
2011-12-12 15:29:51 +00:00
|
|
|
|
|
|
|
protected:
|
2012-03-31 23:27:37 +01:00
|
|
|
virtual void initializeSolver (void);
|
|
|
|
|
|
|
|
virtual void createLinks (void);
|
|
|
|
|
|
|
|
virtual void maxResidualSchedule (void);
|
|
|
|
|
|
|
|
virtual void calculateFactor2VariableMsg (SpLink*) const;
|
|
|
|
|
|
|
|
virtual Params getVar2FactorMsg (const SpLink*) const;
|
|
|
|
|
|
|
|
virtual Params getJointByConditioning (const VarIds&) const;
|
|
|
|
|
|
|
|
virtual void printLinkInformation (void) const;
|
|
|
|
|
|
|
|
SPNodeInfo* ninf (const FgVarNode* var) const
|
|
|
|
{
|
|
|
|
return varsI_[var->getIndex()];
|
|
|
|
}
|
|
|
|
|
|
|
|
SPNodeInfo* ninf (const FgFacNode* fac) const
|
|
|
|
{
|
|
|
|
return facsI_[fac->getIndex()];
|
|
|
|
}
|
2011-12-12 15:29:51 +00:00
|
|
|
|
|
|
|
void calculateAndUpdateMessage (SpLink* link, bool calcResidual = true)
|
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
if (Constants::DEBUG >= 3) {
|
2011-12-12 15:29:51 +00:00
|
|
|
cout << "calculating & updating " << link->toString() << endl;
|
|
|
|
}
|
|
|
|
calculateFactor2VariableMsg (link);
|
|
|
|
if (calcResidual) {
|
|
|
|
link->updateResidual();
|
|
|
|
}
|
|
|
|
link->updateMessage();
|
|
|
|
}
|
|
|
|
|
|
|
|
void calculateMessage (SpLink* link, bool calcResidual = true)
|
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
if (Constants::DEBUG >= 3) {
|
2011-12-12 15:29:51 +00:00
|
|
|
cout << "calculating " << link->toString() << endl;
|
|
|
|
}
|
|
|
|
calculateFactor2VariableMsg (link);
|
|
|
|
if (calcResidual) {
|
|
|
|
link->updateResidual();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void updateMessage (SpLink* link)
|
|
|
|
{
|
|
|
|
link->updateMessage();
|
2012-03-31 23:27:37 +01:00
|
|
|
if (Constants::DEBUG >= 3) {
|
2011-12-12 15:29:51 +00:00
|
|
|
cout << "updating " << link->toString() << endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
struct CompareResidual
|
2011-12-12 15:29:51 +00:00
|
|
|
{
|
|
|
|
inline bool operator() (const SpLink* link1, const SpLink* link2)
|
|
|
|
{
|
|
|
|
return link1->getResidual() > link2->getResidual();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
SpLinkSet links_;
|
|
|
|
unsigned nIters_;
|
|
|
|
vector<SPNodeInfo*> varsI_;
|
|
|
|
vector<SPNodeInfo*> facsI_;
|
|
|
|
const FactorGraph* factorGraph_;
|
|
|
|
|
|
|
|
typedef multiset<SpLink*, CompareResidual> SortedOrder;
|
|
|
|
SortedOrder sortedOrder_;
|
|
|
|
|
|
|
|
typedef unordered_map<SpLink*, SortedOrder::iterator> SpLinkMap;
|
|
|
|
SpLinkMap linkMap_;
|
|
|
|
|
|
|
|
private:
|
2012-03-31 23:27:37 +01:00
|
|
|
void runLoopySolver (void);
|
|
|
|
bool converged (void);
|
2011-12-12 15:29:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // HORUS_FGBPSOLVER_H
|
|
|
|
|