mirror of https://github.com/icsharpcode/ILSpy.git
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
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(); |
|
} |
|
} |
|
}
|
|
|