Jo Shields a575963da9 Imported Upstream version 3.6.0
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
2014-08-13 10:39:27 +01:00

206 lines
6.6 KiB
C#

//
// PathElement`1.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.DataStructures;
using Mono.CodeContracts.Static.Providers;
namespace Mono.CodeContracts.Static.Analysis.HeapAnalysis.Paths {
class PathElement<T> : PathElementBase {
public readonly string Description;
public readonly T Element;
protected string castTo;
protected bool isManagedPointer;
protected bool isStatic;
protected bool isUnmanagedPointer;
#region Overrides of PathElement
public override bool IsStatic
{
get { return this.isStatic; }
}
public override bool IsUnmanagedPointer
{
get { return this.isUnmanagedPointer; }
}
public override bool IsManagedPointer
{
get { return this.isManagedPointer; }
}
public override bool IsParameterRef
{
get { return typeof (T) == typeof (Parameter); }
}
public override string CastTo
{
get { return this.castTo; }
}
public override bool IsAddressOf
{
get { return true; }
}
public override bool TryField (out Field f)
{
if (typeof (T) == typeof (Field)) {
f = (Field) (object) this.Element;
return true;
}
f = default(Field);
return false;
}
public override bool TryGetResultType (out TypeNode type)
{
type = ResultType;
return true;
}
public override TResult Decode<TData, TResult, TVisitor, TLabel> (TLabel label, TVisitor visitor, TData data)
{
if (typeof (T) == typeof (Field)) {
var field = (Field) (object) this.Element;
if (this.isStatic)
return visitor.LoadStaticFieldAddress (label, field, Dummy.Value, data);
return visitor.LoadFieldAddress (label, field, Dummy.Value, Dummy.Value, data);
}
if (typeof (T) == typeof (Local)) {
var local = (Local) (object) this.Element;
return visitor.LoadLocalAddress (label, local, Dummy.Value, data);
}
if (typeof (T) == typeof (Method)) {
var method = (Method) (object) this.Element;
bool isVirtualMethod = this.Func.IsVirtualMethod;
return visitor.Call (label, method, isVirtualMethod, Indexable<TypeNode>.Empty, Dummy.Value, Indexable<Dummy>.Empty, data);
}
if (typeof (T) == typeof (Parameter)) {
var parameter = (Parameter) (object) this.Element;
return visitor.LoadArgAddress (label, parameter, false, Dummy.Value, data);
}
throw new InvalidOperationException ("Field, Local, Method or Parameter expected");
}
public override string ToString ()
{
return this.Description;
}
#endregion
public PathElement (T element, string description, SymFunction c) : base (c)
{
this.Element = element;
this.Description = description;
this.isStatic = false;
this.isUnmanagedPointer = false;
this.isManagedPointer = false;
}
public TypeNode ResultType { get; protected set; }
public virtual bool IsCallerVisible ()
{
return (typeof (T) == typeof (Parameter));
}
#region Overrides of PathElementBase
public override bool TrySetType (TypeNode expectedType, IMetaDataProvider metaDataProvider, out TypeNode resultType)
{
if (typeof (T) == typeof (Parameter)) {
var p = (Parameter) (object) this.Element;
TypeNode type = metaDataProvider.ParameterType (p);
this.isManagedPointer = metaDataProvider.IsManagedPointer (type);
ResultType = resultType = metaDataProvider.ManagedPointer (type);
return true;
}
if (typeof (T) == typeof (Field)) {
var f = (Field) (object) this.Element;
TypeNode type = metaDataProvider.FieldType (f);
this.isStatic = metaDataProvider.IsStatic (f);
this.isManagedPointer = metaDataProvider.IsManagedPointer (type);
ResultType = resultType = metaDataProvider.ManagedPointer (type);
TypeNode declaringType = metaDataProvider.DeclaringType (f);
if (metaDataProvider.IsManagedPointer (expectedType))
expectedType = metaDataProvider.ElementType (expectedType);
expectedType = metaDataProvider.Unspecialized (expectedType);
if (!metaDataProvider.IsStatic (f) && declaringType.Equals (expectedType) &&
(!metaDataProvider.DerivesFrom (expectedType, declaringType) ||
!metaDataProvider.IsProtected (f) && !metaDataProvider.IsPublic (f)))
this.castTo = metaDataProvider.FullName (declaringType);
return true;
}
if (typeof (T) == typeof (Local)) {
var local = (Local) (object) this.Element;
TypeNode type = metaDataProvider.LocalType (local);
this.isManagedPointer = metaDataProvider.IsManagedPointer (type);
ResultType = resultType = metaDataProvider.ManagedPointer (type);
return true;
}
if (typeof (T) == typeof (Method)) {
var method = (Method) (object) this.Element;
ResultType = resultType = !IsAddressOf
? metaDataProvider.ReturnType (method)
: metaDataProvider.ManagedPointer (metaDataProvider.ReturnType (method));
if (metaDataProvider.IsManagedPointer (expectedType))
expectedType = metaDataProvider.ElementType (expectedType);
expectedType = metaDataProvider.Unspecialized (expectedType);
TypeNode declaringType = metaDataProvider.DeclaringType (method);
if (!metaDataProvider.IsStatic (method) && declaringType.Equals (expectedType) &&
(!metaDataProvider.DerivesFrom (expectedType, declaringType)
|| !metaDataProvider.IsProtected (method) && !metaDataProvider.IsPublic (method)))
this.castTo = metaDataProvider.FullName (declaringType);
return true;
}
ResultType = resultType = default(TypeNode);
return false;
}
#endregion
}
}