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

//==============================================================================
//
// 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();
}