diff --git a/Makefile b/Makefile index c4a3ee3828..54bcd3f6d6 100644 --- a/Makefile +++ b/Makefile @@ -238,6 +238,7 @@ VST_OBJ= Saturation/AWPassiveClauseContainer.o\ VS_OBJ = Shell/AIG.o\ Shell/AIGCompressor.o\ + Shell/AIGDefinitionIntroducer.o\ Shell/AIGInliner.o\ Shell/AIGRewriter.o\ Shell/AnswerExtractor.o\ diff --git a/Parse/SMTLIB.cpp b/Parse/SMTLIB.cpp index e9be77d1be..48b8907bb2 100644 --- a/Parse/SMTLIB.cpp +++ b/Parse/SMTLIB.cpp @@ -17,7 +17,7 @@ #include "Kernel/Signature.hpp" #include "Kernel/SortHelper.hpp" -#include "Shell/AIGInliner.hpp" +#include "Shell/AIGDefinitionIntroducer.hpp" #include "Shell/AIGCompressor.hpp" #include "Shell/LispLexer.hpp" #include "Shell/Options.hpp" diff --git a/Shell/AIGDefinitionIntroducer.cpp b/Shell/AIGDefinitionIntroducer.cpp new file mode 100644 index 0000000000..58ce7321e8 --- /dev/null +++ b/Shell/AIGDefinitionIntroducer.cpp @@ -0,0 +1,584 @@ +/** + * @file AIGDefinitionIntroducer.cpp + * Implements class AIGDefinitionIntroducer. + */ + +#include "Lib/SharedSet.hpp" + +#include "Kernel/Clause.hpp" +#include "Kernel/ColorHelper.hpp" +#include "Kernel/Formula.hpp" +#include "Kernel/FormulaUnit.hpp" +#include "Kernel/Problem.hpp" +#include "Kernel/Signature.hpp" +#include "Kernel/SortHelper.hpp" +#include "Kernel/TermIterators.hpp" +#include "Kernel/Unit.hpp" + +#include "Flattening.hpp" +#include "PDUtils.hpp" +#include "SimplifyFalseTrue.hpp" + +#include "AIGDefinitionIntroducer.hpp" + +namespace Shell +{ + +//////////////////////////// +// AIGDefinitionIntroducer +// + +AIGDefinitionIntroducer::AIGDefinitionIntroducer(unsigned threshold) + : _arwr(_fsh.aig()) +{ + CALL("AIGDefinitionIntroducer::AIGDefinitionIntroducer"); + + _namingRefCntThreshold = threshold; + _mergeEquivDefs = false; //not implemented yet +} + +/** + * Add a definition into _defs and return true, or return false + * if @c rhs already as a definition + * + * namingUnit may be zero, that means that the naming unit will be created lazily later + */ +bool AIGDefinitionIntroducer::addAIGName(AIGRef rhs, AIGRef nameAtom, FormulaUnit* namingUnit) +{ + CALL("AIGDefinitionIntroducer::addAIGDefinition"); + ASS(nameAtom.isAtom()); + ASS(!rhs.isPropConst()); + + if(!_defs.insert(rhs, nameAtom)) { + return false; + } + if(namingUnit) { + ALWAYS(_defUnits.insert(rhs, namingUnit)); + } + return true; +} + +/** @c rhs must have a name, this will be returned. */ +AIGRef AIGDefinitionIntroducer::getName(AIGRef rhs) const +{ + CALL("AIGDefinitionIntroducer::getName"); + ASS(hasName(rhs)); + + return _defs.get(rhs); +} + + +bool AIGDefinitionIntroducer::scanDefinition(FormulaUnit* def) +{ + CALL("AIGDefinitionIntroducer::scanDefinition"); + + Literal* lhs; + + AIGRef rhsAig; + + Literal* rhsLit; //valid only from pred equiv + if(PDUtils::isPredicateEquivalence(def, lhs, rhsLit)) { + rhsAig = _fsh.apply(rhsLit); + } + else { + Formula* rhs; + PDUtils::splitDefinition(def, lhs, rhs); + rhsAig = _fsh.apply(rhs).second; + } + + if(rhsAig.isPropConst()) { + return false; + } + + if(lhs->color()!=COLOR_TRANSPARENT) { + return false; + } + + AIGRef lhsAig = _fsh.apply(lhs); + + if(!rhsAig.polarity()) { + rhsAig = rhsAig.neg(); + lhsAig = lhsAig.neg(); + } + + if(!addAIGName(rhsAig, lhsAig, def)) { + //rhs is already defined + AIGRef oldDefTgt; + oldDefTgt = getName(rhsAig); + if(_mergeEquivDefs) { +// _equivs.insert(lhs, oldDefTgt); + NOT_IMPLEMENTED; + } + return false; + } + + _toplevelAIGs.push(rhsAig); + return true; +} + +void AIGDefinitionIntroducer::collectTopLevelAIGsAndDefs(UnitList* units) +{ + CALL("AIGDefinitionIntroducer::collectTopLevelAIGsAndDefs"); + + UnitList::Iterator uit(units); + while(uit.hasNext()) { + Unit* u = uit.next(); + if(u->isClause()) { + continue; + } + FormulaUnit* fu = static_cast(u); + + if(PDUtils::hasDefinitionShape(fu)) { + if(scanDefinition(fu)) { + continue; + } + } + + Formula* form = fu->formula(); + AIGRef formAig = _fsh.apply(form).second; + _toplevelAIGs.push(formAig); + } +} + +AIGDefinitionIntroducer::VarSet* AIGDefinitionIntroducer::getAtomVars(Literal* l) +{ + CALL("AIGDefinitionIntroducer::getAtomVars"); + + static Stack vars; + vars.reset(); + static VariableIterator vit; + vit.reset(l); + while(vit.hasNext()) { + TermList vt = vit.next(); + unsigned var = vt.var(); + vars.push(var); + } + return VarSet::getFromIterator(Stack::Iterator(vars)); +} + +AIGDefinitionIntroducer::NodeInfo& AIGDefinitionIntroducer::getNodeInfo(AIGRef r) +{ + CALL("AIGDefinitionIntroducer::doFirstRefAIGPass"); + + AIGRef rPos = r.getPositive(); + size_t idx = _aigIndexes.get(rPos); + return _refAIGInfos[idx]; +} + +void AIGDefinitionIntroducer::doFirstRefAIGPass() +{ + CALL("AIGDefinitionIntroducer::doFirstRefAIGPass"); + + ASS(_refAIGInfos.isEmpty()); + size_t aigCnt = _refAIGs.size(); + for(size_t i=0; icolor(); + } + else if(r.isPropConst()) { + ni._freeVars = VarSet::getEmpty(); + ni._clr = COLOR_TRANSPARENT; + } + else if(r.isConjunction()) { + NodeInfo& pni1 = getNodeInfo(r.parent(0)); + NodeInfo& pni2 = getNodeInfo(r.parent(1)); + ni._freeVars = pni1._freeVars->getUnion(pni2._freeVars); + ni._clr = ColorHelper::combine(pni1._clr, pni2._clr); + if(ni._clr==COLOR_INVALID) { + USER_ERROR("mixing colors in "+r.toString()); + } + } + else { + ASS(r.isQuantifier()); + NodeInfo& pni = getNodeInfo(r.parent(0)); + VarSet* qVars = r.getQuantifierVars(); + ni._freeVars = pni._freeVars->subtract(qVars); + ni._clr = pni._clr; + } + + unsigned parCnt = r.parentCnt(); + for(unsigned pi = 0; pi argSorts; + argSorts.reset(); + + if(!freeVars->isEmpty()) { + static DHMap varSorts; + varSorts.reset(); + SortHelper::collectVariableSorts(rhs, varSorts); //careful!! this traverses the formula as tree, not as DAG, so may take exponential time! + + VarSet::Iterator vit(*freeVars); + while(vit.hasNext()) { + unsigned var = vit.next(); + args.push(TermList(var, false)); + argSorts.push(varSorts.get(var)); + } + } + + unsigned arity = args.size(); + unsigned pred = env.signature->addFreshPredicate(arity, "sP","aig_name"); + Signature::Symbol* psym = env.signature->getPredicate(pred); + psym->setType(PredicateType::makeType(arity, argSorts.begin(), Sorts::SRT_BOOL)); + if(ni._clr!=COLOR_TRANSPARENT) { + psym->addColor(ni._clr); + } + + Literal* res = Literal::create(pred, arity, true, false, args.begin()); + return res; +} + +void AIGDefinitionIntroducer::introduceName(unsigned aigStackIdx) +{ + CALL("AIGDefinitionIntroducer::introduceName"); + + AIGRef a = getPreNamingAig(aigStackIdx); + NodeInfo& ni = _refAIGInfos[aigStackIdx]; + ASS(!ni._hasName); + ASS(!hasName(a.getPositive())); + + ni._formRefCnt = 1; + Literal* nameLit = getNameLiteral(aigStackIdx); + ni._hasName = true; + ni._name = _fsh.apply(nameLit); + if(a.polarity()) { + ALWAYS(addAIGName(a, ni._name, 0)); + } + else { + ALWAYS(addAIGName(a.neg(), ni._name.neg(), 0)); + } +} + +void AIGDefinitionIntroducer::doSecondRefAIGPass() +{ + CALL("AIGDefinitionIntroducer::doSecondRefAIGPass"); + + TopLevelStack::Iterator tlit(_toplevelAIGs); + while(tlit.hasNext()) { + AIGRef a = tlit.next(); + AIGRef aPos = a.getPositive(); + unsigned stackIdx = _aigIndexes.get(aPos); + NodeInfo& ni = _refAIGInfos[stackIdx]; + ni._formRefCnt++; + ni._inPol[a.polarity()] = true; + } + + size_t aigCnt = _refAIGs.size(); + for(size_t i=aigCnt; i>0;) { + i--; + + AIGRef r = _refAIGs[i]; + NodeInfo& ni = _refAIGInfos[i]; + + if(ni._hasName) { + ni._formRefCnt = 1; + } + if(shouldIntroduceName(i)) { + introduceName(i); + } + + unsigned parCnt = r.parentCnt(); + for(unsigned pi = 0; pifreeVariables(); + if(vars) { + equiv = new QuantifiedFormula(FORALL, vars, equiv); + } + ASS_REP(equiv->freeVariables()->isEmpty(), *equiv); + FormulaUnit* def = new FormulaUnit(equiv, new Inference(Inference::PREDICATE_DEFINITION), Unit::AXIOM); + _newDefs.push(def); + + LOG_UNIT("pp_aigdef_intro", def); + return def; +} + +void AIGDefinitionIntroducer::doThirdRefAIGPass() +{ + CALL("AIGDefinitionIntroducer::doThirdRefAIGPass"); + + size_t aigCnt = _refAIGs.size(); + for(size_t i=0; igetUnion(AIGRewriter::PremiseSet::getSingleton(i)); + ALWAYS(_defsSaturated.insert(r, AIGRewriter::PremRef(ni._name, selfNamingPrems))); + } +} + +void AIGDefinitionIntroducer::scan(UnitList* units) +{ + CALL("AIGDefinitionIntroducer::scan"); + + collectTopLevelAIGsAndDefs(units); + + processTopLevelAIGs(); +} + +void AIGDefinitionIntroducer::processTopLevelAIGs() +{ + CALL("AIGDefinitionIntroducer::processTopLevelAIGs"); + + _refAIGs.loadFromIterator( TopLevelStack::Iterator(_toplevelAIGs) ); + _fsh.aigTransf().makeOrderedAIGGraphStack(_refAIGs); + + doFirstRefAIGPass(); + doSecondRefAIGPass(); + doThirdRefAIGPass(); +// _defsSaturated.loadFromMap(_defs); +// _fsh.aigTransf().saturateMap(_defsSaturated); +} + +Inference* AIGDefinitionIntroducer::getInferenceFromPremIndexes(Unit* orig, AIGRewriter::PremiseSet* premIndexes) +{ + CALL("AIGDefinitionIntroducer::getInferenceFromPremIndexes"); + + if(premIndexes->size()==0) { + return new Inference1(Inference::DEFINITION_FOLDING, orig); + } + if(premIndexes->size()==1) { + unsigned nmPremIdx = premIndexes->sval(); + Unit* nmPrem = _refAIGInfos[nmPremIdx]._namingUnit; + ASS(nmPrem); + return new Inference2(Inference::DEFINITION_FOLDING, nmPrem, orig); + } + + + + UnitList* prems = 0; + UnitList::push(orig, prems); + AIGRewriter::PremiseSet::Iterator psit(*premIndexes); + while(psit.hasNext()) { + unsigned nmPremIdx = psit.next(); + Unit* nmPrem = _refAIGInfos[nmPremIdx]._namingUnit; + UnitList::push(nmPrem, prems); + } + + return new InferenceMany(Inference::DEFINITION_FOLDING, prems); +} + +bool AIGDefinitionIntroducer::apply(FormulaUnit* unit, Unit*& res) +{ + CALL("AIGDefinitionIntroducer::apply(FormulaUnit*,Unit*&)"); + + Formula* f0 = unit->formula(); + + AIGRef f0Aig = _fsh.apply(f0).second; + AIGRewriter::PremiseSet* premIndexes; + AIGRef resAig = _arwr.lev0Deref(f0Aig, _defsSaturated, &premIndexes); + if(f0Aig==resAig) { + return false; + } + Formula* f = _fsh.aigToFormula(resAig); + + if(f->connective()==TRUE) { + res = 0; + LOG("pp_aigdef_apply","orig: " << (*unit) << endl << " simplified to tautology"); + return true; + } + + ASS_REP2(f->freeVariables()->isEmpty(), *f, *unit); + + //TODO add proper inferences + Inference* inf = getInferenceFromPremIndexes(unit, premIndexes); + res = new FormulaUnit(f, inf, unit->inputType()); + LOG("pp_aigdef_apply","orig: " << (*unit) << endl << "intr: " << (*res)); + ASS(!res->isClause()); + res = SimplifyFalseTrue().simplify(static_cast(res)); + res = Flattening::flatten(static_cast(res)); + return true; +} + +///** +// * Introduce definitions into a formula and return modifier formuls. +// * The introducedDefinitions add to the @c introducedDefs list. +// */ +//Formula* AIGDefinitionIntroducer::apply(Formula* f0, UnitList*& introducedDefs) +//{ +// CALL("AIGDefinitionIntroducer::apply(Formula*,UnitList*&)"); +// ASS(_newDefs.isEmpty()); +// +// AIGRef f0Aig = _fsh.apply(f0).second; +// +// _toplevelAIGs.push(f0Aig); +// +// processTopLevelAIGs(); +// +// AIGRef resAig = AIGTransformer::lev0Deref(f0Aig, _defs); +// if(f0Aig==resAig) { +// return f0; +// } +//// LOGV("bug",f0Aig); +//// LOGV("bug",resAig); +// Formula* f = _fsh.aigToFormula(resAig); +// +// +// Stack::Iterator uit(_newDefs); +// while(uit.hasNext()) { +// FormulaUnit* def0 = uit.next(); +// Unit* def; +// if(!apply(def0, def)) { +// def = def0; +// } +// UnitList::push(def, introducedDefs); +// } +// +// return f; +//} + +UnitList* AIGDefinitionIntroducer::getIntroducedFormulas() +{ + CALL("AIGDefinitionIntroducer::getIntroducedFormulas"); + LOG("pp_aigdef_apply","processing definitions"); + + UnitList* res = 0; + + Stack::Iterator uit(_newDefs); + while(uit.hasNext()) { + FormulaUnit* def0 = uit.next(); + Unit* def; + if(!apply(def0, def)) { + def = def0; + } + UnitList::push(def, res); + } + return res; +} + +void AIGDefinitionIntroducer::updateModifiedProblem(Problem& prb) +{ + CALL("AIGDefinitionIntroducer::updateModifiedProblem"); + + prb.invalidateProperty(); +} + +} diff --git a/Shell/AIGDefinitionIntroducer.hpp b/Shell/AIGDefinitionIntroducer.hpp new file mode 100644 index 0000000000..9d4b939721 --- /dev/null +++ b/Shell/AIGDefinitionIntroducer.hpp @@ -0,0 +1,143 @@ +/** + * @file AIGDefinitionIntroducer.hpp + * Defines class AIGDefinitionIntroducer. + */ + +#ifndef __AIGDefinitionIntroducer__ +#define __AIGDefinitionIntroducer__ + +#include "Forwards.hpp" + +#include "Lib/DHMap.hpp" +#include "Lib/DHSet.hpp" +#include "Lib/Stack.hpp" + +#include "AIG.hpp" +#include "AIGRewriter.hpp" + + +namespace Shell { + +class AIGDefinitionIntroducer : public ScanAndApplyFormulaUnitTransformer +{ + typedef SharedSet VarSet; + + struct NodeInfo { + + //these members are filled-in in the doFirstRefAIGPass() function + + /** True if contains quantifiers with {negative,positive} polarity */ + bool _hasQuant[2]; + + bool _hasName; + AIGRef _name; + + VarSet* _freeVars; + /** Color of self or parents. If should be invalid, exception is thrown. */ + Color _clr; + + /** Number of AIG nodes that refer to this node */ + unsigned _directRefCnt; + + //these members are filled-in in the doSecondRefAIGPass() function + + /** True if occurs in toplevel AIGs with {negative,positive} polarity */ + bool _inPol[2]; + /** True if occurs in quantifier AIG nodes with {negative,positive} polarity */ + bool _inQuant[2]; + + /** + * How many times will an AIG node appear in formulas + * after conversion of AIG to formulas + * + * Equals to 1 if node has a name + */ + unsigned _formRefCnt; + + + //these members are used in the third pass + + /** Before the third pass may be zero */ + FormulaUnit* _namingUnit; + }; + + //options + bool _eprPreserving; + unsigned _namingRefCntThreshold; + unsigned _mergeEquivDefs; + + AIGFormulaSharer _fsh; + AIGRewriter _arwr; + + typedef Stack TopLevelStack; + TopLevelStack _toplevelAIGs; + + /** + * All positive AIG refs used in the problem, ordered topologically, + * so that references go only toward the bottom of the stack. + */ + AIGStack _refAIGs; + /** stack of infos corresponding to nodes at corresponding _refAIGs positions */ + Stack _refAIGInfos; + /** Indexes of AIG refs in the _refAIGs stack */ + DHMap _aigIndexes; + + DHMap _defs; + DHMap _defUnits; + + /** + * _defs saturated on the relevand AIGs + */ + AIGRewriter::RefMap _defsSaturated; + +// DHMap _equivs; + + Stack _newDefs; + + bool addAIGName(AIGRef rhs, AIGRef nameAtom, FormulaUnit* namingUnit); + bool hasName(AIGRef rhs) const { return _defs.find(rhs); } + AIGRef getName(AIGRef rhs) const; + bool findName(AIGRef rhs, AIGRef& name) const { return _defs.find(rhs, name); } + + AIGRef getPreNamingAig(unsigned aigStackIdx); + + bool shouldIntroduceName(unsigned aigStackIdx); + Literal* getNameLiteral(unsigned aigStackIdx); + void introduceName(unsigned aigStackIdx); + + bool scanDefinition(FormulaUnit* def); + + //functions called from scan() + void collectTopLevelAIGsAndDefs(UnitList* units); + void processTopLevelAIGs(); + + //functions called from processTopLevelAIGs() + void doFirstRefAIGPass(); + void doSecondRefAIGPass(); + void doThirdRefAIGPass(); + + VarSet* getAtomVars(Literal* l); + NodeInfo& getNodeInfo(AIGRef r); + + FormulaUnit* createNameUnit(AIGRef rhs, AIGRef atomName); + + Inference* getInferenceFromPremIndexes(Unit* orig, AIGRewriter::PremiseSet* premIndexes); + +public: + AIGDefinitionIntroducer(unsigned threshold=4); + + virtual void scan(UnitList* units); + + using ScanAndApplyFormulaUnitTransformer::apply; + virtual bool apply(FormulaUnit* unit, Unit*& res); + + virtual UnitList* getIntroducedFormulas(); + +// Formula* apply(Formula* f, UnitList*& introducedDefs); +protected: + virtual void updateModifiedProblem(Problem& prb); +}; + +} + +#endif // __AIGDefinitionIntroducer__ diff --git a/Shell/AIGInliner.cpp b/Shell/AIGInliner.cpp index 00f6586ab9..1f2ec46ecc 100644 --- a/Shell/AIGInliner.cpp +++ b/Shell/AIGInliner.cpp @@ -510,482 +510,5 @@ bool AIGInliner::apply(FormulaUnit* unit, Unit*& res) } -//////////////////////////// -// AIGDefinitionIntroducer -// - -AIGDefinitionIntroducer::AIGDefinitionIntroducer(unsigned threshold) -{ - CALL("AIGDefinitionIntroducer::AIGDefinitionIntroducer"); - - _namingRefCntThreshold = threshold; - _mergeEquivDefs = false; //not implemented yet -} - -bool AIGDefinitionIntroducer::scanDefinition(FormulaUnit* def) -{ - CALL("AIGDefinitionIntroducer::scanDefinition"); - - Literal* lhs; - - AIGRef rhsAig; - - Literal* rhsLit; //valid only from pred equiv - if(PDUtils::isPredicateEquivalence(def, lhs, rhsLit)) { - rhsAig = _fsh.apply(rhsLit); - } - else { - Formula* rhs; - PDUtils::splitDefinition(def, lhs, rhs); - rhsAig = _fsh.apply(rhs).second; - } - - if(lhs->color()!=COLOR_TRANSPARENT) { - return false; - } - - AIGRef lhsAig = _fsh.apply(lhs); - - if(!rhsAig.polarity()) { - rhsAig = rhsAig.neg(); - lhsAig = lhsAig.neg(); - } - - if(!_defs.insert(rhsAig, lhsAig)) { - //rhs is already defined - AIGRef oldDefTgt; - ALWAYS(_defs.find(rhsAig, oldDefTgt)); - if(_mergeEquivDefs) { -// _equivs.insert(lhs, oldDefTgt); - NOT_IMPLEMENTED; - } - return false; - } - ALWAYS(_defUnits.insert(rhsAig,def)); - - _toplevelAIGs.push(rhsAig); - return true; -} - -void AIGDefinitionIntroducer::collectTopLevelAIGsAndDefs(UnitList* units) -{ - CALL("AIGDefinitionIntroducer::collectTopLevelAIGsAndDefs"); - - UnitList::Iterator uit(units); - while(uit.hasNext()) { - Unit* u = uit.next(); - if(u->isClause()) { - continue; - } - FormulaUnit* fu = static_cast(u); - - if(PDUtils::hasDefinitionShape(fu)) { - if(scanDefinition(fu)) { - continue; - } - } - - Formula* form = fu->formula(); - AIGRef formAig = _fsh.apply(form).second; - _toplevelAIGs.push(formAig); - } -} - -AIGDefinitionIntroducer::VarSet* AIGDefinitionIntroducer::getAtomVars(Literal* l) -{ - CALL("AIGDefinitionIntroducer::getAtomVars"); - - static Stack vars; - vars.reset(); - static VariableIterator vit; - vit.reset(l); - while(vit.hasNext()) { - TermList vt = vit.next(); - unsigned var = vt.var(); - vars.push(var); - } - return VarSet::getFromIterator(Stack::Iterator(vars)); -} - -AIGDefinitionIntroducer::NodeInfo& AIGDefinitionIntroducer::getNodeInfo(AIGRef r) -{ - CALL("AIGDefinitionIntroducer::doFirstRefAIGPass"); - - AIGRef rPos = r.getPositive(); - size_t idx = _aigIndexes.get(rPos); - return _refAIGInfos[idx]; -} - -void AIGDefinitionIntroducer::doFirstRefAIGPass() -{ - CALL("AIGDefinitionIntroducer::doFirstRefAIGPass"); - - ASS(_refAIGInfos.isEmpty()); - size_t aigCnt = _refAIGs.size(); - for(size_t i=0; icolor(); - } - else if(r.isPropConst()) { - ni._freeVars = VarSet::getEmpty(); - ni._clr = COLOR_TRANSPARENT; - } - else if(r.isConjunction()) { - NodeInfo& pni1 = getNodeInfo(r.parent(0)); - NodeInfo& pni2 = getNodeInfo(r.parent(1)); - ni._freeVars = pni1._freeVars->getUnion(pni2._freeVars); - ni._clr = ColorHelper::combine(pni1._clr, pni2._clr); - if(ni._clr==COLOR_INVALID) { - USER_ERROR("mixing colors in "+r.toString()); - } - } - else { - ASS(r.isQuantifier()); - NodeInfo& pni = getNodeInfo(r.parent(0)); - VarSet* qVars = r.getQuantifierVars(); - ni._freeVars = pni._freeVars->subtract(qVars); - ni._clr = pni._clr; - } - - unsigned parCnt = r.parentCnt(); - for(unsigned pi = 0; pi argSorts; - argSorts.reset(); - - if(!freeVars->isEmpty()) { - static DHMap varSorts; - varSorts.reset(); - SortHelper::collectVariableSorts(rhs, varSorts); //careful!! this traverses the formula as tree, not as DAG, so may take exponential time! - - VarSet::Iterator vit(*freeVars); - while(vit.hasNext()) { - unsigned var = vit.next(); - args.push(TermList(var, false)); - argSorts.push(varSorts.get(var)); - } - } - - unsigned arity = args.size(); - unsigned pred = env.signature->addFreshPredicate(arity, "sP","aig_name"); - Signature::Symbol* psym = env.signature->getPredicate(pred); - psym->setType(PredicateType::makeType(arity, argSorts.begin(), Sorts::SRT_BOOL)); - if(ni._clr!=COLOR_TRANSPARENT) { - psym->addColor(ni._clr); - } - - Literal* res = Literal::create(pred, arity, true, false, args.begin()); - return res; -} - -void AIGDefinitionIntroducer::introduceName(unsigned aigStackIdx) -{ - CALL("AIGDefinitionIntroducer::introduceName"); - - AIGRef a = getPreNamingAig(aigStackIdx); - NodeInfo& ni = _refAIGInfos[aigStackIdx]; - ASS(!ni._hasName); - ASS(!_defs.find(a.getPositive())); - - ni._formRefCnt = 1; - Literal* nameLit = getNameLiteral(aigStackIdx); - ni._hasName = true; - ni._name = _fsh.apply(nameLit); - if(a.polarity()) { - ALWAYS(_defs.insert(a, ni._name)); - } - else { - ALWAYS(_defs.insert(a.neg(), ni._name.neg())); - } - - Formula* lhs = new AtomicFormula(nameLit); - Formula* rhs = _fsh.aigToFormula(a); - //TODO: weaken definition based on the way subforula occurrs - Formula* equiv = new BinaryFormula(IFF, lhs, rhs); - //by the behavior of getNameLiteral lhs contains all the free variables of rhs (but unlike rhs contains just one atom) - Formula::VarList* vars = lhs->freeVariables(); - if(vars) { - equiv = new QuantifiedFormula(FORALL, vars, equiv); - } - FormulaUnit* def = new FormulaUnit(equiv, new Inference(Inference::PREDICATE_DEFINITION), Unit::AXIOM); - ALWAYS(_defUnits.insert(a, def)); - _newDefs.push(def); - - LOG_UNIT("pp_aigdef_intro", def); -} - -void AIGDefinitionIntroducer::doSecondRefAIGPass() -{ - CALL("AIGDefinitionIntroducer::doSecondRefAIGPass"); - - TopLevelStack::Iterator tlit(_toplevelAIGs); - while(tlit.hasNext()) { - AIGRef a = tlit.next(); - AIGRef aPos = a.getPositive(); - unsigned stackIdx = _aigIndexes.get(aPos); - NodeInfo& ni = _refAIGInfos[stackIdx]; - ni._formRefCnt++; - ni._inPol[a.polarity()] = true; - } - - size_t aigCnt = _refAIGs.size(); - for(size_t i=aigCnt; i>0;) { - i--; - - AIGRef r = _refAIGs[i]; - NodeInfo& ni = _refAIGInfos[i]; - - if(ni._hasName) { - ni._formRefCnt = 1; - } - if(shouldIntroduceName(i)) { - introduceName(i); - } - - unsigned parCnt = r.parentCnt(); - for(unsigned pi = 0; piconnective()==LITERAL) { return true; } -// -// LOG("pp_aigdef_apply_subform", "checking: "<<(*f)); -// AIGFormulaSharer::ARes ares = _fsh.apply(f); -//// f = ares.first; -// AIGRef a = ares.second; -// LOG("pp_aigdef_apply_subform", " aig: "<formula(); -// Formula* f = rwr.transform(f0); - - AIGRef f0Aig = _fsh.apply(f0).second; - AIGRef resAig = AIGTransformer::lev0Deref(f0Aig, _defs); - if(f0Aig==resAig) { - return false; - } - Formula* f = _fsh.aigToFormula(resAig); - - if(f->connective()==TRUE) { - res = 0; - LOG("pp_aigdef_apply","orig: " << (*unit) << endl << " simplified to tautology"); - return true; - } -// if(f==f0) { -// return false; -// } - //TODO add proper inferences - res = new FormulaUnit(f, new Inference1(Inference::DEFINITION_FOLDING,unit), unit->inputType()); - LOG("pp_aigdef_apply","orig: " << (*unit) << endl << "intr: " << (*res)); - ASS(!res->isClause()); - res = SimplifyFalseTrue().simplify(static_cast(res)); - res = Flattening::flatten(static_cast(res)); - return true; -} - -/** - * Introduce definitions into a formula and return modifier formuls. - * The introducedDefinitions add to the @c introducedDefs list. - */ -Formula* AIGDefinitionIntroducer::apply(Formula* f0, UnitList*& introducedDefs) -{ - CALL("AIGDefinitionIntroducer::apply(Formula*,UnitList*&)"); - ASS(_newDefs.isEmpty()); - - AIGRef f0Aig = _fsh.apply(f0).second; - - _toplevelAIGs.push(f0Aig); - - processTopLevelAIGs(); - - AIGRef resAig = AIGTransformer::lev0Deref(f0Aig, _defs); - if(f0Aig==resAig) { - return f0; - } - Formula* f = _fsh.aigToFormula(resAig); - - - Stack::Iterator uit(_newDefs); - while(uit.hasNext()) { - FormulaUnit* def0 = uit.next(); - Unit* def; - if(!apply(def0, def)) { - def = def0; - } - UnitList::push(def, introducedDefs); - } - - return f; -} - -UnitList* AIGDefinitionIntroducer::getIntroducedFormulas() -{ - CALL("AIGDefinitionIntroducer::getIntroducedFormulas"); - LOG("pp_aigdef_apply","processing definitions"); - - UnitList* res = 0; - - Stack::Iterator uit(_newDefs); - while(uit.hasNext()) { - FormulaUnit* def0 = uit.next(); - Unit* def; - if(!apply(def0, def)) { - def = def0; - } - UnitList::push(def, res); - } - return res; -} - } diff --git a/Shell/AIGInliner.hpp b/Shell/AIGInliner.hpp index 2980e05d19..ca3a10dfa1 100644 --- a/Shell/AIGInliner.hpp +++ b/Shell/AIGInliner.hpp @@ -115,105 +115,6 @@ class AIGInliner : public ScanAndApplyFormulaUnitTransformer { }; -class AIGDefinitionIntroducer : public ScanAndApplyFormulaUnitTransformer -{ - typedef SharedSet VarSet; - - struct NodeInfo { - - //these members are filled-in in the doFirstRefAIGPass() function - - /** True if contains quantifiers with {negative,positive} polarity */ - bool _hasQuant[2]; - bool _hasName; - AIGRef _name; - VarSet* _freeVars; - /** Color of self or parents. If should be invalid, exception is thrown. */ - Color _clr; - - /** Number of AIG nodes that refer to this node */ - unsigned _directRefCnt; - - //these members are filled-in in the doSecondRefAIGPass() function - - /** True if occurs in toplevel AIGs with {negative,positive} polarity */ - bool _inPol[2]; - /** True if occurs in quantifier AIG nodes with {negative,positive} polarity */ - bool _inQuant[2]; - - /** - * How many times will an AIG node appear in formulas - * after conversion of AIG to formulas - * - * Equals to 1 if node has a name - */ - unsigned _formRefCnt; - }; - - struct DefIntroRewriter; - - //options - bool _eprPreserving; - unsigned _namingRefCntThreshold; - unsigned _mergeEquivDefs; - - AIGFormulaSharer _fsh; - - typedef Stack TopLevelStack; - TopLevelStack _toplevelAIGs; - - /** - * All positive AIG refs used in the problem, ordered topologically, - * so that references go only toward the bottom of the stack. - */ - AIGStack _refAIGs; - /** stack of infos corresponding to nodes at corresponding _refAIGs positions */ - Stack _refAIGInfos; - /** Indexes of AIG refs in the _refAIGs stack */ - DHMap _aigIndexes; - - DHMap _defs; - DHMap _defUnits; - -// DHMap _equivs; - - Stack _newDefs; - - AIGRef getPreNamingAig(unsigned aigStackIdx); - - bool shouldIntroduceName(unsigned aigStackIdx); - Literal* getNameLiteral(unsigned aigStackIdx); - void introduceName(unsigned aigStackIdx); - - bool scanDefinition(FormulaUnit* def); - - //functions called from scan() - void collectTopLevelAIGsAndDefs(UnitList* units); - void processTopLevelAIGs(); - - //functions called from processTopLevelAIGs() - void doFirstRefAIGPass(); - void doSecondRefAIGPass(); - - VarSet* getAtomVars(Literal* l); - NodeInfo& getNodeInfo(AIGRef r); - -public: - AIGDefinitionIntroducer(unsigned threshold=4); - - virtual void scan(UnitList* units); - - using ScanAndApplyFormulaUnitTransformer::apply; - virtual bool apply(FormulaUnit* unit, Unit*& res); - - virtual UnitList* getIntroducedFormulas(); - - Formula* apply(Formula* f, UnitList*& introducedDefs); -protected: - virtual void updateModifiedProblem(Problem& prb) {} -}; - - } diff --git a/Shell/Preprocess.cpp b/Shell/Preprocess.cpp index e464536809..67bda19c04 100644 --- a/Shell/Preprocess.cpp +++ b/Shell/Preprocess.cpp @@ -15,6 +15,7 @@ #include "AIG.hpp" #include "AIGCompressor.hpp" +#include "AIGDefinitionIntroducer.hpp" #include "AIGInliner.hpp" #include "AnswerExtractor.hpp" #include "CNF.hpp"