You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
450 lines
13 KiB
450 lines
13 KiB
//==============================================================================
|
|
//
|
|
// Copyright (c) 2002-
|
|
// Authors:
|
|
// * Dave Parker <david.parker@comlab.ox.ac.uk> (University of Oxford)
|
|
//
|
|
//------------------------------------------------------------------------------
|
|
//
|
|
// This file is part of PRISM.
|
|
//
|
|
// PRISM is free software; you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation; either version 2 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// PRISM is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with PRISM; if not, write to the Free Software Foundation,
|
|
// Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
//
|
|
//==============================================================================
|
|
|
|
package explicit;
|
|
|
|
import java.io.File;
|
|
import java.util.ArrayList;
|
|
import java.util.BitSet;
|
|
import java.util.Collections;
|
|
import java.util.Iterator;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
import java.util.function.IntPredicate;
|
|
|
|
import explicit.graphviz.Decorator;
|
|
import parser.State;
|
|
import parser.Values;
|
|
import parser.VarList;
|
|
import prism.ModelType;
|
|
import prism.PrismException;
|
|
import prism.PrismFileLog;
|
|
import prism.PrismLog;
|
|
|
|
/**
|
|
* Interface for (abstract) classes that provide (read-only) access to an explicit-state model.
|
|
*/
|
|
public interface Model
|
|
{
|
|
// Accessors
|
|
|
|
/**
|
|
* Get the type of this model.
|
|
*/
|
|
public ModelType getModelType();
|
|
|
|
/**
|
|
* Get the number of states.
|
|
*/
|
|
public int getNumStates();
|
|
|
|
/**
|
|
* Get the number of initial states.
|
|
*/
|
|
public int getNumInitialStates();
|
|
|
|
/**
|
|
* Get iterator over initial state list.
|
|
*/
|
|
public Iterable<Integer> getInitialStates();
|
|
|
|
/**
|
|
* Get the index of the first initial state
|
|
* (i.e. the one with the lowest index).
|
|
* Returns -1 if there are no initial states.
|
|
*/
|
|
public int getFirstInitialState();
|
|
|
|
/**
|
|
* Check whether a state is an initial state.
|
|
*/
|
|
public boolean isInitialState(int i);
|
|
|
|
/**
|
|
* Get the number of states that are/were deadlocks.
|
|
* (Such states may have been fixed at build-time by adding self-loops)
|
|
*/
|
|
public int getNumDeadlockStates();
|
|
|
|
/**
|
|
* Get iterator over states that are/were deadlocks.
|
|
* (Such states may have been fixed at build-time by adding self-loops)
|
|
*/
|
|
public Iterable<Integer> getDeadlockStates();
|
|
|
|
/**
|
|
* Get list of states that are/were deadlocks.
|
|
* (Such states may have been fixed at build-time by adding self-loops)
|
|
*/
|
|
public StateValues getDeadlockStatesList();
|
|
|
|
/**
|
|
* Get the index of the first state that is/was a deadlock.
|
|
* (i.e. the one with the lowest index).
|
|
* Returns -1 if there are no initial states.
|
|
*/
|
|
public int getFirstDeadlockState();
|
|
|
|
/**
|
|
* Check whether a state is/was deadlock.
|
|
* (Such states may have been fixed at build-time by adding self-loops)
|
|
*/
|
|
public boolean isDeadlockState(int i);
|
|
|
|
/**
|
|
* Get access to a list of states (optionally stored).
|
|
*/
|
|
public List<State> getStatesList();
|
|
|
|
/** Get access to the VarList (optionally stored) */
|
|
public VarList getVarList();
|
|
|
|
/**
|
|
* Get access to a list of constant values (optionally stored).
|
|
*/
|
|
public Values getConstantValues();
|
|
|
|
/**
|
|
* Get the states that satisfy a label in this model (optionally stored).
|
|
* Returns null if there is no label of this name.
|
|
*/
|
|
public BitSet getLabelStates(String name);
|
|
|
|
/**
|
|
* Get the labels that are (optionally) stored.
|
|
* Returns an empty set if there are no labels.
|
|
*/
|
|
public Set<String> getLabels();
|
|
|
|
/** Returns true if a label with the given name is attached to this model */
|
|
public boolean hasLabel(String name);
|
|
|
|
/**
|
|
* Get the total number of transitions in the model.
|
|
*/
|
|
public int getNumTransitions();
|
|
|
|
/**
|
|
* Get an iterator over the successors of state s.
|
|
* Default implementation via the SuccessorsIterator returned
|
|
* from {@code getSuccessors}, ensuring that there are no
|
|
* duplicates.
|
|
*/
|
|
public default Iterator<Integer> getSuccessorsIterator(int s)
|
|
{
|
|
SuccessorsIterator successors = getSuccessors(s);
|
|
return successors.distinct();
|
|
}
|
|
|
|
/**
|
|
* Get a SuccessorsIterator for state s.
|
|
*/
|
|
public SuccessorsIterator getSuccessors(int s);
|
|
|
|
/**
|
|
* Returns true if state s2 is a successor of state s1.
|
|
*/
|
|
public default boolean isSuccessor(int s1, int s2)
|
|
{
|
|
// the code for this method is equivalent to the following stream expression,
|
|
// but kept explicit for performance
|
|
//
|
|
// return getSuccessors(s1).stream().anyMatch(
|
|
// (t) -> {return t == s2;}
|
|
// );
|
|
|
|
SuccessorsIterator it = getSuccessors(s1);
|
|
while (it.hasNext()) {
|
|
int t = it.nextInt();
|
|
if (t == s2)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Check if all the successor states of a state are in a set.
|
|
* @param s The state to check
|
|
* @param set The set to test for inclusion
|
|
*/
|
|
public default boolean allSuccessorsInSet(int s, BitSet set)
|
|
{
|
|
return allSuccessorsMatch(s, set::get);
|
|
}
|
|
|
|
/**
|
|
* Check if any successor states of a state are in a set.
|
|
* @param s The state to check
|
|
* @param set The set to test for inclusion
|
|
*/
|
|
public default boolean someSuccessorsInSet(int s, BitSet set)
|
|
{
|
|
return someSuccessorsMatch(s, set::get);
|
|
}
|
|
|
|
/**
|
|
* Check if all the successor states of a state match the predicate.
|
|
* @param s The state to check
|
|
* @param p the predicate
|
|
*/
|
|
public default boolean allSuccessorsMatch(int s, IntPredicate p)
|
|
{
|
|
// the code for this method is equivalent to the following stream expression,
|
|
// but kept explicit for performance
|
|
//
|
|
// return getSuccessors(s).stream().allMatch(p);
|
|
|
|
SuccessorsIterator it = getSuccessors(s);
|
|
while (it.hasNext()) {
|
|
int t = it.nextInt();
|
|
if (!p.test(t))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Check if any successor states of a state match the predicate.
|
|
* @param s The state to check
|
|
* @param p the predicate
|
|
*/
|
|
public default boolean someSuccessorsMatch(int s, IntPredicate p)
|
|
{
|
|
// the code for this method is equivalent to the following stream expression,
|
|
// but kept explicit for performance
|
|
//
|
|
// return getSuccessors(s).stream().anyMatch(p);
|
|
|
|
SuccessorsIterator it = getSuccessors(s);
|
|
while (it.hasNext()) {
|
|
int t = it.nextInt();
|
|
if (p.test(t))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Find all deadlock states and store this information in the model.
|
|
* If requested (if fix=true) and if needed (i.e. for DTMCs/CTMCs),
|
|
* fix deadlocks by adding self-loops in these states.
|
|
* The set of deadlocks (before any possible fixing) can be obtained from {@link #getDeadlockStates()}.
|
|
* @throws PrismException if the model is unable to fix deadlocks because it is non-mutable.
|
|
*/
|
|
public void findDeadlocks(boolean fix) throws PrismException;
|
|
|
|
/**
|
|
* Checks for deadlocks and throws an exception if any exist.
|
|
*/
|
|
public void checkForDeadlocks() throws PrismException;
|
|
|
|
/**
|
|
* Checks for deadlocks and throws an exception if any exist.
|
|
* States in 'except' (If non-null) are excluded from the check.
|
|
*/
|
|
public void checkForDeadlocks(BitSet except) throws PrismException;
|
|
|
|
/**
|
|
* Export to explicit format readable by PRISM (i.e. a .tra file, etc.).
|
|
*/
|
|
public void exportToPrismExplicit(String baseFilename) throws PrismException;
|
|
|
|
/**
|
|
* Export transition matrix to explicit format readable by PRISM (i.e. a .tra file).
|
|
*/
|
|
public void exportToPrismExplicitTra(String filename) throws PrismException;
|
|
|
|
/**
|
|
* Export transition matrix to explicit format readable by PRISM (i.e. a .tra file).
|
|
*/
|
|
public void exportToPrismExplicitTra(File file) throws PrismException;
|
|
|
|
/**
|
|
* Export transition matrix to explicit format readable by PRISM (i.e. a .tra file).
|
|
*/
|
|
public void exportToPrismExplicitTra(PrismLog log);
|
|
|
|
/**
|
|
* Export to a dot file.
|
|
* @param filename Name of file to export to
|
|
*/
|
|
default void exportToDotFile(String filename) throws PrismException
|
|
{
|
|
try (PrismFileLog log = PrismFileLog.create(filename)) {
|
|
exportToDotFile(log);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file, highlighting states in 'mark'.
|
|
* @param filename Name of file to export to
|
|
* @param mark States to highlight (ignored if null)
|
|
*/
|
|
default void exportToDotFile(String filename, BitSet mark) throws PrismException
|
|
{
|
|
try (PrismFileLog log = PrismFileLog.create(filename)) {
|
|
exportToDotFile(log, mark);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file, decorating states and transitions with the provided decorators
|
|
* @param filename Name of the file to export to
|
|
*/
|
|
default void exportToDotFile(String filename, Iterable<explicit.graphviz.Decorator> decorators) throws PrismException
|
|
{
|
|
try (PrismFileLog log = PrismFileLog.create(filename)) {
|
|
exportToDotFile(log, decorators);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file.
|
|
* @param out PrismLog to export to
|
|
*/
|
|
default void exportToDotFile(PrismLog out)
|
|
{
|
|
exportToDotFile(out, (Iterable<explicit.graphviz.Decorator>)null);
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file, highlighting states in 'mark'.
|
|
* @param out PrismLog to export to
|
|
* @param mark States to highlight (ignored if null)
|
|
*/
|
|
default void exportToDotFile(PrismLog out, BitSet mark) {
|
|
if (mark == null) {
|
|
exportToDotFile(out);
|
|
}
|
|
exportToDotFile(out, Collections.singleton(new explicit.graphviz.MarkStateSetDecorator(mark)));
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file, highlighting states in 'mark'.
|
|
* @param out PrismLog to export to
|
|
* @param mark States to highlight (ignored if null)
|
|
* @param showStates Show state info on nodes?
|
|
*/
|
|
default void exportToDotFile(PrismLog out, BitSet mark, boolean showStates)
|
|
{
|
|
ArrayList<explicit.graphviz.Decorator> decorators = new ArrayList<explicit.graphviz.Decorator>();
|
|
if (showStates) {
|
|
decorators.add(new explicit.graphviz.ShowStatesDecorator(getStatesList()));
|
|
}
|
|
if (mark != null) {
|
|
decorators.add(new explicit.graphviz.MarkStateSetDecorator(mark));
|
|
}
|
|
exportToDotFile(out, decorators);
|
|
}
|
|
|
|
/**
|
|
* Export to a dot file, decorating states and transitions with the provided decorators
|
|
* @param out PrismLog to export to
|
|
*/
|
|
default void exportToDotFile(PrismLog out, Iterable<explicit.graphviz.Decorator> decorators)
|
|
{
|
|
explicit.graphviz.Decoration defaults = new explicit.graphviz.Decoration();
|
|
defaults.attributes().put("shape", "box");
|
|
|
|
// Header
|
|
out.print("digraph " + getModelType() + " {\nnode " + defaults.toString() + ";\n");
|
|
int i, numStates;
|
|
for (i = 0, numStates = getNumStates(); i < numStates; i++) {
|
|
// initialize
|
|
explicit.graphviz.Decoration d = new explicit.graphviz.Decoration(defaults);
|
|
d.setLabel(Integer.toString(i));
|
|
|
|
// run any decorators
|
|
if (decorators != null) {
|
|
for (Decorator decorator : decorators) {
|
|
d = decorator.decorateState(i, d);
|
|
}
|
|
}
|
|
|
|
String decoration = d.toString();
|
|
out.println(i + " " + decoration + ";");
|
|
|
|
// Transitions for state i
|
|
exportTransitionsToDotFile(i, out, decorators);
|
|
}
|
|
|
|
// Footer
|
|
out.print("}\n");
|
|
}
|
|
|
|
/**
|
|
* Export the transitions from state {@code i} in Dot format to {@code out},
|
|
* decorating using the given decorators.
|
|
* <br>
|
|
* The default implementation throws an UnsupportedOperationException,
|
|
* so this method should be overloaded.
|
|
*
|
|
* @param i State index
|
|
* @param out PrismLog for output
|
|
* @param decorators the decorators (may be {@code null})
|
|
*/
|
|
default void exportTransitionsToDotFile(int i, PrismLog out, Iterable<explicit.graphviz.Decorator> decorators)
|
|
{
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
/**
|
|
* Export to a equivalent PRISM language model description.
|
|
*/
|
|
public void exportToPrismLanguage(String filename) throws PrismException;
|
|
|
|
/**
|
|
* Export states list.
|
|
*/
|
|
public void exportStates(int exportType, VarList varList, PrismLog log) throws PrismException;
|
|
|
|
/**
|
|
* Report info/stats about the model as a string.
|
|
*/
|
|
public String infoString();
|
|
|
|
/**
|
|
* Report info/stats about the model, tabulated, as a string.
|
|
*/
|
|
public String infoStringTable();
|
|
|
|
/** Has this model a stored PredecessorRelation? */
|
|
public boolean hasStoredPredecessorRelation();
|
|
|
|
/**
|
|
* If there is a PredecessorRelation stored for this model, return that.
|
|
* Otherwise, create one and return that. If {@code storeIfNew},
|
|
* store it for later use.
|
|
*
|
|
* @param parent a PrismComponent (for obtaining the log)
|
|
* @param storeIfNew if the predecessor relation is newly created, store it
|
|
*/
|
|
public PredecessorRelation getPredecessorRelation(prism.PrismComponent parent, boolean storeIfNew);
|
|
|
|
/** Clear any stored predecessor relation, e.g., because the model was modified */
|
|
public void clearPredecessorRelation();
|
|
|
|
}
|