1 /********************* */
2 /*! \file conjecture_generator.cpp
4 ** Original author: Andrew Reynolds
5 ** Major contributors: none
6 ** Minor contributors (to current version): none
7 ** This file is part of the CVC4 project.
8 ** Copyright (c) 2009-2014 New York University and The University of Iowa
9 ** See the file COPYING in the top-level source directory for licensing
10 ** information.\endverbatim
12 ** \brief conjecture generator class
16 #include "theory/quantifiers/conjecture_generator.h"
17 #include "theory/theory_engine.h"
18 #include "theory/quantifiers/options.h"
19 #include "theory/quantifiers/term_database.h"
20 #include "theory/quantifiers/trigger.h"
21 #include "theory/quantifiers/first_order_model.h"
24 using namespace CVC4::kind
;
25 using namespace CVC4::theory
;
26 using namespace CVC4::theory::quantifiers
;
31 struct sortConjectureScore
{
32 std::vector
< int > d_scores
;
33 bool operator() (unsigned i
, unsigned j
) { return d_scores
[i
]>d_scores
[j
]; }
37 void OpArgIndex::addTerm( ConjectureGenerator
* s
, TNode n
, unsigned index
){
38 if( index
==n
.getNumChildren() ){
39 Assert( n
.hasOperator() );
40 if( std::find( d_ops
.begin(), d_ops
.end(), n
.getOperator() )==d_ops
.end() ){
41 d_ops
.push_back( n
.getOperator() );
42 d_op_terms
.push_back( n
);
45 d_child
[s
->getTermDatabase()->d_arg_reps
[n
][index
]].addTerm( s
, n
, index
+1 );
49 Node
OpArgIndex::getGroundTerm( ConjectureGenerator
* s
, std::vector
< TNode
>& args
) {
51 for( std::map
< TNode
, OpArgIndex
>::iterator it
= d_child
.begin(); it
!= d_child
.end(); ++it
){
52 std::map
< TNode
, Node
>::iterator itf
= s
->d_ground_eqc_map
.find( it
->first
);
53 if( itf
!=s
->d_ground_eqc_map
.end() ){
54 args
.push_back( itf
->second
);
55 Node n
= it
->second
.getGroundTerm( s
, args
);
64 std::vector
< TNode
> args2
;
65 args2
.push_back( d_ops
[0] );
66 args2
.insert( args2
.end(), args
.begin(), args
.end() );
67 return NodeManager::currentNM()->mkNode( d_op_terms
[0].getKind(), args2
);
71 void OpArgIndex::getGroundTerms( ConjectureGenerator
* s
, std::vector
< TNode
>& terms
) {
72 terms
.insert( terms
.end(), d_op_terms
.begin(), d_op_terms
.end() );
73 for( std::map
< TNode
, OpArgIndex
>::iterator it
= d_child
.begin(); it
!= d_child
.end(); ++it
){
74 if( s
->isGroundEqc( it
->first
) ){
75 it
->second
.getGroundTerms( s
, terms
);
82 ConjectureGenerator::ConjectureGenerator( QuantifiersEngine
* qe
, context::Context
* c
) : QuantifiersModule( qe
),
84 d_uequalityEngine(d_notify
, c
, "ConjectureGenerator::ee", false),
85 d_ee_conjectures( c
){
86 d_fullEffortCount
= 0;
87 d_uequalityEngine
.addFunctionKind( kind::APPLY_UF
);
88 d_uequalityEngine
.addFunctionKind( kind::APPLY_CONSTRUCTOR
);
92 void ConjectureGenerator::eqNotifyNewClass( TNode t
){
93 Trace("thm-ee-debug") << "UEE : new equivalence class " << t
<< std::endl
;
94 d_upendingAdds
.push_back( t
);
97 void ConjectureGenerator::eqNotifyPreMerge(TNode t1
, TNode t2
) {
98 //get maintained representatives
101 std::map
< Node
, EqcInfo
* >::iterator it1
= d_eqc_info
.find( t1
);
102 if( it1
!=d_eqc_info
.end() && !it1
->second
->d_rep
.get().isNull() ){
103 rt1
= it1
->second
->d_rep
.get();
105 std::map
< Node
, EqcInfo
* >::iterator it2
= d_eqc_info
.find( t2
);
106 if( it2
!=d_eqc_info
.end() && !it2
->second
->d_rep
.get().isNull() ){
107 rt2
= it2
->second
->d_rep
.get();
109 Trace("thm-ee-debug") << "UEE : equality holds : " << t1
<< " == " << t2
<< std::endl
;
110 Trace("thm-ee-debug") << " ureps : " << rt1
<< " == " << rt2
<< std::endl
;
111 Trace("thm-ee-debug") << " relevant : " << d_pattern_is_relevant
[rt1
] << " " << d_pattern_is_relevant
[rt2
] << std::endl
;
112 Trace("thm-ee-debug") << " normal : " << d_pattern_is_normal
[rt1
] << " " << d_pattern_is_normal
[rt2
] << std::endl
;
113 Trace("thm-ee-debug") << " size : " << d_pattern_fun_sum
[rt1
] << " " << d_pattern_fun_sum
[rt2
] << std::endl
;
115 if( isUniversalLessThan( rt2
, rt1
) ){
117 if( it1
==d_eqc_info
.end() ){
118 ei
= getOrMakeEqcInfo( t1
, true );
126 void ConjectureGenerator::eqNotifyPostMerge(TNode t1
, TNode t2
) {
130 void ConjectureGenerator::eqNotifyDisequal(TNode t1
, TNode t2
, TNode reason
) {
131 Trace("thm-ee-debug") << "UEE : disequality holds : " << t1
<< " != " << t2
<< std::endl
;
136 ConjectureGenerator::EqcInfo::EqcInfo( context::Context
* c
) : d_rep( c
, Node::null() ){
140 ConjectureGenerator::EqcInfo
* ConjectureGenerator::getOrMakeEqcInfo( TNode n
, bool doMake
) {
141 //Assert( getUniversalRepresentative( n )==n );
142 std::map
< Node
, EqcInfo
* >::iterator eqc_i
= d_eqc_info
.find( n
);
143 if( eqc_i
!=d_eqc_info
.end() ){
144 return eqc_i
->second
;
146 EqcInfo
* ei
= new EqcInfo( d_quantEngine
->getSatContext() );
154 void ConjectureGenerator::setUniversalRelevant( TNode n
) {
155 //add pattern information
156 registerPattern( n
, n
.getType() );
157 d_urelevant_terms
[n
] = true;
158 for( unsigned i
=0; i
<n
.getNumChildren(); i
++ ){
159 setUniversalRelevant( n
[i
] );
163 bool ConjectureGenerator::isUniversalLessThan( TNode rt1
, TNode rt2
) {
164 //prefer the one that is (normal, smaller) lexographically
165 Assert( d_pattern_is_relevant
.find( rt1
)!=d_pattern_is_relevant
.end() );
166 Assert( d_pattern_is_relevant
.find( rt2
)!=d_pattern_is_relevant
.end() );
167 Assert( d_pattern_is_normal
.find( rt1
)!=d_pattern_is_normal
.end() );
168 Assert( d_pattern_is_normal
.find( rt2
)!=d_pattern_is_normal
.end() );
169 Assert( d_pattern_fun_sum
.find( rt1
)!=d_pattern_fun_sum
.end() );
170 Assert( d_pattern_fun_sum
.find( rt2
)!=d_pattern_fun_sum
.end() );
172 if( d_pattern_is_relevant
[rt1
] && !d_pattern_is_relevant
[rt2
] ){
173 Trace("thm-ee-debug") << "UEE : LT due to relevant." << std::endl
;
175 }else if( d_pattern_is_relevant
[rt1
]==d_pattern_is_relevant
[rt2
] ){
176 if( d_pattern_is_normal
[rt1
] && !d_pattern_is_normal
[rt2
] ){
177 Trace("thm-ee-debug") << "UEE : LT due to normal." << std::endl
;
179 }else if( d_pattern_is_normal
[rt1
]==d_pattern_is_normal
[rt2
] ){
180 if( d_pattern_fun_sum
[rt1
]<d_pattern_fun_sum
[rt2
] ){
181 Trace("thm-ee-debug") << "UEE : LT due to size." << std::endl
;
182 //decide which representative to use : based on size of the term
184 }else if( d_pattern_fun_sum
[rt1
]==d_pattern_fun_sum
[rt2
] ){
185 //same size : tie goes to term that has already been reported
186 return isReportedCanon( rt1
) && !isReportedCanon( rt2
);
194 bool ConjectureGenerator::isReportedCanon( TNode n
) {
195 return std::find( d_ue_canon
.begin(), d_ue_canon
.end(), n
)==d_ue_canon
.end();
198 void ConjectureGenerator::markReportedCanon( TNode n
) {
199 if( !isReportedCanon( n
) ){
200 d_ue_canon
.push_back( n
);
204 bool ConjectureGenerator::areUniversalEqual( TNode n1
, TNode n2
) {
205 return n1
==n2
|| ( d_uequalityEngine
.hasTerm( n1
) && d_uequalityEngine
.hasTerm( n2
) && d_uequalityEngine
.areEqual( n1
, n2
) );
208 bool ConjectureGenerator::areUniversalDisequal( TNode n1
, TNode n2
) {
209 return n1
!=n2
&& d_uequalityEngine
.hasTerm( n1
) && d_uequalityEngine
.hasTerm( n2
) && d_uequalityEngine
.areDisequal( n1
, n2
, false );
212 TNode
ConjectureGenerator::getUniversalRepresentative( TNode n
, bool add
) {
214 if( d_urelevant_terms
.find( n
)==d_urelevant_terms
.end() ){
215 setUniversalRelevant( n
);
216 //add term to universal equality engine
217 d_uequalityEngine
.addTerm( n
);
218 // addding this term to equality engine will lead to a set of new terms (the new subterms of n)
219 // now, do instantiation-based merging for each of these terms
220 Trace("thm-ee-debug") << "Merge equivalence classes based on instantiations of terms..." << std::endl
;
221 //merge all pending equalities
222 while( !d_upendingAdds
.empty() ){
223 Trace("sg-pending") << "Add " << d_upendingAdds
.size() << " pending terms..." << std::endl
;
224 std::vector
< Node
> pending
;
225 pending
.insert( pending
.end(), d_upendingAdds
.begin(), d_upendingAdds
.end() );
226 d_upendingAdds
.clear();
227 for( unsigned i
=0; i
<pending
.size(); i
++ ){
229 TypeNode tn
= t
.getType();
230 Trace("thm-ee-add") << "UEE : Add universal term " << t
<< std::endl
;
231 std::vector
< Node
> eq_terms
;
232 //if occurs modulo equality at ground level, it is equivalent to representative of ground equality engine
233 TNode gt
= getTermDatabase()->evaluateTerm( t
);
234 if( !gt
.isNull() && gt
!=t
){
235 eq_terms
.push_back( gt
);
237 //get all equivalent terms based on theorem database
238 d_thm_index
.getEquivalentTerms( t
, eq_terms
);
239 if( !eq_terms
.empty() ){
240 Trace("thm-ee-add") << "UEE : Based on ground EE/theorem DB, it is equivalent to " << eq_terms
.size() << " terms : " << std::endl
;
241 //add equivalent terms as equalities to universal engine
242 for( unsigned i
=0; i
<eq_terms
.size(); i
++ ){
243 Trace("thm-ee-add") << " " << eq_terms
[i
] << std::endl
;
244 bool assertEq
= false;
245 if( d_urelevant_terms
.find( eq_terms
[i
] )!=d_urelevant_terms
.end() ){
248 Assert( eq_terms
[i
].getType()==tn
);
249 registerPattern( eq_terms
[i
], tn
);
250 if( isUniversalLessThan( eq_terms
[i
], t
) || ( options::conjectureUeeIntro() && d_pattern_fun_sum
[t
]>=d_pattern_fun_sum
[eq_terms
[i
]] ) ){
251 setUniversalRelevant( eq_terms
[i
] );
257 d_uequalityEngine
.assertEquality( t
.eqNode( eq_terms
[i
] ), true, exp
);
259 Trace("thm-ee-no-add") << "Do not add : " << t
<< " == " << eq_terms
[i
] << std::endl
;
263 Trace("thm-ee-add") << "UEE : No equivalent terms." << std::endl
;
270 if( d_uequalityEngine
.hasTerm( n
) ){
271 Node r
= d_uequalityEngine
.getRepresentative( n
);
272 EqcInfo
* ei
= getOrMakeEqcInfo( r
);
273 if( ei
&& !ei
->d_rep
.get().isNull() ){
274 return ei
->d_rep
.get();
283 Node
ConjectureGenerator::getFreeVar( TypeNode tn
, unsigned i
) {
284 Assert( !tn
.isNull() );
285 while( d_free_var
[tn
].size()<=i
){
286 std::stringstream oss
;
288 std::string typ_name
= oss
.str();
289 while( typ_name
[0]=='(' ){
290 typ_name
.erase( typ_name
.begin() );
292 std::stringstream os
;
293 os
<< typ_name
[0] << i
;
294 Node x
= NodeManager::currentNM()->mkBoundVar( os
.str().c_str(), tn
);
295 d_free_var_num
[x
] = d_free_var
[tn
].size();
296 d_free_var
[tn
].push_back( x
);
298 return d_free_var
[tn
][i
];
303 Node
ConjectureGenerator::getCanonicalTerm( TNode n
, std::map
< TypeNode
, unsigned >& var_count
, std::map
< TNode
, TNode
>& subs
) {
304 if( n
.getKind()==BOUND_VARIABLE
){
305 std::map
< TNode
, TNode
>::iterator it
= subs
.find( n
);
306 if( it
==subs
.end() ){
307 TypeNode tn
= n
.getType();
309 unsigned vn
= var_count
[tn
];
311 subs
[n
] = getFreeVar( tn
, vn
);
317 std::vector
< Node
> children
;
318 if( n
.getKind()!=EQUAL
){
319 if( n
.hasOperator() ){
320 TNode op
= n
.getOperator();
321 if( !d_tge
.isRelevantFunc( op
) ){
324 children
.push_back( op
);
329 for( unsigned i
=0; i
<n
.getNumChildren(); i
++ ){
330 Node cn
= getCanonicalTerm( n
[i
], var_count
, subs
);
334 children
.push_back( cn
);
337 return NodeManager::currentNM()->mkNode( n
.getKind(), children
);
341 bool ConjectureGenerator::isHandledTerm( TNode n
){
342 return !n
.getAttribute(NoMatchAttribute()) && inst::Trigger::isAtomicTrigger( n
) && ( n
.getKind()!=APPLY_UF
|| n
.getOperator().getKind()!=SKOLEM
);
345 Node
ConjectureGenerator::getGroundEqc( TNode r
) {
346 std::map
< TNode
, Node
>::iterator it
= d_ground_eqc_map
.find( r
);
347 return it
!=d_ground_eqc_map
.end() ? it
->second
: Node::null();
350 bool ConjectureGenerator::isGroundEqc( TNode r
) {
351 return d_ground_eqc_map
.find( r
)!=d_ground_eqc_map
.end();
354 bool ConjectureGenerator::isGroundTerm( TNode n
) {
355 return std::find( d_ground_terms
.begin(), d_ground_terms
.end(), n
)!=d_ground_terms
.end();
358 bool ConjectureGenerator::needsCheck( Theory::Effort e
) {
359 // synchonized with instantiation engine
360 return d_quantEngine
->getInstWhenNeedsCheck( e
);
363 bool ConjectureGenerator::hasEnumeratedUf( Node n
) {
364 if( options::conjectureGenGtEnum()>0 ){
365 std::map
< Node
, bool >::iterator it
= d_uf_enum
.find( n
.getOperator() );
366 if( it
==d_uf_enum
.end() ){
367 d_uf_enum
[n
.getOperator()] = true;
368 std::vector
< Node
> lem
;
369 getEnumeratePredUfTerm( n
, options::conjectureGenGtEnum(), lem
);
371 for( unsigned j
=0; j
<lem
.size(); j
++ ){
372 d_quantEngine
->addLemma( lem
[j
], false );
373 d_hasAddedLemma
= true;
382 void ConjectureGenerator::reset_round( Theory::Effort e
) {
386 void ConjectureGenerator::check( Theory::Effort e
, unsigned quant_e
) {
387 if( quant_e
==QuantifiersEngine::QEFFORT_STANDARD
){
389 if( d_fullEffortCount
%optFullCheckFrequency()==0 ){
390 d_hasAddedLemma
= false;
393 if( Trace
.isOn("sg-engine") ){
394 clSet
= double(clock())/double(CLOCKS_PER_SEC
);
395 Trace("sg-engine") << "---Conjecture Engine Round, effort = " << e
<< "---" << std::endl
;
397 eq::EqualityEngine
* ee
= getEqualityEngine();
400 Trace("sg-proc") << "Get eq classes..." << std::endl
;
401 d_op_arg_index
.clear();
402 d_ground_eqc_map
.clear();
403 d_bool_eqc
[0] = Node::null();
404 d_bool_eqc
[1] = Node::null();
405 std::vector
< TNode
> eqcs
;
407 eq::EqClassesIterator eqcs_i
= eq::EqClassesIterator( ee
);
408 while( !eqcs_i
.isFinished() ){
411 if( r
.getType().isBoolean() ){
412 if( areEqual( r
, getTermDatabase()->d_true
) ){
413 d_ground_eqc_map
[r
] = getTermDatabase()->d_true
;
415 }else if( areEqual( r
, getTermDatabase()->d_false
) ){
416 d_ground_eqc_map
[r
] = getTermDatabase()->d_false
;
420 d_em
[r
] = eqcs
.size();
421 eq::EqClassIterator ieqc_i
= eq::EqClassIterator( r
, ee
);
422 while( !ieqc_i
.isFinished() ){
424 if( getTermDatabase()->hasTermCurrent( n
) ){
425 if( isHandledTerm( n
) ){
426 d_op_arg_index
[r
].addTerm( this, n
);
433 Assert( !d_bool_eqc
[0].isNull() );
434 Assert( !d_bool_eqc
[1].isNull() );
435 d_urelevant_terms
.clear();
436 Trace("sg-proc") << "...done get eq classes" << std::endl
;
438 Trace("sg-proc") << "Determine ground EQC..." << std::endl
;
442 for( unsigned i
=0; i
<eqcs
.size(); i
++ ){
444 if( d_ground_eqc_map
.find( r
)==d_ground_eqc_map
.end() ){
445 std::vector
< TNode
> args
;
446 Trace("sg-pat-debug") << "******* Get ground term for " << r
<< std::endl
;
448 if( getTermDatabase()->isInductionTerm( r
) ){
449 n
= d_op_arg_index
[r
].getGroundTerm( this, args
);
454 Trace("sg-pat") << "Ground term for eqc " << r
<< " : " << std::endl
;
455 Trace("sg-pat") << " " << n
<< std::endl
;
456 d_ground_eqc_map
[r
] = n
;
459 Trace("sg-pat-debug") << "...could not find ground term." << std::endl
;
464 //also get ground terms
465 d_ground_terms
.clear();
466 for( unsigned i
=0; i
<eqcs
.size(); i
++ ){
468 d_op_arg_index
[r
].getGroundTerms( this, d_ground_terms
);
470 Trace("sg-proc") << "...done determine ground EQC" << std::endl
;
473 if( Trace
.isOn("sg-gen-eqc") ){
474 for( unsigned i
=0; i
<eqcs
.size(); i
++ ){
477 bool firstTime
= true;
478 bool isFalse
= areEqual( r
, getTermDatabase()->d_false
);
479 eq::EqClassIterator eqc_i
= eq::EqClassIterator( r
, ee
);
480 while( !eqc_i
.isFinished() ){
482 if( getTermDatabase()->hasTermCurrent( n
) && !n
.getAttribute(NoMatchAttribute()) && ( n
.getKind()!=EQUAL
|| isFalse
) ){
484 Trace("sg-gen-eqc") << "e" << d_em
[r
] << " : { " << std::endl
;
487 if( n
.hasOperator() ){
488 Trace("sg-gen-eqc") << " (" << n
.getOperator();
489 getTermDatabase()->computeArgReps( n
);
490 for( unsigned i
=0; i
<getTermDatabase()->d_arg_reps
[n
].size(); i
++ ){
491 Trace("sg-gen-eqc") << " e" << d_em
[getTermDatabase()->d_arg_reps
[n
][i
]];
493 Trace("sg-gen-eqc") << ") :: " << n
<< std::endl
;
495 Trace("sg-gen-eqc") << " " << n
<< std::endl
;
501 Trace("sg-gen-eqc") << "}" << std::endl
;
502 //print out ground term
503 std::map
< TNode
, Node
>::iterator it
= d_ground_eqc_map
.find( r
);
504 if( it
!=d_ground_eqc_map
.end() ){
505 Trace("sg-gen-eqc") << "- Ground term : " << it
->second
<< std::endl
;
511 Trace("sg-proc") << "Compute relevant eqc..." << std::endl
;
512 d_tge
.d_relevant_eqc
[0].clear();
513 d_tge
.d_relevant_eqc
[1].clear();
514 for( unsigned i
=0; i
<eqcs
.size(); i
++ ){
516 std::map
< TNode
, Node
>::iterator it
= d_ground_eqc_map
.find( r
);
518 if( it
==d_ground_eqc_map
.end() ){
521 //based on unproven conjectures? TODO
522 d_tge
.d_relevant_eqc
[index
].push_back( r
);
524 Trace("sg-gen-tg-debug") << "Initial relevant eqc : ";
525 for( unsigned i
=0; i
<d_tge
.d_relevant_eqc
[0].size(); i
++ ){
526 Trace("sg-gen-tg-debug") << "e" << d_em
[d_tge
.d_relevant_eqc
[0][i
]] << " ";
528 Trace("sg-gen-tg-debug") << std::endl
;
529 Trace("sg-proc") << "...done compute relevant eqc" << std::endl
;
532 Trace("sg-proc") << "Collect signature information..." << std::endl
;
533 d_tge
.collectSignatureInformation();
534 if( d_hasAddedLemma
){
535 Trace("sg-proc") << "...added enumeration lemmas." << std::endl
;
537 Trace("sg-proc") << "...done collect signature information" << std::endl
;
541 Trace("sg-proc") << "Build theorem index..." << std::endl
;
544 std::vector
< Node
> provenConj
;
545 quantifiers::FirstOrderModel
* m
= d_quantEngine
->getModel();
546 for( int i
=0; i
<m
->getNumAssertedQuantifiers(); i
++ ){
547 Node q
= m
->getAssertedQuantifier( i
);
548 Trace("thm-db-debug") << "Is " << q
<< " a relevant theorem?" << std::endl
;
550 if( q
[1].getKind()==EQUAL
){
551 bool isSubsume
= false;
553 for( unsigned r
=0; r
<2; r
++ ){
554 TNode nl
= q
[1][r
==0 ? 0 : 1];
555 TNode nr
= q
[1][r
==0 ? 1 : 0];
556 Node eq
= nl
.eqNode( nr
);
557 if( r
==1 || std::find( d_conjectures
.begin(), d_conjectures
.end(), q
)==d_conjectures
.end() ){
558 //must make it canonical
559 std::map
< TypeNode
, unsigned > var_count
;
560 std::map
< TNode
, TNode
> subs
;
561 Trace("sg-proc-debug") << "get canonical " << eq
<< std::endl
;
562 eq
= getCanonicalTerm( eq
, var_count
, subs
);
566 inEe
= d_ee_conjectures
.find( q
[1] )!=d_ee_conjectures
.end();
568 //add to universal equality engine
569 Node nl
= getUniversalRepresentative( eq
[0], true );
570 Node nr
= getUniversalRepresentative( eq
[1], true );
571 if( areUniversalEqual( nl
, nr
) ){
573 //set inactive (will be ignored by other modules)
574 d_quantEngine
->getModel()->setQuantifierActive( q
, false );
577 d_ee_conjectures
[q
[1]] = true;
578 d_uequalityEngine
.assertEquality( nl
.eqNode( nr
), true, exp
);
581 Trace("sg-conjecture") << "*** CONJECTURE : currently proven" << (isSubsume
? " and subsumed" : "");
582 Trace("sg-conjecture") << " : " << q
[1] << std::endl
;
583 provenConj
.push_back( q
);
586 Trace("thm-db-debug") << "Adding theorem to database " << eq
[0] << " == " << eq
[1] << std::endl
;
587 d_thm_index
.addTheorem( eq
[0], eq
[1] );
597 //examine status of other conjectures
598 for( unsigned i
=0; i
<d_conjectures
.size(); i
++ ){
599 Node q
= d_conjectures
[i
];
600 if( std::find( provenConj
.begin(), provenConj
.end(), q
)==provenConj
.end() ){
601 //check each skolem variable
602 bool disproven
= true;
603 //std::vector< Node > sk;
604 //getTermDatabase()->getSkolemConstants( q, sk, true );
605 Trace("sg-conjecture") << " CONJECTURE : ";
606 std::vector
< Node
> ce
;
607 for( unsigned j
=0; j
<getTermDatabase()->d_skolem_constants
[q
].size(); j
++ ){
608 TNode k
= getTermDatabase()->d_skolem_constants
[q
][j
];
609 TNode rk
= getRepresentative( k
);
610 std::map
< TNode
, Node
>::iterator git
= d_ground_eqc_map
.find( rk
);
611 //check if it is a ground term
612 if( git
==d_ground_eqc_map
.end() ){
613 Trace("sg-conjecture") << "ACTIVE : " << q
;
614 if( Trace
.isOn("sg-gen-eqc") ){
615 Trace("sg-conjecture") << " { ";
616 for( unsigned k
=0; k
<getTermDatabase()->d_skolem_constants
[q
].size(); k
++ ){ Trace("sg-conjecture") << getTermDatabase()->d_skolem_constants
[q
][k
] << ( j
==k
? "*" : "" ) << " "; }
617 Trace("sg-conjecture") << "}";
619 Trace("sg-conjecture") << std::endl
;
623 ce
.push_back( git
->second
);
627 Trace("sg-conjecture") << "disproven : " << q
<< " : ";
628 for( unsigned i
=0; i
<ce
.size(); i
++ ){
629 Trace("sg-conjecture") << q
[0][i
] << " -> " << ce
[i
] << " ";
631 Trace("sg-conjecture") << std::endl
;
635 Trace("thm-db") << "Theorem database is : " << std::endl
;
636 d_thm_index
.debugPrint( "thm-db" );
637 Trace("thm-db") << std::endl
;
638 Trace("sg-proc") << "...done build theorem index" << std::endl
;
643 d_pattern_var_id
.clear();
644 d_pattern_var_duplicate
.clear();
645 d_pattern_is_normal
.clear();
646 d_pattern_is_relevant
.clear();
647 d_pattern_fun_id
.clear();
648 d_pattern_fun_sum
.clear();
649 d_rel_patterns
.clear();
650 d_rel_pattern_var_sum
.clear();
651 d_rel_pattern_typ_index
.clear();
652 d_rel_pattern_subs_index
.clear();
654 unsigned rel_term_count
= 0;
655 std::map
< TypeNode
, unsigned > rt_var_max
;
656 std::vector
< TypeNode
> rt_types
;
657 std::map
< TypeNode
, std::map
< int, std::vector
< Node
> > > conj_lhs
;
658 unsigned addedLemmas
= 0;
659 for( unsigned depth
=1; depth
<=3; depth
++ ){
660 Trace("sg-proc") << "Generate relevant LHS at depth " << depth
<< "..." << std::endl
;
661 Trace("sg-rel-term") << "Relevant terms of depth " << depth
<< " : " << std::endl
;
663 d_tge
.d_var_id
.clear();
664 d_tge
.d_var_limit
.clear();
665 d_tge
.reset( depth
, true, TypeNode::null() );
666 while( d_tge
.getNextTerm() ){
668 Node nn
= d_tge
.getTerm();
669 if( !options::conjectureFilterCanonical() || considerTermCanon( nn
, true ) ){
671 Trace("sg-rel-term") << "*** Relevant term : ";
672 d_tge
.debugPrint( "sg-rel-term", "sg-rel-term-debug2" );
673 Trace("sg-rel-term") << std::endl
;
675 for( unsigned r
=0; r
<2; r
++ ){
676 Trace("sg-rel-term-debug") << "...from equivalence classes (" << r
<< ") : ";
677 int index
= d_tge
.d_ccand_eqc
[r
].size()-1;
678 for( unsigned j
=0; j
<d_tge
.d_ccand_eqc
[r
][index
].size(); j
++ ){
679 Trace("sg-rel-term-debug") << "e" << d_em
[d_tge
.d_ccand_eqc
[r
][index
][j
]] << " ";
681 Trace("sg-rel-term-debug") << std::endl
;
683 TypeNode tnn
= nn
.getType();
684 Trace("sg-gen-tg-debug") << "...term is " << nn
<< std::endl
;
685 conj_lhs
[tnn
][depth
].push_back( nn
);
687 //add information about pattern
688 Trace("sg-gen-tg-debug") << "Collect pattern information..." << std::endl
;
689 Assert( std::find( d_rel_patterns
[tnn
].begin(), d_rel_patterns
[tnn
].end(), nn
)==d_rel_patterns
[tnn
].end() );
690 d_rel_patterns
[tnn
].push_back( nn
);
691 //build information concerning the variables in this pattern
693 std::map
< TypeNode
, unsigned > typ_to_subs_index
;
694 std::vector
< TNode
> gsubs_vars
;
695 for( std::map
< TypeNode
, unsigned >::iterator it
= d_tge
.d_var_id
.begin(); it
!= d_tge
.d_var_id
.end(); ++it
){
697 typ_to_subs_index
[it
->first
] = sum
;
699 for( unsigned i
=0; i
<it
->second
; i
++ ){
700 gsubs_vars
.push_back( getFreeVar( it
->first
, i
) );
704 d_rel_pattern_var_sum
[nn
] = sum
;
705 //register the pattern
706 registerPattern( nn
, tnn
);
707 Assert( d_pattern_is_normal
[nn
] );
708 Trace("sg-gen-tg-debug") << "...done collect pattern information" << std::endl
;
710 //record information about types
711 Trace("sg-gen-tg-debug") << "Collect type information..." << std::endl
;
712 PatternTypIndex
* pti
= &d_rel_pattern_typ_index
;
713 for( std::map
< TypeNode
, unsigned >::iterator it
= d_tge
.d_var_id
.begin(); it
!= d_tge
.d_var_id
.end(); ++it
){
714 pti
= &pti
->d_children
[it
->first
][it
->second
];
716 if( rt_var_max
.find( it
->first
)==rt_var_max
.end() || it
->second
>rt_var_max
[it
->first
] ){
717 rt_var_max
[it
->first
] = it
->second
;
720 if( std::find( rt_types
.begin(), rt_types
.end(), tnn
)==rt_types
.end() ){
721 rt_types
.push_back( tnn
);
723 pti
->d_terms
.push_back( nn
);
724 Trace("sg-gen-tg-debug") << "...done collect type information" << std::endl
;
726 Trace("sg-gen-tg-debug") << "Build substitutions for ground EQC..." << std::endl
;
727 std::vector
< TNode
> gsubs_terms
;
728 gsubs_terms
.resize( gsubs_vars
.size() );
729 int index
= d_tge
.d_ccand_eqc
[1].size()-1;
730 for( unsigned j
=0; j
<d_tge
.d_ccand_eqc
[1][index
].size(); j
++ ){
731 TNode r
= d_tge
.d_ccand_eqc
[1][index
][j
];
732 Trace("sg-rel-term-debug") << " Matches for e" << d_em
[r
] << ", which is ground term " << d_ground_eqc_map
[r
] << ":" << std::endl
;
733 std::map
< TypeNode
, std::map
< unsigned, TNode
> > subs
;
734 std::map
< TNode
, bool > rev_subs
;
735 //only get ground terms
737 d_tge
.resetMatching( r
, mode
);
738 while( d_tge
.getNextMatch( r
, subs
, rev_subs
) ){
739 //we will be building substitutions
740 bool firstTime
= true;
741 for( std::map
< TypeNode
, std::map
< unsigned, TNode
> >::iterator it
= subs
.begin(); it
!= subs
.end(); ++it
){
742 unsigned tindex
= typ_to_subs_index
[it
->first
];
743 for( std::map
< unsigned, TNode
>::iterator it2
= it
->second
.begin(); it2
!= it
->second
.end(); ++it2
){
745 Trace("sg-rel-term-debug") << ", ";
748 Trace("sg-rel-term-debug") << " ";
750 Trace("sg-rel-term-debug") << it
->first
<< ":x" << it2
->first
<< " -> " << it2
->second
;
751 Assert( tindex
+it2
->first
<gsubs_terms
.size() );
752 gsubs_terms
[tindex
+it2
->first
] = it2
->second
;
755 Trace("sg-rel-term-debug") << std::endl
;
756 d_rel_pattern_subs_index
[nn
].addSubstitution( r
, gsubs_vars
, gsubs_terms
);
759 Trace("sg-gen-tg-debug") << "...done build substitutions for ground EQC" << std::endl
;
761 Trace("sg-gen-tg-debug") << "> not canonical : " << nn
<< std::endl
;
764 Trace("sg-proc") << "...done generate terms at depth " << depth
<< std::endl
;
765 Trace("sg-stats") << "--------> Total LHS of depth " << depth
<< " : " << rel_term_count
<< std::endl
;
766 //Trace("conjecture-count") << "Total LHS of depth " << depth << " : " << conj_lhs[depth].size() << std::endl;
769 for( unsigned i=0; i<rt_types.size(); i++ ){
770 Trace("sg-term-enum") << "Term enumeration for " << rt_types[i] << " : " << std::endl;
771 Trace("sg-term-enum") << "Ground term : " << rt_types[i].mkGroundTerm() << std::endl;
772 for( unsigned j=0; j<150; j++ ){
773 Trace("sg-term-enum") << " " << getEnumerateTerm( rt_types[i], j ) << std::endl;
778 //consider types from relevant terms
779 for( unsigned rdepth
=0; rdepth
<=depth
; rdepth
++ ){
781 d_tge
.d_var_id
.clear();
782 d_tge
.d_var_limit
.clear();
783 for( std::map
< TypeNode
, unsigned >::iterator it
= rt_var_max
.begin(); it
!= rt_var_max
.end(); ++it
){
784 d_tge
.d_var_id
[ it
->first
] = it
->second
;
785 d_tge
.d_var_limit
[ it
->first
] = it
->second
;
787 std::random_shuffle( rt_types
.begin(), rt_types
.end() );
788 std::map
< TypeNode
, std::vector
< Node
> > conj_rhs
;
789 for( unsigned i
=0; i
<rt_types
.size(); i
++ ){
791 Trace("sg-proc") << "Generate relevant RHS terms of type " << rt_types
[i
] << " at depth " << rdepth
<< "..." << std::endl
;
792 d_tge
.reset( rdepth
, false, rt_types
[i
] );
794 while( d_tge
.getNextTerm() ){
795 Node rhs
= d_tge
.getTerm();
796 if( considerTermCanon( rhs
, false ) ){
797 Trace("sg-rel-prop") << "Relevant RHS : " << rhs
<< std::endl
;
799 Assert( rhs
.getType()==rt_types
[i
] );
800 registerPattern( rhs
, rt_types
[i
] );
802 //consider against all LHS at depth
803 for( unsigned j
=0; j
<conj_lhs
[rt_types
[i
]][depth
].size(); j
++ ){
804 processCandidateConjecture( conj_lhs
[rt_types
[i
]][depth
][j
], rhs
, depth
, rdepth
);
807 conj_rhs
[rt_types
[i
]].push_back( rhs
);
812 flushWaitingConjectures( addedLemmas
, depth
, rdepth
);
813 //consider against all LHS up to depth
815 for( unsigned lhs_depth
= 1; lhs_depth
<=depth
; lhs_depth
++ ){
816 if( (int)addedLemmas
<options::conjectureGenPerRound() ){
817 Trace("sg-proc") << "Consider conjectures at depth (" << lhs_depth
<< ", " << rdepth
<< ")..." << std::endl
;
818 for( std::map
< TypeNode
, std::vector
< Node
> >::iterator it
= conj_rhs
.begin(); it
!= conj_rhs
.end(); ++it
){
819 for( unsigned j
=0; j
<it
->second
.size(); j
++ ){
820 for( unsigned k
=0; k
<conj_lhs
[it
->first
][lhs_depth
].size(); k
++ ){
821 processCandidateConjecture( conj_lhs
[it
->first
][lhs_depth
][k
], it
->second
[j
], lhs_depth
, rdepth
);
825 flushWaitingConjectures( addedLemmas
, lhs_depth
, depth
);
829 if( (int)addedLemmas
>=options::conjectureGenPerRound() ){
833 if( (int)addedLemmas
>=options::conjectureGenPerRound() ){
837 Trace("sg-stats") << "Total conjectures considered : " << d_conj_count
<< std::endl
;
838 if( Trace
.isOn("thm-ee") ){
839 Trace("thm-ee") << "Universal equality engine is : " << std::endl
;
840 eq::EqClassesIterator ueqcs_i
= eq::EqClassesIterator( &d_uequalityEngine
);
841 while( !ueqcs_i
.isFinished() ){
842 TNode r
= (*ueqcs_i
);
843 bool firstTime
= true;
844 TNode rr
= getUniversalRepresentative( r
);
845 Trace("thm-ee") << " " << rr
;
846 Trace("thm-ee") << " : { ";
847 eq::EqClassIterator ueqc_i
= eq::EqClassIterator( r
, &d_uequalityEngine
);
848 while( !ueqc_i
.isFinished() ){
852 Trace("thm-ee") << std::endl
;
855 Trace("thm-ee") << " " << n
<< std::endl
;
859 if( !firstTime
){ Trace("thm-ee") << " "; }
860 Trace("thm-ee") << "}" << std::endl
;
863 Trace("thm-ee") << std::endl
;
865 if( Trace
.isOn("sg-engine") ){
866 double clSet2
= double(clock())/double(CLOCKS_PER_SEC
);
867 Trace("sg-engine") << "Finished conjecture generator, time = " << (clSet2
-clSet
) << std::endl
;
873 unsigned ConjectureGenerator::flushWaitingConjectures( unsigned& addedLemmas
, int ldepth
, int rdepth
) {
874 if( !d_waiting_conjectures_lhs
.empty() ){
875 Trace("sg-proc") << "Generated " << d_waiting_conjectures_lhs
.size() << " conjectures at depth " << ldepth
<< "/" << rdepth
<< "." << std::endl
;
876 if( (int)addedLemmas
<options::conjectureGenPerRound() ){
878 std::vector< unsigned > indices;
879 for( unsigned i=0; i<d_waiting_conjectures_lhs.size(); i++ ){
880 indices.push_back( i );
884 //sort them based on score
885 sortConjectureScore scs;
886 scs.d_scores.insert( scs.d_scores.begin(), d_waiting_conjectures_score.begin(), d_waiting_conjectures_score.end() );
887 std::sort( indices.begin(), indices.end(), scs );
889 //if( doSort && d_waiting_conjectures_score[indices[0]]<optFilterScoreThreshold() ){
891 unsigned prevCount
= d_conj_count
;
892 for( unsigned i
=0; i
<d_waiting_conjectures_lhs
.size(); i
++ ){
893 if( d_waiting_conjectures_score
[i
]>=optFilterScoreThreshold() ){
894 //we have determined a relevant subgoal
895 Node lhs
= d_waiting_conjectures_lhs
[i
];
896 Node rhs
= d_waiting_conjectures_rhs
[i
];
897 if( options::conjectureFilterCanonical() && ( getUniversalRepresentative( lhs
)!=lhs
|| getUniversalRepresentative( rhs
)!=rhs
) ){
900 Trace("sg-engine") << "*** Consider conjecture : " << lhs
<< " == " << rhs
<< std::endl
;
901 Trace("sg-engine-debug") << " score : " << d_waiting_conjectures_score
[i
] << std::endl
;
902 if( optStatsOnly() ){
905 std::vector
< Node
> bvs
;
906 for( std::map
< TypeNode
, unsigned >::iterator it
= d_pattern_var_id
[lhs
].begin(); it
!= d_pattern_var_id
[lhs
].end(); ++it
){
907 for( unsigned i
=0; i
<=it
->second
; i
++ ){
908 bvs
.push_back( getFreeVar( it
->first
, i
) );
913 Node bvl
= NodeManager::currentNM()->mkNode( BOUND_VAR_LIST
, bvs
);
914 rsg
= NodeManager::currentNM()->mkNode( FORALL
, bvl
, lhs
.eqNode( rhs
) );
916 rsg
= lhs
.eqNode( rhs
);
918 rsg
= Rewriter::rewrite( rsg
);
919 d_conjectures
.push_back( rsg
);
920 d_eq_conjectures
[lhs
].push_back( rhs
);
921 d_eq_conjectures
[rhs
].push_back( lhs
);
923 Node lem
= NodeManager::currentNM()->mkNode( OR
, rsg
.negate(), rsg
);
924 d_quantEngine
->addLemma( lem
, false );
925 d_quantEngine
->addRequirePhase( rsg
, false );
927 if( (int)addedLemmas
>=options::conjectureGenPerRound() ){
934 Trace("sg-proc") << "...have now added " << addedLemmas
<< " conjecture lemmas." << std::endl
;
935 if( optStatsOnly() ){
936 Trace("sg-stats") << "Generated " << (d_conj_count
-prevCount
) << " conjectures at depth " << ldepth
<< "/" << rdepth
<< "." << std::endl
;
939 d_waiting_conjectures_lhs
.clear();
940 d_waiting_conjectures_rhs
.clear();
941 d_waiting_conjectures_score
.clear();
942 d_waiting_conjectures
.clear();
947 void ConjectureGenerator::registerQuantifier( Node q
) {
951 void ConjectureGenerator::assertNode( Node n
) {
955 bool ConjectureGenerator::considerTermCanon( Node ln
, bool genRelevant
){
957 //do not consider if it is non-canonical, and either:
958 // (1) we are not generating relevant terms, or
959 // (2) its canonical form is a generalization.
960 TNode lnr
= getUniversalRepresentative( ln
, true );
962 markReportedCanon( ln
);
963 }else if( !genRelevant
|| isGeneralization( lnr
, ln
) ){
964 Trace("sg-gen-consider-term") << "Do not consider term, " << ln
<< " is not canonical representation (which is " << lnr
<< ")." << std::endl
;
968 Trace("sg-gen-tg-debug") << "Will consider term canon " << ln
<< std::endl
;
969 Trace("sg-gen-consider-term-debug") << std::endl
;
973 unsigned ConjectureGenerator::collectFunctions( TNode opat
, TNode pat
, std::map
< TNode
, unsigned >& funcs
,
974 std::map
< TypeNode
, unsigned >& mnvn
, std::map
< TypeNode
, unsigned >& mxvn
){
975 if( pat
.hasOperator() ){
976 funcs
[pat
.getOperator()]++;
977 if( !d_tge
.isRelevantFunc( pat
.getOperator() ) ){
978 d_pattern_is_relevant
[opat
] = false;
981 for( unsigned i
=0; i
<pat
.getNumChildren(); i
++ ){
982 sum
+= collectFunctions( opat
, pat
[i
], funcs
, mnvn
, mxvn
);
986 Assert( pat
.getNumChildren()==0 );
989 if( pat
.getKind()==BOUND_VARIABLE
){
992 d_pattern_var_duplicate
[opat
]++;
995 TypeNode tn
= pat
.getType();
996 unsigned vn
= d_free_var_num
[pat
];
997 std::map
< TypeNode
, unsigned >::iterator it
= mnvn
.find( tn
);
998 if( it
!=mnvn
.end() ){
1000 d_pattern_is_normal
[opat
] = false;
1002 }else if( vn
>mxvn
[tn
] ){
1003 if( vn
!=mxvn
[tn
]+1 ){
1004 d_pattern_is_normal
[opat
] = false;
1009 //first variable of this type
1015 d_pattern_is_relevant
[opat
] = false;
1021 void ConjectureGenerator::registerPattern( Node pat
, TypeNode tpat
) {
1022 if( std::find( d_patterns
[tpat
].begin(), d_patterns
[tpat
].end(), pat
)==d_patterns
[tpat
].end() ){
1023 d_patterns
[TypeNode::null()].push_back( pat
);
1024 d_patterns
[tpat
].push_back( pat
);
1026 Assert( d_pattern_fun_id
.find( pat
)==d_pattern_fun_id
.end() );
1027 Assert( d_pattern_var_id
.find( pat
)==d_pattern_var_id
.end() );
1030 std::map
< TypeNode
, unsigned > mnvn
;
1031 d_pattern_fun_sum
[pat
] = collectFunctions( pat
, pat
, d_pattern_fun_id
[pat
], mnvn
, d_pattern_var_id
[pat
] );
1032 if( d_pattern_is_normal
.find( pat
)==d_pattern_is_normal
.end() ){
1033 d_pattern_is_normal
[pat
] = true;
1035 if( d_pattern_is_relevant
.find( pat
)==d_pattern_is_relevant
.end() ){
1036 d_pattern_is_relevant
[pat
] = true;
1041 bool ConjectureGenerator::isGeneralization( TNode patg
, TNode pat
, std::map
< TNode
, TNode
>& subs
) {
1042 if( patg
.getKind()==BOUND_VARIABLE
){
1043 std::map
< TNode
, TNode
>::iterator it
= subs
.find( patg
);
1044 if( it
!=subs
.end() ){
1045 return it
->second
==pat
;
1051 Assert( patg
.hasOperator() );
1052 if( !pat
.hasOperator() || patg
.getOperator()!=pat
.getOperator() ){
1055 Assert( patg
.getNumChildren()==pat
.getNumChildren() );
1056 for( unsigned i
=0; i
<patg
.getNumChildren(); i
++ ){
1057 if( !isGeneralization( patg
[i
], pat
[i
], subs
) ){
1066 int ConjectureGenerator::calculateGeneralizationDepth( TNode n
, std::vector
< TNode
>& fv
) {
1067 if( n
.getKind()==BOUND_VARIABLE
){
1068 if( std::find( fv
.begin(), fv
.end(), n
)==fv
.end() ){
1076 for( unsigned i
=0; i
<n
.getNumChildren(); i
++ ){
1077 depth
+= calculateGeneralizationDepth( n
[i
], fv
);
1083 Node
ConjectureGenerator::getPredicateForType( TypeNode tn
) {
1084 std::map
< TypeNode
, Node
>::iterator it
= d_typ_pred
.find( tn
);
1085 if( it
==d_typ_pred
.end() ){
1086 TypeNode op_tn
= NodeManager::currentNM()->mkFunctionType( tn
, NodeManager::currentNM()->booleanType() );
1087 Node op
= NodeManager::currentNM()->mkSkolem( "PE", op_tn
, "was created by conjecture ground term enumerator." );
1088 d_typ_pred
[tn
] = op
;
1095 void ConjectureGenerator::getEnumerateUfTerm( Node n
, unsigned num
, std::vector
< Node
>& terms
) {
1096 if( n
.getNumChildren()>0 ){
1097 std::vector
< int > vec
;
1098 for( unsigned i
=0; i
<n
.getNumChildren(); i
++ ){
1105 unsigned last_size
= terms
.size();
1106 while( terms
.size()<num
){
1107 bool success
= true;
1110 vec
.push_back( size_limit
);
1112 //see if we can iterate current
1113 if( vec_sum
<size_limit
&& !getTermDatabase()->getEnumerateTerm( n
[index
].getType(), vec
[index
]+1 ).isNull() ){
1116 vec
.push_back( size_limit
- vec_sum
);
1118 vec_sum
-= vec
[index
];
1121 if( index
==n
.getNumChildren() ){
1127 if( vec
.size()==n
.getNumChildren() ){
1128 Node lc
= getTermDatabase()->getEnumerateTerm( n
[vec
.size()-1].getType(), vec
[vec
.size()-1] );
1130 for( unsigned i
=0; i
<vec
.size(); i
++ ){
1131 Trace("sg-gt-enum-debug") << vec
[i
] << " ";
1133 Trace("sg-gt-enum-debug") << " / " << size_limit
<< std::endl
;
1134 for( unsigned i
=0; i
<n
.getNumChildren(); i
++ ){
1135 Trace("sg-gt-enum-debug") << n
[i
].getType() << " ";
1137 Trace("sg-gt-enum-debug") << std::endl
;
1138 std::vector
< Node
> children
;
1139 children
.push_back( n
.getOperator() );
1140 for( unsigned i
=0; i
<(vec
.size()-1); i
++ ){
1141 Node nn
= getTermDatabase()->getEnumerateTerm( n
[i
].getType(), vec
[i
] );
1142 Assert( !nn
.isNull() );
1143 Assert( nn
.getType()==n
[i
].getType() );
1144 children
.push_back( nn
);
1146 children
.push_back( lc
);
1147 Node n
= NodeManager::currentNM()->mkNode( APPLY_UF
, children
);
1148 Trace("sg-gt-enum") << "Ground term enumerate : " << n
<< std::endl
;
1149 terms
.push_back( n
);
1155 if( terms
.size()>last_size
){
1156 last_size
= terms
.size();
1158 for( unsigned i
=0; i
<vec
.size(); i
++ ){
1168 terms
.push_back( n
);
1172 void ConjectureGenerator::getEnumeratePredUfTerm( Node n
, unsigned num
, std::vector
< Node
>& terms
) {
1173 std::vector
< Node
> uf_terms
;
1174 getEnumerateUfTerm( n
, num
, uf_terms
);
1175 Node p
= getPredicateForType( n
.getType() );
1176 for( unsigned i
=0; i
<uf_terms
.size(); i
++ ){
1177 terms
.push_back( NodeManager::currentNM()->mkNode( APPLY_UF
, p
, uf_terms
[i
] ) );
1181 void ConjectureGenerator::processCandidateConjecture( TNode lhs
, TNode rhs
, unsigned lhs_depth
, unsigned rhs_depth
) {
1182 int score
= considerCandidateConjecture( lhs
, rhs
);
1184 Trace("sg-conjecture") << "* Candidate conjecture : " << lhs
<< " == " << rhs
<< std::endl
;
1185 Trace("sg-conjecture-debug") << " LHS, RHS generalization depth : " << lhs_depth
<< ", " << rhs_depth
<< std::endl
;
1186 Trace("sg-conjecture-debug") << " confirmed = " << d_subs_confirmCount
<< ", #witnesses range = " << d_subs_confirmWitnessRange
.size() << "." << std::endl
;
1187 Trace("sg-conjecture-debug") << " #witnesses for ";
1188 bool firstTime
= true;
1189 for( std::map
< TNode
, std::vector
< TNode
> >::iterator it
= d_subs_confirmWitnessDomain
.begin(); it
!= d_subs_confirmWitnessDomain
.end(); ++it
){
1191 Trace("sg-conjecture-debug") << ", ";
1193 Trace("sg-conjecture-debug") << it
->first
<< " : " << it
->second
.size();
1194 //if( it->second.size()==1 ){
1195 // Trace("sg-conjecture-debug") << " (" << it->second[0] << ")";
1197 Trace("sg-conjecture-debug2") << " (";
1198 for( unsigned j
=0; j
<it
->second
.size(); j
++ ){
1199 if( j
>0 ){ Trace("sg-conjecture-debug2") << " "; }
1200 Trace("sg-conjecture-debug2") << d_ground_eqc_map
[it
->second
[j
]];
1202 Trace("sg-conjecture-debug2") << ")";
1205 Trace("sg-conjecture-debug") << std::endl
;
1206 Trace("sg-conjecture-debug") << " unknown = " << d_subs_unkCount
<< std::endl
;
1207 //Assert( getUniversalRepresentative( rhs )==rhs );
1208 //Assert( getUniversalRepresentative( lhs )==lhs );
1209 d_waiting_conjectures_lhs
.push_back( lhs
);
1210 d_waiting_conjectures_rhs
.push_back( rhs
);
1211 d_waiting_conjectures_score
.push_back( score
);
1212 d_waiting_conjectures
[lhs
].push_back( rhs
);
1213 d_waiting_conjectures
[rhs
].push_back( lhs
);
1217 int ConjectureGenerator::considerCandidateConjecture( TNode lhs
, TNode rhs
) {
1218 Assert( lhs
.getType()==rhs
.getType() );
1220 Trace("sg-cconj-debug") << "Consider candidate conjecture : " << lhs
<< " == " << rhs
<< "?" << std::endl
;
1222 Trace("sg-cconj-debug") << " -> trivial." << std::endl
;
1225 if( lhs
.getKind()==APPLY_CONSTRUCTOR
&& rhs
.getKind()==APPLY_CONSTRUCTOR
){
1226 Trace("sg-cconj-debug") << " -> irrelevant by syntactic analysis." << std::endl
;
1229 //variables of LHS must subsume variables of RHS
1230 for( std::map
< TypeNode
, unsigned >::iterator it
= d_pattern_var_id
[rhs
].begin(); it
!= d_pattern_var_id
[rhs
].end(); ++it
){
1231 std::map
< TypeNode
, unsigned >::iterator itl
= d_pattern_var_id
[lhs
].find( it
->first
);
1232 if( itl
!=d_pattern_var_id
[lhs
].end() ){
1233 if( itl
->second
<it
->second
){
1234 Trace("sg-cconj-debug") << " -> variables of sort " << it
->first
<< " are not subsumed." << std::endl
;
1237 Trace("sg-cconj-debug2") << " variables of sort " << it
->first
<< " are : " << itl
->second
<< " vs " << it
->second
<< std::endl
;
1240 Trace("sg-cconj-debug") << " -> has no variables of sort " << it
->first
<< "." << std::endl
;
1245 //currently active conjecture?
1246 std::map
< Node
, std::vector
< Node
> >::iterator iteq
= d_eq_conjectures
.find( lhs
);
1247 if( iteq
!=d_eq_conjectures
.end() ){
1248 if( std::find( iteq
->second
.begin(), iteq
->second
.end(), rhs
)!=iteq
->second
.end() ){
1249 Trace("sg-cconj-debug") << " -> this conjecture is already active." << std::endl
;
1253 //current a waiting conjecture?
1254 std::map
< Node
, std::vector
< Node
> >::iterator itw
= d_waiting_conjectures
.find( lhs
);
1255 if( itw
!=d_waiting_conjectures
.end() ){
1256 if( std::find( itw
->second
.begin(), itw
->second
.end(), rhs
)!=itw
->second
.end() ){
1257 Trace("sg-cconj-debug") << " -> already are considering this conjecture." << std::endl
;
1261 //check if canonical representation (should be, but for efficiency this is not guarenteed)
1262 //if( options::conjectureFilterCanonical() && ( getUniversalRepresentative( lhs )!=lhs || getUniversalRepresentative( rhs )!=rhs ) ){
1263 // Trace("sg-cconj") << " -> after processing, not canonical." << std::endl;
1268 bool scoreSet
= false;
1270 Trace("sg-cconj") << "Consider possible candidate conjecture : " << lhs
<< " == " << rhs
<< "?" << std::endl
;
1271 //find witness for counterexample, if possible
1272 if( options::conjectureFilterModel() ){
1273 Assert( d_rel_pattern_var_sum
.find( lhs
)!=d_rel_pattern_var_sum
.end() );
1274 Trace("sg-cconj-debug") << "Notify substitutions over " << d_rel_pattern_var_sum
[lhs
] << " variables." << std::endl
;
1275 std::map
< TNode
, TNode
> subs
;
1276 d_subs_confirmCount
= 0;
1277 d_subs_confirmWitnessRange
.clear();
1278 d_subs_confirmWitnessDomain
.clear();
1279 d_subs_unkCount
= 0;
1280 if( !d_rel_pattern_subs_index
[lhs
].notifySubstitutions( this, subs
, rhs
, d_rel_pattern_var_sum
[lhs
] ) ){
1281 Trace("sg-cconj") << " -> found witness that falsifies the conjecture." << std::endl
;
1284 //score is the minimum number of distinct substitutions for a variable
1285 for( std::map
< TNode
, std::vector
< TNode
> >::iterator it
= d_subs_confirmWitnessDomain
.begin(); it
!= d_subs_confirmWitnessDomain
.end(); ++it
){
1286 int num
= (int)it
->second
.size();
1287 if( !scoreSet
|| num
<score
){
1295 Trace("sg-cconj") << " confirmed = " << d_subs_confirmCount
<< ", #witnesses range = " << d_subs_confirmWitnessRange
.size() << "." << std::endl
;
1296 for( std::map
< TNode
, std::vector
< TNode
> >::iterator it
= d_subs_confirmWitnessDomain
.begin(); it
!= d_subs_confirmWitnessDomain
.end(); ++it
){
1297 Trace("sg-cconj") << " #witnesses for " << it
->first
<< " : " << it
->second
.size() << std::endl
;
1303 Trace("sg-cconj") << " -> SUCCESS." << std::endl
;
1304 Trace("sg-cconj") << " score : " << score
<< std::endl
;
1310 bool ConjectureGenerator::notifySubstitution( TNode glhs
, std::map
< TNode
, TNode
>& subs
, TNode rhs
) {
1311 if( Trace
.isOn("sg-cconj-debug") ){
1312 Trace("sg-cconj-debug") << "Ground eqc for LHS : " << glhs
<< ", based on substituion: " << std::endl
;
1313 for( std::map
< TNode
, TNode
>::iterator it
= subs
.begin(); it
!= subs
.end(); ++it
){
1314 Assert( getRepresentative( it
->second
)==it
->second
);
1315 Trace("sg-cconj-debug") << " " << it
->first
<< " -> " << it
->second
<< std::endl
;
1318 Trace("sg-cconj-debug") << "Evaluate RHS : : " << rhs
<< std::endl
;
1319 //get the representative of rhs with substitution subs
1320 TNode grhs
= getTermDatabase()->evaluateTerm( rhs
, subs
, true );
1321 Trace("sg-cconj-debug") << "...done evaluating term, got : " << grhs
<< std::endl
;
1322 if( !grhs
.isNull() ){
1324 Trace("sg-cconj-debug") << "Ground eqc for RHS : " << grhs
<< std::endl
;
1325 //check based on ground terms
1326 std::map
< TNode
, Node
>::iterator itl
= d_ground_eqc_map
.find( glhs
);
1327 if( itl
!=d_ground_eqc_map
.end() ){
1328 std::map
< TNode
, Node
>::iterator itr
= d_ground_eqc_map
.find( grhs
);
1329 if( itr
!=d_ground_eqc_map
.end() ){
1330 Trace("sg-cconj-debug") << "We have ground terms " << itl
->second
<< " and " << itr
->second
<< "." << std::endl
;
1331 if( itl
->second
.isConst() && itr
->second
.isConst() ){
1332 Trace("sg-cconj-debug") << "...disequal constants." << std::endl
;
1333 Trace("sg-cconj-witness") << " Witness of falsification : " << itl
->second
<< " != " << itr
->second
<< ", substutition is : " << std::endl
;
1334 for( std::map
< TNode
, TNode
>::iterator it
= subs
.begin(); it
!= subs
.end(); ++it
){
1335 Trace("sg-cconj-witness") << " " << it
->first
<< " -> " << it
->second
<< std::endl
;
1342 Trace("sg-cconj-debug") << "RHS is identical." << std::endl
;
1343 bool isGroundSubs
= true;
1344 for( std::map
< TNode
, TNode
>::iterator it
= subs
.begin(); it
!= subs
.end(); ++it
){
1345 std::map
< TNode
, Node
>::iterator git
= d_ground_eqc_map
.find( it
->second
);
1346 if( git
==d_ground_eqc_map
.end() ){
1347 isGroundSubs
= false;
1353 Trace("sg-cconj-witness") << " Witnessed " << glhs
<< " == " << grhs
<< ", substutition is : " << std::endl
;
1354 for( std::map
< TNode
, TNode
>::iterator it
= subs
.begin(); it
!= subs
.end(); ++it
){
1355 Trace("sg-cconj-witness") << " " << it
->first
<< " -> " << it
->second
<< std::endl
;
1356 if( std::find( d_subs_confirmWitnessDomain
[it
->first
].begin(), d_subs_confirmWitnessDomain
[it
->first
].end(), it
->second
)==d_subs_confirmWitnessDomain
[it
->first
].end() ){
1357 d_subs_confirmWitnessDomain
[it
->first
].push_back( it
->second
);
1360 d_subs_confirmCount
++;
1361 if( std::find( d_subs_confirmWitnessRange
.begin(), d_subs_confirmWitnessRange
.end(), glhs
)==d_subs_confirmWitnessRange
.end() ){
1362 d_subs_confirmWitnessRange
.push_back( glhs
);
1365 if( optFilterUnknown() ){
1366 Trace("sg-cconj-debug") << "...ground substitution giving terms that are neither equal nor disequal." << std::endl
;
1372 Trace("sg-cconj-debug") << "(could not ground eqc for RHS)." << std::endl
;
1382 void TermGenerator::reset( TermGenEnv
* s
, TypeNode tn
) {
1383 Assert( d_children
.empty() );
1388 Trace("sg-gen-tg-debug2") << "...add to context " << this << std::endl
;
1390 s
->changeContext( true );
1393 bool TermGenerator::getNextTerm( TermGenEnv
* s
, unsigned depth
) {
1394 if( Trace
.isOn("sg-gen-tg-debug2") ){
1395 Trace("sg-gen-tg-debug2") << this << " getNextTerm depth " << depth
<< " : status = " << d_status
<< ", num = " << d_status_num
;
1397 TNode f
= s
->getTgFunc( d_typ
, d_status_num
);
1398 Trace("sg-gen-tg-debug2") << ", f = " << f
;
1399 Trace("sg-gen-tg-debug2") << ", #args = " << s
->d_func_args
[f
].size();
1400 Trace("sg-gen-tg-debug2") << ", childNum = " << d_status_child_num
;
1401 Trace("sg-gen-tg-debug2") << ", #children = " << d_children
.size();
1403 Trace("sg-gen-tg-debug2") << std::endl
;
1408 if( !d_typ
.isNull() ){
1409 if( s
->allowVar( d_typ
) ){
1411 d_status_num
= s
->d_var_id
[d_typ
];
1413 Trace("sg-gen-tg-debug2") << this << " ...return unique var #" << d_status_num
<< std::endl
;
1414 return s
->considerCurrentTerm() ? true : getNextTerm( s
, depth
);
1416 //check allocating new variable
1419 if( s
->d_gen_relevant_terms
){
1422 return getNextTerm( s
, depth
);
1427 return getNextTerm( s
, depth
);
1429 }else if( d_status
==2 ){
1430 //cleanup previous information
1431 //if( d_status_num>=0 ){
1432 // s->d_var_eq_tg[d_status_num].pop_back();
1434 //check if there is another variable
1435 if( (d_status_num
+1)<(int)s
->getNumTgVars( d_typ
) ){
1437 //we have equated two variables
1438 //s->d_var_eq_tg[d_status_num].push_back( d_id );
1439 Trace("sg-gen-tg-debug2") << this << "...consider other var #" << d_status_num
<< std::endl
;
1440 return s
->considerCurrentTerm() ? true : getNextTerm( s
, depth
);
1442 if( s
->d_gen_relevant_terms
){
1446 return getNextTerm( s
, depth
);
1448 }else if( d_status
==4 ){
1450 if( depth
>0 && (d_status_num
+1)<(int)s
->getNumTgFuncs( d_typ
) ){
1452 d_status_child_num
= 0;
1453 Trace("sg-gen-tg-debug2") << this << "...consider function " << s
->getTgFunc( d_typ
, d_status_num
) << std::endl
;
1455 if( !s
->considerCurrentTerm() ){
1457 //don't consider this function
1460 //we have decided on a function application
1462 return getNextTerm( s
, depth
);
1464 //do not choose function applications at depth 0
1466 return getNextTerm( s
, depth
);
1468 }else if( d_status
==5 ){
1469 //iterating over arguments
1470 TNode f
= s
->getTgFunc( d_typ
, d_status_num
);
1471 if( d_status_child_num
<0 ){
1475 return getNextTerm( s
, depth
);
1476 }else if( d_status_child_num
==(int)s
->d_func_args
[f
].size() ){
1477 d_status_child_num
--;
1478 return s
->considerCurrentTermCanon( d_id
) ? true : getNextTerm( s
, depth
);
1481 Assert( d_status_child_num
<(int)s
->d_func_args
[f
].size() );
1482 if( d_status_child_num
==(int)d_children
.size() ){
1483 d_children
.push_back( s
->d_tg_id
);
1484 Assert( s
->d_tg_alloc
.find( s
->d_tg_id
)==s
->d_tg_alloc
.end() );
1485 s
->d_tg_alloc
[d_children
[d_status_child_num
]].reset( s
, s
->d_func_args
[f
][d_status_child_num
] );
1486 return getNextTerm( s
, depth
);
1488 Assert( d_status_child_num
+1==(int)d_children
.size() );
1489 if( s
->d_tg_alloc
[d_children
[d_status_child_num
]].getNextTerm( s
, depth
-1 ) ){
1490 d_status_child_num
++;
1491 return getNextTerm( s
, depth
);
1493 d_children
.pop_back();
1494 d_status_child_num
--;
1495 return getNextTerm( s
, depth
);
1499 }else if( d_status
==1 || d_status
==3 ){
1501 s
->removeVar( d_typ
);
1502 Assert( d_status_num
==(int)s
->d_var_id
[d_typ
] );
1503 //check if there is only one feasible equivalence class. if so, don't make pattern any more specific.
1504 //unsigned i = s->d_ccand_eqc[0].size()-1;
1505 //if( s->d_ccand_eqc[0][i].size()==1 && s->d_ccand_eqc[1][i].empty() ){
1507 // return getNextTerm( s, depth );
1513 return getNextTerm( s
, depth
);
1516 Assert( d_children
.empty() );
1517 Trace("sg-gen-tg-debug2") << "...remove from context " << this << std::endl
;
1518 s
->changeContext( false );
1519 Assert( d_id
==s
->d_tg_id
);
1524 void TermGenerator::resetMatching( TermGenEnv
* s
, TNode eqc
, unsigned mode
) {
1526 d_match_status_child_num
= 0;
1527 d_match_children
.clear();
1528 d_match_children_end
.clear();
1529 d_match_mode
= mode
;
1530 //if this term generalizes, it must generalize a non-ground term
1531 //if( (d_match_mode & ( 1 << 2 ))!=0 && s->isGroundEqc( eqc ) && d_status==5 ){
1532 // d_match_status = -1;
1536 bool TermGenerator::getNextMatch( TermGenEnv
* s
, TNode eqc
, std::map
< TypeNode
, std::map
< unsigned, TNode
> >& subs
, std::map
< TNode
, bool >& rev_subs
) {
1537 if( d_match_status
<0 ){
1540 if( Trace
.isOn("sg-gen-tg-match") ){
1541 Trace("sg-gen-tg-match") << "Matching ";
1542 debugPrint( s
, "sg-gen-tg-match", "sg-gen-tg-match" );
1543 Trace("sg-gen-tg-match") << " with eqc e" << s
->d_cg
->d_em
[eqc
] << "..." << std::endl
;
1544 Trace("sg-gen-tg-match") << " mstatus = " << d_match_status
;
1546 TNode f
= s
->getTgFunc( d_typ
, d_status_num
);
1547 Trace("sg-gen-tg-debug2") << ", f = " << f
;
1548 Trace("sg-gen-tg-debug2") << ", #args = " << s
->d_func_args
[f
].size();
1549 Trace("sg-gen-tg-debug2") << ", mchildNum = " << d_match_status_child_num
;
1550 Trace("sg-gen-tg-debug2") << ", #mchildren = " << d_match_children
.size();
1552 Trace("sg-gen-tg-debug2") << ", current substitution : {";
1553 for( std::map
< TypeNode
, std::map
< unsigned, TNode
> >::iterator itt
= subs
.begin(); itt
!= subs
.end(); ++itt
){
1554 for( std::map
< unsigned, TNode
>::iterator it
= itt
->second
.begin(); it
!= itt
->second
.end(); ++it
){
1555 Trace("sg-gen-tg-debug2") << " " << it
->first
<< " -> e" << s
->d_cg
->d_em
[it
->second
];
1558 Trace("sg-gen-tg-debug2") << " } " << std::endl
;
1562 if( d_match_status
==0 ){
1564 if( (d_match_mode
& ( 1 << 1 ))!=0 ){
1566 if( !s
->isGroundEqc( eqc
) ){
1569 }else if( (d_match_mode
& ( 1 << 2 ))!=0 ){
1570 //only non-ground terms
1571 //if( s->isGroundEqc( eqc ) ){
1575 //store the match : restricted if match_mode.0 = 1
1576 if( (d_match_mode
& ( 1 << 0 ))!=0 ){
1577 std::map
< TNode
, bool >::iterator it
= rev_subs
.find( eqc
);
1578 if( it
==rev_subs
.end() ){
1579 rev_subs
[eqc
] = true;
1584 Assert( subs
[d_typ
].find( d_status_num
)==subs
[d_typ
].end() );
1585 subs
[d_typ
][d_status_num
] = eqc
;
1589 subs
[d_typ
].erase( d_status_num
);
1590 if( (d_match_mode
& ( 1 << 0 ))!=0 ){
1591 rev_subs
.erase( eqc
);
1595 }else if( d_status
==2 ){
1596 if( d_match_status
==0 ){
1598 Assert( d_status_num
<(int)s
->getNumTgVars( d_typ
) );
1599 std::map
< unsigned, TNode
>::iterator it
= subs
[d_typ
].find( d_status_num
);
1600 Assert( it
!=subs
[d_typ
].end() );
1601 return it
->second
==eqc
;
1605 }else if( d_status
==5 ){
1606 //Assert( d_match_children.size()<=d_children.size() );
1607 //enumerating over f-applications in eqc
1608 if( d_match_status_child_num
<0 ){
1610 }else if( d_match_status
==0 ){
1611 //set up next binding
1612 if( d_match_status_child_num
==(int)d_match_children
.size() ){
1613 if( d_match_status_child_num
==0 ){
1615 TNode f
= s
->getTgFunc( d_typ
, d_status_num
);
1616 std::map
< TNode
, TermArgTrie
>::iterator it
= s
->getTermDatabase()->d_func_map_eqc_trie
[f
].d_data
.find( eqc
);
1617 if( it
!=s
->getTermDatabase()->d_func_map_eqc_trie
[f
].d_data
.end() ){
1618 d_match_children
.push_back( it
->second
.d_data
.begin() );
1619 d_match_children_end
.push_back( it
->second
.d_data
.end() );
1622 d_match_status_child_num
--;
1623 return getNextMatch( s
, eqc
, subs
, rev_subs
);
1626 d_match_children
.push_back( d_match_children
[d_match_status_child_num
-1]->second
.d_data
.begin() );
1627 d_match_children_end
.push_back( d_match_children
[d_match_status_child_num
-1]->second
.d_data
.end() );
1631 Assert( d_match_status_child_num
+1==(int)d_match_children
.size() );
1632 if( d_match_children
[d_match_status_child_num
]==d_match_children_end
[d_match_status_child_num
] ){
1633 //no more arguments to bind
1634 d_match_children
.pop_back();
1635 d_match_children_end
.pop_back();
1636 d_match_status_child_num
--;
1637 return getNextMatch( s
, eqc
, subs
, rev_subs
);
1639 if( d_match_status_child_num
==(int)d_children
.size() ){
1640 //successfully matched all children
1641 d_match_children
.pop_back();
1642 d_match_children_end
.pop_back();
1643 d_match_status_child_num
--;
1644 return true;//return d_match_children[d_match_status]!=d_match_children_end[d_match_status];
1647 s
->d_tg_alloc
[d_children
[d_match_status_child_num
]].resetMatching( s
, d_match_children
[d_match_status_child_num
]->first
, d_match_mode
);
1648 return getNextMatch( s
, eqc
, subs
, rev_subs
);
1652 Assert( d_match_status
==1 );
1653 Assert( d_match_status_child_num
+1==(int)d_match_children
.size() );
1654 Assert( d_match_children
[d_match_status_child_num
]!=d_match_children_end
[d_match_status_child_num
] );
1656 if( s
->d_tg_alloc
[d_children
[d_match_status_child_num
]].getNextMatch( s
, d_match_children
[d_match_status_child_num
]->first
, subs
, rev_subs
) ){
1657 d_match_status_child_num
++;
1658 return getNextMatch( s
, eqc
, subs
, rev_subs
);
1661 d_match_children
[d_match_status_child_num
]++;
1662 return getNextMatch( s
, eqc
, subs
, rev_subs
);
1670 unsigned TermGenerator::getDepth( TermGenEnv
* s
) {
1673 for( unsigned i
=0; i
<d_children
.size(); i
++ ){
1674 unsigned d
= s
->d_tg_alloc
[d_children
[i
]].getDepth( s
);
1685 unsigned TermGenerator::calculateGeneralizationDepth( TermGenEnv
* s
, std::map
< TypeNode
, std::vector
< int > >& fvs
) {
1688 for( unsigned i
=0; i
<d_children
.size(); i
++ ){
1689 sum
+= s
->d_tg_alloc
[d_children
[i
]].calculateGeneralizationDepth( s
, fvs
);
1693 Assert( d_status
==2 || d_status
==1 );
1694 std::map
< TypeNode
, std::vector
< int > >::iterator it
= fvs
.find( d_typ
);
1695 if( it
!=fvs
.end() ){
1696 if( std::find( it
->second
.begin(), it
->second
.end(), d_status_num
)!=it
->second
.end() ){
1700 fvs
[d_typ
].push_back( d_status_num
);
1705 unsigned TermGenerator::getGeneralizationDepth( TermGenEnv
* s
) {
1706 //if( s->d_gen_relevant_terms ){
1707 // return s->d_tg_gdepth;
1709 std::map
< TypeNode
, std::vector
< int > > fvs
;
1710 return calculateGeneralizationDepth( s
, fvs
);
1714 Node
TermGenerator::getTerm( TermGenEnv
* s
) {
1715 if( d_status
==1 || d_status
==2 ){
1716 Assert( !d_typ
.isNull() );
1717 return s
->getFreeVar( d_typ
, d_status_num
);
1718 }else if( d_status
==5 ){
1719 Node f
= s
->getTgFunc( d_typ
, d_status_num
);
1720 if( d_children
.size()==s
->d_func_args
[f
].size() ){
1721 std::vector
< Node
> children
;
1722 if( s
->d_tg_func_param
[f
] ){
1723 children
.push_back( f
);
1725 for( unsigned i
=0; i
<d_children
.size(); i
++ ){
1726 Node nc
= s
->d_tg_alloc
[d_children
[i
]].getTerm( s
);
1728 return Node::null();
1730 //Assert( nc.getType()==s->d_func_args[f][i] );
1731 children
.push_back( nc
);
1734 return NodeManager::currentNM()->mkNode( s
->d_func_kind
[f
], children
);
1739 return Node::null();
1742 void TermGenerator::debugPrint( TermGenEnv
* s
, const char * c
, const char * cd
) {
1743 Trace(cd
) << "[*" << d_id
<< "," << d_status
<< "]:";
1744 if( d_status
==1 || d_status
==2 ){
1745 Trace(c
) << s
->getFreeVar( d_typ
, d_status_num
);
1746 }else if( d_status
==5 ){
1747 TNode f
= s
->getTgFunc( d_typ
, d_status_num
);
1748 Trace(c
) << "(" << f
;
1749 for( unsigned i
=0; i
<d_children
.size(); i
++ ){
1751 s
->d_tg_alloc
[d_children
[i
]].debugPrint( s
, c
, cd
);
1753 if( d_children
.size()<s
->d_func_args
[f
].size() ){
1762 void TermGenEnv::collectSignatureInformation() {
1763 d_typ_tg_funcs
.clear();
1765 d_func_kind
.clear();
1766 d_func_args
.clear();
1768 for( std::map
< Node
, TermArgTrie
>::iterator it
= getTermDatabase()->d_func_map_trie
.begin(); it
!= getTermDatabase()->d_func_map_trie
.end(); ++it
){
1769 if( !getTermDatabase()->d_op_map
[it
->first
].empty() ){
1770 Node nn
= getTermDatabase()->d_op_map
[it
->first
][0];
1771 if( d_cg
->isHandledTerm( nn
) && nn
.getKind()!=APPLY_SELECTOR_TOTAL
&& !nn
.getType().isBoolean() ){
1772 bool do_enum
= true;
1773 //check if we have enumerated ground terms
1774 if( nn
.getKind()==APPLY_UF
){
1775 if( !d_cg
->hasEnumeratedUf( nn
) ){
1780 d_funcs
.push_back( it
->first
);
1781 for( unsigned i
=0; i
<nn
.getNumChildren(); i
++ ){
1782 d_func_args
[it
->first
].push_back( nn
[i
].getType() );
1784 d_func_kind
[it
->first
] = nn
.getKind();
1785 d_typ_tg_funcs
[tnull
].push_back( it
->first
);
1786 d_typ_tg_funcs
[nn
.getType()].push_back( it
->first
);
1787 d_tg_func_param
[it
->first
] = ( nn
.getMetaKind() == kind::metakind::PARAMETERIZED
);
1788 Trace("sg-rel-sig") << "Will enumerate function applications of : " << it
->first
<< ", #args = " << d_func_args
[it
->first
].size() << ", kind = " << nn
.getKind() << std::endl
;
1789 getTermDatabase()->computeUfEqcTerms( it
->first
);
1795 for( std::map
< TypeNode
, std::vector
< TNode
> >::iterator it
= d_typ_tg_funcs
.begin(); it
!= d_typ_tg_funcs
.end(); ++it
){
1796 std::random_shuffle( it
->second
.begin(), it
->second
.end() );
1797 if( it
->first
.isNull() ){
1798 Trace("sg-gen-tg-debug") << "In this order : ";
1799 for( unsigned i
=0; i
<it
->second
.size(); i
++ ){
1800 Trace("sg-gen-tg-debug") << it
->second
[i
] << " ";
1802 Trace("sg-gen-tg-debug") << std::endl
;
1807 void TermGenEnv::reset( unsigned depth
, bool genRelevant
, TypeNode tn
) {
1808 Assert( d_tg_alloc
.empty() );
1812 for( unsigned i
=0; i
<2; i
++ ){
1813 d_ccand_eqc
[i
].clear();
1814 d_ccand_eqc
[i
].push_back( d_relevant_eqc
[i
] );
1820 d_tg_gdepth_limit
= depth
;
1821 d_gen_relevant_terms
= genRelevant
;
1822 d_tg_alloc
[0].reset( this, tn
);
1825 bool TermGenEnv::getNextTerm() {
1826 if( d_tg_alloc
[0].getNextTerm( this, d_tg_gdepth_limit
) ){
1827 Assert( (int)d_tg_alloc
[0].getGeneralizationDepth( this )<=d_tg_gdepth_limit
);
1828 if( (int)d_tg_alloc
[0].getGeneralizationDepth( this )!=d_tg_gdepth_limit
){
1829 return getNextTerm();
1839 void TermGenEnv::resetMatching( TNode eqc
, unsigned mode
) {
1840 d_tg_alloc
[0].resetMatching( this, eqc
, mode
);
1844 bool TermGenEnv::getNextMatch( TNode eqc
, std::map
< TypeNode
, std::map
< unsigned, TNode
> >& subs
, std::map
< TNode
, bool >& rev_subs
) {
1845 return d_tg_alloc
[0].getNextMatch( this, eqc
, subs
, rev_subs
);
1849 Node
TermGenEnv::getTerm() {
1850 return d_tg_alloc
[0].getTerm( this );
1853 void TermGenEnv::debugPrint( const char * c
, const char * cd
) {
1854 d_tg_alloc
[0].debugPrint( this, c
, cd
);
1857 unsigned TermGenEnv::getNumTgVars( TypeNode tn
) {
1858 return d_var_id
[tn
];
1861 bool TermGenEnv::allowVar( TypeNode tn
) {
1862 std::map
< TypeNode
, unsigned >::iterator it
= d_var_limit
.find( tn
);
1863 if( it
==d_var_limit
.end() ){
1866 return d_var_id
[tn
]<it
->second
;
1870 void TermGenEnv::addVar( TypeNode tn
) {
1874 void TermGenEnv::removeVar( TypeNode tn
) {
1876 //d_var_eq_tg.pop_back();
1877 //d_var_tg.pop_back();
1880 unsigned TermGenEnv::getNumTgFuncs( TypeNode tn
) {
1881 return d_typ_tg_funcs
[tn
].size();
1884 TNode
TermGenEnv::getTgFunc( TypeNode tn
, unsigned i
) {
1885 return d_typ_tg_funcs
[tn
][i
];
1888 Node
TermGenEnv::getFreeVar( TypeNode tn
, unsigned i
) {
1889 return d_cg
->getFreeVar( tn
, i
);
1892 bool TermGenEnv::considerCurrentTerm() {
1893 Assert( !d_tg_alloc
.empty() );
1895 //if generalization depth is too large, don't consider it
1896 unsigned i
= d_tg_alloc
.size();
1897 Trace("sg-gen-tg-debug") << "Consider term ";
1898 d_tg_alloc
[0].debugPrint( this, "sg-gen-tg-debug", "sg-gen-tg-debug" );
1899 Trace("sg-gen-tg-debug") << "? curr term size = " << d_tg_alloc
.size() << ", last status = " << d_tg_alloc
[i
-1].d_status
;
1900 Trace("sg-gen-tg-debug") << std::endl
;
1902 if( d_tg_gdepth_limit
>=0 && d_tg_alloc
[0].getGeneralizationDepth( this )>(unsigned)d_tg_gdepth_limit
){
1903 Trace("sg-gen-consider-term") << "-> generalization depth of ";
1904 d_tg_alloc
[0].debugPrint( this, "sg-gen-consider-term", "sg-gen-tg-debug" );
1905 Trace("sg-gen-consider-term") << " is too high " << d_tg_gdepth
<< " " << d_tg_alloc
[0].getGeneralizationDepth( this ) << ", do not consider." << std::endl
;
1911 if( d_tg_alloc[i-1].d_status==1 ){
1912 }else if( d_tg_alloc[i-1].d_status==2 ){
1913 }else if( d_tg_alloc[i-1].d_status==5 ){
1915 Trace("sg-gen-tg-debug") << "Bad tg: " << &d_tg_alloc[i-1] << std::endl;
1919 //if equated two variables, first check if context-independent TODO
1920 //----end optimizations
1923 //check based on which candidate equivalence classes match
1924 if( d_gen_relevant_terms
){
1925 Trace("sg-gen-tg-debug") << "Filter based on relevant ground EQC";
1926 Trace("sg-gen-tg-debug") << ", #eqc to try = " << d_ccand_eqc
[0][i
-1].size() << "/" << d_ccand_eqc
[1][i
-1].size() << std::endl
;
1928 Assert( d_ccand_eqc
[0].size()>=2 );
1929 Assert( d_ccand_eqc
[0].size()==d_ccand_eqc
[1].size() );
1930 Assert( d_ccand_eqc
[0].size()==d_tg_id
+1 );
1931 Assert( d_tg_id
==d_tg_alloc
.size() );
1932 for( unsigned r
=0; r
<2; r
++ ){
1933 d_ccand_eqc
[r
][i
].clear();
1936 //re-check feasibility of EQC
1937 for( unsigned r
=0; r
<2; r
++ ){
1938 for( unsigned j
=0; j
<d_ccand_eqc
[r
][i
-1].size(); j
++ ){
1939 std::map
< TypeNode
, std::map
< unsigned, TNode
> > subs
;
1940 std::map
< TNode
, bool > rev_subs
;
1943 mode
= d_cg
->optReqDistinctVarPatterns() ? ( 1 << 0 ) : 0;
1944 mode
= mode
| (1 << 2 );
1948 d_tg_alloc
[0].resetMatching( this, d_ccand_eqc
[r
][i
-1][j
], mode
);
1949 if( d_tg_alloc
[0].getNextMatch( this, d_ccand_eqc
[r
][i
-1][j
], subs
, rev_subs
) ){
1950 d_ccand_eqc
[r
][i
].push_back( d_ccand_eqc
[r
][i
-1][j
] );
1954 for( unsigned r
=0; r
<2; r
++ ){
1955 Trace("sg-gen-tg-debug") << "Current eqc of type " << r
<< " : ";
1956 for( unsigned j
=0; j
<d_ccand_eqc
[r
][i
].size(); j
++ ){
1957 Trace("sg-gen-tg-debug") << "e" << d_cg
->d_em
[d_ccand_eqc
[r
][i
][j
]] << " ";
1959 Trace("sg-gen-tg-debug") << std::endl
;
1961 if( options::conjectureFilterActiveTerms() && d_ccand_eqc
[0][i
].empty() ){
1962 Trace("sg-gen-consider-term") << "Do not consider term of form ";
1963 d_tg_alloc
[0].debugPrint( this, "sg-gen-consider-term", "sg-gen-consider-term-debug" );
1964 Trace("sg-gen-consider-term") << " since no relevant EQC matches it." << std::endl
;
1967 if( options::conjectureFilterModel() && d_ccand_eqc
[1][i
].empty() ){
1968 Trace("sg-gen-consider-term") << "Do not consider term of form ";
1969 d_tg_alloc
[0].debugPrint( this, "sg-gen-consider-term", "sg-gen-consider-term-debug" );
1970 Trace("sg-gen-consider-term") << " since no ground EQC matches it." << std::endl
;
1974 Trace("sg-gen-tg-debug") << "Will consider term ";
1975 d_tg_alloc
[0].debugPrint( this, "sg-gen-tg-debug", "sg-gen-tg-debug" );
1976 Trace("sg-gen-tg-debug") << std::endl
;
1977 Trace("sg-gen-consider-term-debug") << std::endl
;
1981 void TermGenEnv::changeContext( bool add
) {
1983 for( unsigned r
=0; r
<2; r
++ ){
1984 d_ccand_eqc
[r
].push_back( std::vector
< TNode
>() );
1988 for( unsigned r
=0; r
<2; r
++ ){
1989 d_ccand_eqc
[r
].pop_back();
1992 Assert( d_tg_alloc
.find( d_tg_id
)!=d_tg_alloc
.end() );
1993 d_tg_alloc
.erase( d_tg_id
);
1997 bool TermGenEnv::considerCurrentTermCanon( unsigned tg_id
){
1998 Assert( tg_id
<d_tg_alloc
.size() );
1999 if( options::conjectureFilterCanonical() ){
2000 //check based on a canonicity of the term (if there is one)
2001 Trace("sg-gen-tg-debug") << "Consider term canon ";
2002 d_tg_alloc
[0].debugPrint( this, "sg-gen-tg-debug", "sg-gen-tg-debug" );
2003 Trace("sg-gen-tg-debug") << ", tg is [" << tg_id
<< "]..." << std::endl
;
2005 Node ln
= d_tg_alloc
[tg_id
].getTerm( this );
2006 Trace("sg-gen-tg-debug") << "Term is " << ln
<< std::endl
;
2007 return d_cg
->considerTermCanon( ln
, d_gen_relevant_terms
);
2012 bool TermGenEnv::isRelevantFunc( Node f
) {
2013 return std::find( d_funcs
.begin(), d_funcs
.end(), f
)!=d_funcs
.end();
2015 TermDb
* TermGenEnv::getTermDatabase() {
2016 return d_cg
->getTermDatabase();
2018 Node
TermGenEnv::getGroundEqc( TNode r
) {
2019 return d_cg
->getGroundEqc( r
);
2021 bool TermGenEnv::isGroundEqc( TNode r
){
2022 return d_cg
->isGroundEqc( r
);
2024 bool TermGenEnv::isGroundTerm( TNode n
){
2025 return d_cg
->isGroundTerm( n
);
2029 void SubstitutionIndex::addSubstitution( TNode eqc
, std::vector
< TNode
>& vars
, std::vector
< TNode
>& terms
, unsigned i
) {
2030 if( i
==vars
.size() ){
2033 Assert( d_var
.isNull() || d_var
==vars
[i
] );
2035 d_children
[terms
[i
]].addSubstitution( eqc
, vars
, terms
, i
+1 );
2039 bool SubstitutionIndex::notifySubstitutions( ConjectureGenerator
* s
, std::map
< TNode
, TNode
>& subs
, TNode rhs
, unsigned numVars
, unsigned i
) {
2041 Assert( d_children
.empty() );
2042 return s
->notifySubstitution( d_var
, subs
, rhs
);
2044 Assert( i
==0 || !d_children
.empty() );
2045 for( std::map
< TNode
, SubstitutionIndex
>::iterator it
= d_children
.begin(); it
!= d_children
.end(); ++it
){
2046 Trace("sg-cconj-debug2") << "Try " << d_var
<< " -> " << it
->first
<< " (" << i
<< "/" << numVars
<< ")" << std::endl
;
2047 subs
[d_var
] = it
->first
;
2048 if( !it
->second
.notifySubstitutions( s
, subs
, rhs
, numVars
, i
+1 ) ){
2057 void TheoremIndex::addTheorem( std::vector
< TNode
>& lhs_v
, std::vector
< unsigned >& lhs_arg
, TNode rhs
){
2058 if( lhs_v
.empty() ){
2059 if( std::find( d_terms
.begin(), d_terms
.end(), rhs
)==d_terms
.end() ){
2060 d_terms
.push_back( rhs
);
2063 unsigned index
= lhs_v
.size()-1;
2064 if( lhs_arg
[index
]==lhs_v
[index
].getNumChildren() ){
2067 addTheorem( lhs_v
, lhs_arg
, rhs
);
2070 addTheoremNode( lhs_v
[index
][lhs_arg
[index
]-1], lhs_v
, lhs_arg
, rhs
);
2075 void TheoremIndex::addTheoremNode( TNode curr
, std::vector
< TNode
>& lhs_v
, std::vector
< unsigned >& lhs_arg
, TNode rhs
){
2076 Trace("thm-db-debug") << "Adding conjecture for subterm " << curr
<< "..." << std::endl
;
2077 if( curr
.hasOperator() ){
2078 lhs_v
.push_back( curr
);
2079 lhs_arg
.push_back( 0 );
2080 d_children
[curr
.getOperator()].addTheorem( lhs_v
, lhs_arg
, rhs
);
2082 Assert( curr
.getKind()==kind::BOUND_VARIABLE
);
2083 TypeNode tn
= curr
.getType();
2084 Assert( d_var
[tn
].isNull() || d_var
[tn
]==curr
);
2086 d_children
[curr
].addTheorem( lhs_v
, lhs_arg
, rhs
);
2090 void TheoremIndex::getEquivalentTerms( std::vector
< TNode
>& n_v
, std::vector
< unsigned >& n_arg
,
2091 std::map
< TNode
, TNode
>& smap
, std::vector
< TNode
>& vars
, std::vector
< TNode
>& subs
,
2092 std::vector
< Node
>& terms
) {
2093 Trace("thm-db-debug") << "Get equivalent terms " << n_v
.size() << " " << n_arg
.size() << std::endl
;
2095 Trace("thm-db-debug") << "Number of terms : " << d_terms
.size() << std::endl
;
2096 //apply substutitions to RHS's
2097 for( unsigned i
=0; i
<d_terms
.size(); i
++ ){
2098 Node n
= d_terms
[i
].substitute( vars
.begin(), vars
.end(), subs
.begin(), subs
.end() );
2099 terms
.push_back( n
);
2102 unsigned index
= n_v
.size()-1;
2103 if( n_arg
[index
]==n_v
[index
].getNumChildren() ){
2106 getEquivalentTerms( n_v
, n_arg
, smap
, vars
, subs
, terms
);
2109 getEquivalentTermsNode( n_v
[index
][n_arg
[index
]-1], n_v
, n_arg
, smap
, vars
, subs
, terms
);
2114 void TheoremIndex::getEquivalentTermsNode( Node curr
, std::vector
< TNode
>& n_v
, std::vector
< unsigned >& n_arg
,
2115 std::map
< TNode
, TNode
>& smap
, std::vector
< TNode
>& vars
, std::vector
< TNode
>& subs
,
2116 std::vector
< Node
>& terms
) {
2117 Trace("thm-db-debug") << "Get equivalent based on subterm " << curr
<< "..." << std::endl
;
2118 if( curr
.hasOperator() ){
2119 Trace("thm-db-debug") << "Check based on operator..." << std::endl
;
2120 std::map
< TNode
, TheoremIndex
>::iterator it
= d_children
.find( curr
.getOperator() );
2121 if( it
!=d_children
.end() ){
2122 n_v
.push_back( curr
);
2123 n_arg
.push_back( 0 );
2124 it
->second
.getEquivalentTerms( n_v
, n_arg
, smap
, vars
, subs
, terms
);
2126 Trace("thm-db-debug") << "...done check based on operator" << std::endl
;
2128 TypeNode tn
= curr
.getType();
2129 std::map
< TypeNode
, TNode
>::iterator itt
= d_var
.find( tn
);
2130 if( itt
!=d_var
.end() ){
2131 Trace("thm-db-debug") << "Check for substitution with " << itt
->second
<< "..." << std::endl
;
2132 Assert( curr
.getType()==itt
->second
.getType() );
2133 //add to substitution if possible
2134 bool success
= false;
2135 std::map
< TNode
, TNode
>::iterator it
= smap
.find( itt
->second
);
2136 if( it
==smap
.end() ){
2137 smap
[itt
->second
] = curr
;
2138 vars
.push_back( itt
->second
);
2139 subs
.push_back( curr
);
2141 }else if( it
->second
==curr
){
2144 //also check modulo equality (in universal equality engine)
2146 Trace("thm-db-debug") << "...check for substitution with " << itt
->second
<< ", success = " << success
<< "." << std::endl
;
2148 d_children
[itt
->second
].getEquivalentTerms( n_v
, n_arg
, smap
, vars
, subs
, terms
);
2153 void TheoremIndex::debugPrint( const char * c
, unsigned ind
) {
2154 for( std::map
< TNode
, TheoremIndex
>::iterator it
= d_children
.begin(); it
!= d_children
.end(); ++it
){
2155 for( unsigned i
=0; i
<ind
; i
++ ){ Trace(c
) << " "; }
2156 Trace(c
) << it
->first
<< std::endl
;
2157 it
->second
.debugPrint( c
, ind
+1 );
2159 if( !d_terms
.empty() ){
2160 for( unsigned i
=0; i
<ind
; i
++ ){ Trace(c
) << " "; }
2162 for( unsigned i
=0; i
<d_terms
.size(); i
++ ){
2163 Trace(c
) << " " << d_terms
[i
];
2165 Trace(c
) << " }" << std::endl
;
2167 //if( !d_var.isNull() ){
2168 // for( unsigned i=0; i<ind; i++ ){ Trace(c) << " "; }
2169 // Trace(c) << "var:" << d_var << std::endl;
2173 bool ConjectureGenerator::optReqDistinctVarPatterns() { return false; }
2174 bool ConjectureGenerator::optFilterUnknown() { return true; } //may change
2175 int ConjectureGenerator::optFilterScoreThreshold() { return 1; }
2176 unsigned ConjectureGenerator::optFullCheckFrequency() { return 1; }
2178 bool ConjectureGenerator::optStatsOnly() { return false; }