1 /******************************************************************************
2 * Top contributors (to current version):
3 * Andrew Reynolds, Mathias Preiner, Tim King
5 * This file is part of the cvc5 project.
7 * Copyright (c) 2009-2021 by the authors listed in the file AUTHORS
8 * in the top-level source directory and their institutional affiliations.
9 * All rights reserved. See the file COPYING in the top-level source
10 * directory for licensing information.
11 * ****************************************************************************
16 #include "cvc4_private.h"
18 #ifndef CVC5__THEORY__THEORY_MODEL_BUILDER_H
19 #define CVC5__THEORY__THEORY_MODEL_BUILDER_H
21 #include <unordered_map>
22 #include <unordered_set>
24 #include "theory/theory_model.h"
32 /** TheoryEngineModelBuilder class
34 * This is the class used by TheoryEngine
35 * for constructing TheoryModel objects, which is the class
36 * that represents models for a set of assertions.
38 * A call to TheoryEngineModelBuilder::buildModel(...) is made
39 * after a full effort check passes with no theory solvers
40 * adding lemmas or conflicts, and theory combination passes
41 * with no splits on shared terms. If buildModel is successful,
42 * this will set up the data structures in TheoryModel to represent
43 * a model for the current set of assertions.
45 class TheoryEngineModelBuilder
47 typedef std::unordered_map
<Node
, Node
, NodeHashFunction
> NodeMap
;
48 typedef std::unordered_set
<Node
, NodeHashFunction
> NodeSet
;
51 TheoryEngineModelBuilder();
52 virtual ~TheoryEngineModelBuilder() {}
54 * Should be called only on models m after they have been prepared
55 * (e.g. using ModelManager). In other words, the equality engine of model
56 * m contains all relevant information from each theory that is needed
57 * for building a model. This class is responsible simply for ensuring
58 * that all equivalence classes of the equality engine of m are assigned
61 * This constructs the model m, via the following steps:
62 * (1) builder-specified pre-processing,
63 * (2) find the equivalence classes of m's
64 * equality engine that initially contain constants,
65 * (3) assign constants to all equivalence classes
66 * of m's equality engine, through alternating
67 * iterations of evaluation and enumeration,
68 * (4) builder-specific processing, which includes assigning total
69 * interpretations to uninterpreted functions.
71 * This function returns false if any of the above
72 * steps results in a lemma sent on an output channel.
73 * Lemmas may be sent on an output channel by this
74 * builder in steps (2) or (5), for instance, if the model we
75 * are building fails to satisfy a quantified formula.
77 * @param m The model to build
78 * @return true if the model was successfully built.
80 bool buildModel(TheoryModel
* m
);
84 * This is called when m is a model that will be returned to the user. This
85 * method checks the internal consistency of the model if we are in a debug
88 void postProcessModel(bool incomplete
, TheoryModel
* m
);
92 //-----------------------------------virtual functions
93 /** pre-process build model
94 * Do pre-processing specific to this model builder.
95 * Called in step (2) of the build construction,
98 virtual bool preProcessBuildModel(TheoryModel
* m
);
99 /** process build model
100 * Do processing specific to this model builder.
101 * Called in step (5) of the build construction,
103 * By default, this assigns values to each function
104 * that appears in m's equality engine.
106 virtual bool processBuildModel(TheoryModel
* m
);
108 * Check assertions and printing debug information for the model.
109 * Calls after step (5) described above is complete.
111 virtual void debugModel(TheoryModel
* m
) {}
112 //-----------------------------------end virtual functions
114 /** Debug check model.
116 * This throws an assertion failure if the model contains an equivalence
117 * class with two terms t1 and t2 such that t1^M != t2^M.
119 void debugCheckModel(TheoryModel
* m
);
121 /** Evaluate equivalence class
123 * If this method returns a non-null node c, then c is a constant and some
124 * term in the equivalence class of r evaluates to c based on the current
125 * state of the model m.
127 Node
evaluateEqc(TheoryModel
* m
, TNode r
);
128 /** is n an assignable expression?
130 * A term n is an assignable expression if its value is unconstrained by a
131 * standard model. Examples of assignable terms are:
133 * - applications of array select,
134 * - applications of datatype selectors,
135 * - applications of uninterpreted functions.
136 * Assignable terms must be first-order, that is, all instances of the above
137 * terms are not assignable if they have a higher-order (function) type.
139 bool isAssignable(TNode n
);
140 /** add assignable subterms
141 * Adds all assignable subterms of n to tm's equality engine.
143 void addAssignableSubterms(TNode n
, TheoryModel
* tm
, NodeSet
& cache
);
144 /** normalize representative r
146 * This returns a term that is equivalent to r's
147 * interpretation in the model m. It may do so
148 * by rewriting the application of r's operator to the
149 * result of normalizing each of r's children, if
150 * each child is constant.
152 Node
normalize(TheoryModel
* m
, TNode r
, bool evalOnly
);
153 /** assign constant representative
155 * Called when equivalence class eqc is assigned a constant
156 * representative const_rep.
158 * eqc should be a representative of tm's equality engine.
160 void assignConstantRep(TheoryModel
* tm
, Node eqc
, Node const_rep
);
163 * This adds to type_list the list of types that tn is built from.
164 * For example, if tn is (Array Int Bool) and type_list is empty,
165 * then we append ( Int, Bool, (Array Int Bool) ) to type_list.
169 std::vector
<TypeNode
>& type_list
,
170 std::unordered_set
<TypeNode
, TypeNodeHashFunction
>& visiting
);
171 /** assign function f based on the model m.
172 * This construction is based on "table form". For example:
178 * f = (lambda xy. (ite (and (= x 0) (= y 1)) 1
179 * (ite (and (= x 0) (= y 2)) 2
180 * (ite (and (= x 1) (= y 1)) 3 ...)))
182 void assignFunction(TheoryModel
* m
, Node f
);
183 /** assign function f based on the model m.
184 * This construction is based on "dag form". For example:
190 * f = (lambda xy. (ite (= x 0) (ite (= y 1) 1
191 * (ite (= y 2) 2 ...))
192 * (ite (= x 1) (ite (= y 1) 3 ...)
195 * where the above is represented as a directed acyclic graph (dag).
196 * This construction is accomplished by assigning values to (f c)
197 * terms before f, e.g.
198 * (f 0) = (lambda y. (ite (= y 1) 1
199 * (ite (= y 2) 2 ...))
200 * (f 1) = (lambda y. (ite (= y 1) 3 ...))
202 * f = (lambda xy. (ite (= x 0) ((f 0) y)
203 * (ite (= x 1) ((f 1) y) ...))
205 void assignHoFunction(TheoryModel
* m
, Node f
);
208 * Assign all unassigned functions in the model m (those returned by
209 * TheoryModel::getFunctionsToAssign),
210 * using the two functions above. Currently:
211 * If ufHo is disabled, we call assignFunction for all functions.
212 * If ufHo is enabled, we call assignHoFunction.
214 void assignFunctions(TheoryModel
* m
);
218 * A temporary cache mapping terms to their
219 * normalized form, used during buildModel.
221 NodeMap d_normalizedCache
;
222 /** mapping from terms to the constant associated with their equivalence class
224 std::map
<Node
, Node
> d_constantReps
;
226 /** Theory engine model builder assigner class
228 * This manages the assignment of values to terms of a given type.
229 * In particular, it is a wrapper around a type enumerator that is restricted
230 * by a set of values that it cannot generate, called an "assignment exclusion
236 Assigner() : d_te(nullptr), d_isActive(false) {}
238 * Initialize this assigner to generate values of type tn, with properties
239 * tep and assignment exclusion set aes.
241 void initialize(TypeNode tn
,
242 TypeEnumeratorProperties
* tep
,
243 const std::vector
<Node
>& aes
);
244 /** get the next term in the enumeration
246 * This method returns the next legal term based on type enumeration, where
247 * a term is legal it does not belong to the assignment exclusion set of
248 * this assigner. If no more terms exist, this method returns null. This
249 * should never be the case due to the conditions ensured by theory solvers
250 * for finite types. If it is the case, we give an assertion failure.
252 Node
getNextAssignment();
253 /** The type enumerator */
254 std::unique_ptr
<TypeEnumerator
> d_te
;
255 /** The assignment exclusion set of this Assigner */
256 std::vector
<Node
> d_assignExcSet
;
258 * Is active, flag set to true when all values in d_assignExcSet are
263 /** Is the given Assigner ready to assign values?
265 * This returns true if all values in the assignment exclusion set of a have
266 * a known value according to the state of this model builder (via a lookup
267 * in d_constantReps). It updates the assignment exclusion vector of a to
268 * these values whenever possible.
270 bool isAssignerActive(TheoryModel
* tm
, Assigner
& a
);
271 //------------------------------------for codatatypes
272 /** is v an excluded codatatype value?
274 * If this returns true, then v is a value
275 * that cannot be used during enumeration in step (4)
276 * of model construction.
278 * repSet is the set of representatives of the same type as v,
279 * assertedReps is a map from representatives t,
280 * eqc is the equivalence class that v reside.
282 * This function is used to avoid alpha-equivalent
283 * assignments for codatatype terms, described in
284 * Reynolds/Blanchette CADE 2015. In particular,
285 * this function returns true if v is in
286 * the set V^{x}_I from page 9, where x is eqc
287 * and I is the model we are building.
289 bool isExcludedCdtValue(Node v
,
290 std::set
<Node
>* repSet
,
291 std::map
<Node
, Node
>& assertedReps
,
293 /** is codatatype value match
295 * This returns true if v is r{ eqc -> t } for some t.
296 * If this function returns true, then t above is
299 bool isCdtValueMatch(Node v
, Node r
, Node eqc
, Node
& eqc_m
);
300 //------------------------------------end for codatatypes
302 //---------------------------------for debugging finite model finding
303 /** does type tn involve an uninterpreted sort? */
304 bool involvesUSort(TypeNode tn
);
305 /** is v an excluded value based on uninterpreted sorts?
306 * This gives an assertion failure in the case that v contains
307 * an uninterpreted constant whose index is out of the bounds
308 * specified by eqc_usort_count.
310 bool isExcludedUSortValue(std::map
<TypeNode
, unsigned>& eqc_usort_count
,
312 std::map
<Node
, bool>& visited
);
313 //---------------------------------end for debugging finite model finding
315 }; /* class TheoryEngineModelBuilder */
317 } // namespace theory
320 #endif /* CVC5__THEORY__THEORY_MODEL_BUILDER_H */