// // MarkNSObjects.cs // // Authors: // Jb Evain (jbevain@novell.com) // Sebastien Pouliot // // (C) 2009 Novell, Inc. // Copyright (C) 2011-2014 Xamarin, 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 Mono.Cecil; using Mono.Linker; using Mono.Tuner; namespace Xamarin.Linker.Steps { public class MarkNSObjects : BaseSubStep { static string ProductAssembly; public override SubStepTargets Targets { get { return SubStepTargets.Type; } } public override void ProcessType (TypeDefinition type) { if (ProductAssembly == null) ProductAssembly = (Profile.Current as BaseProfile).ProductAssembly; bool nsobject = type.IsNSObject (); if (!nsobject && !type.IsNativeObject ()) return; if (!IsProductType (type)) { // we need to annotate the parent type(s) of a nested type // otherwise the sweeper will not keep the parents (nor the childs) if (type.IsNested) { var parent = type.DeclaringType; while (parent != null) { Annotations.Mark (parent); parent = parent.DeclaringType; } } Annotations.Mark (type); Annotations.SetPreserve (type, TypePreserve.All); } else { PreserveIntPtrConstructor (type); if (nsobject) PreserveExportedMethods (type); } } void PreserveExportedMethods (TypeDefinition type) { if (!type.HasMethods) return; foreach (var method in type.Methods) { if (!IsExportedMethod (method)) continue; if (!IsOverridenInUserCode (method)) continue; PreserveMethod (type, method); } } bool IsOverridenInUserCode (MethodDefinition method) { if (!method.IsVirtual) return false; var overrides = Annotations.GetOverrides (method); if (overrides == null || overrides.Count == 0) return false; foreach (MethodDefinition @override in overrides) if (!IsProductMethod (@override)) return true; return false; } bool IsExportedMethod (MethodDefinition method) { return HasExportAttribute (method); } bool HasExportAttribute (ICustomAttributeProvider provider) { if (!provider.HasCustomAttributes) return false; foreach (CustomAttribute attribute in provider.CustomAttributes) if (attribute.Constructor.DeclaringType.Inherits (Namespaces.Foundation, "ExportAttribute")) return true; return false; } void PreserveIntPtrConstructor (TypeDefinition type) { if (!type.HasMethods) return; foreach (MethodDefinition constructor in type.GetConstructors ()) { if (!constructor.HasParameters) continue; if (constructor.Parameters.Count != 1 || !constructor.Parameters [0].ParameterType.Is ("System", "IntPtr")) continue; PreserveMethod (type, constructor); break; // only one .ctor can match this } } void PreserveMethod (TypeDefinition type, MethodDefinition method) { Annotations.AddPreservedMethod (type, method); } static bool IsProductMethod (MethodDefinition method) { return (method.DeclaringType.Module.Assembly.Name.Name == ProductAssembly); } static bool IsProductType (TypeDefinition type) { var name = type.Module.Assembly.Name.Name; return name == ProductAssembly || name == "System.Net.Http"; } } }