#include #include #include "ConstraintTree.h" #include "Util.h" void CTNode::addChild (CTNode* n, bool updateLevels) { if (updateLevels) { updateChildLevels (n, level_ + 1); } CTChilds::iterator chIt = findChild (n); if (chIt != childs_.end()) { assert ((*chIt)->symbol() == n->symbol()); const CTChilds& childsToAdd = n->childs(); for (CTChilds::const_iterator it = childsToAdd.begin(); it != childsToAdd.end(); ++ it) { (*chIt)->addChild (*it, false); } delete n; } else { // childs_.push_back (n); childs_.insert (n); } } void CTNode::removeChild (CTNode* child) { CTChilds::iterator it; it = findChild (child); if (it == childs_.end()) { cout << "oops" << endl; } assert (it != childs_.end()); childs_.erase (it); } void CTNode::removeChilds (void) { childs_.clear(); } void CTNode::removeAndDeleteChild (CTNode* child) { removeChild (child); CTNode::deleteSubtree (child); } void CTNode::removeAndDeleteAllChilds (void) { for (CTChilds::const_iterator chIt = childs_.begin(); chIt != childs_.end(); ++ chIt) { deleteSubtree (*chIt); } childs_.clear(); } SymbolSet CTNode::childSymbols (void) const { SymbolSet symbols; for (CTChilds::const_iterator chIt = childs_.begin(); chIt != childs_.end(); ++ chIt) { symbols.insert ((*chIt)->symbol()); } return symbols; } void CTNode::updateChildLevels (CTNode* n, unsigned level) { n->setLevel (level); const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { updateChildLevels (*chIt, level + 1); } } CTNode* CTNode::copySubtree (const CTNode* n) { CTNode* newNode = new CTNode (*n); const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { newNode->addChild (copySubtree (*chIt)); } return newNode; } void CTNode::deleteSubtree (CTNode* n) { assert (n); const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { deleteSubtree (*chIt); } delete n; } ostream& operator<< (ostream &out, const CTNode& n) { // out << "(" << n.level() << ") " ; out << n.symbol(); return out; } ConstraintTree::ConstraintTree (unsigned nrLvs) { for (unsigned i = 0; i < nrLvs; i++) { logVars_.push_back (LogVar (i)); } root_ = new CTNode (0, 0); logVarSet_ = LogVarSet (logVars_); } ConstraintTree::ConstraintTree (const LogVars& logVars) { root_ = new CTNode (0, 0); logVars_ = logVars; logVarSet_ = LogVarSet (logVars); } ConstraintTree::ConstraintTree ( const LogVars& logVars, const Tuples& tuples) { root_ = new CTNode (0, 0); logVars_ = logVars; logVarSet_ = LogVarSet (logVars); for (unsigned i = 0; i < tuples.size(); i++) { addTuple (tuples[i]); } } ConstraintTree::ConstraintTree (const ConstraintTree& ct) { root_ = CTNode::copySubtree (ct.root_); logVars_ = ct.logVars_; logVarSet_ = ct.logVarSet_; } ConstraintTree::~ConstraintTree (void) { CTNode::deleteSubtree (root_); } void ConstraintTree::addTuple (const Tuple& tuple) { CTNode* prevNode = root_; for (unsigned i = 0; i < tuple.size(); i++) { CTNode tmp (tuple[i], 0); CTChilds::const_iterator it = prevNode->findChild (&tmp); if (it == prevNode->childs().end()) { CTNode* newNode = new CTNode (tuple[i], i + 1); prevNode->addChild (newNode, false); prevNode = newNode; } else { prevNode = *it; } } /* CTNode* prevNode = root_; CTChilds currNodes = root_->childs(); for (unsigned i = 0; i < tuple.size(); i++) { CTChilds::const_iterator it = currNodes.begin(); while (it != currNodes.end() && (*it)->symbol() != tuple[i]) { ++ it; } if (it == currNodes.end()) { CTNode* newNode = new CTNode (tuple[i], i + 1); prevNode->addChild (newNode); prevNode = newNode; currNodes.clear(); } else { prevNode = *it; currNodes = (*it)->childs(); } } */ } bool ConstraintTree::containsTuple (const Tuple& tuple) { CTNode* prevNode = root_; for (unsigned i = 0; i < tuple.size(); i++) { CTNode tmp (tuple[i], 0); CTChilds::const_iterator it = prevNode->findChild (&tmp); if (it == prevNode->childs().end()) { return false; } else { prevNode = *it; } } return true; } void ConstraintTree::moveToTop (const LogVars& lvs) { for (unsigned i = 0; i < lvs.size(); i++) { LogVars::iterator it = std::find (logVars_.begin(), logVars_.end(), lvs[i]); assert (it != logVars_.end()); unsigned pos = std::distance (logVars_.begin(), it); for (unsigned j = pos; j > i; j--) { swapLogVar (logVars_[j-1]); } } } void ConstraintTree::moveToBottom (const LogVars& lvs) { for (int i = lvs.size() - 1; i >= 0; i--) { LogVars::iterator it = std::find (logVars_.begin(), logVars_.end(), lvs[i]); assert (it != logVars_.end()); unsigned pos = std::distance (logVars_.begin(), it); unsigned stop = logVars_.size() - (lvs.size() - i - 1); for (unsigned j = pos; j < stop - 1; j++) { swapLogVar (logVars_[j]); } } } void ConstraintTree::join (ConstraintTree* ct, bool assertWhenNotFound) { if (logVarSet_.empty()) { delete root_; root_ = CTNode::copySubtree (ct->root()); logVars_ = ct->logVars(); logVarSet_ = ct->logVarSet(); return; } LogVarSet intersect = logVarSet_ & ct->logVarSet_; if (intersect.empty()) { CTNodes leafs = getNodesAtLevel (getLevel (logVars_.back())); const CTChilds& childs = ct->root()->childs(); for (CTNodes::const_iterator leafIt = leafs.begin(); leafIt != leafs.end(); ++ leafIt) { for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { (*leafIt)->addChild (CTNode::copySubtree (*chIt)); } } Util::addToVector (logVars_, ct->logVars_); logVarSet_ |= ct->logVarSet_; } else { moveToTop (intersect.elements()); ct->moveToTop (intersect.elements()); Tuples tuples; CTNodes appendNodes; getTuples (ct->root(), Tuples(), intersect.size(), tuples, appendNodes); CTNodes::const_iterator appendIt = appendNodes.begin(); for (unsigned i = 0; i < tuples.size(); ++ i, ++ appendIt) { bool tupleFounded = join (root_, tuples[i], 0, *appendIt); if (assertWhenNotFound) { tupleFounded; // hack to avoid gcc warning assert (tupleFounded); } } LogVars newLvs (ct->logVars().begin() + intersect.size(), ct->logVars().end()); Util::addToVector (logVars_, newLvs); logVarSet_ |= LogVarSet (newLvs); } } unsigned ConstraintTree::getLevel (LogVar X) const { LogVars::const_iterator it = std::find (logVars_.begin(), logVars_.end(), X); assert (it != logVars_.end()); unsigned level = std::distance (logVars_.begin(), it); level += 1; // root is in level 0, first logVar is in level 1 return level; } void ConstraintTree::rename (LogVar X_old, LogVar X_new) { assert (logVarSet_.contains (X_old)); assert (logVarSet_.contains (X_new) == false); logVarSet_ -= X_old; logVarSet_ |= X_new; for (unsigned i = 0; i < logVars_.size(); i++) { if (logVars_[i] == X_old) { logVars_[i] = X_new; return; } } assert (false); } void ConstraintTree::applySubstitution (const Substitution& theta) { LogVars discardedLvs = theta.getDiscardedLogVars(); for (unsigned i = 0; i < discardedLvs.size(); i++) { remove(discardedLvs[i]); } for (unsigned i = 0; i < logVars_.size(); i++) { logVars_[i] = theta.newNameFor (logVars_[i]); } logVarSet_ = LogVarSet (logVars_); } void ConstraintTree::project (const LogVarSet& X) { assert (logVarSet_.contains (X)); remove ((logVarSet_ - X)); } void ConstraintTree::remove (const LogVarSet& X) { assert (logVarSet_.contains (X)); if (X.empty()) { return; } moveToBottom (X.elements()); unsigned level = getLevel (X.front()) - 1; CTNodes nodes = getNodesAtLevel (level); for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { (*it)->removeAndDeleteAllChilds(); } logVars_.resize (logVars_.size() - X.size()); logVarSet_ -= X; } bool ConstraintTree::ConstraintTree::isSingleton (LogVar X) { const CTNodes& nodes = getNodesAtLevel (getLevel (X)); Symbol symb = nodes.front()->symbol(); for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { if ((*it)->symbol() != symb) { return false; } } return true; } LogVarSet ConstraintTree::singletons (void) { LogVarSet singletons; for (unsigned i = 0; i < logVars_.size(); i++) { if (isSingleton (logVars_[i])) { singletons.insert (logVars_[i]); } } return singletons; } TupleSet ConstraintTree::tupleSet (unsigned stopLevel) const { assert (root_->isRoot()); Tuples tuples; if (stopLevel == 0) { stopLevel = logVars_.size(); } getTuples (root_, Tuples(), stopLevel, tuples, CTNodes() = {}); return TupleSet (tuples); } TupleSet ConstraintTree::tupleSet (const LogVars& lvs) { Tuples tuples; moveToTop (lvs); unsigned stopLevel = lvs.size(); getTuples (root_, Tuples(), stopLevel, tuples, CTNodes() = {}); return TupleSet (tuples); } void ConstraintTree::exportToGraphViz ( const char* fileName, bool showLogVars) const { ofstream out (fileName); if (!out.is_open()) { cerr << "error: cannot open file to write at " ; cerr << "ConstraintTree::exportToDotFile()" << endl; abort(); } out << "digraph {" << endl; ConstraintTree copy (*this); // copy.moveToTop (copy.logVarSet_.elements()); CTNodes nodes = getNodesBelow (copy.root_); out << "\"" << copy.root_ << "\"" << " [label=\"R\"]" << endl; for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { out << "\"" << *it << "\""; out << " [label=\"" << **it << "\"]" ; out << endl; } for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { const CTChilds& childs = (*it)->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { out << "\"" << *it << "\"" ; out << " -> " ; out << "\"" << *chIt << "\"" << endl ; } } if (showLogVars) { out << "Root [label=\"\", shape=plaintext]" << endl; for (unsigned i = 0; i < copy.logVars_.size(); i++) { out << copy.logVars_[i] << " [label=" ; out << copy.logVars_[i] << ", " ; out << "shape=plaintext, fontsize=14]" << endl; } out << "Root -> " << copy.logVars_[0]; out << " [style=invis]" << endl; for (unsigned i = 0; i < copy.logVars_.size() - 1; i++) { out << copy.logVars_[i] << " -> " << copy.logVars_[i + 1]; out << " [style=invis]" << endl; } } out << "}" << endl; out.close(); } bool ConstraintTree::isCountNormalized (const LogVarSet& Ys) { assert (logVarSet_.contains (Ys)); if (Ys.empty()) { return true; } if (Ys.size() == logVars_.size()) { assert (LogVarSet (logVars_) == LogVarSet (Ys)); return true; } LogVarSet Zs = logVarSet_ - LogVarSet (Ys); moveToTop (Zs.elements()); CTNodes nodes = getNodesAtLevel (Zs.size()); unsigned count = countTuples (*nodes.begin()); for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { if (countTuples (*it) != count) { return false; } } return true; } unsigned ConstraintTree::getConditionalCount (const LogVarSet& Ys) { assert (isCountNormalized (Ys)); if (Ys.empty()) { return 1; } if (Ys.size() == logVars_.size()) { assert (LogVarSet (Ys) == LogVarSet (logVars_)); return countTuples (root_); } LogVarSet Zs = logVarSet_ - Ys; moveToTop (Zs.elements()); CTNode* n = root_; unsigned l = 0; while (l != Zs.size()) { n = *(n->childs().begin()); l ++; } return countTuples (n); } TinySet ConstraintTree::getConditionalCounts (const LogVarSet& Ys) { TinySet counts; assert (logVarSet_.contains (Ys)); if (Ys.empty()) { counts.insert (1); } else if (Ys.size() == logVars_.size()) { assert (LogVarSet (logVars_) == LogVarSet (Ys)); counts.insert (countTuples (root_)); } else { LogVarSet Zs = logVarSet_ - LogVarSet (Ys); moveToTop (Zs.elements()); CTNodes nodes = getNodesAtLevel (Zs.size()); for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { counts.insert (countTuples (*it)); } } return counts; } bool ConstraintTree::isCarteesianProduct (const LogVarSet& Xs) const { assert (logVarSet_.contains (Xs)); if (Xs.size() <= 1) { return true; } for (unsigned i = 1; i < Xs.size(); i++) { CTNodes nodesit = getNodesAtLevel (i); vector nodes (nodesit.begin(), nodesit.end()); // TODO for (unsigned j = 1; j < nodes.size(); j++) { if (nodes[j-1]->nrChilds() != nodes[ j ]->nrChilds()) { return false; } if (nodes[j-1]->childSymbols() != nodes[ j ]->childSymbols()) { return false; } } } return true; } std::pair ConstraintTree::split ( const Tuple& tuple, unsigned stopLevel) { ConstraintTree tempCt (logVars_, {tuple}); return split (&tempCt, stopLevel); } std::pair ConstraintTree::split ( const ConstraintTree* ct, unsigned stopLevel) const { assert (stopLevel > 0); assert (stopLevel <= logVars_.size()); assert (stopLevel <= ct->logVars_.size()); CTNodes commNodes; ConstraintTree* exclCt = new ConstraintTree (*this); split (exclCt->root(), ct->root(), commNodes, stopLevel); ConstraintTree* commCt = new ConstraintTree (logVars_); for (CTNodes::const_iterator it = commNodes.begin(); it != commNodes.end(); ++ it) { commCt->root()->addChild (*it); } // cout << commCt->tupleSet() << " + " ; // cout << exclCt->tupleSet() << " = " ; // cout << tupleSet() << endl << endl; // if (((commCt->tupleSet() | exclCt->tupleSet()) == tupleSet()) == false) { // exportToGraphViz ("_fail.dot", true); // commCt->exportToGraphViz ("_fail_comm.dot", true); // exclCt->exportToGraphViz ("_fail_excl.dot", true); // } // assert ((commCt->tupleSet() | exclCt->tupleSet()) == tupleSet()); // assert ((exclCt->tupleSet (stopLevel) & ct->tupleSet (stopLevel)).empty()); return {commCt, exclCt}; } ConstraintTrees ConstraintTree::countNormalize (const LogVarSet& Ys) { assert (logVarSet_.contains (Ys)); LogVarSet Zs = logVarSet_ - Ys; if (Ys.empty() || Zs.empty()) { return { new ConstraintTree (*this) }; } moveToTop (Zs.elements()); ConstraintTrees cts; unordered_map countMap; unsigned stopLevel = getLevel (Zs.back()); const CTChilds& childs = root_->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { const vector>& res = countNormalize (*chIt, stopLevel); for (unsigned j = 0; j < res.size(); j++) { unordered_map::iterator it = countMap.find (res[j].second); if (it == countMap.end()) { ConstraintTree* newCt = new ConstraintTree (logVars_); it = countMap.insert (make_pair (res[j].second, newCt)).first; cts.push_back (newCt); } it->second->root_->addChild (res[j].first); } } return cts; } ConstraintTrees ConstraintTree::jointCountNormalize ( ConstraintTree* commCt, ConstraintTree* exclCt, LogVar X, LogVar X_new1, LogVar X_new2) { unsigned N = getConditionalCount (X); // cout << "My tuples: " << tupleSet() << endl; // cout << "CommCt tuples: " << commCt->tupleSet() << endl; // cout << "ExclCt tuples: " << exclCt->tupleSet() << endl; // cout << "Counted Lv: " << X << endl; // cout << "X_new1: " << X_new1 << endl; // cout << "X_new2: " << X_new2 << endl; // cout << "Original N: " << N << endl; // cout << endl; ConstraintTrees normCts1 = commCt->countNormalize (X); vector counts1 (normCts1.size()); for (unsigned i = 0; i < normCts1.size(); i++) { counts1[i] = normCts1[i]->getConditionalCount (X); // cout << "normCts1[" << i << "] #" << counts1[i] ; // cout << " " << normCts1[i]->tupleSet() << endl; } ConstraintTrees normCts2 = exclCt->countNormalize (X); vector counts2 (normCts2.size()); for (unsigned i = 0; i < normCts2.size(); i++) { counts2[i] = normCts2[i]->getConditionalCount (X); // cout << "normCts2[" << i << "] #" << counts2[i] ; // cout << " " << normCts2[i]->tupleSet() << endl; } // cout << endl; ConstraintTree* excl1 = 0; for (unsigned i = 0; i < normCts1.size(); i++) { if (counts1[i] == N) { excl1 = normCts1[i]; normCts1.erase (normCts1.begin() + i); counts1.erase (counts1.begin() + i); // cout << "joint-count(" << N << ",0)" << endl; break; } } ConstraintTree* excl2 = 0; for (unsigned i = 0; i < normCts2.size(); i++) { if (counts2[i] == N) { excl2 = normCts2[i]; normCts2.erase (normCts2.begin() + i); counts2.erase (counts2.begin() + i); // cout << "joint-count(0," << N << ")" << endl; break; } } for (unsigned i = 0; i < normCts1.size(); i++) { unsigned j; for (j = 0; counts1[i] + counts2[j] != N; j++) ; // cout << "joint-count(" << counts1[i] ; // cout << "," << counts2[j] << ")" << endl; const CTChilds& childs = normCts2[j]->root_->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { normCts1[i]->root_->addChild (CTNode::copySubtree (*chIt)); } delete normCts2[j]; } ConstraintTrees cts = normCts1; commCt->rename (X, X_new1); exclCt->rename (X, X_new2); for (unsigned i = 0; i < cts.size(); i++) { cts[i]->remove (X); cts[i]->join (commCt); cts[i]->join (exclCt); } if (excl1 != 0) { cts.push_back (excl1); } if (excl2 != 0) { cts.push_back (excl2); } return cts; } bool ConstraintTree::identical ( const ConstraintTree* ct1, const ConstraintTree* ct2, unsigned stopLevel) { TupleSet ts1 = ct1->tupleSet (stopLevel); TupleSet ts2 = ct2->tupleSet (stopLevel); return ts1 == ts2; } bool ConstraintTree::overlap ( const ConstraintTree* ct1, const ConstraintTree* ct2, unsigned stopLevel) { const CTChilds& childs1 = ct1->root_->childs(); const CTChilds& childs2 = ct2->root_->childs(); for (CTChilds::const_iterator chIt1 = childs1.begin(); chIt1 != childs1.end(); ++ chIt1) { for (CTChilds::const_iterator chIt2 = childs2.begin(); chIt2 != childs2.end(); ++ chIt2) { if (overlap (*chIt1, *chIt2, stopLevel)) { return true; } } } return false; } LogVars ConstraintTree::expand (LogVar X) { moveToBottom ({X}); assert (isCountNormalized (X)); CTNodes nodes = getNodesAtLevel (logVars_.size() - 1); unsigned nrSymbols = getConditionalCount (X); for (CTNodes::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { Symbols symbols; const CTChilds& childs = (*it)->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { symbols.push_back ((*chIt)->symbol()); } (*it)->removeAndDeleteAllChilds(); CTNode* prev = *it; assert (symbols.size() == nrSymbols); for (unsigned j = 0; j < nrSymbols; j++) { CTNode* newNode = new CTNode (symbols[j], (*it)->level() + j); prev->addChild (newNode); prev = newNode; } } LogVars newLvs; logVars_.pop_back(); for (unsigned i = 0; i < nrSymbols; i++) { logVars_.push_back (LogVar (logVarSet_.back() + 1)); newLvs.push_back (LogVar (logVarSet_.back() + 1)); logVarSet_.insert (LogVar (logVarSet_.back() + 1)); } logVarSet_ -= X; return newLvs; } ConstraintTrees ConstraintTree::ground (LogVar X) { moveToTop ({X}); ConstraintTrees cts; const CTChilds& nodes = root_->childs(); for (CTChilds::const_iterator it = nodes.begin(); it != nodes.end(); ++ it) { CTNode* copy = CTNode::copySubtree (*it); copy->setSymbol ((*it)->symbol()); ConstraintTree* newCt = new ConstraintTree (logVars_); newCt->root()->addChild (copy); cts.push_back (newCt); } return cts; } unsigned ConstraintTree::countTuples (const CTNode* n) const { if (n->isLeaf()) { return 1; } unsigned sum = 0; const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { sum += countTuples (*chIt); } return sum; } CTNodes ConstraintTree::getNodesBelow (CTNode* fromHere) const { CTNodes nodes; queue queue; queue.push (fromHere); while (queue.empty() == false) { CTNode* node = queue.front(); nodes.push_back (node); for (CTChilds::const_iterator chIt = node->childs().begin(); chIt != node->childs().end(); ++ chIt) { queue.push (*chIt); } queue.pop(); } return nodes; } CTNodes ConstraintTree::getLeafsBelow (CTNode* fromHere) const { CTNodes nodes; queue queue; queue.push (fromHere); while (queue.empty() == false) { CTNode* node = queue.front(); if (node->isLeaf()) { nodes.push_back (node); } for (CTChilds::const_iterator chIt = node->childs().begin(); chIt != node->childs().end(); ++ chIt) { queue.push (*chIt); } queue.pop(); } return nodes; } CTNodes ConstraintTree::getNodesAtLevel (unsigned level) const { assert (level <= logVars_.size()); CTNodes nodes; queue queue; queue.push (root_); while (queue.empty() == false) { CTNode* node = queue.front(); if (node->level() == level) { nodes.push_back (node); } else { for (CTChilds::const_iterator chIt = node->childs().begin(); chIt != node->childs().end(); ++ chIt) { queue.push (*chIt); } } queue.pop(); } return nodes; } void ConstraintTree::addChildsOnBottom (CTNode* n1, const CTNode* n2) { if (n1->isLeaf()) { for (CTChilds::const_iterator chIt = n2->childs().begin(); chIt != n2->childs().end(); ++ chIt) { n1->addChild (CTNode::copySubtree (*chIt)); } } else { for (CTChilds::const_iterator chIt = n1->childs().begin(); chIt != n1->childs().end(); ++ chIt) { addChildsOnBottom (*chIt, n2); } } } void ConstraintTree::swapLogVar (LogVar X) { LogVars::iterator it; it = std::find (logVars_.begin(),logVars_.end(), X); assert (it != logVars_.end()); unsigned pos = std::distance (logVars_.begin(), it); const CTNodes& nodes = getNodesAtLevel (pos); for (CTNodes::const_iterator nodeIt = nodes.begin(); nodeIt != nodes.end(); ++ nodeIt) { CTChilds childsCopy = (*nodeIt)->childs(); (*nodeIt)->removeChilds(); for (CTChilds::const_iterator ccIt = childsCopy.begin(); ccIt != childsCopy.end(); ++ ccIt) { const CTChilds& grandsons = (*ccIt)->childs(); for (CTChilds::const_iterator gsIt = grandsons.begin(); gsIt != grandsons.end(); ++ gsIt) { CTNode* childCopy = new CTNode (**ccIt); const CTChilds& greatGrandsons = (*gsIt)->childs(); for (CTChilds::const_iterator ggsIt = greatGrandsons.begin(); ggsIt != greatGrandsons.end(); ++ ggsIt) { childCopy->addChild (*ggsIt, false); } childCopy->setLevel (childCopy->level() + 1); (*gsIt)->removeChilds(); (*gsIt)->addChild (childCopy, false); (*gsIt)->setLevel ((*gsIt)->level() - 1); (*nodeIt)->addChild ((*gsIt), false); } delete (*ccIt); } } std::swap (logVars_[pos], logVars_[pos + 1]); } bool ConstraintTree::join ( CTNode* currNode, const Tuple& tuple, unsigned currIdx, CTNode* appendNode) { bool tupleFounded = false; CTNode tmp (tuple[currIdx], 0); CTChilds::const_iterator it = currNode->findChild (&tmp); if (it != currNode->childs().end()) { if (currIdx == tuple.size() - 1) { addChildsOnBottom (currNode, appendNode); return true; } else { tupleFounded = join (*it, tuple, currIdx + 1, appendNode); } } return tupleFounded; } bool ConstraintTree::indenticalSubtrees ( const CTNode* n1, const CTNode* n2, bool compare) const { if (compare) { if (n1->symbol() != n2->symbol()) { return false; } } const CTChilds& childs1 = n1->childs(); const CTChilds& childs2 = n2->childs(); if (childs1.size() != childs2.size()) { return false; } CTChilds::const_iterator chIt1 = childs1.begin(); CTChilds::const_iterator chIt2 = childs2.begin(); while (chIt1 != childs1.end()) { if (indenticalSubtrees (*chIt1, *chIt2, true) == false) { return false; } ++ chIt1; ++ chIt2; } return true; } void ConstraintTree::getTuples ( CTNode* n, Tuples currTuples, unsigned stopLevel, Tuples& tuplesCollected, CTNodes& continuationNodes) const { if (n->isRoot() == false) { if (currTuples.size() == 0) { currTuples.push_back ({ n->symbol()}); } else { for (unsigned i = 0; i < currTuples.size(); i++) { currTuples[i].push_back (n->symbol()); } } if (n->level() == stopLevel) { for (unsigned i = 0; i < currTuples.size(); i++) { tuplesCollected.push_back (currTuples[i]); continuationNodes.push_back (n); } return; } } const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { getTuples (*chIt, currTuples, stopLevel, tuplesCollected, continuationNodes); } } unsigned ConstraintTree::size (void) const { return countTuples (root_); } unsigned ConstraintTree::nrSymbols (LogVar X) { moveToTop ({X}); return root_->childs().size(); } vector> ConstraintTree::countNormalize ( const CTNode* n, unsigned stopLevel) { if (n->level() == stopLevel) { return vector>() = { make_pair (CTNode::copySubtree (n), countTuples (n)) }; } vector> res; const CTChilds& childs = n->childs(); for (CTChilds::const_iterator chIt = childs.begin(); chIt != childs.end(); ++ chIt) { const vector>& lowerRes = countNormalize (*chIt, stopLevel); for (unsigned j = 0; j < lowerRes.size(); j++) { CTNode* newNode = new CTNode (*n); newNode->addChild (lowerRes[j].first); res.push_back (make_pair (newNode, lowerRes[j].second)); } } return res; } void ConstraintTree::split ( CTNode* n1, CTNode* n2, CTNodes& nodes, unsigned stopLevel) { CTChilds& childs1 = n1->childs(); CTChilds& childs2 = n2->childs(); CTNodes toRemove; for (CTChilds::const_iterator chIt1 = childs1.begin(); chIt1 != childs1.end(); ++ chIt1) { CTNode* intersectNode = 0; for (CTChilds::const_iterator chIt2 = childs2.begin(); chIt2 != childs2.end(); ++ chIt2) { if ((*chIt1)->symbol() == (*chIt2)->symbol()) { intersectNode = *chIt2; break; } } if (intersectNode == 0) { continue; } if ((*chIt1)->level() == stopLevel) { CTNode* newNode = CTNode::copySubtree (*chIt1); nodes.push_back (newNode); toRemove.push_back (*chIt1); } else { CTNodes lowerNodes; split ((*chIt1), intersectNode, lowerNodes, stopLevel); if (lowerNodes.empty() == false) { CTNode* newNode = new CTNode (**chIt1); for (CTNodes::const_iterator lowerIt = lowerNodes.begin(); lowerIt != lowerNodes.end(); ++ lowerIt) { newNode->addChild (*lowerIt); } nodes.push_back (newNode); } if ((*chIt1)->isLeaf()) { toRemove.push_back (*chIt1); } } } for (CTNodes::const_iterator it = toRemove.begin(); it != toRemove.end(); ++ it) { n1->removeAndDeleteChild (*it); } } bool ConstraintTree::overlap ( const CTNode* n1, const CTNode* n2, unsigned stopLevel) { if (n1->isRoot() == false) { if (n1->level() == stopLevel) { return n1->symbol() == n2->symbol(); } if (n1->symbol() != n2->symbol()) { return false; } } const CTChilds& childsI = n1->childs(); const CTChilds& childsJ = n2->childs(); for (CTChilds::const_iterator chIt1 = childsI.begin(); chIt1 != childsI.end(); ++ chIt1) { for (CTChilds::const_iterator chIt2 = childsJ.begin(); chIt2 != childsJ.end(); ++ chIt2) { if (overlap (*chIt1, *chIt2, stopLevel)) { return true; } } } return false; }