2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
// Author:
|
|
|
|
// Jb Evain (jbevain@gmail.com)
|
|
|
|
//
|
2016-11-10 13:04:39 +00:00
|
|
|
// Copyright (c) 2008 - 2015 Jb Evain
|
|
|
|
// Copyright (c) 2008 - 2011 Novell, Inc.
|
2014-08-13 10:39:27 +01:00
|
|
|
//
|
2016-11-10 13:04:39 +00:00
|
|
|
// Licensed under the MIT/X11 license.
|
2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
|
|
|
|
using System;
|
2016-11-10 13:04:39 +00:00
|
|
|
using System.Threading;
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
using Mono.Collections.Generic;
|
|
|
|
|
|
|
|
namespace Mono.Cecil.Cil {
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
public sealed class MethodBody {
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
readonly internal MethodDefinition method;
|
|
|
|
|
|
|
|
internal ParameterDefinition this_parameter;
|
|
|
|
internal int max_stack_size;
|
|
|
|
internal int code_size;
|
|
|
|
internal bool init_locals;
|
|
|
|
internal MetadataToken local_var_token;
|
|
|
|
|
|
|
|
internal Collection<Instruction> instructions;
|
|
|
|
internal Collection<ExceptionHandler> exceptions;
|
|
|
|
internal Collection<VariableDefinition> variables;
|
|
|
|
|
|
|
|
public MethodDefinition Method {
|
|
|
|
get { return method; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public int MaxStackSize {
|
|
|
|
get { return max_stack_size; }
|
|
|
|
set { max_stack_size = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public int CodeSize {
|
|
|
|
get { return code_size; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool InitLocals {
|
|
|
|
get { return init_locals; }
|
|
|
|
set { init_locals = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public MetadataToken LocalVarToken {
|
|
|
|
get { return local_var_token; }
|
|
|
|
set { local_var_token = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public Collection<Instruction> Instructions {
|
2016-11-10 13:04:39 +00:00
|
|
|
get { return instructions ?? (instructions = new InstructionCollection (method)); }
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public bool HasExceptionHandlers {
|
|
|
|
get { return !exceptions.IsNullOrEmpty (); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public Collection<ExceptionHandler> ExceptionHandlers {
|
|
|
|
get { return exceptions ?? (exceptions = new Collection<ExceptionHandler> ()); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool HasVariables {
|
|
|
|
get { return !variables.IsNullOrEmpty (); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public Collection<VariableDefinition> Variables {
|
|
|
|
get { return variables ?? (variables = new VariableDefinitionCollection ()); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public ParameterDefinition ThisParameter {
|
|
|
|
get {
|
|
|
|
if (method == null || method.DeclaringType == null)
|
|
|
|
throw new NotSupportedException ();
|
|
|
|
|
|
|
|
if (!method.HasThis)
|
|
|
|
return null;
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
if (this_parameter == null)
|
|
|
|
Interlocked.CompareExchange (ref this_parameter, CreateThisParameter (method), null);
|
2014-08-13 10:39:27 +01:00
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
return this_parameter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static ParameterDefinition CreateThisParameter (MethodDefinition method)
|
|
|
|
{
|
|
|
|
var parameter_type = method.DeclaringType as TypeReference;
|
|
|
|
|
|
|
|
if (parameter_type.HasGenericParameters) {
|
|
|
|
var instance = new GenericInstanceType (parameter_type);
|
|
|
|
for (int i = 0; i < parameter_type.GenericParameters.Count; i++)
|
|
|
|
instance.GenericArguments.Add (parameter_type.GenericParameters [i]);
|
|
|
|
|
|
|
|
parameter_type = instance;
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
}
|
2016-11-10 13:04:39 +00:00
|
|
|
|
|
|
|
if (parameter_type.IsValueType || parameter_type.IsPrimitive)
|
|
|
|
parameter_type = new ByReferenceType (parameter_type);
|
|
|
|
|
|
|
|
return new ParameterDefinition (parameter_type, method);
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public MethodBody (MethodDefinition method)
|
|
|
|
{
|
|
|
|
this.method = method;
|
|
|
|
}
|
|
|
|
|
|
|
|
public ILProcessor GetILProcessor ()
|
|
|
|
{
|
|
|
|
return new ILProcessor (this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
sealed class VariableDefinitionCollection : Collection<VariableDefinition> {
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
internal VariableDefinitionCollection ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
internal VariableDefinitionCollection (int capacity)
|
|
|
|
: base (capacity)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnAdd (VariableDefinition item, int index)
|
|
|
|
{
|
|
|
|
item.index = index;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnInsert (VariableDefinition item, int index)
|
|
|
|
{
|
|
|
|
item.index = index;
|
|
|
|
|
|
|
|
for (int i = index; i < size; i++)
|
|
|
|
items [i].index = i + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnSet (VariableDefinition item, int index)
|
|
|
|
{
|
|
|
|
item.index = index;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnRemove (VariableDefinition item, int index)
|
|
|
|
{
|
|
|
|
item.index = -1;
|
|
|
|
|
|
|
|
for (int i = index + 1; i < size; i++)
|
|
|
|
items [i].index = i - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class InstructionCollection : Collection<Instruction> {
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
readonly MethodDefinition method;
|
|
|
|
|
|
|
|
internal InstructionCollection (MethodDefinition method)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
2016-11-10 13:04:39 +00:00
|
|
|
this.method = method;
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
internal InstructionCollection (MethodDefinition method, int capacity)
|
2014-08-13 10:39:27 +01:00
|
|
|
: base (capacity)
|
|
|
|
{
|
2016-11-10 13:04:39 +00:00
|
|
|
this.method = method;
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnAdd (Instruction item, int index)
|
|
|
|
{
|
|
|
|
if (index == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
var previous = items [index - 1];
|
|
|
|
previous.next = item;
|
|
|
|
item.previous = previous;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnInsert (Instruction item, int index)
|
|
|
|
{
|
|
|
|
if (size == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
var current = items [index];
|
|
|
|
if (current == null) {
|
|
|
|
var last = items [index - 1];
|
|
|
|
last.next = item;
|
|
|
|
item.previous = last;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var previous = current.previous;
|
|
|
|
if (previous != null) {
|
|
|
|
previous.next = item;
|
|
|
|
item.previous = previous;
|
|
|
|
}
|
|
|
|
|
|
|
|
current.previous = item;
|
|
|
|
item.next = current;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnSet (Instruction item, int index)
|
|
|
|
{
|
|
|
|
var current = items [index];
|
|
|
|
|
|
|
|
item.previous = current.previous;
|
|
|
|
item.next = current.next;
|
|
|
|
|
|
|
|
current.previous = null;
|
|
|
|
current.next = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnRemove (Instruction item, int index)
|
|
|
|
{
|
|
|
|
var previous = item.previous;
|
|
|
|
if (previous != null)
|
|
|
|
previous.next = item.next;
|
|
|
|
|
|
|
|
var next = item.next;
|
|
|
|
if (next != null)
|
|
|
|
next.previous = item.previous;
|
|
|
|
|
2016-11-10 13:04:39 +00:00
|
|
|
RemoveSequencePoint (item);
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
item.previous = null;
|
|
|
|
item.next = null;
|
|
|
|
}
|
2016-11-10 13:04:39 +00:00
|
|
|
|
|
|
|
void RemoveSequencePoint (Instruction instruction)
|
|
|
|
{
|
|
|
|
var debug_info = method.debug_info;
|
|
|
|
if (debug_info == null || !debug_info.HasSequencePoints)
|
|
|
|
return;
|
|
|
|
|
|
|
|
var sequence_points = debug_info.sequence_points;
|
|
|
|
for (int i = 0; i < sequence_points.Count; i++) {
|
|
|
|
if (sequence_points [i].Offset == instruction.offset) {
|
|
|
|
sequence_points.RemoveAt (i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
}
|