//-----------------------------------------------------------------------------
//
// Copyright (C) Microsoft Corporation. All Rights Reserved.
//
//-----------------------------------------------------------------------------
using Microsoft.Contracts;
namespace Microsoft.AbstractInterpretationFramework.Collections
{
using System.Collections;
/// Represents a functional collection of key/value pairs.
/// 2
public interface IFunctionalMap : System.Collections.ICollection, System.Collections.IEnumerable
{
/// Adds an element with the provided key and value to the .
/// The to use as the value of the element to add.
/// The to use as the key of the element to add.
/// 2
IFunctionalMap! Add(object! key, object value);
///
/// Set the value of the key (that is already in the map)
///
IFunctionalMap! Set(object! key, object value);
/// Determines whether the contains an element with the specified key.
/// true if the contains an element with the key; otherwise, false.
/// The key to locate in the .
/// 2
[Pure]
bool Contains(object! key);
/// Returns an for the .
/// An for the .
/// 2
[Pure] [GlobalAccess(false)] [Escapes(true,false)]
new System.Collections.IDictionaryEnumerator GetEnumerator();
/// Gets an containing the keys of the .
/// An containing the keys of the .
/// 2
System.Collections.ICollection Keys { get; }
/// Removes the element with the specified key from the .
/// The key of the element to remove.
/// 2
IFunctionalMap! Remove(object! key);
/// Gets an containing the values in the .
/// An containing the values in the .
/// 2
System.Collections.ICollection Values { get; }
object this [object! key] { get; /*set;*/ }
}
///
/// An implementation of the
///
/// interface with a as the backing store.
///
class FunctionalHashtable : IFunctionalMap
{
private readonly Hashtable! h;
///
/// Cannot directly construct an instance of a FunctionalHashtbl.
///
private FunctionalHashtable()
{
this.h = new Hashtable();
// base();
}
///
/// Cannot directly construct an instance of a FunctionalHashtbl.
///
private FunctionalHashtable(Hashtable! h)
{
this.h = h;
// base();
}
private static readonly IFunctionalMap! empty = new FunctionalHashtable();
public static IFunctionalMap! Empty { get { return empty; } }
public IFunctionalMap! Add(object! key, object value)
{
Hashtable r = h.Clone() as Hashtable;
assume r != null;
r.Add(key, value);
return new FunctionalHashtable(r);
}
public IFunctionalMap! Set(object! key, object value)
{
Hashtable r = h.Clone() as Hashtable;
assume r != null;
assert this.Contains(key); // The entry must be defined
r[key] = value;
return new FunctionalHashtable(r);
}
[Pure]
public bool Contains(object! key)
{
return h.Contains(key);
}
[Pure] [GlobalAccess(false)] [Escapes(true,false)]
IEnumerator! IEnumerable.GetEnumerator()
{
return h.GetEnumerator();
}
[Pure] [GlobalAccess(false)] [Escapes(true,false)]
IDictionaryEnumerator IFunctionalMap.GetEnumerator()
{
return h.GetEnumerator();
}
public ICollection Keys
{
get { return h.Keys; }
}
public IFunctionalMap! Remove(object! key)
{
Hashtable r = h.Clone() as Hashtable;
assume r != null;
r.Remove(key);
return new FunctionalHashtable(r);
}
public ICollection Values
{
get { return h.Values; }
}
public object this[object! key]
{
get { return h[key]; }
}
public int Count
{
[Pure] get { return h.Count; }
}
public bool IsSynchronized
{
[Pure] get { return h.IsSynchronized; }
}
public object! SyncRoot
{
[Pure] get { return h.SyncRoot; }
}
public void CopyTo(System.Array! a, int index)
{
h.CopyTo(a, index);
}
}
public struct Pair/**/
{
private object first;
private object second;
public object First { get { return first; } }
public object Second { get { return second; } }
public Pair(object first, object second)
{
this.first = first;
this.second = second;
}
public override bool Equals(object obj)
{
if (obj == null) return false;
if (!(obj is Pair)) return false;
Pair other = (Pair)obj;
return object.Equals(this.first, other.first) && object.Equals(this.second, other.second);
}
public override int GetHashCode()
{
int h = this.first == null ? 0 : this.first.GetHashCode();
h ^= this.second == null ? 0 : this.second.GetHashCode();
return h;
}
}
}
namespace Microsoft.AbstractInterpretationFramework.Collections.Generic
{
using System.Collections.Generic;
public struct Pair
{
private T1 first;
private T2 second;
public T1 First { get { return first; } }
public T2 Second { get { return second; } }
public Pair(T1 first, T2 second)
{
this.first = first;
this.second = second;
}
public override bool Equals(object obj)
{
if (obj == null) return false;
if (!(obj is Pair)) return false;
Pair other = (Pair)obj;
return object.Equals(this.first, other.first) && object.Equals(this.second, other.second);
}
public override int GetHashCode()
{
int h = this.first == null ? 0 : this.first.GetHashCode();
h ^= this.second == null ? 0 : this.second.GetHashCode();
return h;
}
public override string! ToString()
{
return string.Format("({0},{1})", first, second);
}
}
public struct Triple
{
private T1 first;
private T2 second;
private T3 third;
public T1 First { get { return first; } }
public T2 Second { get { return second; } }
public T3 Third { get { return third; } }
public Triple(T1 first, T2 second, T3 third)
{
this.first = first;
this.second = second;
this.third = third;
}
public override bool Equals(object obj)
{
if (obj == null) return false;
if (!(obj is Triple)) return false;
Triple other = (Triple)obj;
return object.Equals(this.first, other.first) && object.Equals(this.second, other.second) && object.Equals(this.third, other.third);
}
public override int GetHashCode()
{
int h = this.first == null ? 0 : this.first.GetHashCode();
h ^= this.second == null ? 0 : this.second.GetHashCode();
h ^= this.third == null ? 0 : this.third.GetHashCode();
return h;
}
public override string! ToString()
{
return string.Format("({0},{1},{2})", first, second, third);
}
}
}