.NET Decompiler with support for PDB generation, ReadyToRun, Metadata (&more) - cross-platform!
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

167 lines
4.4 KiB

// Copyright (c) 2016 Daniel Grunwald
//
// 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.Diagnostics;
namespace ICSharpCode.Decompiler.IL
{
/// <summary>
/// The collection of variables in a <c>ILFunction</c>.
/// </summary>
public class ILVariableCollection : ICollection<ILVariable>, IReadOnlyList<ILVariable>
{
readonly ILFunction scope;
readonly List<ILVariable> list = new List<ILVariable>();
internal ILVariableCollection(ILFunction scope)
{
this.scope = scope;
}
/// <summary>
/// Gets a variable given its <c>IndexInFunction</c>.
/// </summary>
public ILVariable this[int index] {
get {
return list[index];
}
}
public bool Add(ILVariable item)
{
if (item.Function != null)
{
if (item.Function == scope)
return false;
else
throw new ArgumentException("Variable already belongs to another scope");
}
item.Function = scope;
item.IndexInFunction = list.Count;
list.Add(item);
return true;
}
void ICollection<ILVariable>.Add(ILVariable item)
{
Add(item);
}
public void Clear()
{
foreach (var v in list)
{
v.Function = null;
}
list.Clear();
}
public bool Contains(ILVariable item)
{
Debug.Assert(item.Function != scope || list[item.IndexInFunction] == item);
return item.Function == scope;
}
public bool Remove(ILVariable item)
{
if (item.Function != scope)
return false;
Debug.Assert(list[item.IndexInFunction] == item);
RemoveAt(item.IndexInFunction);
return true;
}
void RemoveAt(int index)
{
list[index].Function = null;
// swap-remove index
list[index] = list[list.Count - 1];
list[index].IndexInFunction = index;
list.RemoveAt(list.Count - 1);
}
/// <summary>
/// Remove variables that have StoreCount == LoadCount == AddressCount == 0.
/// </summary>
public void RemoveDead()
{
for (int i = 0; i < list.Count;)
{
if (ShouldRemoveVariable(list[i]))
{
RemoveAt(i);
}
else
{
i++;
}
}
static bool ShouldRemoveVariable(ILVariable v)
{
if (!v.IsDead)
return false;
// Note: we cannot remove display-class locals from the collection,
// even if they are unused - which is always the case, if TDCU succeeds,
// because they are necessary for PDB generation to produce correct results.
if (v.Kind == VariableKind.DisplayClassLocal)
return false;
// Do not remove parameter variables, as these are defined even if unused.
if (v.Kind == VariableKind.Parameter)
{
// However, remove unused this-parameters of delegates, expression trees, etc.
// These will be replaced with the top-level function's this-parameter.
if (v.Index == -1 && v.Function.Kind != ILFunctionKind.TopLevelFunction)
return true;
return false;
}
return true;
}
}
public int Count {
get { return list.Count; }
}
public void CopyTo(ILVariable[] array, int arrayIndex)
{
list.CopyTo(array, arrayIndex);
}
bool ICollection<ILVariable>.IsReadOnly {
get { return false; }
}
public List<ILVariable>.Enumerator GetEnumerator()
{
return list.GetEnumerator();
}
IEnumerator<ILVariable> IEnumerable<ILVariable>.GetEnumerator()
{
return GetEnumerator();
}
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
{
return GetEnumerator();
}
}
}