//-----------------------------------------------------------------------------
//
// Copyright (C) Microsoft Corporation. All Rights Reserved.
//
//-----------------------------------------------------------------------------
// This file specifies the expression language used by the Abstract
// Interpretation Framework.
//
// expressions e ::= x variables
// | f(e1,...,en) uninterpreted functions
// | \x:t.e lambda expressions
//
// types t ::= b user-defined/built-in base types
// | t1 * ... * tn -> t' function type
namespace Microsoft.AbstractInterpretationFramework
{
using System.Collections;
using Microsoft.Contracts;
//----------------------------- Expressions -----------------------------
///
/// An interface for expressions. This expression language is specified
/// by interfaces to allow the client to be able to use their existing
/// AST nodes as AIF expressions.
///
/// This only serves as a place for operations on expressions. Clients
/// should implement directly either IVariable or IFunApp.
///
public interface IExpr
{
///
/// Execute a visit over the expression.
///
/// The expression visitor.
/// The result of the visit.
[Pure][Reads(ReadsAttribute.Reads.Owned)] object DoVisit(ExprVisitor! visitor);
// TODO: Type checking of the expressions.
}
///
/// An interface for variables.
///
/// This interface should be implemented by the client.
///
public interface IVariable : IExpr
{
string! Name { get; } // Each client must define the name for variables
}
///
/// An interface for function applications.
///
/// This interface should be implemented by the client.
///
public interface IFunApp : IExpr
{
IFunctionSymbol! FunctionSymbol { [Pure][Reads(ReadsAttribute.Reads.Owned)] get; }
IList/**/! Arguments
{
[Pure][Reads(ReadsAttribute.Reads.Owned)][Rep] get
ensures result.IsReadOnly;
;
}
///
/// Provides a method to create a new uninterpreted function
/// with the same function symbol but with the arguments with
/// args.
///
/// The new arguments.
/// A copy of the function with the new arguments.
IFunApp! CloneWithArguments(IList/**/! args)
//TODO requires this.Arguments.Count == args.Count;
;
}
///
/// An interface for anonymous functions (i.e., lambda expressions)
///
public interface IFunction : IExpr
{
IVariable! Param { [Pure][Reads(ReadsAttribute.Reads.Owned)] get; }
AIType! ParamType { [Pure][Reads(ReadsAttribute.Reads.Owned)] get; }
IExpr! Body { [Pure][Reads(ReadsAttribute.Reads.Owned)] get; }
IFunction! CloneWithBody(IExpr! body);
}
///
/// An abstract class that provides an interface for expression visitors.
///
public abstract class ExprVisitor
{
public abstract object Default(IExpr! expr);
public virtual object VisitVariable(IVariable! var)
{
return Default(var);
}
public virtual object VisitFunApp(IFunApp! funapp)
{
return Default(funapp);
}
public virtual object VisitFunction(IFunction! fun)
{
return Default(fun);
}
}
///
/// A utility class for dealing with expressions.
///
public sealed class ExprUtil
{
///
/// Yield an expression that is 'inexpr' with 'var' replaced by 'subst'.
///
/// The expression to substitute.
/// The variable to substitute for.
/// The expression to substitute into.
public static IExpr! Substitute(IExpr! subst, IVariable! var, IExpr! inexpr)
{
IExpr result = null;
if (inexpr is IVariable)
{
result = inexpr.Equals(var) ? subst : inexpr;
}
else if (inexpr is IFunApp)
{
IFunApp! funapp = (IFunApp!)inexpr;
IList newargs = null;
newargs = new ArrayList{ IExpr! arg in funapp.Arguments; Substitute(subst, var, arg) };
result = funapp.CloneWithArguments(newargs);
}
else if (inexpr is IFunction)
{
IFunction! fun = (IFunction!)inexpr;
if (fun.Param.Equals(var))
result = fun;
else
result = fun.CloneWithBody(Substitute(subst, var, fun.Body));
}
else
{
assert false;
}
return result;
}
//
// Poor man's pattern matching.
//
// The methods below implement pattern matching for AI expressions.
//
// Example Usage:
// Match(e, Prop.Imp,
// (Matcher)delegate (IExpr e) { return Match(e, Prop.And, out x, out y); }
// out z)
// which sees if 'e' matches Prop.Imp(Prop.And(x,y),z) binding x,y,z to the subtrees.
//
public delegate bool Matcher(IExpr! expr);
private static IFunApp/*?*/ MatchFunctionSymbol(IExpr! expr, IFunctionSymbol! f)
{
IFunApp app = expr as IFunApp;
if (app != null)
{
if (app.FunctionSymbol.Equals(f))
return app;
else
return null;
}
else
return null;
}
public static bool Match(IExpr! expr, IFunctionSymbol! f, params Matcher[]! subs)
{
IFunApp app = MatchFunctionSymbol(expr,f);
if (app != null)
{
int i = 0; // Note ***0***
foreach (Matcher! s in subs)
{
if (!s((IExpr!)app.Arguments[i])) { return false; }
i++;
}
return true;
}
else { return false; }
}
// Unary Binding
public static bool Match(IExpr! expr, IFunctionSymbol! f, out IExpr arg0, params Matcher[]! subs)
{
arg0 = null;
IFunApp app = MatchFunctionSymbol(expr,f);
if (app != null)
{
arg0 = (IExpr!)app.Arguments[0];
int i = 1; // Note ***1***
foreach (Matcher! s in subs)
{
if (!s((IExpr!)app.Arguments[i])) { return false; }
i++;
}
return true;
}
else { return false; }
}
// Binary Binding
public static bool Match(IExpr! expr, IFunctionSymbol! f, Matcher! sub0, out IExpr arg1, params Matcher[]! subs)
{
arg1 = null;
IFunApp app = MatchFunctionSymbol(expr,f);
if (app != null)
{
if (!sub0((IExpr!)app.Arguments[0])) { return false; }
arg1 = (IExpr!)app.Arguments[1];
int i = 2; // Note ***2***
foreach (Matcher! s in subs)
{
if (!s((IExpr!)app.Arguments[i])) { return false; }
i++;
}
return true;
}
else { return false; }
}
public static bool Match(IExpr! expr, IFunctionSymbol! f, out IExpr arg0, out IExpr arg1, params Matcher[]! subs)
{
arg0 = null;
arg1 = null;
IFunApp app = MatchFunctionSymbol(expr,f);
if (app != null)
{
arg0 = (IExpr!)app.Arguments[0];
arg1 = (IExpr!)app.Arguments[1];
int i = 2; // Note ***2***
foreach (Matcher! s in subs)
{
if (!s((IExpr!)app.Arguments[i])) { return false; }
i++;
}
return true;
}
else { return false; }
}
// Ternary Binding
public static bool Match(IExpr! expr, IFunctionSymbol! f, out IExpr arg0, out IExpr arg1, out IExpr arg2, params Matcher[]! subs)
{
arg0 = null;
arg1 = null;
arg2 = null;
IFunApp app = MatchFunctionSymbol(expr,f);
if (app != null)
{
arg0 = (IExpr!)app.Arguments[0];
arg1 = (IExpr!)app.Arguments[1];
arg2 = (IExpr!)app.Arguments[2];
int i = 3; // Note ***3***
foreach (Matcher! s in subs)
{
if (!s((IExpr!)app.Arguments[i])) { return false; }
i++;
}
return true;
}
else { return false; }
}
///
/// Not intended to be instantiated.
///
private ExprUtil() { }
}
//------------------------------ Symbols --------------------------------
///
/// An interface for function symbols. Constants are represented by
/// 0-ary function symbols.
///
/// This interface should be implemented by abstract domains, but client
/// expressions need keep track of function symbols.
///
public interface IFunctionSymbol
{
AIType! AIType { [Pure][Reads(ReadsAttribute.Reads.Owned)][Rep][ResultNotNewlyAllocated]
get; }
}
///
/// The type of the arguments to ExprUtil.Match, a poor man's pattern
/// matching.
///
public interface IMatchable
{
}
//-------------------------------- Types --------------------------------
///
/// Types.
///
public interface AIType
{
}
///
/// Function type constructor.
///
public sealed class FunctionType : AIType
{
/*[Own]*/ private readonly IList/**/! argTypes;
/*[Own]*/ private readonly AIType! retType;
public FunctionType(params AIType[]! types)
requires types.Length >= 2;
{
AIType type = types[types.Length-1];
assume type != null;
this.retType = type;
ArrayList argTypes = new ArrayList();
for (int i = 0; i < types.Length-1; i++)
{
type = types[i];
assume type != null;
argTypes.Add(types);
}
this.argTypes = ArrayList.ReadOnly(argTypes);
}
public IList/**/! Arguments
{
[Pure][Reads(ReadsAttribute.Reads.Owned)][Rep]
get
ensures result.IsReadOnly;
{
return argTypes;
}
}
public int Arity
{
[Pure][Reads(ReadsAttribute.Reads.Owned)] get { return argTypes.Count; }
}
public AIType! ReturnType
{
[Pure][Reads(ReadsAttribute.Reads.Owned)] get { return retType; }
}
/* TODO Do we have the invariant that two functions are equal iff they're the same object.
public override bool Equals(object o)
{
if (o != null && o is FunctionType)
{
FunctionType other = (FunctionType) o;
if (Arity == other.Arity
&& ReturnType.Equals(other.ReturnType))
{
for (int i = 0; i < Arity; i++)
{
if (!argTypes[i].Equals(other.argTypes[i]))
return false;
}
return true;
}
else
return false;
}
else
return false;
}
*/
}
//------------------------------ Queries -------------------------------
public enum Answer { Yes, No, Maybe };
///
/// An interface that specifies a queryable object that can answer
/// whether a predicate holds.
///
public interface IQueryable
{
///
/// Answers the query whether the given predicate holds.
///
/// The given predicate.
/// Yes, No, or Maybe.
Answer CheckPredicate(IExpr! pred);
///
/// A simplified interface for disequalities. One can always
/// implement this by calling CheckPredicate, but it may be
/// more efficient with this method.
///
Answer CheckVariableDisequality(IVariable! var1, IVariable! var2);
}
public static class QueryUtil
{
public static Answer Negate(Answer ans)
{
switch (ans)
{
case Answer.Yes:
return Answer.No;
case Answer.No:
return Answer.Yes;
default:
return Answer.Maybe;
}
}
}
//----------------------------- Exceptions -----------------------------
public class TypeError : CheckedException
{
}
}