95fdb59ea6
Former-commit-id: b39a328747c2f3414dc52e009fb6f0aa80ca2492
216 lines
7.0 KiB
C#
216 lines
7.0 KiB
C#
//
|
|
// Mono.Documentation/exceptions.cs
|
|
//
|
|
// Authors:
|
|
// Jonathan Pryor (jonpryor@vt.edu)
|
|
//
|
|
// (C) 2008 Novell, Inc.
|
|
//
|
|
// 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.Collections.ObjectModel;
|
|
using System.Linq;
|
|
|
|
using Mono.Cecil;
|
|
using Mono.Cecil.Cil;
|
|
using Mono.Documentation.Updater;
|
|
|
|
namespace Mono.Documentation {
|
|
|
|
[Flags]
|
|
public enum ExceptionLocations {
|
|
Member = 0x0,
|
|
Assembly = 0x1,
|
|
DependentAssemblies = 0x2,
|
|
AddedMembers = 0x4,
|
|
}
|
|
|
|
public class ExceptionSources {
|
|
internal ExceptionSources (TypeReference exception)
|
|
{
|
|
Exception = exception;
|
|
Sources = new HashSet<MemberReference> ();
|
|
}
|
|
|
|
public TypeReference Exception { get; private set; }
|
|
public HashSet<MemberReference> Sources;
|
|
}
|
|
|
|
|
|
public class ExceptionLookup {
|
|
|
|
SlashDocMemberFormatter xdoc = new SlashDocMemberFormatter ();
|
|
|
|
// xdoc(MemberRef) -> xdoc(TypeRef) -> ExceptionSource
|
|
// where ExceptionSource.Exception == xdoc(TypeRef)
|
|
Dictionary<string, Dictionary<string, ExceptionSources>> db = new Dictionary<string, Dictionary<string, ExceptionSources>> ();
|
|
|
|
ExceptionLocations locations;
|
|
|
|
public ExceptionLookup (ExceptionLocations locations)
|
|
{
|
|
this.locations = locations;
|
|
}
|
|
|
|
public IEnumerable<ExceptionSources> this [MemberReference member] {
|
|
get {
|
|
if (member == null)
|
|
throw new ArgumentNullException ("member");
|
|
|
|
var memberDef = member.Resolve ();
|
|
if (memberDef == null) {
|
|
ArrayType array = member.DeclaringType as ArrayType;
|
|
if (array != null && array.Rank > 1) {
|
|
// Multi-dimensional array; the member is runtime generated,
|
|
// doesn't "really" exist (in a form that we can resolve),
|
|
// so we can't do anything further.
|
|
return new ExceptionSources[0];
|
|
}
|
|
return new ExceptionSources[0];
|
|
//throw new NotSupportedException (string.Format (
|
|
// "Unable to resolve member {0}::{1}.",
|
|
// member.DeclaringType.FullName, member.Name));
|
|
}
|
|
string memberDecl = xdoc.GetDeclaration (member);
|
|
Dictionary<string, ExceptionSources> e;
|
|
if (!db.TryGetValue (memberDecl, out e)) {
|
|
e = new Dictionary<string, ExceptionSources> ();
|
|
db.Add (memberDecl, e);
|
|
var bodies = GetMethodBodies (member);
|
|
foreach (var body in bodies) {
|
|
if (body == null)
|
|
continue;
|
|
FillExceptions (body, e);
|
|
}
|
|
}
|
|
return e.Values;
|
|
}
|
|
}
|
|
|
|
MethodBody[] GetMethodBodies (MemberReference member)
|
|
{
|
|
if (member is MethodReference) {
|
|
return new[]{ (((MethodReference) member).Resolve ()).Body };
|
|
}
|
|
if (member is PropertyReference) {
|
|
PropertyDefinition prop = (PropertyDefinition) member;
|
|
return new[]{
|
|
prop.GetMethod != null ? prop.GetMethod.Body : null,
|
|
prop.SetMethod != null ? prop.SetMethod.Body : null,
|
|
};
|
|
}
|
|
if (member is FieldReference)
|
|
return new MethodBody[]{};
|
|
if (member is EventReference) {
|
|
EventDefinition ev = (EventDefinition) member;
|
|
return new[]{
|
|
ev.AddMethod != null ? ev.AddMethod.Body : null,
|
|
ev.InvokeMethod != null ? ev.InvokeMethod.Body : null,
|
|
ev.RemoveMethod != null ? ev.RemoveMethod.Body : null,
|
|
};
|
|
}
|
|
throw new NotSupportedException ("Unsupported member type: " + member.GetType().FullName);
|
|
}
|
|
|
|
void FillExceptions (MethodBody body, Dictionary<string, ExceptionSources> exceptions)
|
|
{
|
|
for (int i = 0; i < body.Instructions.Count; ++i) {
|
|
Instruction instruction = body.Instructions [i];
|
|
switch (instruction.OpCode.Code) {
|
|
case Code.Call:
|
|
case Code.Callvirt: {
|
|
if ((locations & ExceptionLocations.Assembly) == 0 &&
|
|
(locations & ExceptionLocations.DependentAssemblies) == 0)
|
|
break;
|
|
MemberReference memberRef = ((MemberReference) instruction.Operand);
|
|
if (((locations & ExceptionLocations.Assembly) != 0 &&
|
|
body.Method.DeclaringType.Scope.Name == memberRef.DeclaringType.Scope.Name) ||
|
|
((locations & ExceptionLocations.DependentAssemblies) != 0 &&
|
|
body.Method.DeclaringType.Scope.Name != memberRef.DeclaringType.Scope.Name)) {
|
|
|
|
IEnumerable<ExceptionSources> memberExceptions = this [memberRef];
|
|
AddExceptions (body, instruction,
|
|
memberExceptions.Select (es => es.Exception),
|
|
memberExceptions.SelectMany (es => es.Sources),
|
|
exceptions);
|
|
}
|
|
break;
|
|
}
|
|
case Code.Newobj: {
|
|
MethodReference ctor = (MethodReference) instruction.Operand;
|
|
if (IsExceptionConstructor (ctor)) {
|
|
AddExceptions (body, instruction,
|
|
new TypeReference[]{ctor.DeclaringType},
|
|
new MemberReference[]{body.Method},
|
|
exceptions);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void AddExceptions (MethodBody body, Instruction instruction, IEnumerable<TypeReference> add, IEnumerable<MemberReference> sources,
|
|
Dictionary<string, ExceptionSources> exceptions)
|
|
{
|
|
var handlers = body.ExceptionHandlers.Cast<ExceptionHandler> ()
|
|
.Where (eh => instruction.Offset >= eh.TryStart.Offset &&
|
|
instruction.Offset <= eh.TryEnd.Offset);
|
|
foreach (var ex in add) {
|
|
if (!handlers.Any (h => IsExceptionCaught (ex, h.CatchType))) {
|
|
ExceptionSources s;
|
|
string eName = xdoc.GetDeclaration (ex);
|
|
if (!exceptions.TryGetValue (eName, out s)) {
|
|
s = new ExceptionSources (ex);
|
|
exceptions.Add (eName, s);
|
|
}
|
|
foreach (var m in sources)
|
|
s.Sources.Add (m);
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool IsExceptionConstructor (MethodReference ctor)
|
|
{
|
|
return GetBases (ctor.DeclaringType)
|
|
.Any (t => t.FullName == "System.Exception");
|
|
}
|
|
|
|
bool IsExceptionCaught (TypeReference exception, TypeReference catcher)
|
|
{
|
|
return GetBases (exception).Select (e => xdoc.GetDeclaration (e))
|
|
.Union (GetBases (catcher).Select (e => xdoc.GetDeclaration (e)))
|
|
.Any ();
|
|
}
|
|
|
|
static IEnumerable<TypeReference> GetBases (TypeReference type)
|
|
{
|
|
yield return type;
|
|
TypeDefinition def = type.Resolve ();
|
|
while (def != null && def.BaseType != null) {
|
|
yield return def.BaseType;
|
|
def = def.BaseType.Resolve ();
|
|
}
|
|
}
|
|
}
|
|
}
|