From: Tianyi Liang Date: Fri, 7 Mar 2014 03:27:37 +0000 (-0600) Subject: adds incremental for strings; clean-up codes X-Git-Tag: cvc5-1.0.0~7044^2 X-Git-Url: https://git.libre-soc.org/?a=commitdiff_plain;h=2301bcdf463cb7ee042d8b796c2cea4c9493fe16;p=cvc5.git adds incremental for strings; clean-up codes --- diff --git a/src/theory/strings/regexp_operation.cpp b/src/theory/strings/regexp_operation.cpp index 573aabe81..9d5c1c7e4 100644 --- a/src/theory/strings/regexp_operation.cpp +++ b/src/theory/strings/regexp_operation.cpp @@ -546,102 +546,6 @@ Node RegExpOpr::mkAllExceptOne( char exp_c ) { return NodeManager::currentNM()->mkNode( kind::REGEXP_UNION, vec_nodes ); } -Node RegExpOpr::complement( Node r ) { - Trace("strings-regexp-compl") << "RegExp-Compl starts with " << mkString( r ) << std::endl; - Node retNode = r; - if( d_compl_cache.find( r ) != d_compl_cache.end() ) { - retNode = d_compl_cache[r]; - } else { - int k = r.getKind(); - switch( k ) { - case kind::STRING_TO_REGEXP: - { - if(r[0].isConst()) { - if(r[0] == d_emptyString) { - retNode = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, d_sigma, d_sigma_star ); - } else { - std::vector< Node > vec_nodes; - vec_nodes.push_back( NodeManager::currentNM()->mkNode( kind::STRING_TO_REGEXP, d_emptyString ) ); - CVC4::String s = r[0].getConst(); - for(unsigned i=0; imkNode( kind::STRING_TO_REGEXP, - NodeManager::currentNM()->mkConst( s.substr(0, i) ) ); - tmp = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, tmph, tmp ); - } - tmp = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, tmp, d_sigma_star ); - vec_nodes.push_back( tmp ); - } - Node tmp = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, r, d_sigma, d_sigma_star ); - vec_nodes.push_back( tmp ); - retNode = NodeManager::currentNM()->mkNode( kind::REGEXP_UNION, vec_nodes ); - } - } else { - Trace("strings-error") << "Unsupported string variable: " << r[0] << " in complement of RegExp." << std::endl; - //AlwaysAssert( false ); - //return Node::null(); - } - } - break; - case kind::REGEXP_CONCAT: - { - std::vector< Node > vec_nodes; - for(unsigned i=0; imkNode( kind::REGEXP_CONCAT, r[j], tmp ); - } - if(i != r.getNumChildren() - 1) { - tmp = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, tmp, - NodeManager::currentNM()->mkNode( kind::REGEXP_STAR, d_sigma ) ); - } - vec_nodes.push_back( tmp ); - } - retNode = NodeManager::currentNM()->mkNode( kind::REGEXP_UNION, vec_nodes ); - } - break; - case kind::REGEXP_UNION: - { - std::vector< Node > vec_nodes; - for(unsigned i=0; imkNode( kind::REGEXP_INTER, vec_nodes ); - } - break; - case kind::REGEXP_INTER: - { - std::vector< Node > vec_nodes; - for(unsigned i=0; imkNode( kind::REGEXP_UNION, vec_nodes ); - } - break; - case kind::REGEXP_STAR: - { - retNode = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, d_sigma, d_sigma_star ); - Node tmp = complement( r[0] ); - tmp = NodeManager::currentNM()->mkNode( kind::REGEXP_CONCAT, r, tmp ); - retNode = NodeManager::currentNM()->mkNode( kind::REGEXP_INTER, retNode, tmp ); - } - break; - default: - //TODO: special sym: sigma, none, all - Trace("strings-error") << "Unsupported term: " << mkString( r ) << " in complement of RegExp." << std::endl; - //AlwaysAssert( false ); - //return Node::null(); - } - d_compl_cache[r] = retNode; - } - Trace("strings-regexp-compl") << "RegExp-Compl returns : " << mkString( retNode ) << std::endl; - return retNode; -} - //simplify void RegExpOpr::simplify(Node t, std::vector< Node > &new_nodes, bool polarity) { Trace("strings-regexp-simpl") << "RegExp-Simpl starts with " << t << ", polarity=" << polarity << std::endl; @@ -772,17 +676,10 @@ void RegExpOpr::simplifyNRegExp( Node s, Node r, std::vector< Node > &new_nodes //internal Node s1 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, s, d_zero, b1); Node s2 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, s, b1, NodeManager::currentNM()->mkNode(kind::MINUS, lens, b1)); - //Node s1 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); - //Node s2 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); - //Node b2v = NodeManager::currentNM()->mkNode(kind::BOUND_VAR_LIST, s1, s2); - //Node s12 = s.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s1, s2)); - //Node s1len = b1.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, s1)); Node s1r1 = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s1, r[0]).negate(); Node s2r2 = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s2, r).negate(); conc = NodeManager::currentNM()->mkNode(kind::OR, s1r1, s2r2); - //conc = NodeManager::currentNM()->mkNode(kind::AND, s12, s1len, conc); - //conc = NodeManager::currentNM()->mkNode(kind::EXISTS, b2v, conc); conc = NodeManager::currentNM()->mkNode(kind::IMPLIES, g1, conc); conc = NodeManager::currentNM()->mkNode(kind::FORALL, b1v, conc); conc = NodeManager::currentNM()->mkNode(kind::AND, sne, conc); diff --git a/src/theory/strings/regexp_operation.h b/src/theory/strings/regexp_operation.h index f9ae0a0ca..7b41c1764 100644 --- a/src/theory/strings/regexp_operation.h +++ b/src/theory/strings/regexp_operation.h @@ -23,6 +23,7 @@ #include "util/hash.h" #include "theory/theory.h" #include "theory/rewriter.h" +//#include "context/cdhashmap.h" namespace CVC4 { namespace theory { @@ -30,6 +31,7 @@ namespace strings { class RegExpOpr { typedef std::pair< Node, CVC4::String > PairDvStr; + private: Node d_emptyString; Node d_true; @@ -58,9 +60,9 @@ private: public: RegExpOpr(); + bool checkConstRegExp( Node r ); void simplify(Node t, std::vector< Node > &new_nodes, bool polarity); - Node complement( Node r ); int delta( Node r ); Node derivativeSingle( Node r, CVC4::String c ); bool guessLength( Node r, int &co ); diff --git a/src/theory/strings/theory_strings.cpp b/src/theory/strings/theory_strings.cpp index 072d28f9d..f819d46fb 100644 --- a/src/theory/strings/theory_strings.cpp +++ b/src/theory/strings/theory_strings.cpp @@ -2,7 +2,7 @@ /*! \file theory_strings.cpp ** \verbatim ** Original author: Tianyi Liang - ** Major contributors: none + ** Major contributors: Andrew Reynolds ** Minor contributors (to current version): Morgan Deters ** This file is part of the CVC4 project. ** Copyright (c) 2009-2013 New York University and The University of Iowa @@ -37,18 +37,27 @@ TheoryStrings::TheoryStrings(context::Context* c, context::UserContext* u, Outpu : Theory(THEORY_STRINGS, c, u, out, valuation, logicInfo), d_notify( *this ), d_equalityEngine(d_notify, c, "theory::strings::TheoryStrings"), - d_conflict( c, false ), + d_conflict(c, false), d_infer(c), d_infer_exp(c), d_nf_pairs(c), + d_loop_antec(u), + d_length_inst(u), d_length_nodes(c), - //d_var_lmin( c ), - //d_var_lmax( c ), - d_str_pos_ctn( c ), - d_str_neg_ctn( c ), - d_reg_exp_mem( c ), - d_regexp_deriv_processed( c ), - d_curr_cardinality( c, 0 ) + d_str_pos_ctn(c), + d_str_neg_ctn(c), + d_neg_ctn_eqlen(u), + d_neg_ctn_ulen(u), + d_pos_ctn_cached(u), + d_neg_ctn_cached(u), + d_reg_exp_mem(c), + d_regexp_ucached(u), + d_regexp_ccached(c), + d_regexp_ant(c), + d_input_vars(u), + d_input_var_lsum(u), + d_cardinality_lits(u), + d_curr_cardinality(c, 0) { // The kinds we are treating as function application in congruence //d_equalityEngine.addFunctionKind(kind::STRING_IN_REGEXP); @@ -67,9 +76,7 @@ TheoryStrings::TheoryStrings(context::Context* c, context::UserContext* u, Outpu d_true = NodeManager::currentNM()->mkConst( true ); d_false = NodeManager::currentNM()->mkConst( false ); - d_all_warning = true; - d_regexp_incomplete = false; - d_opt_regexp_gcd = true; + //d_opt_regexp_gcd = true; } TheoryStrings::~TheoryStrings() { @@ -184,7 +191,6 @@ bool TheoryStrings::propagate(TNode literal) { Debug("strings-propagate") << "TheoryStrings::propagate(" << literal << "): already in conflict" << std::endl; return false; } - Trace("strings-prop") << "strPropagate " << literal << std::endl; // Propagate out bool ok = d_out->propagate(literal); if (!ok) { @@ -235,10 +241,7 @@ Node TheoryStrings::explain( TNode literal ){ void TheoryStrings::presolve() { Trace("strings-presolve") << "TheoryStrings::Presolving : get fmf options " << (options::stringFMF() ? "true" : "false") << std::endl; - Trace("strings-presolve") << "TheoryStrings::Presolving : get unroll depth options " << options::stringRegExpUnrollDepth() << std::endl; d_opt_fmf = options::stringFMF(); - d_regexp_max_depth = options::stringRegExpUnrollDepth(); - d_regexp_unroll_depth = options::stringRegExpUnrollDepth(); } @@ -400,15 +403,14 @@ void TheoryStrings::preRegisterTerm(TNode n) { Debug("strings-prereg") << "TheoryStrings::preRegisterTerm() " << n << endl; //collectTerms( n ); switch (n.getKind()) { - case kind::EQUAL: - d_equalityEngine.addTriggerEquality(n); - break; - case kind::STRING_IN_REGEXP: - //do not add trigger here - //d_equalityEngine.addTriggerPredicate(n); - break; - case kind::STRING_SUBSTR_TOTAL: - { + case kind::EQUAL: + d_equalityEngine.addTriggerEquality(n); + break; + case kind::STRING_IN_REGEXP: + //do not add trigger here + //d_equalityEngine.addTriggerPredicate(n); + break; + case kind::STRING_SUBSTR_TOTAL: { Node lenxgti = NodeManager::currentNM()->mkNode( kind::GEQ, NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, n[0] ), NodeManager::currentNM()->mkNode( kind::PLUS, n[1], n[2] ) ); @@ -421,40 +423,36 @@ void TheoryStrings::preRegisterTerm(TNode n) { Node x_eq_123 = n[0].eqNode( NodeManager::currentNM()->mkNode( kind::STRING_CONCAT, sk1, n, sk3 ) ); Node len_sk1_eq_i = n[1].eqNode( NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, sk1 ) ); Node lenc = n[2].eqNode( NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, n ) ); - Node lemma = Rewriter::rewrite( NodeManager::currentNM()->mkNode( kind::ITE, cond, + Node lemma = Rewriter::rewrite(NodeManager::currentNM()->mkNode(kind::ITE, cond, NodeManager::currentNM()->mkNode( kind::AND, x_eq_123, len_sk1_eq_i, lenc ), - n.eqNode(NodeManager::currentNM()->mkConst( ::CVC4::String("") )) ) ); + n.eqNode(d_emptyString))); Trace("strings-lemma") << "Strings::Lemma SUBSTR : " << lemma << std::endl; d_out->lemma(lemma); + //d_equalityEngine.addTerm(n); } - //d_equalityEngine.addTerm(n); - default: - if(n.getType().isString() && n.getKind()!=kind::STRING_CONCAT && n.getKind()!=kind::CONST_STRING ) { - if( std::find( d_length_intro_vars.begin(), d_length_intro_vars.end(), n )==d_length_intro_vars.end() ) { + default: { + if(n.getType().isString() && n.getKind()!=kind::STRING_CONCAT && n.getKind()!=kind::CONST_STRING ) { Node n_len = NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, n); - Node n_len_eq_z = n_len.eqNode( d_zero ); + Node n_len_eq_z = n.eqNode(d_emptyString); //n_len.eqNode( d_zero ); n_len_eq_z = Rewriter::rewrite( n_len_eq_z ); Node n_len_geq_zero = NodeManager::currentNM()->mkNode( kind::OR, n_len_eq_z, NodeManager::currentNM()->mkNode( kind::GT, n_len, d_zero) ); Trace("strings-lemma") << "Strings::Lemma LENGTH >= 0 : " << n_len_geq_zero << std::endl; d_out->lemma(n_len_geq_zero); d_out->requirePhase( n_len_eq_z, true ); - } - // FMF - if( n.getKind() == kind::VARIABLE ) {//options::stringFMF() && - if( std::find(d_in_vars.begin(), d_in_vars.end(), n) == d_in_vars.end() ) { - d_in_vars.push_back( n ); + // FMF + if( n.getKind() == kind::VARIABLE ) {//options::stringFMF() && + d_input_vars.insert(n); } - } - } - if (n.getType().isBoolean()) { - // Get triggered for both equal and dis-equal - d_equalityEngine.addTriggerPredicate(n); - } else { - // Function applications/predicates - d_equalityEngine.addTerm(n); - } - break; + } + if (n.getType().isBoolean()) { + // Get triggered for both equal and dis-equal + d_equalityEngine.addTriggerPredicate(n); + } else { + // Function applications/predicates + d_equalityEngine.addTerm(n); + } + } } } @@ -488,6 +486,7 @@ void TheoryStrings::check(Effort e) { //must record string in regular expressions if ( atom.getKind() == kind::STRING_IN_REGEXP ) { d_reg_exp_mem.push_back( assertion ); + //d_equalityEngine.assertPredicate(atom, polarity, fact); } else if (atom.getKind() == kind::STRING_STRCTN) { if(polarity) { d_str_pos_ctn.push_back( atom ); @@ -968,7 +967,7 @@ bool TheoryStrings::processLoop(std::vector< Node > &antec, } Node ant = mkExplain( antec ); if(d_loop_antec.find(ant) == d_loop_antec.end()) { - d_loop_antec[ant] = true; + d_loop_antec.insert(ant); Node str_in_re; if( s_zy == t_yz && @@ -1013,12 +1012,10 @@ bool TheoryStrings::processLoop(std::vector< Node > &antec, conc = NodeManager::currentNM()->mkNode( kind::AND, vec_conc );//, len_x_gt_len_y } // normal case - if( !options::stringExp() ) { - //set its antecedant to ant, to say when it is relevant - d_reg_exp_ant[str_in_re] = ant; - //unroll the str in re constraint once - unrollStar( str_in_re ); - } + //set its antecedant to ant, to say when it is relevant + d_regexp_ant[str_in_re] = ant; + //unroll the str in re constraint once + // unrollStar( str_in_re ); sendLemma( ant, conc, "LOOP-BREAK" ); ++(d_statistics.d_loop_lemmas); @@ -1459,7 +1456,12 @@ bool TheoryStrings::processDeq( Node ni, Node nj ) { std::vector< Node > antec_new_lits; antec.insert( antec.end(), d_normal_forms_exp[ni].begin(), d_normal_forms_exp[ni].end() ); antec.insert( antec.end(), d_normal_forms_exp[nj].begin(), d_normal_forms_exp[nj].end() ); - antec.push_back( ni.eqNode( nj ).negate() ); + //check disequal + if( areDisequal( ni, nj ) ){ + antec.push_back( ni.eqNode( nj ).negate() ); + }else{ + antec_new_lits.push_back( ni.eqNode( nj ).negate() ); + } antec_new_lits.push_back( li.eqNode( lj ).negate() ); std::vector< Node > conc; Node sk1 = NodeManager::currentNM()->mkSkolem( "x_dsplit_$$", ni.getType(), "created for disequality normalization" ); @@ -1786,11 +1788,10 @@ bool TheoryStrings::checkSimple() { if( d_length_inst.find(n)==d_length_inst.end() ) { //Node nr = d_equalityEngine.getRepresentative( n ); //if( d_length_nodes.find(nr)==d_length_nodes.end() ) { - d_length_inst[n] = true; + d_length_inst.insert(n); Trace("strings-debug") << "get n: " << n << endl; Node sk = NodeManager::currentNM()->mkSkolem( "lsym_$$", n.getType(), "created for length" ); d_statistics.d_new_skolems += 1; - d_length_intro_vars.push_back( sk ); Node eq = NodeManager::currentNM()->mkNode( kind::EQUAL, sk, n ); eq = Rewriter::rewrite(eq); Trace("strings-lemma") << "Strings::Lemma LENGTH Term : " << eq << std::endl; @@ -1978,10 +1979,10 @@ void TheoryStrings::checkDeqNF() { for( unsigned j=0; jmkSkolem( "s_unroll_$$", x.getType(), "created for unrolling" ); - Node sk_xp= NodeManager::currentNM()->mkSkolem( "x_unroll_$$", x.getType(), "created for unrolling" ); - d_statistics.d_new_skolems += 2; - std::vector< Node >cc; - - // must also call preprocessing on unr1 - Node unr1 = Rewriter::rewrite( NodeManager::currentNM()->mkNode( kind::STRING_IN_REGEXP, sk_s, r[0] ) ); - if(unr1.getKind() == kind::EQUAL) { - sk_s = unr1[0] == sk_s ? unr1[1] : unr1[2]; - Node unr0 = sk_s.eqNode( d_emptyString ).negate(); - cc.push_back(unr0); - } else { - std::vector< Node > urc; - d_regexp_opr.simplify(unr1, urc, true); - Node unr0 = sk_s.eqNode( d_emptyString ).negate(); - cc.push_back(unr0); //cc.push_back(urc1); - cc.insert(cc.end(), urc.begin(), urc.end()); - } - - Node unr2 = x.eqNode( mkConcat( sk_s, sk_xp ) ); - Node unr3 = NodeManager::currentNM()->mkNode( kind::STRING_IN_REGEXP, sk_xp, r ); - d_reg_exp_unroll_depth[unr3] = depth + 1; - if( d_reg_exp_ant.find( atom )!=d_reg_exp_ant.end() ){ - d_reg_exp_ant[unr3] = d_reg_exp_ant[atom]; - } - - //|x|>|xp| - Node len_x_gt_len_xp = NodeManager::currentNM()->mkNode( kind::GT, - NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, x ), - NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, sk_xp ) ); - - cc.push_back(unr2); cc.push_back(unr3); cc.push_back(len_x_gt_len_xp); - - Node unr = NodeManager::currentNM()->mkNode( kind::AND, cc ); - Node lem = NodeManager::currentNM()->mkNode( kind::OR, xeqe, unr ); - Node ant = atom; - if( d_reg_exp_ant.find( atom )!=d_reg_exp_ant.end() ) { - ant = d_reg_exp_ant[atom]; - } - sendLemma( ant, lem, "Unroll" ); - ++(d_statistics.d_unroll_lemmas); - return true; - }else{ - Trace("strings-regexp") << "Strings::regexp: Stop unrolling " << atom << " the max (" << depth << ") is reached." << std::endl; - return false; +Node TheoryStrings::mkRegExpAntec(Node atom, Node ant) { + if(d_regexp_ant.find(atom) == d_regexp_ant.end()) { + return Rewriter::rewrite( NodeManager::currentNM()->mkNode(kind::AND, ant, atom) ); + } else { + Node n = d_regexp_ant[atom]; + return Rewriter::rewrite( NodeManager::currentNM()->mkNode(kind::AND, ant, n) ); } } bool TheoryStrings::checkMemberships() { - bool is_unk = false; bool addedLemma = false; + std::vector< Node > processed; + std::vector< Node > cprocessed; for( unsigned i=0; imkNode(kind::AND, antec, atom); - Node conc = Node::null(); - sendLemma(antec, conc, "RegExp Delta Conflict"); - addedLemma = true; - flag = false; - d_regexp_deriv_processed[atom] = true; - } - } else if(splitRegExp( x, r, atom )) { - addedLemma = true; flag = false; - d_regexp_deriv_processed[ atom ] = true; - } - } + flag = checkPDerivative(x, r, atom, addedLemma, processed, cprocessed); } else { - //TODO: will be removed soon. keep it for consistence if(! options::stringExp()) { - is_unk = true; - break; + throw LogicException("Strings Incomplete (due to Negative Membership) by default, try --strings-exp option."); } } if(flag) { - std::vector< Node > nvec; - d_regexp_opr.simplify(atom, nvec, polarity); - Node conc = nvec.size()==1 ? nvec[0] : - NodeManager::currentNM()->mkNode(kind::AND, nvec); - conc = Rewriter::rewrite(conc); - sendLemma( assertion, conc, "REGEXP" ); - addedLemma = true; - d_regexp_cache[assertion] = true; + //check if the term is atomic + Node xr = getRepresentative( x ); + Trace("strings-regexp") << xr << " is rep of " << x << std::endl; + Assert( d_normal_forms.find( xr )!=d_normal_forms.end() ); + //TODO + if( true || r.getKind()!=kind::REGEXP_STAR || ( d_normal_forms[xr].size()==1 && x.getKind()!=kind::STRING_CONCAT ) ){ + Trace("strings-regexp") << "Unroll/simplify membership of atomic term " << xr << std::endl; + //if so, do simple unrolling + std::vector< Node > nvec; + d_regexp_opr.simplify(atom, nvec, polarity); + Node antec = assertion; + if(d_regexp_ant.find(assertion) != d_regexp_ant.end()) { + antec = d_regexp_ant[assertion]; + for(std::vector< Node >::const_iterator itr=nvec.begin(); itrgetKind() == kind::STRING_IN_REGEXP) { + if(d_regexp_ant.find( *itr ) == d_regexp_ant.end()) { + d_regexp_ant[ *itr ] = antec; + } + } + } + } + Node conc = nvec.size()==1 ? nvec[0] : + NodeManager::currentNM()->mkNode(kind::AND, nvec); + conc = Rewriter::rewrite(conc); + sendLemma( antec, conc, "REGEXP" ); + addedLemma = true; + processed.push_back( assertion ); + //d_regexp_ucached[assertion] = true; + } else { + Trace("strings-regexp") << "Unroll/simplify membership of non-atomic term " << xr << " = "; + for( unsigned j=0; j antec; + std::vector< Node > antecn; + antec.insert( antec.begin(), d_normal_forms_exp[xr].begin(), d_normal_forms_exp[xr].end() ); + if( x!=xr ){ + antec.push_back( x.eqNode( xr ) ); + } + antecn.push_back( assertion ); + Node ant = mkExplain( antec, antecn ); + Trace("strings-regexp-debug") << "Construct conclusion..." << std::endl; + Node conc; + if( polarity ){ + if( d_normal_forms[xr].size()==0 ){ + conc = d_true; + }else if( d_normal_forms[xr].size()==1 ){ + Trace("strings-regexp-debug") << "Case 1\n"; + conc = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, d_normal_forms[xr][0], r); + }else{ + Trace("strings-regexp-debug") << "Case 2\n"; + std::vector< Node > conc_c; + Node s11 = NodeManager::currentNM()->mkSkolem( "s11_$$", NodeManager::currentNM()->stringType(), "created for re" ); + Node s12 = NodeManager::currentNM()->mkSkolem( "s12_$$", NodeManager::currentNM()->stringType(), "created for re" ); + Node s21 = NodeManager::currentNM()->mkSkolem( "s21_$$", NodeManager::currentNM()->stringType(), "created for re" ); + Node s22 = NodeManager::currentNM()->mkSkolem( "s22_$$", NodeManager::currentNM()->stringType(), "created for re" ); + conc = p1.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s11, s12)); + conc_c.push_back(conc); + conc = p2.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s21, s22)); + conc_c.push_back(conc); + conc = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s11, r); + conc_c.push_back(conc); + conc = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s12, s21), r[0]); + conc_c.push_back(conc); + conc = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s22, r); + conc_c.push_back(conc); + conc = Rewriter::rewrite(NodeManager::currentNM()->mkNode(kind::AND, conc_c)); + Node eqz = Rewriter::rewrite(x.eqNode(d_emptyString)); + conc = NodeManager::currentNM()->mkNode(kind::OR, eqz, conc); + d_pending_req_phase[eqz] = true; + } + }else{ + if( d_normal_forms[xr].size()==0 ){ + conc = d_false; + }else if( d_normal_forms[xr].size()==1 ){ + Trace("strings-regexp-debug") << "Case 3\n"; + conc = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, d_normal_forms[xr][0], r).negate(); + }else{ + Trace("strings-regexp-debug") << "Case 4\n"; + Node len1 = NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, p1); + Node len2 = NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, p2); + Node bi = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->integerType()); + Node bj = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->integerType()); + Node b1v = NodeManager::currentNM()->mkNode(kind::BOUND_VAR_LIST, bi, bj); + Node g1 = NodeManager::currentNM()->mkNode(kind::AND, + NodeManager::currentNM()->mkNode(kind::GEQ, bi, d_zero), + NodeManager::currentNM()->mkNode(kind::GEQ, len1, bi), + NodeManager::currentNM()->mkNode(kind::GEQ, bj, d_zero), + NodeManager::currentNM()->mkNode(kind::GEQ, len2, bj)); + Node s11 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, p1, d_zero, bi); + Node s12 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, p1, bi, NodeManager::currentNM()->mkNode(kind::MINUS, len1, bi)); + Node s21 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, p2, d_zero, bj); + Node s22 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, p2, bj, NodeManager::currentNM()->mkNode(kind::MINUS, len2, bj)); + Node cc1 = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s11, r).negate(); + Node cc2 = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s12, s21), r[0]).negate(); + Node cc3 = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, s22, r).negate(); + conc = NodeManager::currentNM()->mkNode(kind::OR, cc1, cc2, cc3); + conc = NodeManager::currentNM()->mkNode(kind::IMPLIES, g1, conc); + conc = NodeManager::currentNM()->mkNode(kind::FORALL, b1v, conc); + conc = NodeManager::currentNM()->mkNode(kind::AND, x.eqNode(d_emptyString).negate(), conc); + } + } + if( conc!=d_true ){ + ant = mkRegExpAntec(assertion, ant); + sendLemma(ant, conc, "REGEXP CSTAR"); + addedLemma = true; + if( conc==d_false ){ + d_regexp_ccached.insert( assertion ); + }else{ + cprocessed.push_back( assertion ); + } + }else{ + d_regexp_ccached.insert(assertion); + } + } } } if(d_conflict) { @@ -2341,97 +2382,67 @@ bool TheoryStrings::checkMemberships() { } } if( addedLemma ){ + if( !d_conflict ){ + for( unsigned i=0; isetIncomplete(); - } return false; } } -//TODO remove -bool TheoryStrings::checkMemberships2() { - bool is_unk = false; - bool addedLemma = false; - for( unsigned i=0; i &processed, std::vector< Node > &cprocessed) { + /*if(d_opt_regexp_gcd) { + if(d_membership_length.find(atom) == d_membership_length.end()) { + addedLemma = addMembershipLength(atom); + d_membership_length[atom] = true; } else { - //TODO: negative membership - Node x = atom[0]; - Node r = atom[1]; - Assert( r.getKind()==kind::REGEXP_STAR ); - if( areEqual( x, d_emptyString ) ) { - Node ant = NodeManager::currentNM()->mkNode( kind::AND, assertion, x.eqNode( d_emptyString ) ); + Trace("strings-regexp") << "Membership length is already added." << std::endl; + } + }*/ + if(areEqual(x, d_emptyString)) { + int rdel = d_regexp_opr.delta(r); + if(rdel == 1) { + d_regexp_ccached.insert(atom); + } else if(rdel == 2) { + Node antec = mkRegExpAntec(atom, x.eqNode(d_emptyString)); + Node conc = Node::null(); + sendLemma(antec, conc, "RegExp Delta CONFLICT"); + addedLemma = true; + d_regexp_ccached.insert(atom); + return false; + } + } else { + Node xr = getRepresentative( x ); + if(x != xr) { + Node n = NodeManager::currentNM()->mkNode(kind::STRING_IN_REGEXP, xr, r); + Node nn = Rewriter::rewrite( n ); + if(nn == d_true) { + d_regexp_ccached.insert(atom); + return false; + } else if(nn == d_false) { + Node antec = mkRegExpAntec(atom, x.eqNode(xr)); Node conc = Node::null(); - sendLemma( ant, conc, "RegExp Empty Conflict" ); + sendLemma(antec, conc, "RegExp Delta CONFLICT"); addedLemma = true; - } else { - Trace("strings-regexp") << "RegEx is incomplete due to " << assertion << "." << std::endl; - is_unk = true; + d_regexp_ccached.insert(atom); + return false; } } - } - if( addedLemma ){ - doPendingLemmas(); - return true; - }else{ - if( is_unk ){ - //if(!d_opt_fmf) { - // d_opt_fmf = true; - //d_regexp_unroll_depth += 2; - // Node t = getNextDecisionRequest(); - // return !t.isNull(); - //} else { - Trace("strings-regexp") << "Strings::regexp: possibly incomplete." << std::endl; - //d_regexp_incomplete = true; - d_out->setIncomplete(); - //} + Node sREant = mkRegExpAntec(atom, d_true); + if(splitRegExp( x, r, sREant )) { + addedLemma = true; + processed.push_back( atom ); + return false; } - return false; } + return true; } bool TheoryStrings::checkContains() { @@ -2454,14 +2465,14 @@ bool TheoryStrings::checkPosContains() { Node x = atom[0]; Node s = atom[1]; if( !areEqual( s, d_emptyString ) && !areEqual( s, x ) ) { - if(d_str_pos_ctn_rewritten.find(atom) == d_str_pos_ctn_rewritten.end()) { + if(d_pos_ctn_cached.find(atom) == d_pos_ctn_cached.end()) { Node sk1 = NodeManager::currentNM()->mkSkolem( "sc1_$$", s.getType(), "created for contain" ); Node sk2 = NodeManager::currentNM()->mkSkolem( "sc2_$$", s.getType(), "created for contain" ); d_statistics.d_new_skolems += 2; Node eq = Rewriter::rewrite( x.eqNode( NodeManager::currentNM()->mkNode( kind::STRING_CONCAT, sk1, s, sk2 ) ) ); sendLemma( atom, eq, "POS-INC" ); addedLemma = true; - d_str_pos_ctn_rewritten[ atom ] = true; + d_pos_ctn_cached.insert( atom ); } else { Trace("strings-ctn") << "... is already rewritten." << std::endl; } @@ -2479,7 +2490,6 @@ bool TheoryStrings::checkPosContains() { } bool TheoryStrings::checkNegContains() { - bool is_unk = false; bool addedLemma = false; for( unsigned i=0; imkNode( kind::AND, atom.negate(), eq ) ); Node xneqs = x.eqNode(s).negate(); - d_str_ctn_eqlen[ atom ] = true; + d_neg_ctn_eqlen.insert( atom ); sendLemma( antc, xneqs, "NEG-CTN-EQL" ); addedLemma = true; } } else if(!areDisequal(lenx, lens)) { - Node s = lenx < lens ? lenx : lens; - Node eq = s.eqNode( lenx < lens ? lens : lenx ); - if(d_str_neg_ctn_ulen.find(eq) == d_str_neg_ctn_ulen.end()) { - d_str_neg_ctn_ulen[ eq ] = true; + if(d_neg_ctn_ulen.find(atom) == d_neg_ctn_ulen.end()) { + d_neg_ctn_ulen.insert( atom ); sendSplit(lenx, lens, "NEG-CTN-SP"); addedLemma = true; } } else { - if(d_str_neg_ctn_rewritten.find(atom) == d_str_neg_ctn_rewritten.end()) { + if(d_neg_ctn_cached.find(atom) == d_neg_ctn_cached.end()) { Node b1 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->integerType()); Node b1v = NodeManager::currentNM()->mkNode(kind::BOUND_VAR_LIST, b1); Node g1 = Rewriter::rewrite( NodeManager::currentNM()->mkNode( kind::AND, NodeManager::currentNM()->mkNode( kind::GEQ, b1, d_zero ), @@ -2528,10 +2536,6 @@ bool TheoryStrings::checkNegContains() { Node s2 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, x, NodeManager::currentNM()->mkNode( kind::PLUS, b1, b2 ), d_one); Node s5 = NodeManager::currentNM()->mkNode(kind::STRING_SUBSTR_TOTAL, s, b2, d_one); - //Node s1 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); - //Node s3 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); - //Node s4 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); - //Node s6 = NodeManager::currentNM()->mkBoundVar(NodeManager::currentNM()->stringType()); Node b2v = NodeManager::currentNM()->mkNode(kind::BOUND_VAR_LIST, b2);//, s1, s3, s4, s6); std::vector< Node > vec_nodes; @@ -2540,22 +2544,8 @@ bool TheoryStrings::checkNegContains() { cc = NodeManager::currentNM()->mkNode( kind::GEQ, lens, b2 ); vec_nodes.push_back(cc); - //cc = x.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s1, s2, s3)); - //vec_nodes.push_back(cc); - //cc = s.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_CONCAT, s4, s5, s6)); - //vec_nodes.push_back(cc); cc = s2.eqNode(s5).negate(); vec_nodes.push_back(cc); - //cc = NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, s1).eqNode(NodeManager::currentNM()->mkNode( kind::PLUS, b1, b2 )); - //vec_nodes.push_back(cc); - //cc = NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, s4).eqNode(b2); - //vec_nodes.push_back(cc); - - // charAt length - //cc = d_one.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, s2)); - //vec_nodes.push_back(cc); - //cc = d_one.eqNode(NodeManager::currentNM()->mkNode(kind::STRING_LENGTH, s5)); - //vec_nodes.push_back(cc); Node conc = Rewriter::rewrite( NodeManager::currentNM()->mkNode(kind::AND, vec_nodes) ); Node xlss = NodeManager::currentNM()->mkNode( kind::GT, lens, lenx ); @@ -2564,15 +2554,14 @@ bool TheoryStrings::checkNegContains() { conc = NodeManager::currentNM()->mkNode( kind::IMPLIES, g1, conc ); conc = NodeManager::currentNM()->mkNode( kind::FORALL, b1v, conc ); - d_str_neg_ctn_rewritten[ atom ] = true; + d_neg_ctn_cached.insert( atom ); sendLemma( atom.negate(), conc, "NEG-CTN-BRK" ); //d_pending_req_phase[xlss] = true; addedLemma = true; } } } else { - Trace("strings-ctn") << "Strings Incomplete (due to Negative Contain) by default." << std::endl; - is_unk = true; + throw LogicException("Strings Incomplete (due to Negative Contain) by default, try --strings-exp option."); } } } @@ -2581,10 +2570,6 @@ bool TheoryStrings::checkNegContains() { doPendingLemmas(); return true; } else { - if( is_unk ){ - Trace("strings-ctn") << "Strings::CTN: possibly incomplete." << std::endl; - d_out->setIncomplete(); - } return false; } } @@ -2628,6 +2613,14 @@ bool TheoryStrings::splitRegExp( Node x, Node r, Node ant ) { // TODO cstr in vre Assert(x != d_emptyString); Trace("strings-regexp-split") << "TheoryStrings::splitRegExp: x=" << x << ", r= " << r << std::endl; + //if(x.isConst()) { + // Node n = NodeManager::currentNM()->mkNode( kind::STRING_IN_REGEXP, x, r ); + // Node r = Rewriter::rewrite( n ); + // if(n != r) { + // sendLemma(ant, r, "REGEXP REWRITE"); + // return true; + // } + //} CVC4::String s = getHeadConst( x ); if( !s.isEmptyString() && d_regexp_opr.checkConstRegExp( r ) ) { Node conc = Node::null(); @@ -2677,27 +2670,30 @@ bool TheoryStrings::splitRegExp( Node x, Node r, Node ant ) { Node TheoryStrings::getNextDecisionRequest() { if(d_opt_fmf && !d_conflict) { + Node in_var_lsum = d_input_var_lsum.get(); //Trace("strings-fmf-debug") << "Strings::FMF: Assertion Level = " << d_valuation.getAssertionLevel() << std::endl; //initialize the term we will minimize - if( d_in_var_lsum.isNull() && !d_in_vars.empty() ){ + if( in_var_lsum.isNull() && !d_input_vars.empty() ){ Trace("strings-fmf-debug") << "Input variables: "; std::vector< Node > ll; - for(std::vector< Node >::iterator itr = d_in_vars.begin(); - itr != d_in_vars.end(); ++itr) { + for(NodeSet::const_iterator itr = d_input_vars.begin(); + itr != d_input_vars.end(); ++itr) { Trace("strings-fmf-debug") << " " << (*itr) ; ll.push_back( NodeManager::currentNM()->mkNode( kind::STRING_LENGTH, *itr ) ); } Trace("strings-fmf-debug") << std::endl; - d_in_var_lsum = ll.size()==1 ? ll[0] : NodeManager::currentNM()->mkNode( kind::PLUS, ll ); - d_in_var_lsum = Rewriter::rewrite( d_in_var_lsum ); + in_var_lsum = ll.size()==1 ? ll[0] : NodeManager::currentNM()->mkNode( kind::PLUS, ll ); + in_var_lsum = Rewriter::rewrite( in_var_lsum ); + d_input_var_lsum.set( in_var_lsum ); } - if( !d_in_var_lsum.isNull() ){ + if( !in_var_lsum.isNull() ){ //Trace("strings-fmf") << "Get next decision request." << std::endl; //check if we need to decide on something int decideCard = d_curr_cardinality.get(); if( d_cardinality_lits.find( decideCard )!=d_cardinality_lits.end() ){ bool value; - if( d_valuation.hasSatValue( d_cardinality_lits[ d_curr_cardinality.get() ], value ) ) { + Node cnode = d_cardinality_lits[ d_curr_cardinality.get() ]; + if( d_valuation.hasSatValue( cnode, value ) ) { if( !value ){ d_curr_cardinality.set( d_curr_cardinality.get() + 1 ); decideCard = d_curr_cardinality.get(); @@ -2712,7 +2708,7 @@ Node TheoryStrings::getNextDecisionRequest() { } if( decideCard!=-1 ){ if( d_cardinality_lits.find( decideCard )==d_cardinality_lits.end() ){ - Node lit = NodeManager::currentNM()->mkNode( kind::LEQ, d_in_var_lsum, NodeManager::currentNM()->mkConst( Rational( decideCard ) ) ); + Node lit = NodeManager::currentNM()->mkNode( kind::LEQ, in_var_lsum, NodeManager::currentNM()->mkConst( Rational( decideCard ) ) ); lit = Rewriter::rewrite( lit ); d_cardinality_lits[decideCard] = lit; Node lem = NodeManager::currentNM()->mkNode( kind::OR, lit, lit.negate() ); @@ -2720,8 +2716,9 @@ Node TheoryStrings::getNextDecisionRequest() { d_out->lemma( lem ); d_out->requirePhase( lit, true ); } - Trace("strings-fmf") << "Strings::FMF: Decide positive on " << d_cardinality_lits[ decideCard ] << std::endl; - return d_cardinality_lits[ decideCard ]; + Node lit = d_cardinality_lits[ decideCard ]; + Trace("strings-fmf") << "Strings::FMF: Decide positive on " << lit << std::endl; + return lit; } } } @@ -2738,51 +2735,25 @@ Node TheoryStrings::mkSplitEq( const char * c, const char * info, Node lhs, Node Node eq = lhs.eqNode( mkConcat( rhs, sk ) ); eq = Rewriter::rewrite( eq ); if( lgtZero ) { - d_var_lgtz[sk] = true; Node sk_gt_zero = NodeManager::currentNM()->mkNode( kind::EQUAL, sk, d_emptyString).negate(); Trace("strings-lemma") << "Strings::Lemma SK-NON-EMPTY: " << sk_gt_zero << std::endl; d_lemma_cache.push_back( sk_gt_zero ); } - //store it in proper map - if( options::stringFMF() ){ - d_var_split_graph_lhs[sk] = lhs; - d_var_split_graph_rhs[sk] = rhs; - //d_var_split_eq[sk] = eq; - - //int mpl = getMaxPossibleLength( sk ); - Trace("strings-progress") << "Strings::Progress: Because of " << eq << std::endl; - Trace("strings-progress") << "Strings::Progress: \t" << lhs << "(up:" << getMaxPossibleLength(lhs) << ") is removed" << std::endl; - Trace("strings-progress") << "Strings::Progress: \t" << sk << "(up:" << getMaxPossibleLength(sk) << ") is added" << std::endl; - } return eq; } -int TheoryStrings::getMaxPossibleLength( Node x ) { - if( d_var_split_graph_lhs.find( x )==d_var_split_graph_lhs.end() ){ - if( x.getKind()==kind::CONST_STRING ){ - return x.getConst().size(); - }else{ - return d_curr_cardinality.get(); - } - }else{ - return getMaxPossibleLength( d_var_split_graph_lhs[x] ) - 1; - } -} - // Stats TheoryStrings::Statistics::Statistics(): d_splits("TheoryStrings::NumOfSplitOnDemands", 0), d_eq_splits("TheoryStrings::NumOfEqSplits", 0), d_deq_splits("TheoryStrings::NumOfDiseqSplits", 0), d_loop_lemmas("TheoryStrings::NumOfLoops", 0), - d_unroll_lemmas("TheoryStrings::NumOfUnrolls", 0), d_new_skolems("TheoryStrings::NumOfNewSkolems", 0) { StatisticsRegistry::registerStat(&d_splits); StatisticsRegistry::registerStat(&d_eq_splits); StatisticsRegistry::registerStat(&d_deq_splits); StatisticsRegistry::registerStat(&d_loop_lemmas); - StatisticsRegistry::registerStat(&d_unroll_lemmas); StatisticsRegistry::registerStat(&d_new_skolems); } @@ -2791,7 +2762,6 @@ TheoryStrings::Statistics::~Statistics(){ StatisticsRegistry::unregisterStat(&d_eq_splits); StatisticsRegistry::unregisterStat(&d_deq_splits); StatisticsRegistry::unregisterStat(&d_loop_lemmas); - StatisticsRegistry::unregisterStat(&d_unroll_lemmas); StatisticsRegistry::unregisterStat(&d_new_skolems); } diff --git a/src/theory/strings/theory_strings.h b/src/theory/strings/theory_strings.h index bf8a3d894..cbfa481c3 100644 --- a/src/theory/strings/theory_strings.h +++ b/src/theory/strings/theory_strings.h @@ -25,6 +25,7 @@ #include "theory/strings/regexp_operation.h" #include "context/cdchunk_list.h" +#include "context/cdhashset.h" namespace CVC4 { namespace theory { @@ -40,6 +41,8 @@ class TheoryStrings : public Theory { typedef context::CDHashMap NodeListMap; typedef context::CDHashMap NodeBoolMap; typedef context::CDHashMap NodeIntMap; + typedef context::CDHashMap NodeNodeMap; + typedef context::CDHashSet NodeSet; public: TheoryStrings(context::Context* c, context::UserContext* u, OutputChannel& out, Valuation valuation, const LogicInfo& logicInfo); @@ -117,7 +120,6 @@ private: Node d_zero; Node d_one; // Options - bool d_all_warning; bool d_opt_fmf; bool d_opt_regexp_gcd; // Helper functions @@ -135,7 +137,6 @@ private: eq::EqualityEngine d_equalityEngine; /** Are we in conflict */ context::CDO d_conflict; - std::vector< Node > d_length_intro_vars; //list of pairs of nodes to merge std::map< Node, Node > d_pending_exp; std::vector< Node > d_pending; @@ -154,7 +155,7 @@ private: bool isNormalFormPair( Node n1, Node n2 ); bool isNormalFormPair2( Node n1, Node n2 ); // loop ant - std::map< Node, bool > d_loop_antec; + NodeSet d_loop_antec; ///////////////////////////////////////////////////////////////////////////// // MODEL GENERATION @@ -192,7 +193,7 @@ private: std::map< Node, EqcInfo* > d_eqc_info; EqcInfo * getOrMakeEqcInfo( Node eqc, bool doMake = true ); //maintain which concat terms have the length lemma instantiated - std::map< Node, bool > d_length_inst; + NodeSet d_length_inst; NodeBoolMap d_length_nodes; private: void mergeCstVec(std::vector< Node > &vec_strings); @@ -220,7 +221,8 @@ private: bool processDeq( Node n1, Node n2 ); int processReverseDeq( std::vector< Node >& nfi, std::vector< Node >& nfj, Node ni, Node nj ); int processSimpleDeq( std::vector< Node >& nfi, std::vector< Node >& nfj, Node ni, Node nj, unsigned& index, bool isRev ); - bool unrollStar( Node atom ); + //bool unrollStar( Node atom ); + Node mkRegExpAntec(Node atom, Node ant); bool checkSimple(); bool checkNormalForms(); @@ -229,7 +231,7 @@ private: bool checkCardinality(); bool checkInductiveEquations(); bool checkMemberships(); - bool checkMemberships2(); + bool checkPDerivative(Node x, Node r, Node atom, bool &addedLemma, std::vector< Node > &processed, std::vector< Node > &cprocessed); bool checkContains(); bool checkPosContains(); bool checkNegContains(); @@ -279,41 +281,27 @@ protected: void separateByLength( std::vector< Node >& n, std::vector< std::vector< Node > >& col, std::vector< Node >& lts ); void printConcat( std::vector< Node >& n, const char * c ); - // Measurement private: - //NodeIntMap d_var_lmin; - //NodeIntMap d_var_lmax; - std::map< Node, Node > d_var_split_graph_lhs; - std::map< Node, Node > d_var_split_graph_rhs; - std::map< Node, bool > d_var_lgtz; Node mkSplitEq( const char * c, const char * info, Node lhs, Node rhs, bool lgtZero ); - int getMaxPossibleLength( Node x ); // Special String Functions NodeList d_str_pos_ctn; NodeList d_str_neg_ctn; - std::map< Node, bool > d_str_ctn_eqlen; - std::map< Node, bool > d_str_neg_ctn_ulen; - std::map< Node, bool > d_str_pos_ctn_rewritten; - std::map< Node, bool > d_str_neg_ctn_rewritten; + NodeSet d_neg_ctn_eqlen; + NodeSet d_neg_ctn_ulen; + NodeSet d_pos_ctn_cached; + NodeSet d_neg_ctn_cached; // Regular Expression private: // regular expression memberships NodeList d_reg_exp_mem; - std::map< Node, bool > d_regexp_cache; - // antecedant for why reg exp membership must be true - std::map< Node, Node > d_reg_exp_ant; - std::map< Node, bool > d_reg_exp_unroll; - std::map< Node, int > d_reg_exp_unroll_depth; - bool d_regexp_incomplete; - int d_regexp_unroll_depth; - int d_regexp_max_depth; + NodeSet d_regexp_ucached; + NodeSet d_regexp_ccached; + // antecedant for why regexp membership must be true + NodeNodeMap d_regexp_ant; // membership length - std::map< Node, bool > d_membership_length; - // regular expression derivative - std::map< Node, bool > d_reg_exp_deriv; - NodeBoolMap d_regexp_deriv_processed; + //std::map< Node, bool > d_membership_length; // regular expression operations RegExpOpr d_regexp_opr; @@ -324,9 +312,9 @@ private: // Finite Model Finding private: - std::vector< Node > d_in_vars; - Node d_in_var_lsum; - std::map< int, Node > d_cardinality_lits; + NodeSet d_input_vars; + context::CDO< Node > d_input_var_lsum; + context::CDHashMap< int, Node > d_cardinality_lits; context::CDO< int > d_curr_cardinality; public: //for finite model finding @@ -341,7 +329,6 @@ public: IntStat d_eq_splits; IntStat d_deq_splits; IntStat d_loop_lemmas; - IntStat d_unroll_lemmas; IntStat d_new_skolems; Statistics(); ~Statistics(); diff --git a/src/theory/strings/theory_strings_preprocess.cpp b/src/theory/strings/theory_strings_preprocess.cpp index 15958def8..b2988d54a 100644 --- a/src/theory/strings/theory_strings_preprocess.cpp +++ b/src/theory/strings/theory_strings_preprocess.cpp @@ -24,10 +24,6 @@ namespace theory { namespace strings { StringsPreprocess::StringsPreprocess() { - std::vector< TypeNode > argTypes; - argTypes.push_back(NodeManager::currentNM()->stringType()); - argTypes.push_back(NodeManager::currentNM()->integerType()); - //Constants d_zero = NodeManager::currentNM()->mkConst( ::CVC4::Rational(0) ); } diff --git a/src/theory/theory_engine.cpp b/src/theory/theory_engine.cpp index 63024e5d5..89711ac02 100644 --- a/src/theory/theory_engine.cpp +++ b/src/theory/theory_engine.cpp @@ -418,6 +418,10 @@ void TheoryEngine::check(Theory::Effort effort) { } else if(options::produceModels()) { // must build model at this point d_curr_model_builder->buildModel(d_curr_model, true); + } + Trace("theory::assertions-model") << endl; + if (Trace.isOn("theory::assertions-model")) { + printAssertions("theory::assertions-model"); } }