// // SetDomain.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 System.IO; using Mono.CodeContracts.Static.DataStructures; namespace Mono.CodeContracts.Static.Lattices { struct SetDomain : IAbstractDomain> where T : IEquatable { public static readonly SetDomain TopValue = new SetDomain (ImmutableSet.Empty ()); public static readonly SetDomain BottomValue = new SetDomain ((IImmutableSet) null); readonly IImmutableSet set; SetDomain (IImmutableSet set) { this.set = set; } public SetDomain (Func keyConverter) { set = ImmutableSet.Empty (keyConverter); } public SetDomain Top { get { return TopValue; } } public SetDomain Bottom { get { return BottomValue; } } public bool IsTop { get { return set != null && set.Count == 0; } } public bool IsBottom { get { return set == null; } } public SetDomain Join (SetDomain that) { SetDomain result; if (this.TryTrivialJoin (that, out result)) return result; return new SetDomain (set.Intersect (that.set)); } public SetDomain Join (SetDomain that, bool widening, out bool weaker) { if (set == that.set) { weaker = false; return this; } if (IsBottom) { weaker = !that.IsBottom; return that; } if (that.IsBottom || IsTop) { weaker = false; return this; } if (that.IsTop) { weaker = !IsTop; return that; } var join = set.Intersect (that.set); weaker = join.Count < set.Count; return new SetDomain (join); } public SetDomain Widen (SetDomain that) { //no widening - finite lattice return Join (that); } public SetDomain Meet (SetDomain that) { SetDomain result; if (this.TryTrivialMeet (that, out result)) return result; return new SetDomain (set.Union (that.set)); } public bool LessEqual (SetDomain that) { if (IsBottom) return true; if (that.IsBottom) return false; return that.set.IsContainedIn (set); } public SetDomain ImmutableVersion () { return this; } public SetDomain Clone () { return this; } public SetDomain With (T elem) { return new SetDomain (set.Add (elem)); } public SetDomain Without (T elem) { return new SetDomain (set.Remove (elem)); } public bool Contains (T item) { return set.Contains (item); } public void Dump (TextWriter tw) { if (IsBottom) tw.WriteLine ("Bot"); else if (IsTop) tw.WriteLine ("Top"); else set.Dump (tw); } } }