#include "CFactorGraph.h" #include "Factor.h" bool CFactorGraph::checkForIdenticalFactors = true; CFactorGraph::CFactorGraph (const FactorGraph& fg) { groundFg_ = &fg; freeColor_ = 0; const VarNodes& varNodes = fg.varNodes(); varSignatures_.reserve (varNodes.size()); for (unsigned i = 0; i < varNodes.size(); i++) { unsigned c = (varNodes[i]->neighbors().size() * 2) + 1; varSignatures_.push_back (Signature (c)); } const FactorNodes& facNodes = fg.factorNodes(); factorSignatures_.reserve (facNodes.size()); for (unsigned i = 0; i < facNodes.size(); i++) { unsigned c = facNodes[i]->neighbors().size() + 1; factorSignatures_.push_back (Signature (c)); } varColors_.resize (varNodes.size()); factorColors_.resize (facNodes.size()); setInitialColors(); createGroups(); } CFactorGraph::~CFactorGraph (void) { for (unsigned i = 0; i < varClusters_.size(); i++) { delete varClusters_[i]; } for (unsigned i = 0; i < facClusters_.size(); i++) { delete facClusters_[i]; } } void CFactorGraph::setInitialColors (void) { // create the initial variable colors VarColorMap colorMap; const VarNodes& varNodes = groundFg_->varNodes(); for (unsigned i = 0; i < varNodes.size(); i++) { unsigned dsize = varNodes[i]->range(); VarColorMap::iterator it = colorMap.find (dsize); if (it == colorMap.end()) { it = colorMap.insert (make_pair ( dsize, vector (dsize+1,-1))).first; } unsigned idx; if (varNodes[i]->hasEvidence()) { idx = varNodes[i]->getEvidence(); } else { idx = dsize; } vector& stateColors = it->second; if (stateColors[idx] == -1) { stateColors[idx] = getFreeColor(); } setColor (varNodes[i], stateColors[idx]); } const FactorNodes& facNodes = groundFg_->factorNodes(); if (checkForIdenticalFactors) { unsigned groupCount = 1; for (unsigned i = 0; i < facNodes.size(); i++) { Factor* f1 = facNodes[i]->factor(); if (f1->distId() != Util::maxUnsigned()) { continue; } f1->setDistId (groupCount); for (unsigned j = i + 1; j < facNodes.size(); j++) { Factor* f2 = facNodes[j]->factor(); if (f2->distId() != Util::maxUnsigned()) { continue; } if (f1->size() == f2->size() && f1->ranges() == f2->ranges() && f1->params() == f2->params()) { f2->setDistId (groupCount); } } groupCount ++; } } // create the initial factor colors DistColorMap distColors; for (unsigned i = 0; i < facNodes.size(); i++) { unsigned distId = facNodes[i]->factor()->distId(); DistColorMap::iterator it = distColors.find (distId); if (it == distColors.end()) { it = distColors.insert (make_pair (distId, getFreeColor())).first; } setColor (facNodes[i], it->second); } } void CFactorGraph::createGroups (void) { VarSignMap varGroups; FacSignMap factorGroups; unsigned nIters = 0; bool groupsHaveChanged = true; const VarNodes& varNodes = groundFg_->varNodes(); const FactorNodes& facNodes = groundFg_->factorNodes(); while (groupsHaveChanged || nIters == 1) { nIters ++; unsigned prevFactorGroupsSize = factorGroups.size(); factorGroups.clear(); // set a new color to the factors with the same signature for (unsigned i = 0; i < facNodes.size(); i++) { const Signature& signature = getSignature (facNodes[i]); FacSignMap::iterator it = factorGroups.find (signature); if (it == factorGroups.end()) { it = factorGroups.insert (make_pair (signature, FactorNodes())).first; } it->second.push_back (facNodes[i]); } for (FacSignMap::iterator it = factorGroups.begin(); it != factorGroups.end(); it++) { Color newColor = getFreeColor(); FactorNodes& groupMembers = it->second; for (unsigned i = 0; i < groupMembers.size(); i++) { setColor (groupMembers[i], newColor); } } // set a new color to the variables with the same signature unsigned prevVarGroupsSize = varGroups.size(); varGroups.clear(); for (unsigned i = 0; i < varNodes.size(); i++) { const Signature& signature = getSignature (varNodes[i]); VarSignMap::iterator it = varGroups.find (signature); if (it == varGroups.end()) { it = varGroups.insert (make_pair (signature, VarNodes())).first; } it->second.push_back (varNodes[i]); } for (VarSignMap::iterator it = varGroups.begin(); it != varGroups.end(); it++) { Color newColor = getFreeColor(); VarNodes& groupMembers = it->second; for (unsigned i = 0; i < groupMembers.size(); i++) { setColor (groupMembers[i], newColor); } } groupsHaveChanged = prevVarGroupsSize != varGroups.size() || prevFactorGroupsSize != factorGroups.size(); } //printGroups (varGroups, factorGroups); createClusters (varGroups, factorGroups); } void CFactorGraph::createClusters ( const VarSignMap& varGroups, const FacSignMap& factorGroups) { varClusters_.reserve (varGroups.size()); for (VarSignMap::const_iterator it = varGroups.begin(); it != varGroups.end(); it++) { const VarNodes& groupVars = it->second; VarCluster* vc = new VarCluster (groupVars); for (unsigned i = 0; i < groupVars.size(); i++) { vid2VarCluster_.insert (make_pair (groupVars[i]->varId(), vc)); } varClusters_.push_back (vc); } facClusters_.reserve (factorGroups.size()); for (FacSignMap::const_iterator it = factorGroups.begin(); it != factorGroups.end(); it++) { FactorNode* groupFactor = it->second[0]; const VarNodes& neighs = groupFactor->neighbors(); VarClusterSet varClusters; varClusters.reserve (neighs.size()); for (unsigned i = 0; i < neighs.size(); i++) { VarId vid = neighs[i]->varId(); varClusters.push_back (vid2VarCluster_.find (vid)->second); } facClusters_.push_back (new FacCluster (it->second, varClusters)); } } const Signature& CFactorGraph::getSignature (const VarNode* varNode) { Signature& sign = varSignatures_[varNode->getIndex()]; vector::iterator it = sign.colors.begin(); const FactorNodes& neighs = varNode->neighbors(); for (unsigned i = 0; i < neighs.size(); i++) { *it = getColor (neighs[i]); it ++; *it = neighs[i]->factor()->indexOf (varNode->varId()); it ++; } *it = getColor (varNode); return sign; } const Signature& CFactorGraph::getSignature (const FactorNode* facNode) { Signature& sign = factorSignatures_[facNode->getIndex()]; vector::iterator it = sign.colors.begin(); const VarNodes& neighs = facNode->neighbors(); for (unsigned i = 0; i < neighs.size(); i++) { *it = getColor (neighs[i]); it ++; } *it = getColor (facNode); return sign; } FactorGraph* CFactorGraph::getCompressedFactorGraph (void) { FactorGraph* fg = new FactorGraph(); for (unsigned i = 0; i < varClusters_.size(); i++) { VarNode* var = varClusters_[i]->getGroundVarNodes()[0]; VarNode* newVar = new VarNode (var); varClusters_[i]->setRepresentativeVariable (newVar); fg->addVariable (newVar); } for (unsigned i = 0; i < facClusters_.size(); i++) { const VarClusterSet& myVarClusters = facClusters_[i]->getVarClusters(); Vars myGroundVars; myGroundVars.reserve (myVarClusters.size()); for (unsigned j = 0; j < myVarClusters.size(); j++) { VarNode* v = myVarClusters[j]->getRepresentativeVariable(); myGroundVars.push_back (v); } Factor* newFactor = new Factor (myGroundVars, facClusters_[i]->getGroundFactors()[0]->params()); FactorNode* fn = new FactorNode (newFactor); facClusters_[i]->setRepresentativeFactor (fn); fg->addFactor (fn); for (unsigned j = 0; j < myGroundVars.size(); j++) { fg->addEdge (fn, static_cast (myGroundVars[j])); } } fg->setIndexes(); return fg; } unsigned CFactorGraph::getGroundEdgeCount ( const FacCluster* fc, const VarCluster* vc) const { const FactorNodes& clusterGroundFactors = fc->getGroundFactors(); VarNode* varNode = vc->getGroundVarNodes()[0]; unsigned count = 0; for (unsigned i = 0; i < clusterGroundFactors.size(); i++) { if (clusterGroundFactors[i]->factor()->indexOf (varNode->varId()) != -1) { count ++; } } // CVarNodes vars = vc->getGroundVarNodes(); // for (unsigned i = 1; i < vars.size(); i++) { // VarNode* var = vc->getGroundVarNodes()[i]; // unsigned count2 = 0; // for (unsigned i = 0; i < clusterGroundFactors.size(); i++) { // if (clusterGroundFactors[i]->getPosition (var) != -1) { // count2 ++; // } // } // if (count != count2) { cout << "oops!" << endl; abort(); } // } return count; } void CFactorGraph::printGroups ( const VarSignMap& varGroups, const FacSignMap& factorGroups) const { unsigned count = 1; cout << "variable groups:" << endl; for (VarSignMap::const_iterator it = varGroups.begin(); it != varGroups.end(); it++) { const VarNodes& groupMembers = it->second; if (groupMembers.size() > 0) { cout << count << ": " ; for (unsigned i = 0; i < groupMembers.size(); i++) { cout << groupMembers[i]->label() << " " ; } count ++; cout << endl; } } count = 1; cout << endl << "factor groups:" << endl; for (FacSignMap::const_iterator it = factorGroups.begin(); it != factorGroups.end(); it++) { const FactorNodes& groupMembers = it->second; if (groupMembers.size() > 0) { cout << ++count << ": " ; for (unsigned i = 0; i < groupMembers.size(); i++) { cout << groupMembers[i]->getLabel() << " " ; } count ++; cout << endl; } } }