#develop (short for SharpDevelop) is a free IDE for .NET programming languages.
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.
 
 
 
 
 
 

248 lines
8.6 KiB

//
// ABIComparer.cs
//
// Author:
// Mike Krüger <mkrueger@xamarin.com>
//
// Copyright (c) 2013 Xamarin Inc. (http://xamarin.com)
//
// 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 ICSharpCode.NRefactory.TypeSystem;
using System.Collections.Generic;
using System.Linq;
namespace ICSharpCode.NRefactory.Analysis
{
/// <summary>
/// Used to check the compatibility state of two compilations.
/// </summary>
public enum AbiCompatibility
{
/// <summary>
/// The ABI is equal
/// </summary>
Equal,
/// <summary>
/// Some items got added, but the ABI remains to be compatible
/// </summary>
Bigger,
/// <summary>
/// The ABI has changed
/// </summary>
Incompatible
}
[Serializable]
public sealed class AbiEventArgs : EventArgs
{
public string Message { get; set; }
public AbiEventArgs(string message)
{
this.Message = message;
}
}
/// <summary>
/// The Abi comparer checks the public API of two compilation and determines the compatibility state.
/// </summary>
public class AbiComparer
{
public bool StopOnIncompatibility {
get;
set;
}
void CheckContstraints(IType otype, ITypeParameter p1, ITypeParameter p2, ref AbiCompatibility compatibility)
{
if (p1.DirectBaseTypes.Count () != p2.DirectBaseTypes.Count () ||
p1.HasReferenceTypeConstraint != p2.HasReferenceTypeConstraint ||
p1.HasValueTypeConstraint != p2.HasValueTypeConstraint ||
p1.HasDefaultConstructorConstraint != p2.HasDefaultConstructorConstraint) {
OnIncompatibilityFound (new AbiEventArgs (string.Format (TranslateString ("Type parameter constraints of type {0} have changed."), otype.FullName)));
compatibility = AbiCompatibility.Incompatible;
}
}
void CheckContstraints(IMethod omethod, ITypeParameter p1, ITypeParameter p2, ref AbiCompatibility compatibility)
{
if (p1.DirectBaseTypes.Count () != p2.DirectBaseTypes.Count () ||
p1.HasReferenceTypeConstraint != p2.HasReferenceTypeConstraint ||
p1.HasValueTypeConstraint != p2.HasValueTypeConstraint ||
p1.HasDefaultConstructorConstraint != p2.HasDefaultConstructorConstraint) {
OnIncompatibilityFound (new AbiEventArgs (string.Format (TranslateString ("Type parameter constraints of method {0} have changed."), omethod.FullName)));
compatibility = AbiCompatibility.Incompatible;
}
}
void CheckTypes (ITypeDefinition oType, ITypeDefinition nType, ref AbiCompatibility compatibility)
{
int oldMemberCount = 0;
Predicate<IUnresolvedMember> pred = null;
if (oType.Kind == TypeKind.Class || oType.Kind == TypeKind.Struct)
pred = m => (m.IsPublic || m.IsProtected) && !m.IsOverride && !m.IsSynthetic;
for (int i = 0; i < oType.TypeParameterCount; i++) {
CheckContstraints (oType, oType.TypeParameters[i], nType.TypeParameters[i], ref compatibility);
if (compatibility == AbiCompatibility.Incompatible && StopOnIncompatibility)
return;
}
foreach (var member in oType.GetMembers (pred, GetMemberOptions.IgnoreInheritedMembers)) {
var newMember = nType.GetMembers (m => member.UnresolvedMember.Name == m.Name && m.IsPublic == member.IsPublic && m.IsProtected == member.IsProtected);
var equalMember = newMember.FirstOrDefault (m => SignatureComparer.Ordinal.Equals (member, m));
if (equalMember == null) {
compatibility = AbiCompatibility.Incompatible;
if (StopOnIncompatibility)
return;
continue;
}
var om = member as IMethod;
if (om != null) {
for (int i = 0; i < om.TypeParameters.Count; i++) {
CheckContstraints (om, om.TypeParameters[i], ((IMethod)equalMember).TypeParameters[i], ref compatibility);
if (compatibility == AbiCompatibility.Incompatible && StopOnIncompatibility)
return;
}
}
oldMemberCount++;
}
if (compatibility == AbiCompatibility.Bigger && oType.Kind != TypeKind.Interface)
return;
if (oldMemberCount != nType.GetMembers (pred, GetMemberOptions.IgnoreInheritedMembers).Count ()) {
if (oType.Kind == TypeKind.Interface) {
OnIncompatibilityFound (new AbiEventArgs (string.Format (TranslateString ("Interafce {0} has changed."), oType.FullName)));
compatibility = AbiCompatibility.Incompatible;
} else {
if (compatibility == AbiCompatibility.Equal)
compatibility = AbiCompatibility.Bigger;
}
}
}
void CheckNamespace(INamespace oNs, INamespace nNs, ref AbiCompatibility compatibility)
{
foreach (var type in oNs.Types) {
if (!type.IsPublic && !type.IsProtected)
continue;
var newType = nNs.GetTypeDefinition (type.Name, type.TypeParameterCount);
if (newType == null) {
OnIncompatibilityFound (new AbiEventArgs (string.Format (TranslateString ("Type definition {0} is missing."), type.FullName)));
compatibility = AbiCompatibility.Incompatible;
if (StopOnIncompatibility)
return;
continue;
}
CheckTypes (type, newType, ref compatibility);
if (compatibility == AbiCompatibility.Incompatible && StopOnIncompatibility)
return;
}
if (compatibility == AbiCompatibility.Bigger)
return;
foreach (var type in nNs.Types) {
if (!type.IsPublic && !type.IsProtected)
continue;
if (oNs.GetTypeDefinition (type.Name, type.TypeParameterCount) == null) {
if (compatibility == AbiCompatibility.Equal)
compatibility = AbiCompatibility.Bigger;
return;
}
}
}
static bool ContainsPublicTypes(INamespace testNs)
{
var stack = new Stack<INamespace> ();
stack.Push (testNs);
while (stack.Count > 0) {
var ns = stack.Pop ();
if (ns.Types.Any (t => t.IsPublic))
return true;
foreach (var child in ns.ChildNamespaces)
stack.Push (child);
}
return false;
}
/// <summary>
/// Check the specified oldProject and newProject if they're compatible.
/// </summary>
/// <param name="oldProject">Old project.</param>
/// <param name="newProject">New project.</param>
public AbiCompatibility Check (ICompilation oldProject, ICompilation newProject)
{
var oldStack = new Stack<INamespace> ();
var newStack = new Stack<INamespace> ();
oldStack.Push (oldProject.MainAssembly.RootNamespace);
newStack.Push (newProject.MainAssembly.RootNamespace);
AbiCompatibility compatibility = AbiCompatibility.Equal;
while (oldStack.Count > 0) {
var oNs = oldStack.Pop ();
var nNs = newStack.Pop ();
CheckNamespace (oNs, nNs, ref compatibility);
if (compatibility == AbiCompatibility.Incompatible && StopOnIncompatibility)
return AbiCompatibility.Incompatible;
foreach (var child in oNs.ChildNamespaces) {
var newChild = nNs.GetChildNamespace (child.Name);
if (newChild == null) {
OnIncompatibilityFound (new AbiEventArgs (string.Format (TranslateString ("Namespace {0} is missing."), child.FullName)));
if (StopOnIncompatibility)
return AbiCompatibility.Incompatible;
continue;
}
oldStack.Push (child);
newStack.Push (newChild);
}
// check if namespaces are added
if (compatibility != AbiCompatibility.Bigger) {
foreach (var child in nNs.ChildNamespaces) {
if (oNs.GetChildNamespace (child.Name) == null) {
if (compatibility == AbiCompatibility.Equal && ContainsPublicTypes (child))
compatibility = AbiCompatibility.Bigger;
break;
}
}
}
}
return compatibility;
}
public virtual string TranslateString(string str)
{
return str;
}
public event EventHandler<AbiEventArgs> IncompatibilityFound;
protected virtual void OnIncompatibilityFound(AbiEventArgs e)
{
var handler = IncompatibilityFound;
if (handler != null)
handler(this, e);
}
}
}