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.
 
 
 
 
 
 

1067 lines
38 KiB

//==============================================================================
//
// Copyright (c) 2002-
// Authors:
// * Dave Parker <david.parker@comlab.ox.ac.uk> (University of Oxford, formerly University of Birmingham)
//
//------------------------------------------------------------------------------
//
// 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 parser.ast;
import jltl2ba.SimpleLTL;
import param.BigRational;
import parser.*;
import parser.visitor.*;
import prism.ModelType;
import prism.PrismException;
import prism.PrismLangException;
import parser.type.*;
// Abstract class for PRISM language expressions
public abstract class Expression extends ASTElement
{
/**
* Is this expression constant?
*/
public abstract boolean isConstant();
/**
* Is this expression a proposition?
* (i.e. something that can be evaluated on a single state in isolation)
*/
public abstract boolean isProposition();
/**
* Evaluate this expression, return result.
* Note: assumes that type checking has been done already.
*/
public abstract Object evaluate(EvaluateContext ec) throws PrismLangException;
/**
* Evaluate this expression exactly, return the result as a BigRational.
* Note: assumes that type checking has been done already.
*/
public abstract BigRational evaluateExact(EvaluateContext ec) throws PrismLangException;
/**
* Get "name" of the result of this expression (used for y-axis of any graphs plotted)
*/
public String getResultName()
{
return "Result";
}
/**
* Does this expression equate to a single value, rather than a different value for all states,
* when evaluated during model checking?
*/
public abstract boolean returnsSingleValue();
// Overrided version of deepCopy() from superclass ASTElement (to reduce casting).
/**
* Perform a deep copy.
*/
public abstract Expression deepCopy();
// Utility methods:
/**
* Check expression (property) for validity with respect to a particular model type
* (i.e. whether not it is a property that can be model checked for that model type).
*/
public void checkValid(ModelType modelType) throws PrismLangException
{
CheckValid visitor = new CheckValid(modelType);
accept(visitor);
}
/**
* Determine whether expression is a valid "simple" path formula , i.e. a formula
* that could occur in the P operator of a PCTL/CSL formula (not LTL, PCTL*).
* This is defined as a single instance of a temporal operator (X, U, F, etc.),
* possibly negated. Strictly speaking, negations are not allowed in PCTL/CSL
* but they can always be converted to a dual formula which is.
*/
public boolean isSimplePathFormula() throws PrismLangException
{
// One (or more) top-level negations is allowed.
// Top-level parentheses also OK.
if (this instanceof ExpressionUnaryOp) {
ExpressionUnaryOp expr = (ExpressionUnaryOp) this;
int op = expr.getOperator();
if (op == ExpressionUnaryOp.NOT || op == ExpressionUnaryOp.PARENTH) {
return expr.getOperand().isSimplePathFormula();
} else {
return false;
}
}
// Otherwise, must be a temporal operator.
else if (this instanceof ExpressionTemporal) {
ExpressionTemporal expr = (ExpressionTemporal) this;
// And children, if present, must be state (not path) formulas
if (expr.getOperand1() != null && !(expr.getOperand1().getType() instanceof TypeBool)) {
return false;
}
if (expr.getOperand2() != null && !(expr.getOperand2().getType() instanceof TypeBool)) {
return false;
}
return true;
}
// Default: false.
return false;
}
/**
* Returns {@code true} if this expression is a path formula.
* If {@code allowNestedOperators==false} then we don't allow
* nested P, R, ... operators.
* @param allowNestedOperators allow nested P, R, ... operators?
*/
public boolean isPathFormula(boolean allowNestedOperators)
{
try {
if (getType() == null) {
this.typeCheck();
}
if (getType() == TypePathBool.getInstance() ||
getType() == TypeBool.getInstance()) {
if (!allowNestedOperators) {
if (this.computeProbNesting() >= 1) {
return false;
}
}
return true;
}
return false;
} catch (PrismLangException e) {
return false;
}
}
/**
* Convert a property expression (an LTL formula) into the classes used by
* the jltl2ba (and jltl2dstar) libraries.
*/
public jltl2ba.SimpleLTL convertForJltl2ba() throws PrismLangException
{
ConvertForJltl2ba converter = new ConvertForJltl2ba();
return converter.convert(this);
}
/**
* Evaluate this expression, using no constant or variable values.
* Note: assumes that type checking has been done already.
*/
public Object evaluate() throws PrismLangException
{
return evaluate(new EvaluateContextValues(null, null));
}
/**
* Evaluate this expression, based on values for constants (but not variables).
* Constant values are supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public Object evaluate(Values constantValues) throws PrismLangException
{
return evaluate(new EvaluateContextValues(constantValues, null));
}
/**
* Evaluate this expression, based on values for constants/variables.
* Each set of values is supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public Object evaluate(Values constantValues, Values varValues) throws PrismLangException
{
return evaluate(new EvaluateContextValues(constantValues, varValues));
}
/**
* Evaluate this expression, based on values for variables (but not constants).
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public Object evaluate(State state) throws PrismLangException
{
return evaluate(new EvaluateContextState(state));
}
/**
* Evaluate this expression, based on values for constants/variables.
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that type checking has been done.
*/
public Object evaluate(Values constantValues, State state) throws PrismLangException
{
return evaluate(new EvaluateContextState(constantValues, state));
}
/**
* Evaluate this expression, based on values for some variables (but not constants).
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public Object evaluate(State substate, int[] varMap) throws PrismLangException
{
return evaluate(new EvaluateContextSubstate(substate, varMap));
}
/**
* Evaluate this expression, based on values for constants and some variables.
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that type checking has been done.
*/
public Object evaluate(Values constantValues, State substate, int[] varMap) throws PrismLangException
{
return evaluate(new EvaluateContextSubstate(constantValues, substate, varMap));
}
/**
* Evaluate this expression as an integer.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
*/
public int evaluateInt(EvaluateContext ec) throws PrismLangException
{
Object o = evaluate(ec);
if (o instanceof Integer) {
return ((Integer) o).intValue();
}
if (o instanceof Boolean) {
return ((Boolean) o).booleanValue() ? 1 : 0;
}
throw new PrismLangException("Cannot evaluate to an integer", this);
}
/**
* Evaluate this expression as an integer, using no constant or variable values.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Note: assumes that type checking has been done already.
*/
public int evaluateInt() throws PrismLangException
{
return evaluateInt(new EvaluateContextValues(null, null));
}
/**
* Evaluate this expression as an integer, based on values for constants (but not variables).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public int evaluateInt(Values constantValues) throws PrismLangException
{
return evaluateInt(new EvaluateContextValues(constantValues, null));
}
/**
* Evaluate this expression as an integer, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Each set of values is supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public int evaluateInt(Values constantValues, Values varValues) throws PrismLangException
{
return evaluateInt(new EvaluateContextValues(constantValues, varValues));
}
/**
* Evaluate this expression as an integer, based on values for variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public int evaluateInt(State state) throws PrismLangException
{
return evaluateInt(new EvaluateContextState(state));
}
/**
* Evaluate this expression as an integer, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that type checking has been done.
*/
public int evaluateInt(Values constantValues, State state) throws PrismLangException
{
return evaluateInt(new EvaluateContextState(constantValues, state));
}
/**
* Evaluate this expression as an integer, based on values for some variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public int evaluateInt(State substate, int[] varMap) throws PrismLangException
{
return evaluateInt(new EvaluateContextSubstate(substate, varMap));
}
/**
* Evaluate this expression as an integer, based on values for constants and some variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that type checking has been done.
*/
public int evaluateInt(Values constantValues, State substate, int[] varMap) throws PrismLangException
{
return evaluateInt(new EvaluateContextSubstate(constantValues, substate, varMap));
}
/**
* Evaluate this expression as a double.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
*/
public double evaluateDouble(EvaluateContext ec) throws PrismLangException
{
Object o = evaluate(ec);
if (o instanceof Integer) {
return ((Integer) o).intValue();
}
if (o instanceof Double) {
return ((Double) o).doubleValue();
}
if (o instanceof Boolean) {
return ((Boolean) o).booleanValue() ? 1.0 : 0.0;
}
throw new PrismLangException("Cannot evaluate to a double", this);
}
/**
* Evaluate this expression as a double, using no constant or variable values.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Note: assumes that type checking has been done already.
*/
public double evaluateDouble() throws PrismLangException
{
return evaluateDouble(new EvaluateContextValues(null, null));
}
/**
* Evaluate this expression as a double, based on values for constants (but not variables).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Constant values are supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public double evaluateDouble(Values constantValues) throws PrismLangException
{
return evaluateDouble(new EvaluateContextValues(constantValues, null));
}
/**
* Evaluate this expression as a double, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Each set of values is supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public double evaluateDouble(Values constantValues, Values varValues) throws PrismLangException
{
return evaluateDouble(new EvaluateContextValues(constantValues, varValues));
}
/**
* Evaluate this expression as a double, based on values for variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public double evaluateDouble(State state) throws PrismLangException
{
return evaluateDouble(new EvaluateContextState(state));
}
/**
* Evaluate this expression as a double, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that type checking has been done.
*/
public double evaluateDouble(Values constantValues, State state) throws PrismLangException
{
return evaluateDouble(new EvaluateContextState(constantValues, state));
}
/**
* Evaluate this expression as a double, based on values for some variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public double evaluateDouble(State substate, int[] varMap) throws PrismLangException
{
return evaluateDouble(new EvaluateContextSubstate(substate, varMap));
}
/**
* Evaluate this expression as a double, based on values for constants and some variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0.0/1.0).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that type checking has been done.
*/
public double evaluateDouble(Values constantValues, State substate, int[] varMap) throws PrismLangException
{
return evaluateDouble(new EvaluateContextSubstate(constantValues, substate, varMap));
}
/**
* Evaluate this expression as a boolean.
* Any typing issues cause an exception.
*/
public boolean evaluateBoolean(EvaluateContext ec) throws PrismLangException
{
Object o = evaluate(ec);
if (!(o instanceof Boolean)) {
throw new PrismLangException("Cannot evaluate to a boolean", this);
}
return ((Boolean) o).booleanValue();
}
/**
* Evaluate this expression as a boolean, using no constant or variable values.
* Any typing issues cause an exception.
* Note: assumes that type checking has been done already.
*/
public boolean evaluateBoolean() throws PrismLangException
{
return evaluateBoolean(new EvaluateContextValues(null, null));
}
/**
* Evaluate this expression as a boolean, based on values for constants (but not variables).
* Any typing issues cause an exception.
* Constant values are supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public boolean evaluateBoolean(Values constantValues) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextValues(constantValues, null));
}
/**
* Evaluate this expression as a boolean, based on values for constants/variables.
* Any typing issues cause an exception.
* Each set of values is supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public boolean evaluateBoolean(Values constantValues, Values varValues) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextValues(constantValues, varValues));
}
/**
* Evaluate this expression as a boolean, based on values for variables (but not constants).
* Any typing issues cause an exception.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public boolean evaluateBoolean(State state) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextState(state));
}
/**
* Evaluate this expression as a boolean, based on values for constants/variables.
* Any typing issues cause an exception.
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that type checking has been done.
*/
public boolean evaluateBoolean(Values constantValues, State state) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextState(constantValues, state));
}
/**
* Evaluate this expression as a boolean, based on values for some variables (but not constants).
* Any typing issues cause an exception.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public boolean evaluateBoolean(State substate, int[] varMap) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextSubstate(substate, varMap));
}
/**
* Evaluate this expression as a boolean, based on values for constants and some variables.
* Any typing issues cause an exception.
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that type checking has been done.
*/
public boolean evaluateBoolean(Values constantValues, State substate, int[] varMap) throws PrismLangException
{
return evaluateBoolean(new EvaluateContextSubstate(constantValues, substate, varMap));
}
/**
* Evaluate this expression exactly to a BigRational, using no constant or variable values.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Note: assumes that type checking has been done already.
*/
public BigRational evaluateExact() throws PrismLangException
{
return evaluateExact(new EvaluateContextValues(null, null));
}
/**
* Evaluate this expression exactly to a BigRational, based on values for constants (but not variables).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public BigRational evaluateExact(Values constantValues) throws PrismLangException
{
return evaluateExact(new EvaluateContextValues(constantValues, null));
}
/**
* Evaluate this expression exactly to a BigRational, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Each set of values is supplied as a Values object.
* Note: assumes that type checking has been done already.
*/
public BigRational evaluateExact(Values constantValues, Values varValues) throws PrismLangException
{
return evaluateExact(new EvaluateContextValues(constantValues, varValues));
}
/**
* Evaluate this expression exactly to a BigRational, based on values for variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public BigRational evaluateExact(State state) throws PrismLangException
{
return evaluateExact(new EvaluateContextState(state));
}
/**
* Evaluate this expression as an integer, based on values for constants/variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, i.e. array of variable values.
* Note: assumes that type checking has been done.
*/
public BigRational evaluateExact(Values constantValues, State state) throws PrismLangException
{
return evaluateExact(new EvaluateContextState(constantValues, state));
}
/**
* Evaluate this expression exactly to a BigRational, based on values for some variables (but not constants).
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that constants have been evaluated and type checking has been done.
*/
public BigRational evaluateExact(State substate, int[] varMap) throws PrismLangException
{
return evaluateExact(new EvaluateContextSubstate(substate, varMap));
}
/**
* Evaluate this expression exactly to a BigRational, based on values for constants and some variables.
* Any typing issues cause an exception (but: we do allow conversion of boolean to 0/1).
* Constant values are supplied as a Values object.
* Variable values are supplied as a State object, indexed over a subset of all variables,
* and a mapping from indices (over all variables) to this subset (-1 if not in subset).
* If any variables required for evaluation are missing, this will fail with an exception.
* Note: assumes that type checking has been done.
*/
public BigRational evaluateExact(Values constantValues, State substate, int[] varMap) throws PrismLangException
{
return evaluateExact(new EvaluateContextSubstate(constantValues, substate, varMap));
}
// Static constructors for convenience
public static ExpressionLiteral True()
{
return new ExpressionLiteral(TypeBool.getInstance(), true);
}
public static ExpressionLiteral False()
{
return new ExpressionLiteral(TypeBool.getInstance(), false);
}
public static ExpressionLiteral Int(int i)
{
return new ExpressionLiteral(TypeInt.getInstance(), i);
}
public static ExpressionLiteral Double(double d)
{
return new ExpressionLiteral(TypeDouble.getInstance(), d);
}
public static ExpressionLiteral Literal(Object o) throws PrismLangException
{
if (o instanceof Integer) {
return Int(((Integer) o).intValue());
} else if (o instanceof Double) {
return Double(((Double) o).doubleValue());
} else if (o instanceof Boolean) {
return (((Boolean) o).booleanValue() ? True() : False());
} else {
throw new PrismLangException("Unknown object type " + o.getClass());
}
}
public static ExpressionUnaryOp Not(Expression expr)
{
return new ExpressionUnaryOp(ExpressionUnaryOp.NOT, expr);
}
public static ExpressionBinaryOp And(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.AND, expr1, expr2);
}
public static ExpressionBinaryOp Or(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.OR, expr1, expr2);
}
public static ExpressionBinaryOp Iff(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.IFF, expr1, expr2);
}
public static ExpressionBinaryOp Implies(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.IMPLIES, expr1, expr2);
}
public static ExpressionBinaryOp Plus(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.PLUS, expr1, expr2);
}
public static ExpressionUnaryOp Minus(Expression expr)
{
return new ExpressionUnaryOp(ExpressionUnaryOp.MINUS, expr);
}
public static ExpressionBinaryOp Minus(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.MINUS, expr1, expr2);
}
public static ExpressionBinaryOp Times(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.TIMES, expr1, expr2);
}
public static ExpressionBinaryOp Divide(Expression expr1, Expression expr2)
{
return new ExpressionBinaryOp(ExpressionBinaryOp.DIVIDE, expr1, expr2);
}
public static ExpressionUnaryOp Parenth(Expression expr)
{
return new ExpressionUnaryOp(ExpressionUnaryOp.PARENTH, expr);
}
public static ExpressionTemporal Next(Expression expr) {
return new ExpressionTemporal(ExpressionTemporal.P_X, null, expr);
}
// Static testers for convenience
public static boolean isTrue(Expression expr)
{
return expr instanceof ExpressionLiteral && ((ExpressionLiteral) expr).getValue().equals(true);
}
public static boolean isFalse(Expression expr)
{
return expr instanceof ExpressionLiteral && ((ExpressionLiteral) expr).getValue().equals(false);
}
public static boolean isInt(Expression expr)
{
return expr instanceof ExpressionLiteral && expr.getType() instanceof TypeInt;
}
public static boolean isDouble(Expression expr)
{
return expr instanceof ExpressionLiteral && expr.getType() instanceof TypeDouble;
}
public static boolean isNot(Expression expr)
{
return expr instanceof ExpressionUnaryOp && ((ExpressionUnaryOp) expr).getOperator() == ExpressionUnaryOp.NOT;
}
public static boolean isAnd(Expression expr)
{
return expr instanceof ExpressionBinaryOp
&& ((ExpressionBinaryOp) expr).getOperator() == ExpressionBinaryOp.AND;
}
public static boolean isOr(Expression expr)
{
return expr instanceof ExpressionBinaryOp && ((ExpressionBinaryOp) expr).getOperator() == ExpressionBinaryOp.OR;
}
public static boolean isIff(Expression expr)
{
return expr instanceof ExpressionBinaryOp
&& ((ExpressionBinaryOp) expr).getOperator() == ExpressionBinaryOp.IFF;
}
public static boolean isImplies(Expression expr)
{
return expr instanceof ExpressionBinaryOp
&& ((ExpressionBinaryOp) expr).getOperator() == ExpressionBinaryOp.IMPLIES;
}
public static boolean isParenth(Expression expr)
{
return expr instanceof ExpressionUnaryOp
&& ((ExpressionUnaryOp) expr).getOperator() == ExpressionUnaryOp.PARENTH;
}
public static boolean isRelOp(Expression expr)
{
return expr instanceof ExpressionBinaryOp
&& ExpressionBinaryOp.isRelOp(((ExpressionBinaryOp) expr).getOperator());
}
/**
* Test if an expression is a function of type {@code nameCode}.
*/
public static boolean isFunc(Expression expr, int nameCode)
{
return expr instanceof ExpressionFunc && ((ExpressionFunc) expr).getNameCode() == nameCode;
}
/**
* Test if an expression is a quantitative property (P=?, R=? or S=?)
*/
public static boolean isQuantitative(Expression expr)
{
if (expr instanceof ExpressionProb) {
return ((ExpressionProb) expr).getProb() == null;
}
else if (expr instanceof ExpressionReward) {
return ((ExpressionReward) expr).getReward() == null;
}
else if (expr instanceof ExpressionSS) {
return ((ExpressionSS) expr).getProb() == null;
}
return false;
}
/**
* Test if an expression is a reachability path formula (F phi), possibly with a time bound.
*/
public static boolean isReach(Expression expr)
{
if (expr instanceof ExpressionTemporal) {
if (((ExpressionTemporal) expr).getOperator() == ExpressionTemporal.P_F) {
return ((ExpressionTemporal) expr).getOperand2().isProposition();
}
}
return false;
}
/**
* Test if an expression contains time bounds on temporal operators
*/
public static boolean containsTemporalTimeBounds(Expression expr)
{
try {
// check for time bounds, don't recurse into P/R/SS subformulas
expr.accept(new ExpressionTraverseNonNested()
{
public void visitPre(ExpressionTemporal e) throws PrismLangException
{
if (e.hasBounds())
throw new PrismLangException("");
}
});
} catch (PrismLangException e) {
return true;
}
return false;
}
/**
* Test if an expression contains a multi(...) property within
*/
public static boolean containsMultiObjective(Expression expr)
{
try {
ASTTraverse astt = new ASTTraverse()
{
public void visitPost(ExpressionFunc e) throws PrismLangException
{
if (e.getNameCode() == ExpressionFunc.MULTI)
throw new PrismLangException("Found one", e);
}
};
expr.accept(astt);
} catch (PrismLangException e) {
return true;
}
return false;
}
/**
* Test if an expression is an LTL formula and is in positive normal form,
* i.e. where negation only occurs at the level of state formulae.
* This means that the operators => and <=> are also disallowed.
*/
public static boolean isPositiveNormalFormLTL(Expression expr)
{
// State formulae (negated or otherwise) are OK
if (expr.getType() instanceof TypeBool)
return true;
// Otherwise recurse, looking for negations...
else if (expr instanceof ExpressionUnaryOp) {
ExpressionUnaryOp exprUnOp = (ExpressionUnaryOp) expr;
int op = exprUnOp.getOperator();
switch (op) {
// Negation is not allowed
// (since we already checked for state formulae that include negation)
case ExpressionUnaryOp.NOT:
return false;
default:
return isPositiveNormalFormLTL(exprUnOp.getOperand());
}
}
else if (expr instanceof ExpressionBinaryOp) {
ExpressionBinaryOp exprBinOp = (ExpressionBinaryOp) expr;
int op = exprBinOp.getOperator();
switch (op) {
// => and <=> are not allowed
case ExpressionBinaryOp.IMPLIES:
case ExpressionBinaryOp.IFF:
return false;
default:
return isPositiveNormalFormLTL(exprBinOp.getOperand1()) && isPositiveNormalFormLTL(exprBinOp.getOperand2());
}
}
else if (expr instanceof ExpressionTemporal) {
ExpressionTemporal exprTemp = (ExpressionTemporal) expr;
if (exprTemp.getOperand1() != null && !isPositiveNormalFormLTL(exprTemp.getOperand1())) {
return false;
}
if (exprTemp.getOperand2() != null && !isPositiveNormalFormLTL(exprTemp.getOperand2())) {
return false;
}
return true;
}
// If we get here, it is probably not even LTL
return false;
}
/**
* Test if an expression is a co-safe LTL formula, detected syntactically
* (i.e. if it is in positive normal form and only uses X, F and U).
*/
public static boolean isCoSafeLTLSyntactic(Expression expr)
{
return isCoSafeLTLSyntactic(expr, false);
}
/**
* Test if an expression is a co-safe LTL formula, detected syntactically
* (i.e. if it is in positive normal form and only uses X, F and U).
* If {@code convert} is true, the expression is first converted into positive normal form,
* and then it is checked whether it only uses X, F and U.
* For example, a => ! (G b) would return true if (and only if) {@code convert} was true.
*/
public static boolean isCoSafeLTLSyntactic(Expression expr, boolean convert)
{
// Convert to or check for positive normal form
if (convert) {
expr = BooleanUtils.convertLTLToPositiveNormalForm(expr.deepCopy());
} else {
if (!isPositiveNormalFormLTL(expr))
return false;
}
// Check temporal operators
try {
ASTTraverse astt = new ASTTraverse()
{
public void visitPost(ExpressionTemporal e) throws PrismLangException
{
if (e.getOperator() == ExpressionTemporal.P_X)
return;
if (e.getOperator() == ExpressionTemporal.P_F)
return;
if (e.getOperator() == ExpressionTemporal.P_U)
return;
throw new PrismLangException("Found non-X/F/U", e);
}
};
expr.accept(astt);
} catch (PrismLangException e) {
return false;
}
// All good
return true;
}
/**
* Converts an Expression that is a simple path formula to a canonical form:
* Either a single non-negated next-step operator
* or a single until-operator, optionally preceded by a single negation.
* Parentheses are removed.
* @param expr the simple path formula
* @return the canonical expression
*/
public static Expression convertSimplePathFormulaToCanonicalForm(Expression expr) throws PrismLangException
{
boolean negated = false;
if (!expr.isSimplePathFormula()) {
throw new PrismLangException("Expression is not a simple path formula.");
}
// Negation/parentheses
while (expr instanceof ExpressionUnaryOp) {
ExpressionUnaryOp exprUnary = (ExpressionUnaryOp) expr;
// remove parentheses
if (exprUnary.getOperator() == ExpressionUnaryOp.PARENTH) {
expr = exprUnary.getOperand();
}
// deal with negation
else if (exprUnary.getOperator() == ExpressionUnaryOp.NOT) {
negated = !negated;
expr = exprUnary.getOperand();
} else {
throw new PrismLangException("Expression is not a simple path formula: Unexpected unary operator "+exprUnary.getOperatorSymbol());
}
}
// Temporal operator
if (expr instanceof ExpressionTemporal) {
ExpressionTemporal exprTemp = (ExpressionTemporal) expr;
// Next
if (exprTemp.getOperator() == ExpressionTemporal.P_X) {
if (negated) {
if (exprTemp.hasBounds()) throw new PrismLangException("Next-Step operator should not have bounds!");
// ! X expr <=> X ! expr
return new ExpressionTemporal(ExpressionTemporal.P_X, null,
Expression.Not(Expression.Parenth(exprTemp.getOperand2())));
} else {
// X expr
return exprTemp;
}
} else if (exprTemp.getOperator() == ExpressionTemporal.P_U) {
// Until
expr = exprTemp;
} else {
// other operators: convert
expr = exprTemp.convertToUntilForm();
}
} else {
throw new PrismLangException("Expression is not a simple path formula: Unsupported expression "+expr.toString());
}
if (negated) {
if (expr instanceof ExpressionUnaryOp &&
((ExpressionUnaryOp)expr).getOperator() == ExpressionUnaryOp.NOT) {
// remove the outer negation
return ((ExpressionUnaryOp)expr).getOperand();
} else {
// negate
return Expression.Not(expr);
}
}
return expr;
}
/**
* Create a property expression (an LTL formula) from the classes used by the jltl2ba (and jltl2dstar) libraries.
*/
public static Expression createFromJltl2ba(SimpleLTL ltl) throws PrismException
{
switch (ltl.kind) {
case AND:
return Expression.And(createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
case AP:
return new ExpressionLabel(ltl.ap);
case EQUIV:
return Expression.Iff(createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
case FALSE:
return Expression.False();
case FINALLY:
return new ExpressionTemporal(ExpressionTemporal.P_F, null, createFromJltl2ba(ltl.left));
case GLOBALLY:
return new ExpressionTemporal(ExpressionTemporal.P_G, null, createFromJltl2ba(ltl.left));
case IMPLIES:
return Expression.Implies(createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
case NEXT:
return new ExpressionTemporal(ExpressionTemporal.P_X, null, createFromJltl2ba(ltl.left));
case NOT:
return Expression.Not(createFromJltl2ba(ltl.left));
case OR:
return Expression.Or(createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
case RELEASE:
return new ExpressionTemporal(ExpressionTemporal.P_R, createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
case TRUE:
return Expression.True();
case UNTIL:
return new ExpressionTemporal(ExpressionTemporal.P_U, createFromJltl2ba(ltl.left), createFromJltl2ba(ltl.right));
default:
throw new PrismException("Cannot convert jltl2ba formula " + ltl);
}
}
}
//------------------------------------------------------------------------------