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

255 lines
8.0 KiB
C#

//
// HeapAnalysis.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.Collections.Generic;
using System.IO;
using Mono.CodeContracts.Static.AST;
using Mono.CodeContracts.Static.AST.Visitors;
using Mono.CodeContracts.Static.Analysis.HeapAnalysis.SymbolicGraph;
using Mono.CodeContracts.Static.ControlFlow;
using Mono.CodeContracts.Static.DataFlowAnalysis;
using Mono.CodeContracts.Static.DataStructures;
using Mono.CodeContracts.Static.Providers;
namespace Mono.CodeContracts.Static.Analysis.HeapAnalysis {
class HeapAnalysis : IAnalysis<APC, Domain, IILVisitor<APC, int, int, Domain, Domain>, Dummy> {
private readonly Dictionary<Pair<APC, APC>, IImmutableMap<SymbolicValue, Sequence<SymbolicValue>>> forwardRenamings =
new Dictionary<Pair<APC, APC>, IImmutableMap<SymbolicValue, Sequence<SymbolicValue>>> ();
public readonly Dictionary<APC, IMergeInfo> MergeInfoCache = new Dictionary<APC, IMergeInfo> ();
public readonly DoubleDictionary<APC, APC, Dummy> RenamePoints = new DoubleDictionary<APC, APC, Dummy> ();
private readonly ICodeLayer<int, int, IStackContextProvider, Dummy> stackLayer;
private IFixPointInfo<APC, Domain> fixPointInfo;
public HeapAnalysis (ICodeLayer<int, int, IStackContextProvider, Dummy> stackLayer)
{
this.stackLayer = stackLayer;
}
public IStackContextProvider StackContextProvider
{
get { return this.stackLayer.ILDecoder.ContextProvider; }
}
public IMetaDataProvider MetaDataProvider
{
get { return this.stackLayer.MetaDataProvider; }
}
public IContractProvider ContractProvider
{
get { return this.stackLayer.ContractProvider; }
}
public Method CurrentMethod
{
get { return this.StackContextProvider.MethodContext.CurrentMethod; }
}
IILVisitor<APC, int, int, Domain, Domain> IAnalysis<APC, Domain, IILVisitor<APC, int, int, Domain, Domain>, Dummy>.
GetVisitor ()
{
return GetVisitor ();
}
public Domain Join (Pair<APC, APC> edge, Domain newstate, Domain prevstate, out bool weaker, bool widen)
{
if (DebugOptions.Debug)
{
Console.WriteLine ("-----OPT Join at {0}", edge);
Console.WriteLine ("-----Existing state:");
prevstate.Dump (Console.Out);
Console.WriteLine ("-----New state:");
newstate.Dump (Console.Out);
}
IMergeInfo mi;
Domain domain = prevstate.Join (newstate, widen, out weaker, out mi);
if (weaker) {
IMergeInfo mi2;
if (this.MergeInfoCache.TryGetValue (edge.Value, out mi2) && mi2 == null)
this.MergeInfoCache [edge.Value] = mi;
} else
this.MergeInfoCache [edge.Value] = mi;
if (DebugOptions.Debug)
{
Console.WriteLine ("-----Result state: changed = {0} (widen = {1})", weaker ? 1 : 0, widen ? 1 : 0);
domain.Dump (Console.Out);
Console.WriteLine ("----------------------------------------------");
}
return domain;
}
public Domain ImmutableVersion (Domain arg)
{
return arg.ImmutableVersion ();
}
public Domain MutableVersion (Domain arg)
{
if (arg.IsBottom)
return arg;
return arg.Clone ();
}
public Domain EdgeConversion (APC @from, APC to, bool isJoinPoint, Dummy data, Domain state)
{
if (isJoinPoint) {
this.RenamePoints [from, to] = Dummy.Value;
if (!this.MergeInfoCache.ContainsKey (to))
this.MergeInfoCache.Add (to, null);
}
if (DebugOptions.Debug)
{
Console.WriteLine ("----Edge conversion on {0}->{1}------", from, to);
state.Dump (Console.Out);
Console.WriteLine ("-------------------------------------");
}
return state;
}
public bool IsBottom (APC pc, Domain state)
{
return state.IsBottom;
}
public Predicate<APC> SaveFixPointInfo (IFixPointInfo<APC, Domain> fixPointInfo)
{
this.fixPointInfo = fixPointInfo;
return pc => true;
}
public void Dump (Pair<Domain, TextWriter> pair)
{
pair.Key.Dump (pair.Value);
}
private IILVisitor<APC, int, int, Domain, Domain> GetVisitor ()
{
return new AnalysisDecoder (this);
}
public Domain InitialValue ()
{
return new Domain (this);
}
public IILDecoder<APC, SymbolicValue, SymbolicValue, IValueContextProvider<SymbolicValue>, IImmutableMap<SymbolicValue, Sequence<SymbolicValue>>>
GetDecoder<Context>(IILDecoder<APC, int, int, Context, Dummy> underlying)
where Context : IStackContextProvider
{
return new ValueDecoder<Context> (this, underlying);
}
public bool IsUnreachable (APC pc)
{
Domain domain;
if (!this.fixPointInfo.PreStateLookup (pc, out domain) || domain.IsBottom)
return true;
return false;
}
public IImmutableMap<SymbolicValue, Sequence<SymbolicValue>> EdgeRenaming (Pair<APC, APC> edge, bool isJoinPoint)
{
IImmutableMap<SymbolicValue, Sequence<SymbolicValue>> forwardRenaming;
if (this.forwardRenamings.TryGetValue (edge, out forwardRenaming))
return forwardRenaming;
IImmutableMap<SymbolicValue, Sequence<SymbolicValue>> renaming = null;
Domain afterBegin;
PostStateLookup (edge.Key, out afterBegin);
if (afterBegin == null || afterBegin.IsBottom)
return null;
Domain beforeEnd;
PreStateLookup (edge.Value, out beforeEnd);
if (beforeEnd != null) {
IImmutableMap<SymValue, Sequence<SymValue>> forward;
if (!TryComputeFromJoinCache (afterBegin, beforeEnd, edge.Value, out forward)) {
IImmutableMap<SymValue, SymValue> backward;
if (!afterBegin.LessEqual (beforeEnd, out forward, out backward))
throw new InvalidOperationException ("Should never happen");
if (isJoinPoint && forward == null)
forward = afterBegin.GetForwardIdentityMap ();
}
if (forward != null) {
renaming = ImmutableIntKeyMap<SymbolicValue, Sequence<SymbolicValue>>.Empty (SymbolicValue.GetUniqueKey);
foreach (SymValue sv in forward.Keys) {
Sequence<SymbolicValue> targets = null;
foreach (SymValue target in forward [sv].AsEnumerable ())
targets = targets.Cons (new SymbolicValue (target));
if (targets != null)
renaming = renaming.Add (new SymbolicValue (sv), targets);
}
}
}
this.forwardRenamings.Add (edge, renaming);
return renaming;
}
private bool TryComputeFromJoinCache (Domain inDomain, Domain outDomain, APC joinPoint, out IImmutableMap<SymValue, Sequence<SymValue>> forward)
{
forward = null;
IMergeInfo mi;
if (this.MergeInfoCache.TryGetValue (joinPoint, out mi) && mi != null && outDomain.IsResultEGraph (mi)) {
if (inDomain.IsGraph1 (mi)) {
forward = mi.ForwardG1Map;
return true;
}
if (inDomain.IsGraph2 (mi)) {
forward = mi.ForwardG2Map;
return true;
}
}
return false;
}
public Domain GetPreState (APC pc)
{
Domain domain;
PreStateLookup (pc, out domain);
return domain;
}
public bool PreStateLookup (APC pc, out Domain ifFound)
{
return this.fixPointInfo.PreStateLookup (pc, out ifFound);
}
public bool PostStateLookup (APC pc, out Domain ifFound)
{
return this.fixPointInfo.PostStateLookup (pc, out ifFound);
}
}
}