2012-03-22 11:33:24 +00:00
|
|
|
#include <queue>
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
#include <fstream>
|
|
|
|
|
2012-03-22 11:33:24 +00:00
|
|
|
#include "ConstraintTree.h"
|
|
|
|
#include "Util.h"
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
CTNode::addChild (CTNode* child, bool updateLevels)
|
|
|
|
{
|
|
|
|
if (updateLevels) {
|
|
|
|
updateChildLevels (child, level_ + 1);
|
|
|
|
}
|
|
|
|
bool found = false;
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs_.begin();
|
|
|
|
chIt != childs_.end(); ++ chIt) {
|
|
|
|
if (child->symbol() == (*chIt)->symbol()) {
|
|
|
|
CTNodes childChildsCopy = child->childs();
|
|
|
|
for (CTNodes::const_iterator ccIt = childChildsCopy.begin();
|
|
|
|
ccIt != childChildsCopy.end(); ++ ccIt) {
|
|
|
|
(*chIt)->addChild (*ccIt, false);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (found) {
|
|
|
|
delete child;
|
|
|
|
} else {
|
|
|
|
childs_.push_back (child);
|
2012-04-15 03:19:15 +01:00
|
|
|
// childs_.insert (child);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
CTNode::removeChild (CTNode* child)
|
|
|
|
{
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes::iterator it;
|
|
|
|
it = std::find (childs_.begin(), childs_.end(), child);
|
2012-03-22 11:33:24 +00:00
|
|
|
assert (it != childs_.end());
|
|
|
|
childs_.erase (it);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-04-11 15:36:50 +01:00
|
|
|
void
|
|
|
|
CTNode::removeChilds (void)
|
|
|
|
{
|
|
|
|
childs_.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
void
|
|
|
|
CTNode::removeAndDeleteChild (CTNode* child)
|
|
|
|
{
|
|
|
|
removeChild (child);
|
|
|
|
CTNode::deleteSubtree (child);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
CTNode::removeAndDeleteAllChilds (void)
|
|
|
|
{
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs_.begin();
|
|
|
|
chIt != childs_.end(); ++ chIt) {
|
|
|
|
deleteSubtree (*chIt);
|
2012-03-31 23:27:37 +01:00
|
|
|
}
|
|
|
|
childs_.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-22 11:33:24 +00:00
|
|
|
SymbolSet
|
|
|
|
CTNode::childSymbols (void) const
|
|
|
|
{
|
|
|
|
SymbolSet symbols;
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs_.begin();
|
|
|
|
chIt != childs_.end(); ++ chIt) {
|
|
|
|
symbols.insert ((*chIt)->symbol());
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
return symbols;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
CTNode::updateChildLevels (CTNode* n, unsigned level)
|
|
|
|
{
|
|
|
|
n->setLevel (level);
|
|
|
|
const CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
updateChildLevels (*chIt, level + 1);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
CTNode*
|
|
|
|
CTNode::copySubtree (const CTNode* n)
|
|
|
|
{
|
|
|
|
CTNode* newNode = new CTNode (*n);
|
|
|
|
const CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
newNode->addChild (copySubtree (*chIt));
|
2012-03-31 23:27:37 +01:00
|
|
|
}
|
|
|
|
return newNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
CTNode::deleteSubtree (CTNode* n)
|
|
|
|
{
|
|
|
|
assert (n);
|
|
|
|
const CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
deleteSubtree (*chIt);
|
2012-03-31 23:27:37 +01:00
|
|
|
}
|
|
|
|
delete n;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-22 11:33:24 +00:00
|
|
|
ostream& operator<< (ostream &out, const CTNode& n)
|
|
|
|
{
|
|
|
|
// out << "(" << n.level() << ") " ;
|
|
|
|
out << n.symbol();
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-31 23:27:37 +01:00
|
|
|
ConstraintTree::ConstraintTree (unsigned nrLvs)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < nrLvs; i++) {
|
|
|
|
logVars_.push_back (LogVar (i));
|
|
|
|
}
|
|
|
|
root_ = new CTNode (0, 0);
|
|
|
|
logVarSet_ = LogVarSet (logVars_);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-22 11:33:24 +00:00
|
|
|
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)
|
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
root_ = CTNode::copySubtree (ct.root_);
|
2012-03-22 11:33:24 +00:00
|
|
|
logVars_ = ct.logVars_;
|
|
|
|
logVarSet_ = ct.logVarSet_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ConstraintTree::~ConstraintTree (void)
|
|
|
|
{
|
2012-03-31 23:27:37 +01:00
|
|
|
CTNode::deleteSubtree (root_);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
ConstraintTree::addTuple (const Tuple& tuple)
|
|
|
|
{
|
|
|
|
CTNode* prevNode = root_;
|
|
|
|
CTNodes currNodes = root_->childs();
|
|
|
|
for (unsigned i = 0; i < tuple.size(); i++) {
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes::const_iterator it = currNodes.begin();
|
|
|
|
while (it != currNodes.end() && (*it)->symbol() != tuple[i]) {
|
|
|
|
++ it;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
if (it == currNodes.end()) {
|
2012-03-22 11:33:24 +00:00
|
|
|
CTNode* newNode = new CTNode (tuple[i], i + 1);
|
|
|
|
prevNode->addChild (newNode);
|
|
|
|
prevNode = newNode;
|
|
|
|
currNodes.clear();
|
|
|
|
} else {
|
2012-04-15 03:19:15 +01:00
|
|
|
prevNode = *it;
|
|
|
|
currNodes = (*it)->childs();
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
ConstraintTree::containsTuple (const Tuple& tuple)
|
|
|
|
{
|
|
|
|
queue<CTNode*> queue;
|
|
|
|
queue.push (root_);
|
|
|
|
|
|
|
|
while (queue.empty() == false) {
|
|
|
|
CTNode* n = queue.front();
|
|
|
|
if (n == root_ || n->symbol() == (tuple[n->level() - 1])) {
|
|
|
|
if (n->level() == tuple.size()) {
|
|
|
|
return true;
|
|
|
|
} else {
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = n->childs().begin();
|
|
|
|
chIt != n->childs().end(); ++ chIt) {
|
|
|
|
queue.push (*chIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
queue.pop();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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)
|
2012-03-31 23:27:37 +01:00
|
|
|
{
|
|
|
|
if (logVarSet_.empty()) {
|
|
|
|
delete root_;
|
|
|
|
root_ = CTNode::copySubtree (ct->root());
|
|
|
|
logVars_ = ct->logVars();
|
|
|
|
logVarSet_ = ct->logVarSet();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-03-22 11:33:24 +00:00
|
|
|
LogVarSet intersect = logVarSet_ & ct->logVarSet_;
|
|
|
|
if (intersect.empty()) {
|
|
|
|
CTNodes leafs = getNodesAtLevel (getLevel (logVars_.back()));
|
2012-04-15 03:19:15 +01:00
|
|
|
const CTNodes& childs = ct->root()->childs();
|
|
|
|
for (CTNodes::const_iterator leafIt = leafs.begin();
|
|
|
|
leafIt != leafs.end(); ++ leafIt) {
|
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
(*leafIt)->addChild (CTNode::copySubtree (*chIt));
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
2012-03-31 23:27:37 +01:00
|
|
|
Util::addToVector (logVars_, ct->logVars_);
|
2012-03-22 11:33:24 +00:00
|
|
|
logVarSet_ |= ct->logVarSet_;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
moveToBottom (intersect.elements());
|
|
|
|
ct->moveToTop (intersect.elements());
|
|
|
|
|
|
|
|
unsigned level = getLevel (intersect.front());
|
|
|
|
CTNodes nodes = getNodesAtLevel (level);
|
|
|
|
|
|
|
|
Tuples tuples;
|
2012-03-31 23:27:37 +01:00
|
|
|
CTNodes continNodes;
|
2012-03-22 11:33:24 +00:00
|
|
|
getTuples (ct->root(),
|
|
|
|
Tuples(),
|
|
|
|
intersect.size(),
|
|
|
|
tuples,
|
2012-03-31 23:27:37 +01:00
|
|
|
continNodes);
|
2012-03-22 11:33:24 +00:00
|
|
|
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes::const_iterator continIt;
|
|
|
|
continIt = continNodes.begin();
|
|
|
|
for (unsigned i = 0; i < tuples.size(); i++, ++ continIt) {
|
2012-03-22 11:33:24 +00:00
|
|
|
bool tupleFounded = false;
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator nodeIt = nodes.begin();
|
|
|
|
nodeIt != nodes.end(); ++ nodeIt) {
|
|
|
|
tupleFounded |= join (*nodeIt, tuples[i], 0, *continIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
if (assertWhenNotFound) {
|
|
|
|
assert (tupleFounded);
|
|
|
|
}
|
|
|
|
}
|
2012-03-31 23:27:37 +01:00
|
|
|
|
|
|
|
LogVars newLvs (ct->logVars().begin() + intersect.size(),
|
|
|
|
ct->logVars().end());
|
|
|
|
Util::addToVector (logVars_, newLvs);
|
|
|
|
logVarSet_ |= LogVarSet (newLvs);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2012-04-03 11:58:21 +01:00
|
|
|
LogVars discardedLvs = theta.getDiscardedLogVars();
|
|
|
|
for (unsigned i = 0; i < discardedLvs.size(); i++) {
|
|
|
|
remove(discardedLvs[i]);
|
|
|
|
}
|
2012-03-22 11:33:24 +00:00
|
|
|
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);
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
(*it)->removeAndDeleteAllChilds();
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
logVars_.resize (logVars_.size() - X.size());
|
|
|
|
logVarSet_ -= X;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
ConstraintTree::ConstraintTree::isSingleton (LogVar X)
|
|
|
|
{
|
|
|
|
SymbolSet symbols;
|
|
|
|
CTNodes nodes = getNodesAtLevel (getLevel (X));
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
symbols.insert ((*it)->symbol());
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
return symbols.size() == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
out << "\"" << *it << "\"";
|
|
|
|
out << " [label=\"" << **it << "\"]" ;
|
2012-03-22 11:33:24 +00:00
|
|
|
out << endl;
|
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
const CTNodes& childs = (*it)->childs();
|
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
out << "\"" << **it << "\"" ;
|
2012-03-22 11:33:24 +00:00
|
|
|
out << " -> " ;
|
2012-04-15 03:19:15 +01:00
|
|
|
out << "\"" << *chIt << "\"" << endl ;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
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());
|
2012-04-15 03:19:15 +01:00
|
|
|
unsigned count = countTuples (*nodes.begin());
|
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
if (countTuples (*it) != count) {
|
2012-03-22 11:33:24 +00:00
|
|
|
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()) {
|
2012-04-15 03:19:15 +01:00
|
|
|
n = *(n->childs().begin());
|
2012-03-22 11:33:24 +00:00
|
|
|
l ++;
|
|
|
|
}
|
|
|
|
return countTuples (n);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TinySet<unsigned>
|
|
|
|
ConstraintTree::getConditionalCounts (const LogVarSet& Ys)
|
|
|
|
{
|
|
|
|
TinySet<unsigned> 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());
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
counts.insert (countTuples (*it));
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
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++) {
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes nodesit = getNodesAtLevel (i);
|
|
|
|
vector<CTNode*> nodes (nodesit.begin(), nodesit.end()); // TODO
|
2012-03-22 11:33:24 +00:00
|
|
|
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*,ConstraintTree*>
|
|
|
|
ConstraintTree::split (
|
|
|
|
const Tuple& tuple,
|
|
|
|
unsigned stopLevel)
|
|
|
|
{
|
|
|
|
ConstraintTree tempCt (logVars_, {tuple});
|
|
|
|
return split (&tempCt, stopLevel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::pair<ConstraintTree*, ConstraintTree*>
|
|
|
|
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_);
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = commNodes.begin();
|
|
|
|
it != commNodes.end(); ++ it) {
|
|
|
|
commCt->root()->addChild (*it);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << commCt->tupleSet() << " + " ;
|
|
|
|
// cout << exclCt->tupleSet() << " = " ;
|
|
|
|
// cout << tupleSet() << endl << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
// if (((commCt->tupleSet() | exclCt->tupleSet()) == tupleSet()) == false) {
|
|
|
|
// exportToGraphViz ("_fail.dot", true);
|
|
|
|
// commCt->exportToGraphViz ("_fail_comm.dot", true);
|
|
|
|
// exclCt->exportToGraphViz ("_fail_excl.dot", true);
|
|
|
|
// }
|
2012-03-31 23:27:37 +01:00
|
|
|
// assert ((commCt->tupleSet() | exclCt->tupleSet()) == tupleSet());
|
|
|
|
// assert ((exclCt->tupleSet (stopLevel) & ct->tupleSet (stopLevel)).empty());
|
2012-03-22 11:33:24 +00:00
|
|
|
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<unsigned, ConstraintTree*> countMap;
|
|
|
|
unsigned stopLevel = getLevel (Zs.back());
|
|
|
|
const CTNodes& childs = root_->childs();
|
|
|
|
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
2012-03-22 11:33:24 +00:00
|
|
|
const vector<pair<CTNode*, unsigned>>& res
|
2012-04-15 03:19:15 +01:00
|
|
|
= countNormalize (*chIt, stopLevel);
|
2012-03-22 11:33:24 +00:00
|
|
|
for (unsigned j = 0; j < res.size(); j++) {
|
|
|
|
unordered_map<unsigned, ConstraintTree*>::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);
|
2012-03-31 23:27:37 +01:00
|
|
|
// 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;
|
2012-03-22 11:33:24 +00:00
|
|
|
|
|
|
|
ConstraintTrees normCts1 = commCt->countNormalize (X);
|
|
|
|
vector<unsigned> counts1 (normCts1.size());
|
|
|
|
for (unsigned i = 0; i < normCts1.size(); i++) {
|
|
|
|
counts1[i] = normCts1[i]->getConditionalCount (X);
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << "normCts1[" << i << "] #" << counts1[i] ;
|
|
|
|
// cout << " " << normCts1[i]->tupleSet() << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConstraintTrees normCts2 = exclCt->countNormalize (X);
|
|
|
|
vector<unsigned> counts2 (normCts2.size());
|
|
|
|
for (unsigned i = 0; i < normCts2.size(); i++) {
|
|
|
|
counts2[i] = normCts2[i]->getConditionalCount (X);
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << "normCts2[" << i << "] #" << counts2[i] ;
|
|
|
|
// cout << " " << normCts2[i]->tupleSet() << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
|
|
|
|
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);
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << "joint-count(" << N << ",0)" << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
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);
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << "joint-count(0," << N << ")" << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < normCts1.size(); i++) {
|
|
|
|
unsigned j;
|
|
|
|
for (j = 0; counts1[i] + counts2[j] != N; j++) ;
|
2012-03-31 23:27:37 +01:00
|
|
|
// cout << "joint-count(" << counts1[i] ;
|
|
|
|
// cout << "," << counts2[j] << ")" << endl;
|
2012-03-22 11:33:24 +00:00
|
|
|
const CTNodes& childs = normCts2[j]->root_->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
normCts1[i]->root_->addChild (CTNode::copySubtree (*chIt));
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-03-31 23:27:37 +01:00
|
|
|
delete normCts2[j];
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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 CTNodes& childs1 = ct1->root_->childs();
|
|
|
|
const CTNodes& childs2 = ct2->root_->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt1 = childs1.begin();
|
|
|
|
chIt1 != childs1.end(); ++ chIt1) {
|
|
|
|
for (CTNodes::const_iterator chIt2 = childs2.begin();
|
|
|
|
chIt2 != childs2.end(); ++ chIt2) {
|
|
|
|
if (overlap (*chIt1, *chIt2, stopLevel)) {
|
2012-03-22 11:33:24 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
LogVars
|
|
|
|
ConstraintTree::expand (LogVar X)
|
|
|
|
{
|
|
|
|
moveToBottom ({X});
|
|
|
|
assert (isCountNormalized (X));
|
|
|
|
CTNodes nodes = getNodesAtLevel (logVars_.size() - 1);
|
|
|
|
unsigned nrSymbols = getConditionalCount (X);
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
2012-03-22 11:33:24 +00:00
|
|
|
Symbols symbols;
|
2012-04-15 03:19:15 +01:00
|
|
|
const CTNodes& childs = (*it)->childs();
|
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
symbols.push_back ((*chIt)->symbol());
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
(*it)->removeAndDeleteAllChilds();
|
|
|
|
CTNode* prev = *it;
|
2012-03-22 11:33:24 +00:00
|
|
|
assert (symbols.size() == nrSymbols);
|
|
|
|
for (unsigned j = 0; j < nrSymbols; j++) {
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNode* newNode = new CTNode (symbols[j], (*it)->level() + j);
|
2012-03-22 11:33:24 +00:00
|
|
|
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 CTNodes& nodes = root_->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator it = nodes.begin();
|
|
|
|
it != nodes.end(); ++ it) {
|
|
|
|
CTNode* copy = CTNode::copySubtree (*it);
|
|
|
|
copy->setSymbol ((*it)->symbol());
|
2012-03-22 11:33:24 +00:00
|
|
|
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 CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
sum += countTuples (*chIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
return sum;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CTNodes
|
|
|
|
ConstraintTree::getNodesBelow (CTNode* fromHere) const
|
|
|
|
{
|
|
|
|
CTNodes nodes;
|
|
|
|
queue<CTNode*> queue;
|
|
|
|
queue.push (fromHere);
|
|
|
|
while (queue.empty() == false) {
|
|
|
|
CTNode* node = queue.front();
|
|
|
|
nodes.push_back (node);
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = node->childs().begin();
|
|
|
|
chIt != node->childs().end(); ++ chIt) {
|
|
|
|
queue.push (*chIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
queue.pop();
|
|
|
|
}
|
|
|
|
return nodes;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CTNodes
|
|
|
|
ConstraintTree::getNodesAtLevel (unsigned level) const
|
|
|
|
{
|
|
|
|
assert (level <= logVars_.size());
|
|
|
|
CTNodes nodes;
|
|
|
|
queue<CTNode*> queue;
|
|
|
|
queue.push (root_);
|
|
|
|
while (queue.empty() == false) {
|
|
|
|
CTNode* node = queue.front();
|
|
|
|
if (node->level() == level) {
|
|
|
|
nodes.push_back (node);
|
|
|
|
} else {
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = node->childs().begin();
|
|
|
|
chIt != node->childs().end(); ++ chIt) {
|
|
|
|
queue.push (*chIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
queue.pop();
|
|
|
|
}
|
|
|
|
return nodes;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
ConstraintTree::swapLogVar (LogVar X)
|
|
|
|
{
|
2012-04-15 03:19:15 +01:00
|
|
|
LogVars::iterator it;
|
|
|
|
it = std::find (logVars_.begin(),logVars_.end(), X);
|
2012-03-22 11:33:24 +00:00
|
|
|
assert (it != logVars_.end());
|
|
|
|
unsigned pos = std::distance (logVars_.begin(), it);
|
|
|
|
const CTNodes& nodes = getNodesAtLevel (pos);
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator nodeIt = nodes.begin();
|
|
|
|
nodeIt != nodes.end(); ++ nodeIt) {
|
|
|
|
CTNodes childsCopy = (*nodeIt)->childs();
|
|
|
|
(*nodeIt)->removeChilds();
|
|
|
|
for (CTNodes::const_iterator ccIt = childsCopy.begin();
|
|
|
|
ccIt != childsCopy.end(); ++ ccIt) {
|
|
|
|
const CTNodes& grandsons = (*ccIt)->childs();
|
|
|
|
for (CTNodes::const_iterator gsIt = grandsons.begin();
|
|
|
|
gsIt != grandsons.end(); ++ gsIt) {
|
|
|
|
CTNode* childCopy = new CTNode (**ccIt);
|
|
|
|
const CTNodes& greatGrandsons = (*gsIt)->childs();
|
|
|
|
for (CTNodes::const_iterator ggsIt = greatGrandsons.begin();
|
|
|
|
ggsIt != greatGrandsons.end(); ++ ggsIt) {
|
|
|
|
childCopy->addChild (*ggsIt, false);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
childCopy->setLevel (childCopy->level() + 1);
|
|
|
|
(*gsIt)->removeChilds();
|
|
|
|
(*gsIt)->addChild (childCopy, false);
|
|
|
|
(*gsIt)->setLevel ((*gsIt)->level() - 1);
|
|
|
|
(*nodeIt)->addChild ((*gsIt), false);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
delete (*ccIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
std::swap (logVars_[pos], logVars_[pos + 1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
ConstraintTree::join (
|
|
|
|
CTNode* n,
|
|
|
|
const Tuple& tuple,
|
|
|
|
unsigned currIdx,
|
|
|
|
CTNode* appendNode)
|
|
|
|
{
|
|
|
|
bool tupleFounded = false;
|
|
|
|
if (n->symbol() == tuple[currIdx]) {
|
|
|
|
if (currIdx == tuple.size() - 1) {
|
|
|
|
const CTNodes& childs = appendNode->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
n->addChild (CTNode::copySubtree (*chIt));
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
tupleFounded |= join (*chIt, tuple, currIdx + 1, appendNode);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tupleFounded;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
ConstraintTree::indenticalSubtrees (
|
|
|
|
const CTNode* n1,
|
|
|
|
const CTNode* n2,
|
|
|
|
bool compare) const
|
|
|
|
{
|
|
|
|
if (compare) {
|
|
|
|
if (n1->symbol() != n2->symbol()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const CTNodes& childs1 = n1->childs();
|
|
|
|
const CTNodes& childs2 = n2->childs();
|
|
|
|
if (childs1.size() != childs2.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes::const_iterator chIt1 = childs1.begin();
|
|
|
|
CTNodes::const_iterator chIt2 = childs2.begin();
|
|
|
|
while (chIt1 != childs1.end()) {
|
|
|
|
if (indenticalSubtrees (*chIt1, *chIt2, true) == false) {
|
2012-03-22 11:33:24 +00:00
|
|
|
return false;
|
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
++ chIt1;
|
|
|
|
++ chIt2;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
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 CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
|
|
|
getTuples (*chIt, currTuples, stopLevel, tuplesCollected,
|
|
|
|
continuationNodes);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
ConstraintTree::size (void) const
|
|
|
|
{
|
|
|
|
return countTuples (root_);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
ConstraintTree::nrSymbols (LogVar X)
|
|
|
|
{
|
|
|
|
moveToTop ({X});
|
|
|
|
return root_->childs().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
vector<pair<CTNode*, unsigned>>
|
|
|
|
ConstraintTree::countNormalize (
|
|
|
|
const CTNode* n,
|
|
|
|
unsigned stopLevel)
|
|
|
|
{
|
|
|
|
if (n->level() == stopLevel) {
|
|
|
|
return vector<pair<CTNode*, unsigned>>() = {
|
2012-03-31 23:27:37 +01:00
|
|
|
make_pair (CTNode::copySubtree (n), countTuples (n))
|
2012-03-22 11:33:24 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
vector<pair<CTNode*, unsigned>> res;
|
|
|
|
const CTNodes& childs = n->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt = childs.begin();
|
|
|
|
chIt != childs.end(); ++ chIt) {
|
2012-03-22 11:33:24 +00:00
|
|
|
const vector<pair<CTNode*, unsigned>>& lowerRes =
|
2012-04-15 03:19:15 +01:00
|
|
|
countNormalize (*chIt, stopLevel);
|
2012-03-22 11:33:24 +00:00
|
|
|
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)
|
|
|
|
{
|
|
|
|
CTNodes& childs1 = n1->childs();
|
|
|
|
CTNodes& childs2 = n2->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt1 = childs1.begin();
|
|
|
|
chIt1 != childs1.end(); ++ chIt1) {
|
2012-03-22 11:33:24 +00:00
|
|
|
CTNode* intersectNode = 0;
|
2012-04-15 03:19:15 +01:00
|
|
|
for (CTNodes::const_iterator chIt2 = childs2.begin();
|
|
|
|
chIt2 != childs2.end(); ++ chIt2) {
|
|
|
|
if ((*chIt1)->symbol() == (*chIt2)->symbol()) {
|
|
|
|
intersectNode = *chIt2;
|
2012-03-22 11:33:24 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (intersectNode == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-04-15 03:19:15 +01:00
|
|
|
if ((*chIt1)->level() == stopLevel) {
|
|
|
|
CTNode* newNode = CTNode::copySubtree (*chIt1);
|
2012-03-22 11:33:24 +00:00
|
|
|
nodes.push_back (newNode);
|
2012-04-15 03:19:15 +01:00
|
|
|
(*chIt1)->setSymbol (Symbol::invalid());
|
2012-03-22 11:33:24 +00:00
|
|
|
} else {
|
|
|
|
CTNodes lowerNodes;
|
2012-04-15 03:19:15 +01:00
|
|
|
split ((*chIt1), intersectNode, lowerNodes, stopLevel);
|
2012-03-22 11:33:24 +00:00
|
|
|
if (lowerNodes.empty() == false) {
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNode* newNode = new CTNode (**chIt1);
|
|
|
|
for (CTNodes::const_iterator lowerIt = lowerNodes.begin();
|
|
|
|
lowerIt != lowerNodes.end(); ++ lowerIt) {
|
|
|
|
newNode->addChild (*lowerIt);
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
nodes.push_back (newNode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-15 03:19:15 +01:00
|
|
|
CTNodes::const_iterator chIt = childs1.begin();
|
|
|
|
while (chIt != childs1.end()) {
|
|
|
|
bool inc = true;
|
|
|
|
if ((*chIt)->symbol() == Symbol::invalid()) {
|
|
|
|
n1->removeAndDeleteChild ((*chIt));
|
|
|
|
inc = false;
|
|
|
|
} else if ((*chIt)->isLeaf() &&
|
|
|
|
(*chIt)->level() != stopLevel) {
|
|
|
|
n1->removeAndDeleteChild ((*chIt));
|
|
|
|
inc = false;
|
|
|
|
}
|
|
|
|
if (inc) {
|
|
|
|
++ chIt;
|
2012-03-22 11:33:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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 CTNodes& childsI = n1->childs();
|
2012-04-15 03:19:15 +01:00
|
|
|
const CTNodes& childsJ = n2->childs();
|
|
|
|
for (CTNodes::const_iterator chIt1 = childsI.begin();
|
|
|
|
chIt1 != childsI.end(); ++ chIt1) {
|
|
|
|
for (CTNodes::const_iterator chIt2 = childsJ.begin();
|
|
|
|
chIt2 != childsJ.end(); ++ chIt2) {
|
|
|
|
if (overlap (*chIt1, *chIt2, stopLevel)) {
|
2012-03-22 11:33:24 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|