// // ExpressionAssertDischarger.cs // // Authors: // Alexander Chebaturkin (chebaturkin@gmail.com) // // Copyright (C) 2011 Alexander Chebaturkin // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // using System; using Mono.CodeContracts.Static.AST; using Mono.CodeContracts.Static.AST.Visitors; using Mono.CodeContracts.Static.ControlFlow; using Mono.CodeContracts.Static.Lattices; namespace Mono.CodeContracts.Static.Analysis.NonNull { struct ExpressionAssertDischarger : ISymbolicExpressionVisitor> where E : IEquatable where V : IEquatable { private readonly Analysis analysis; private readonly APC pc; public ExpressionAssertDischarger(Analysis analysis, APC pc) { this.analysis = analysis; this.pc = pc; } private IExpressionContextProvider ContextProvider { get { return this.analysis.ContextProvider; } } #region Implementation of IExpressionILVisitor private FlatDomain Recurse(bool polarity, E expr) { return this.ContextProvider.ExpressionContext.Decode, ExpressionAssertDischarger>(expr, this, polarity); } public FlatDomain Binary(E orig, BinaryOperator op, V dest, E operand1, E operand2, bool polarity) { switch (op) { case BinaryOperator.Ceq: case BinaryOperator.Cobjeq: if (this.ContextProvider.ExpressionContext.IsZero (operand2) || this.ContextProvider.ExpressionContext.IsZero (operand1)) return this.Recurse (!polarity, operand1); return ProofOutcome.Top; case BinaryOperator.Cne_Un: if (this.ContextProvider.ExpressionContext.IsZero(operand2) || this.ContextProvider.ExpressionContext.IsZero(operand1)) return this.Recurse (polarity, operand1); return ProofOutcome.Top; default: return this.SymbolicConstant (orig, this.ContextProvider.ExpressionContext.Unrefine (orig), polarity); } } public FlatDomain Isinst(E orig, TypeNode type, V dest, E obj, bool polarity) { if (!polarity) return this.analysis.IsNull (this.pc, dest); FlatDomain outcome = this.analysis.IsNonNull(this.pc, dest); return outcome.IsTrue() ? outcome : this.Recurse (true, obj); } public FlatDomain LoadNull(E orig, V dest, bool polarity) { return polarity ? ProofOutcome.False : ProofOutcome.True; } public FlatDomain LoadConst(E orig, TypeNode type, object constant, V dest, bool polarity) { var isConstantEqualZero = constant is int && (int) constant == 0; return (isConstantEqualZero != polarity) ? ProofOutcome.True : ProofOutcome.False; } public FlatDomain Sizeof(E pc, TypeNode type, V dest, bool polarity) { return polarity ? ProofOutcome.True : ProofOutcome.False; } public FlatDomain Unary(E orig, UnaryOperator op, bool unsigned, V dest, E source, bool polarity) { switch (op) { case UnaryOperator.Conv_i: case UnaryOperator.Conv_i1: case UnaryOperator.Conv_i2: case UnaryOperator.Conv_i4: case UnaryOperator.Conv_i8: case UnaryOperator.Conv_u: case UnaryOperator.Conv_u1: case UnaryOperator.Conv_u2: case UnaryOperator.Conv_u4: case UnaryOperator.Conv_u8: return this.Recurse (polarity, source); case UnaryOperator.Neg: return this.Recurse(polarity, source); case UnaryOperator.Not: return this.Recurse(!polarity, source); default: return this.SymbolicConstant (orig, this.ContextProvider.ExpressionContext.Unrefine (orig), polarity); } } #endregion #region Implementation of ISymbolicExpressionVisitor public FlatDomain SymbolicConstant(E pc, V variable, bool polarity) { return polarity ? this.analysis.IsNonNull (this.pc, variable) : this.analysis.IsNull(this.pc, variable); } #endregion } }