a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
404 lines
16 KiB
C#
404 lines
16 KiB
C#
//
|
|
// Interval.cs
|
|
//
|
|
// Authors:
|
|
// Alexander Chebaturkin (chebaturkin@gmail.com)
|
|
//
|
|
// Copyright (C) 2012 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 System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
|
|
using Mono.CodeContracts.Static.AST;
|
|
using Mono.CodeContracts.Static.DataStructures;
|
|
using Mono.CodeContracts.Static.Lattices;
|
|
using Mono.CodeContracts.Static.Providers;
|
|
|
|
namespace Mono.CodeContracts.Static.Analysis.Numerical {
|
|
/// <summary>
|
|
/// Represents a closed interval of <see cref="Rational"/> values.
|
|
/// </summary>
|
|
public class Interval : IntervalBase<Interval, Rational>, IEquatable<Interval> {
|
|
static Interval cached_top_value;
|
|
static Interval cached_bottom_value;
|
|
|
|
readonly bool is_bottom;
|
|
readonly bool is_top;
|
|
|
|
Interval (Rational lowerBound, Rational upperBound)
|
|
: base (lowerBound, upperBound)
|
|
{
|
|
if (lowerBound.IsMinusInfinity && upperBound.IsPlusInfinity ||
|
|
lowerBound.IsMinusInfinity && upperBound.IsMinusInfinity ||
|
|
lowerBound.IsPlusInfinity && upperBound.IsPlusInfinity) {
|
|
LowerBound = Rational.MinusInfinity;
|
|
UpperBound = Rational.PlusInfinity;
|
|
is_top = true;
|
|
}
|
|
|
|
is_bottom = LowerBound > UpperBound;
|
|
}
|
|
|
|
public static Interval TopValue
|
|
{
|
|
get
|
|
{
|
|
if (ReferenceEquals (cached_top_value, null))
|
|
cached_top_value = new Interval (Rational.MinusInfinity, Rational.PlusInfinity);
|
|
return cached_top_value;
|
|
}
|
|
}
|
|
|
|
public static Interval BottomValue
|
|
{
|
|
get
|
|
{
|
|
if (ReferenceEquals (cached_bottom_value, null))
|
|
cached_bottom_value = new Interval (Rational.PlusInfinity,
|
|
Rational.MinusInfinity);
|
|
return cached_bottom_value;
|
|
}
|
|
}
|
|
|
|
public override Interval Top { get { return TopValue; } }
|
|
public override Interval Bottom { get { return BottomValue; } }
|
|
|
|
public override bool IsTop { get { return is_top; } }
|
|
public override bool IsBottom { get { return is_bottom; } }
|
|
|
|
public override bool LessEqual (Interval that)
|
|
{
|
|
bool result;
|
|
if (this.TryTrivialLessEqual (that, out result))
|
|
return result;
|
|
|
|
//less equal <==> is included in
|
|
return LowerBound >= that.LowerBound && UpperBound <= that.UpperBound;
|
|
}
|
|
|
|
public bool LessEqual (IEnumerable<Interval> right)
|
|
{
|
|
return right.Any (LessEqual);
|
|
}
|
|
|
|
public override Interval Join (Interval that, bool widening, out bool weaker)
|
|
{
|
|
weaker = false;
|
|
|
|
return widening ? Widen (that) : Join (that);
|
|
}
|
|
|
|
public override Interval Join (Interval that)
|
|
{
|
|
Interval result;
|
|
if (this.TryTrivialJoin (that, out result))
|
|
return result;
|
|
|
|
return For (Rational.Min (LowerBound, that.LowerBound),
|
|
Rational.Max (UpperBound, that.UpperBound));
|
|
}
|
|
|
|
public override Interval Widen (Interval that)
|
|
{
|
|
Interval result;
|
|
if (this.TryTrivialJoin (that, out result))
|
|
return result;
|
|
|
|
return For (LowerBound < that.LowerBound
|
|
? ThresholdDB.GetPrevious (LowerBound)
|
|
: that.LowerBound,
|
|
UpperBound > that.UpperBound
|
|
? ThresholdDB.GetNext (UpperBound)
|
|
: that.UpperBound);
|
|
}
|
|
|
|
public override Interval Meet (Interval that)
|
|
{
|
|
Interval result;
|
|
if (this.TryTrivialMeet (that, out result))
|
|
return result;
|
|
|
|
return For (
|
|
Rational.Max (LowerBound, that.LowerBound),
|
|
Rational.Min (UpperBound, that.UpperBound));
|
|
}
|
|
|
|
public static Interval For (Rational lowerBound, Rational upperBound)
|
|
{
|
|
return new Interval (lowerBound, upperBound);
|
|
}
|
|
|
|
public static Interval For (Rational lowerBound, long upperBound)
|
|
{
|
|
return For (lowerBound, Rational.For (upperBound));
|
|
}
|
|
|
|
public static Interval For (long lowerBound, long upperBound)
|
|
{
|
|
return For (Rational.For (lowerBound), Rational.For (upperBound));
|
|
}
|
|
|
|
public static Interval For (long lower, Rational upperBound)
|
|
{
|
|
return For (Rational.For (lower), upperBound);
|
|
}
|
|
|
|
public static Interval For (Rational value)
|
|
{
|
|
return For (value, value);
|
|
}
|
|
|
|
public static Interval For (long value)
|
|
{
|
|
return For (Rational.For (value));
|
|
}
|
|
|
|
public static Interval operator + (Interval l, Interval r)
|
|
{
|
|
if (l.IsBottom || r.IsBottom)
|
|
return BottomValue;
|
|
|
|
Rational lower, upper;
|
|
if (l.IsTop || r.IsTop
|
|
|| !Rational.TryAdd (l.LowerBound, r.LowerBound, out lower)
|
|
|| !Rational.TryAdd (l.UpperBound, r.UpperBound, out upper))
|
|
return TopValue;
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
public static Interval operator - (Interval l, Interval r)
|
|
{
|
|
if (l.IsBottom || r.IsBottom)
|
|
return BottomValue;
|
|
|
|
Rational lower, upper;
|
|
if (l.IsTop || r.IsTop
|
|
|| !Rational.TrySubtract (l.LowerBound, r.UpperBound, out lower)
|
|
|| !Rational.TrySubtract (l.UpperBound, r.LowerBound, out upper))
|
|
return TopValue;
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
public static Interval operator / (Interval l, Interval r)
|
|
{
|
|
if (l.IsBottom || r.IsBottom)
|
|
return BottomValue;
|
|
|
|
Rational a, b, c, d;
|
|
if (l.IsTop || r.IsTop || r.Includes (0)
|
|
|| !Rational.TryDivide (l.LowerBound, r.UpperBound, out a)
|
|
|| !Rational.TryDivide (l.LowerBound, r.LowerBound, out b)
|
|
|| !Rational.TryDivide (l.UpperBound, r.UpperBound, out c)
|
|
|| !Rational.TryDivide (l.UpperBound, r.LowerBound, out d))
|
|
|
|
return TopValue;
|
|
|
|
var lower = Rational.Min (Rational.Min (a, b), Rational.Min (c, d));
|
|
var upper = Rational.Max (Rational.Max (a, b), Rational.Max (c, d));
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
public static Interval operator * (Interval l, Interval r)
|
|
{
|
|
if (l.IsBottom || r.IsBottom)
|
|
return BottomValue;
|
|
|
|
Rational lower, upper;
|
|
if (l.IsTop || r.IsTop
|
|
|| !Rational.TryMultiply (l.LowerBound, r.LowerBound, out lower)
|
|
|| !Rational.TryMultiply (l.UpperBound, r.UpperBound, out upper))
|
|
return TopValue;
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
public static Interval operator - (Interval l)
|
|
{
|
|
if (!l.IsNormal ())
|
|
return l;
|
|
|
|
Rational lower;
|
|
Rational upper;
|
|
if (!Rational.TryUnaryMinus (l.UpperBound, out lower) ||
|
|
!Rational.TryUnaryMinus (l.LowerBound, out upper))
|
|
return TopValue;
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
public static bool AreConsecutiveIntegers (Interval prev, Interval next)
|
|
{
|
|
if (!prev.IsNormal () || !next.IsNormal () ||
|
|
!prev.UpperBound.IsInteger || !next.LowerBound.IsInteger)
|
|
return false;
|
|
|
|
return prev.UpperBound + Rational.One == next.LowerBound;
|
|
}
|
|
|
|
public static Interval ApplyConversion (ExpressionOperator conv, Interval intv)
|
|
{
|
|
if (intv.is_bottom)
|
|
return intv;
|
|
|
|
switch (conv) {
|
|
case ExpressionOperator.ConvertToInt32:
|
|
return intv.RefineWithTypeRanges (int.MinValue, int.MaxValue);
|
|
default:
|
|
return intv;
|
|
}
|
|
}
|
|
|
|
public bool TryGetSingletonFiniteInt32 (out int value)
|
|
{
|
|
int lower;
|
|
int upper;
|
|
if (IsFiniteAndInt32 (out lower, out upper) && lower == upper)
|
|
return true.With (lower, out value);
|
|
|
|
return false.Without (out value);
|
|
}
|
|
|
|
public bool Includes (long x)
|
|
{
|
|
return this.IsNormal () && LowerBound <= x && x <= UpperBound;
|
|
}
|
|
|
|
public bool Includes (int x)
|
|
{
|
|
return this.IsNormal () && LowerBound <= x && x <= UpperBound;
|
|
}
|
|
|
|
public bool OverlapsWith (Interval that)
|
|
{
|
|
return !Meet (that).IsBottom;
|
|
}
|
|
|
|
public bool OnTheLeftOf (Interval that)
|
|
{
|
|
if (!this.IsNormal () || !that.IsNormal ())
|
|
return false;
|
|
|
|
return UpperBound <= that.LowerBound;
|
|
}
|
|
|
|
public override Interval ImmutableVersion ()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
public override Interval Clone ()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
public override bool Equals (object obj)
|
|
{
|
|
if (ReferenceEquals (null, obj))
|
|
return false;
|
|
if (ReferenceEquals (this, obj))
|
|
return true;
|
|
return Equals ((Interval) obj);
|
|
}
|
|
|
|
public bool Equals (Interval that)
|
|
{
|
|
if (ReferenceEquals (this, that))
|
|
return true;
|
|
if (ReferenceEquals (that, null))
|
|
return false;
|
|
|
|
return LowerBound == that.LowerBound && UpperBound == that.UpperBound;
|
|
}
|
|
|
|
public override int GetHashCode ()
|
|
{
|
|
return (LowerBound.GetHashCode () * 397) ^ UpperBound.GetHashCode ();
|
|
}
|
|
|
|
public override void Dump (TextWriter tw)
|
|
{
|
|
tw.WriteLine (ToString ());
|
|
}
|
|
|
|
protected override bool IsFiniteBound (Rational n)
|
|
{
|
|
return n.IsInteger;
|
|
}
|
|
|
|
bool IsFiniteAndInt32 (out int lower, out int upper)
|
|
{
|
|
if (IsFinite && LowerBound.IsInt32 && UpperBound.IsInt32) {
|
|
try {
|
|
lower = (int) LowerBound;
|
|
upper = (int) UpperBound;
|
|
return true;
|
|
}
|
|
catch (ArithmeticException) {
|
|
}
|
|
}
|
|
|
|
return false.With (0, out lower).
|
|
With (0, out upper);
|
|
}
|
|
|
|
Interval RefineWithTypeRanges (int min, int max)
|
|
{
|
|
var lower = LowerBound.IsInfinity || !LowerBound.IsInRange (min, max)
|
|
? Rational.MinusInfinity
|
|
: LowerBound.PreviousInt32;
|
|
|
|
var upper = UpperBound.IsInfinity || !UpperBound.IsInRange (min, max)
|
|
? Rational.PlusInfinity
|
|
: UpperBound.NextInt32;
|
|
|
|
return For (lower, upper);
|
|
}
|
|
|
|
internal static class Ranges {
|
|
static Interval int8Range;
|
|
static Interval int32Range;
|
|
static Interval int64Range;
|
|
|
|
public static Interval Int8Range { get { return int8Range ?? (int8Range = For (sbyte.MinValue, sbyte.MaxValue)); } }
|
|
public static Interval Int32Range { get { return int32Range ?? (int32Range = For (int.MinValue, int.MaxValue)); } }
|
|
public static Interval Int64Range { get { return int64Range ?? (int64Range = For (int.MinValue, int.MaxValue)); } }
|
|
|
|
public static Interval GetIntervalForType (TypeNode type, IMetaDataProvider provider)
|
|
{
|
|
if (provider.Equal (provider.System_Int8, type))
|
|
return Int8Range;
|
|
if (provider.Equal (provider.System_Int32, type))
|
|
return Int32Range;
|
|
if (provider.Equal (provider.System_Int64, type))
|
|
return Int64Range;
|
|
|
|
return TopValue;
|
|
}
|
|
}
|
|
}
|
|
} |