We assume that you grasped the way that ProB setups up the initial states of a B machine as outlined in Tutorial Setup Phases, and have understood why animation is difficult as outlined in Tutorial Understanding the Complexity of B Animation.
First, it is important to understand that as of version 1.3, ProB uses three representations for sets and relations:
The third representation allows ProB to deal with larger sets and relations (starting at around 100,000 elements performance will start to slow down). However, this representation can only be used if the constraint solver knows exactly all elements of a set or relation. The list representation will be used if only part of the set or relation is known. The symbolic representation can be used to represent certain sets (and set comprehensions) symbolically (such as large intervals, and to some very limited extent also infinite sets).
To give an idea about the performance related to the new AVL-based representation, take the substitution
numbers := numbers - ran(%n.(n:cur..limit/cur|cur*n))
For example, if the PROPERTIES contains the following predicates
then all values will be fully known and the new AVL-based representation will be used. However, for example for
we would only have partial information about x (which the constraint solver will encode using the list [1|->_, 2|->_] where the underscore marks as of yet unknown bits.
Basically, the ProB constraint solver works as follows:
The priority is a rough estimate of the number of possible solutions; predicates with the lowest estimate will be "executed" first.
The following picture provides a very rough picture of this process:
Let us use the following B machine as starting point:
MACHINE Propagation CONSTANTS low,up,f,fi,invert PROPERTIES f:low..up --> 0..1 & invert: 0..1 --> 0..1 & fi = (f ; invert) & invert = {0|->1,1|->0} & low = 0 & up = 3 & f = {0 |-> 1, 1|->1, 2|->0, 3|->1} VARIABLES xx INVARIANT xx:low..up INITIALISATION xx:=low OPERATIONS val <-- Sense = BEGIN val := f(xx) END; Inc = IF xx<up THEN xx := xx+1 ELSE xx := low END END
After loading the file, we get offered one possible way to setup the constants:
In this case, ProB performed no enumeration at all. It could immediately determine the values of all constants. We can inspect the behaviour of the constraint solver by choosing "Advanced Preferences..." in the Preferences menu and the turning the "Provide various tracing information on the terminal/console" on. If we now reopen the machine, the following information will be printed (amongst others) on the console:
... % checking_properties ====> low = 0 -->> low val/1: int(0) ====> up = 3 -->> up val/1: int(3) ====> invert = {0 |-> 1,1 |-> 0} -->> invert val/1: AVL.size=2 ====> f = {0 |-> 1,1 |-> 1,2 |-> 0,3 |-> 1} -->> f val/1: AVL.size=4 ====> f : low .. up --> 0 .. 1 ====> invert : 0 .. 1 --> 0 .. 1 ====> fi = f ; invert -->> fi val/1: AVL.size=4 -->> SUCCESS; all constants valued % start_enumerating_constants % grounding_wait_flags_for_constants(wfx(_57893,_66619,_56937)) % found_enumeration_of_constants(10) ...
The arrow -->> shows us when an AVL-representation was found for a constant. The ====> tells us when a predicate is interpreted. You can also see that all constants have been valued before enumeration was started. You can also see that ProB does not treat the predicates in the same order as you have typed them in the PROPERTIES clause: indeed, ProB sorts the properties (e.g., moving equalities earlier) before starting the interpretation. As of version 1.3.2 the PROPERTIES are also partitioned, to improve performance and make it easier to locate inconsistencies.
After selecting SETUP_CONSTANTS and INITIALISATION, we obtain the following state, and we can ensure ourselves that the values found for f and fi are correct (by looking at the "State Properties" pane):
To make the example more challenging, let us increase up to 100 and remove the equality for f. In other words, the PROPERTIES will now look like this:
f:low..up --> 0..1 & invert: 0..1 --> 0..1 & fi = (f ; invert) & invert = {0|->1,1|->0} & low = 0 & up = 100
Saving and reopening the machine, results in the following picture:
You can see that ProB had no problem with this machine. Still, one may wonder why it only proposes four solutions for the constants. Indeed, there should be 2^101 = 2.54e+30 different solutions (over 2 thousand billion billion billion solutions). In fact, as quite often there are a lot of solutions for the constants, ProB (luckily) does not compute all of them. Indeed, there is a cut-off point after which it will no longer search for more solutions. In this case, the orange button "max" appears in the "Enabled Operatoins" pane. The cut-off itself is controlled by a preference. You can either
There is a similar preference to control how many solutions are found for ways to execute an operation (SET_PREF_MAX_OPERATIONS). (As a side note: these preferences can also be set to 0. This means that you will have to use "Execute an Operation..." in the Animate menu to add transitions one by one.)
One may wonder what happens if there are no solutions. Will not ProB have to examine all of these solutions? The answer is: sometimes yes. Let us add the predicate f=fi to the PROPERTIES, which are now unsatisfiable. If we save and reopen the machine, we get the following picture:
If we answer yes, the Properties will be debugged by adding conjuncts one-by-one. We obtain the following:
This will hopefully help us pinpoint the error in our properties. See Tutorial Troubleshooting the Setup for more ways to locate problems in the properties. After clicking Done, we get the following; observe the orange timeout button in the "State Properties" pane:
Note that one can click on the orange Timeout button: this will offer us the option to try to find solutions without a timeout. You should use this option with care: it can lock up your computer for a long time. You should be able to stop the computation by selecting ProB's terminal window and hitting CONTROL-C. Hopefully, the following message will appear in the terminal:
^CProlog interruption (h for help)?
You can now type A followed by RETURN. This will typically raise an error message, which you can dismiss. Afterwards, you should be able to continue working.
Finally, one can control the time after which a timeout will trigger using the preference "Time out for computing enabled transitions (in ms)" in the Animation Preferences. You can also control it using a definition for SET_PREF_TIME_OUT in the DEFINITIONS clause.
Let us replace f=fi by the following property:
!x.(x:low..up => f(x)=x mod 2)
If we save and reopen, we obtain the following picture:
ProB had no trouble finding a solution here. Indeed, it expanded the universally quantified formula before starting enumeration of f. Here, it is important to understand when a universally quantified formula !x.(P=>Q) is expanded:
Take for example,
!x.(x:dom(f) => f(x)=x mod 2)
Here, the first case does not apply, as f is not yet fully known. Luckily, the formula is in of the second form and ProB will still be able to quickly find the single solution.
However, if we use
!x.(#y.(x|->y:f) => f(x)=x mod 2)
then none of the cases apply, and ProB will delay checking the formula until it knows f. As there are 2^101 possibilities, this will lead to a timeout.
Finally, in the current version of ProB (1.3) it is better to use !x.(x:low..up => f(x)=x mod 2) rather than the seemingly equivalent !x.(x>=low & x<=up => f(x)=x mod 2). Especially for large values of MAXINT and MININT, the former version is currently more efficient. (But we are planning to overcome this issue.)