1 | // Copyright (c) 2010-2013 AlphaSierraPapa for the SharpDevelop Team |
---|
2 | // |
---|
3 | // Permission is hereby granted, free of charge, to any person obtaining a copy of this |
---|
4 | // software and associated documentation files (the "Software"), to deal in the Software |
---|
5 | // without restriction, including without limitation the rights to use, copy, modify, merge, |
---|
6 | // publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons |
---|
7 | // to whom the Software is furnished to do so, subject to the following conditions: |
---|
8 | // |
---|
9 | // The above copyright notice and this permission notice shall be included in all copies or |
---|
10 | // substantial portions of the Software. |
---|
11 | // |
---|
12 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, |
---|
13 | // INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR |
---|
14 | // PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE |
---|
15 | // FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR |
---|
16 | // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
---|
17 | // DEALINGS IN THE SOFTWARE. |
---|
18 | |
---|
19 | using System; |
---|
20 | using System.Collections.Generic; |
---|
21 | using System.Diagnostics; |
---|
22 | using System.Linq; |
---|
23 | using System.Runtime.CompilerServices; |
---|
24 | using System.Runtime.InteropServices; |
---|
25 | using System.Threading; |
---|
26 | using ICSharpCode.NRefactory.Documentation; |
---|
27 | using ICSharpCode.NRefactory.Semantics; |
---|
28 | using ICSharpCode.NRefactory.TypeSystem.Implementation; |
---|
29 | using ICSharpCode.NRefactory.Utils; |
---|
30 | using Mono.Cecil; |
---|
31 | |
---|
32 | |
---|
33 | namespace ICSharpCode.NRefactory.TypeSystem |
---|
34 | { |
---|
35 | /// <summary> |
---|
36 | /// Allows loading an IProjectContent from an already compiled assembly. |
---|
37 | /// </summary> |
---|
38 | /// <remarks>Instance methods are not thread-safe; you need to create multiple instances of CecilLoader |
---|
39 | /// if you want to load multiple project contents in parallel.</remarks> |
---|
40 | public sealed class CecilLoader : AssemblyLoader |
---|
41 | { |
---|
42 | /// <summary> |
---|
43 | /// Version number of the cecil loader. |
---|
44 | /// Should be incremented when fixing bugs in the cecil loader so that project contents cached on disk |
---|
45 | /// (which might be incorrect due to the bug) are re-created. |
---|
46 | /// </summary> |
---|
47 | const int cecilLoaderVersion = 1; |
---|
48 | |
---|
49 | #region Options |
---|
50 | // Most options are defined in the AssemblyLoader base class |
---|
51 | |
---|
52 | /// <summary> |
---|
53 | /// Specifies whether to use lazy loading. The default is false. |
---|
54 | /// If this property is set to true, the CecilLoader will not copy all the relevant information |
---|
55 | /// out of the Cecil object model, but will maintain references to the Cecil objects. |
---|
56 | /// This speeds up the loading process and avoids loading unnecessary information, but it causes |
---|
57 | /// the Cecil objects to stay in memory (which can significantly increase memory usage). |
---|
58 | /// It also prevents serialization of the Cecil-loaded type system. |
---|
59 | /// </summary> |
---|
60 | /// <remarks> |
---|
61 | /// Because the type system can be used on multiple threads, but Cecil is not |
---|
62 | /// thread-safe for concurrent read access, the CecilLoader will lock on the <see cref="ModuleDefinition"/> instance |
---|
63 | /// for every delay-loading operation. |
---|
64 | /// If you access the Cecil objects directly in your application, you may need to take the same lock. |
---|
65 | /// </remarks> |
---|
66 | public bool LazyLoad { get; set; } |
---|
67 | |
---|
68 | /// <summary> |
---|
69 | /// This delegate gets executed whenever an entity was loaded. |
---|
70 | /// </summary> |
---|
71 | /// <remarks> |
---|
72 | /// This callback may be to build a dictionary that maps between |
---|
73 | /// entities and cecil objects. |
---|
74 | /// Warning: if delay-loading is used and the type system is accessed by multiple threads, |
---|
75 | /// the callback may be invoked concurrently on multiple threads. |
---|
76 | /// </remarks> |
---|
77 | [CLSCompliant(false)] |
---|
78 | public Action<IUnresolvedEntity, MemberReference> OnEntityLoaded { get; set; } |
---|
79 | |
---|
80 | /// <summary> |
---|
81 | /// Gets a value indicating whether this instance stores references to the cecil objects. |
---|
82 | /// </summary> |
---|
83 | /// <value> |
---|
84 | /// <c>true</c> if this instance has references to the cecil objects; otherwise, <c>false</c>. |
---|
85 | /// </value> |
---|
86 | public bool HasCecilReferences { get { return typeSystemTranslationTable != null; } } |
---|
87 | #endregion |
---|
88 | |
---|
89 | ModuleDefinition currentModule; |
---|
90 | CecilUnresolvedAssembly currentAssembly; |
---|
91 | |
---|
92 | /// <summary> |
---|
93 | /// Initializes a new instance of the <see cref="ICSharpCode.NRefactory.TypeSystem.CecilLoader"/> class. |
---|
94 | /// </summary> |
---|
95 | public CecilLoader() |
---|
96 | { |
---|
97 | } |
---|
98 | |
---|
99 | /// <summary> |
---|
100 | /// Initializes a new instance of the <see cref="ICSharpCode.NRefactory.TypeSystem.CecilLoader"/> class. |
---|
101 | /// </summary> |
---|
102 | /// <param name='createCecilReferences'> |
---|
103 | /// If true references to the cecil objects are hold. In this case the cecil loader can do a type system -> cecil mapping. |
---|
104 | /// </param> |
---|
105 | [Obsolete("The built-in entity<->cecil mapping is obsolete. Use the OnEntityLoaded callback instead!")] |
---|
106 | public CecilLoader(bool createCecilReferences) : this() |
---|
107 | { |
---|
108 | if (createCecilReferences) |
---|
109 | typeSystemTranslationTable = new Dictionary<object, object> (); |
---|
110 | } |
---|
111 | |
---|
112 | /// <summary> |
---|
113 | /// Creates a nested CecilLoader for lazy-loading. |
---|
114 | /// </summary> |
---|
115 | private CecilLoader(CecilLoader loader) |
---|
116 | { |
---|
117 | // use a shared typeSystemTranslationTable |
---|
118 | this.typeSystemTranslationTable = loader.typeSystemTranslationTable; |
---|
119 | this.IncludeInternalMembers = loader.IncludeInternalMembers; |
---|
120 | this.LazyLoad = loader.LazyLoad; |
---|
121 | this.OnEntityLoaded = loader.OnEntityLoaded; |
---|
122 | this.currentModule = loader.currentModule; |
---|
123 | this.currentAssembly = loader.currentAssembly; |
---|
124 | // don't use interning - the interning provider is most likely not thread-safe |
---|
125 | this.interningProvider = InterningProvider.Dummy; |
---|
126 | // don't use cancellation for delay-loaded members |
---|
127 | } |
---|
128 | |
---|
129 | #region Load From AssemblyDefinition |
---|
130 | /// <summary> |
---|
131 | /// Loads the assembly definition into a project content. |
---|
132 | /// </summary> |
---|
133 | /// <returns>Unresolved type system representing the assembly</returns> |
---|
134 | [CLSCompliant(false)] |
---|
135 | public IUnresolvedAssembly LoadAssembly(AssemblyDefinition assemblyDefinition) |
---|
136 | { |
---|
137 | if (assemblyDefinition == null) |
---|
138 | throw new ArgumentNullException("assemblyDefinition"); |
---|
139 | return LoadModule(assemblyDefinition.MainModule); |
---|
140 | } |
---|
141 | |
---|
142 | /// <summary> |
---|
143 | /// Loads the module definition into a project content. |
---|
144 | /// </summary> |
---|
145 | /// <returns>Unresolved type system representing the assembly</returns> |
---|
146 | [CLSCompliant(false)] |
---|
147 | public IUnresolvedAssembly LoadModule(ModuleDefinition moduleDefinition) |
---|
148 | { |
---|
149 | if (moduleDefinition == null) |
---|
150 | throw new ArgumentNullException("moduleDefinition"); |
---|
151 | |
---|
152 | this.currentModule = moduleDefinition; |
---|
153 | |
---|
154 | // Read assembly and module attributes |
---|
155 | IList<IUnresolvedAttribute> assemblyAttributes = new List<IUnresolvedAttribute>(); |
---|
156 | IList<IUnresolvedAttribute> moduleAttributes = new List<IUnresolvedAttribute>(); |
---|
157 | AssemblyDefinition assemblyDefinition = moduleDefinition.Assembly; |
---|
158 | if (assemblyDefinition != null) { |
---|
159 | AddAttributes(assemblyDefinition, assemblyAttributes); |
---|
160 | } |
---|
161 | AddAttributes(moduleDefinition, moduleAttributes); |
---|
162 | |
---|
163 | assemblyAttributes = interningProvider.InternList(assemblyAttributes); |
---|
164 | moduleAttributes = interningProvider.InternList(moduleAttributes); |
---|
165 | |
---|
166 | this.currentAssembly = new CecilUnresolvedAssembly(assemblyDefinition != null ? assemblyDefinition.Name.FullName : moduleDefinition.Name, this.DocumentationProvider); |
---|
167 | currentAssembly.Location = moduleDefinition.FullyQualifiedName; |
---|
168 | currentAssembly.AssemblyAttributes.AddRange(assemblyAttributes); |
---|
169 | currentAssembly.ModuleAttributes.AddRange(assemblyAttributes); |
---|
170 | |
---|
171 | // Register type forwarders: |
---|
172 | foreach (ExportedType type in moduleDefinition.ExportedTypes) { |
---|
173 | if (type.IsForwarder) { |
---|
174 | int typeParameterCount; |
---|
175 | string ns = type.Namespace; |
---|
176 | string name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(type.Name, out typeParameterCount); |
---|
177 | ns = interningProvider.Intern(ns); |
---|
178 | name = interningProvider.Intern(name); |
---|
179 | var typeRef = new GetClassTypeReference(GetAssemblyReference(type.Scope), ns, name, typeParameterCount); |
---|
180 | typeRef = interningProvider.Intern(typeRef); |
---|
181 | var key = new TopLevelTypeName(ns, name, typeParameterCount); |
---|
182 | currentAssembly.AddTypeForwarder(key, typeRef); |
---|
183 | } |
---|
184 | } |
---|
185 | |
---|
186 | // Create and register all types: |
---|
187 | CecilLoader cecilLoaderCloneForLazyLoading = LazyLoad ? new CecilLoader(this) : null; |
---|
188 | List<TypeDefinition> cecilTypeDefs = new List<TypeDefinition>(); |
---|
189 | List<DefaultUnresolvedTypeDefinition> typeDefs = new List<DefaultUnresolvedTypeDefinition>(); |
---|
190 | foreach (TypeDefinition td in moduleDefinition.Types) { |
---|
191 | this.CancellationToken.ThrowIfCancellationRequested(); |
---|
192 | if (this.IncludeInternalMembers || (td.Attributes & TypeAttributes.VisibilityMask) == TypeAttributes.Public) { |
---|
193 | string name = td.Name; |
---|
194 | if (name.Length == 0) |
---|
195 | continue; |
---|
196 | |
---|
197 | if (this.LazyLoad) { |
---|
198 | var t = new LazyCecilTypeDefinition(cecilLoaderCloneForLazyLoading, td); |
---|
199 | currentAssembly.AddTypeDefinition(t); |
---|
200 | RegisterCecilObject(t, td); |
---|
201 | } else { |
---|
202 | var t = CreateTopLevelTypeDefinition(td); |
---|
203 | cecilTypeDefs.Add(td); |
---|
204 | typeDefs.Add(t); |
---|
205 | currentAssembly.AddTypeDefinition(t); |
---|
206 | // The registration will happen after the members are initialized |
---|
207 | } |
---|
208 | } |
---|
209 | } |
---|
210 | // Initialize the type's members: |
---|
211 | for (int i = 0; i < typeDefs.Count; i++) { |
---|
212 | InitTypeDefinition(cecilTypeDefs[i], typeDefs[i]); |
---|
213 | } |
---|
214 | |
---|
215 | AddToTypeSystemTranslationTable(this.currentAssembly, assemblyDefinition); |
---|
216 | // Freezing the assembly here is important: |
---|
217 | // otherwise it will be frozen when a compilation is first created |
---|
218 | // from it. But freezing has the effect of changing some collection instances |
---|
219 | // (to ReadOnlyCollection). This hidden mutation was causing a crash |
---|
220 | // when the FastSerializer was saving the assembly at the same time as |
---|
221 | // the first compilation was created from it. |
---|
222 | // By freezing the assembly now, we ensure it is usable on multiple |
---|
223 | // threads without issues. |
---|
224 | currentAssembly.Freeze(); |
---|
225 | |
---|
226 | var result = this.currentAssembly; |
---|
227 | this.currentAssembly = null; |
---|
228 | this.currentModule = null; |
---|
229 | return result; |
---|
230 | } |
---|
231 | |
---|
232 | /// <summary> |
---|
233 | /// Sets the current module. |
---|
234 | /// This causes ReadTypeReference() to use <see cref="DefaultAssemblyReference.CurrentAssembly"/> for references |
---|
235 | /// in that module. |
---|
236 | /// </summary> |
---|
237 | [CLSCompliant(false)] |
---|
238 | public void SetCurrentModule(ModuleDefinition module) |
---|
239 | { |
---|
240 | this.currentModule = module; |
---|
241 | } |
---|
242 | |
---|
243 | /// <summary> |
---|
244 | /// Loads a type from Cecil. |
---|
245 | /// </summary> |
---|
246 | /// <param name="typeDefinition">The Cecil TypeDefinition.</param> |
---|
247 | /// <returns>ITypeDefinition representing the Cecil type.</returns> |
---|
248 | [CLSCompliant(false)] |
---|
249 | public IUnresolvedTypeDefinition LoadType(TypeDefinition typeDefinition) |
---|
250 | { |
---|
251 | if (typeDefinition == null) |
---|
252 | throw new ArgumentNullException("typeDefinition"); |
---|
253 | var td = CreateTopLevelTypeDefinition(typeDefinition); |
---|
254 | InitTypeDefinition(typeDefinition, td); |
---|
255 | return td; |
---|
256 | } |
---|
257 | #endregion |
---|
258 | |
---|
259 | #region IUnresolvedAssembly implementation |
---|
260 | [Serializable, FastSerializerVersion(cecilLoaderVersion)] |
---|
261 | sealed class CecilUnresolvedAssembly : DefaultUnresolvedAssembly, IDocumentationProvider |
---|
262 | { |
---|
263 | readonly IDocumentationProvider documentationProvider; |
---|
264 | |
---|
265 | public CecilUnresolvedAssembly(string fullAssemblyName, IDocumentationProvider documentationProvider) |
---|
266 | : base(fullAssemblyName) |
---|
267 | { |
---|
268 | this.documentationProvider = documentationProvider; |
---|
269 | } |
---|
270 | |
---|
271 | DocumentationComment IDocumentationProvider.GetDocumentation(IEntity entity) |
---|
272 | { |
---|
273 | if (documentationProvider != null) |
---|
274 | return documentationProvider.GetDocumentation(entity); |
---|
275 | else |
---|
276 | return null; |
---|
277 | } |
---|
278 | } |
---|
279 | #endregion |
---|
280 | |
---|
281 | #region Load Assembly From Disk |
---|
282 | public override IUnresolvedAssembly LoadAssemblyFile(string fileName) |
---|
283 | { |
---|
284 | if (fileName == null) |
---|
285 | throw new ArgumentNullException("fileName"); |
---|
286 | var param = new ReaderParameters { AssemblyResolver = new DummyAssemblyResolver() }; |
---|
287 | ModuleDefinition module = ModuleDefinition.ReadModule(fileName, param); |
---|
288 | return LoadModule(module); |
---|
289 | } |
---|
290 | |
---|
291 | // used to prevent Cecil from loading referenced assemblies |
---|
292 | sealed class DummyAssemblyResolver : IAssemblyResolver |
---|
293 | { |
---|
294 | public AssemblyDefinition Resolve(AssemblyNameReference name) |
---|
295 | { |
---|
296 | return null; |
---|
297 | } |
---|
298 | |
---|
299 | public AssemblyDefinition Resolve(string fullName) |
---|
300 | { |
---|
301 | return null; |
---|
302 | } |
---|
303 | |
---|
304 | public AssemblyDefinition Resolve(AssemblyNameReference name, ReaderParameters parameters) |
---|
305 | { |
---|
306 | return null; |
---|
307 | } |
---|
308 | |
---|
309 | public AssemblyDefinition Resolve(string fullName, ReaderParameters parameters) |
---|
310 | { |
---|
311 | return null; |
---|
312 | } |
---|
313 | } |
---|
314 | #endregion |
---|
315 | |
---|
316 | #region Read Type Reference |
---|
317 | /// <summary> |
---|
318 | /// Reads a type reference. |
---|
319 | /// </summary> |
---|
320 | /// <param name="type">The Cecil type reference that should be converted into |
---|
321 | /// a type system type reference.</param> |
---|
322 | /// <param name="typeAttributes">Attributes associated with the Cecil type reference. |
---|
323 | /// This is used to support the 'dynamic' type.</param> |
---|
324 | [CLSCompliant(false)] |
---|
325 | public ITypeReference ReadTypeReference(TypeReference type, ICustomAttributeProvider typeAttributes = null) |
---|
326 | { |
---|
327 | int typeIndex = 0; |
---|
328 | return CreateType(type, typeAttributes, ref typeIndex); |
---|
329 | } |
---|
330 | |
---|
331 | ITypeReference CreateType(TypeReference type, ICustomAttributeProvider typeAttributes, ref int typeIndex) |
---|
332 | { |
---|
333 | while (type is OptionalModifierType || type is RequiredModifierType) { |
---|
334 | type = ((TypeSpecification)type).ElementType; |
---|
335 | } |
---|
336 | if (type == null) { |
---|
337 | return SpecialType.UnknownType; |
---|
338 | } |
---|
339 | |
---|
340 | if (type is Mono.Cecil.ByReferenceType) { |
---|
341 | typeIndex++; |
---|
342 | return interningProvider.Intern( |
---|
343 | new ByReferenceTypeReference( |
---|
344 | CreateType( |
---|
345 | (type as Mono.Cecil.ByReferenceType).ElementType, |
---|
346 | typeAttributes, ref typeIndex))); |
---|
347 | } else if (type is Mono.Cecil.PointerType) { |
---|
348 | typeIndex++; |
---|
349 | return interningProvider.Intern( |
---|
350 | new PointerTypeReference( |
---|
351 | CreateType( |
---|
352 | (type as Mono.Cecil.PointerType).ElementType, |
---|
353 | typeAttributes, ref typeIndex))); |
---|
354 | } else if (type is Mono.Cecil.ArrayType) { |
---|
355 | typeIndex++; |
---|
356 | return interningProvider.Intern( |
---|
357 | new ArrayTypeReference( |
---|
358 | CreateType( |
---|
359 | (type as Mono.Cecil.ArrayType).ElementType, |
---|
360 | typeAttributes, ref typeIndex), |
---|
361 | (type as Mono.Cecil.ArrayType).Rank)); |
---|
362 | } else if (type is GenericInstanceType) { |
---|
363 | GenericInstanceType gType = (GenericInstanceType)type; |
---|
364 | ITypeReference baseType = CreateType(gType.ElementType, typeAttributes, ref typeIndex); |
---|
365 | ITypeReference[] para = new ITypeReference[gType.GenericArguments.Count]; |
---|
366 | for (int i = 0; i < para.Length; ++i) { |
---|
367 | typeIndex++; |
---|
368 | para[i] = CreateType(gType.GenericArguments[i], typeAttributes, ref typeIndex); |
---|
369 | } |
---|
370 | return interningProvider.Intern(new ParameterizedTypeReference(baseType, para)); |
---|
371 | } else if (type is GenericParameter) { |
---|
372 | GenericParameter typeGP = (GenericParameter)type; |
---|
373 | return TypeParameterReference.Create(typeGP.Owner is MethodDefinition ? SymbolKind.Method : SymbolKind.TypeDefinition, typeGP.Position); |
---|
374 | } else if (type.IsNested) { |
---|
375 | ITypeReference typeRef = CreateType(type.DeclaringType, typeAttributes, ref typeIndex); |
---|
376 | int partTypeParameterCount; |
---|
377 | string namepart = ReflectionHelper.SplitTypeParameterCountFromReflectionName(type.Name, out partTypeParameterCount); |
---|
378 | namepart = interningProvider.Intern(namepart); |
---|
379 | return interningProvider.Intern(new NestedTypeReference(typeRef, namepart, partTypeParameterCount)); |
---|
380 | } else { |
---|
381 | string ns = interningProvider.Intern(type.Namespace ?? string.Empty); |
---|
382 | string name = type.Name; |
---|
383 | if (name == null) |
---|
384 | throw new InvalidOperationException("type.Name returned null. Type: " + type.ToString()); |
---|
385 | |
---|
386 | if (name == "Object" && ns == "System" && HasDynamicAttribute(typeAttributes, typeIndex)) { |
---|
387 | return SpecialType.Dynamic; |
---|
388 | } else { |
---|
389 | int typeParameterCount; |
---|
390 | name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(name, out typeParameterCount); |
---|
391 | name = interningProvider.Intern(name); |
---|
392 | if (currentAssembly != null) { |
---|
393 | IUnresolvedTypeDefinition c = currentAssembly.GetTypeDefinition(ns, name, typeParameterCount); |
---|
394 | if (c != null) |
---|
395 | return c; |
---|
396 | } |
---|
397 | return interningProvider.Intern(new GetClassTypeReference(GetAssemblyReference(type.Scope), ns, name, typeParameterCount)); |
---|
398 | } |
---|
399 | } |
---|
400 | } |
---|
401 | |
---|
402 | IAssemblyReference GetAssemblyReference(IMetadataScope scope) |
---|
403 | { |
---|
404 | if (scope == null || scope == currentModule) |
---|
405 | return DefaultAssemblyReference.CurrentAssembly; |
---|
406 | else |
---|
407 | return interningProvider.Intern(new DefaultAssemblyReference(scope.Name)); |
---|
408 | } |
---|
409 | |
---|
410 | static bool HasDynamicAttribute(ICustomAttributeProvider attributeProvider, int typeIndex) |
---|
411 | { |
---|
412 | if (attributeProvider == null || !attributeProvider.HasCustomAttributes) |
---|
413 | return false; |
---|
414 | foreach (CustomAttribute a in attributeProvider.CustomAttributes) { |
---|
415 | TypeReference type = a.AttributeType; |
---|
416 | if (type.Name == "DynamicAttribute" && type.Namespace == "System.Runtime.CompilerServices") { |
---|
417 | if (a.ConstructorArguments.Count == 1) { |
---|
418 | CustomAttributeArgument[] values = a.ConstructorArguments[0].Value as CustomAttributeArgument[]; |
---|
419 | if (values != null && typeIndex < values.Length && values[typeIndex].Value is bool) |
---|
420 | return (bool)values[typeIndex].Value; |
---|
421 | } |
---|
422 | return true; |
---|
423 | } |
---|
424 | } |
---|
425 | return false; |
---|
426 | } |
---|
427 | #endregion |
---|
428 | |
---|
429 | #region Read Attributes |
---|
430 | #region Assembly Attributes |
---|
431 | static readonly ITypeReference assemblyVersionAttributeTypeRef = typeof(System.Reflection.AssemblyVersionAttribute).ToTypeReference(); |
---|
432 | |
---|
433 | void AddAttributes(AssemblyDefinition assembly, IList<IUnresolvedAttribute> outputList) |
---|
434 | { |
---|
435 | if (assembly.HasCustomAttributes) { |
---|
436 | AddCustomAttributes(assembly.CustomAttributes, outputList); |
---|
437 | } |
---|
438 | if (assembly.HasSecurityDeclarations) { |
---|
439 | AddSecurityAttributes(assembly.SecurityDeclarations, outputList); |
---|
440 | } |
---|
441 | |
---|
442 | // AssemblyVersionAttribute |
---|
443 | if (assembly.Name.Version != null) { |
---|
444 | var assemblyVersion = new DefaultUnresolvedAttribute(assemblyVersionAttributeTypeRef, new[] { KnownTypeReference.String }); |
---|
445 | assemblyVersion.PositionalArguments.Add(CreateSimpleConstantValue(KnownTypeReference.String, assembly.Name.Version.ToString())); |
---|
446 | outputList.Add(interningProvider.Intern(assemblyVersion)); |
---|
447 | } |
---|
448 | } |
---|
449 | |
---|
450 | IConstantValue CreateSimpleConstantValue(ITypeReference type, object value) |
---|
451 | { |
---|
452 | return interningProvider.Intern(new SimpleConstantValue(type, interningProvider.InternValue(value))); |
---|
453 | } |
---|
454 | #endregion |
---|
455 | |
---|
456 | #region Module Attributes |
---|
457 | void AddAttributes(ModuleDefinition module, IList<IUnresolvedAttribute> outputList) |
---|
458 | { |
---|
459 | if (module.HasCustomAttributes) { |
---|
460 | AddCustomAttributes(module.CustomAttributes, outputList); |
---|
461 | } |
---|
462 | } |
---|
463 | #endregion |
---|
464 | |
---|
465 | #region Parameter Attributes |
---|
466 | static readonly IUnresolvedAttribute inAttribute = new DefaultUnresolvedAttribute(typeof(InAttribute).ToTypeReference()); |
---|
467 | static readonly IUnresolvedAttribute outAttribute = new DefaultUnresolvedAttribute(typeof(OutAttribute).ToTypeReference()); |
---|
468 | |
---|
469 | void AddAttributes(ParameterDefinition parameter, DefaultUnresolvedParameter targetParameter) |
---|
470 | { |
---|
471 | if (!targetParameter.IsOut) { |
---|
472 | if (parameter.IsIn) |
---|
473 | targetParameter.Attributes.Add(inAttribute); |
---|
474 | if (parameter.IsOut) |
---|
475 | targetParameter.Attributes.Add(outAttribute); |
---|
476 | } |
---|
477 | if (parameter.HasCustomAttributes) { |
---|
478 | AddCustomAttributes(parameter.CustomAttributes, targetParameter.Attributes); |
---|
479 | } |
---|
480 | if (parameter.HasMarshalInfo) { |
---|
481 | targetParameter.Attributes.Add(ConvertMarshalInfo(parameter.MarshalInfo)); |
---|
482 | } |
---|
483 | } |
---|
484 | #endregion |
---|
485 | |
---|
486 | #region Method Attributes |
---|
487 | static readonly ITypeReference dllImportAttributeTypeRef = typeof(DllImportAttribute).ToTypeReference(); |
---|
488 | static readonly SimpleConstantValue trueValue = new SimpleConstantValue(KnownTypeReference.Boolean, true); |
---|
489 | static readonly SimpleConstantValue falseValue = new SimpleConstantValue(KnownTypeReference.Boolean, false); |
---|
490 | static readonly ITypeReference callingConventionTypeRef = typeof(CallingConvention).ToTypeReference(); |
---|
491 | static readonly IUnresolvedAttribute preserveSigAttribute = new DefaultUnresolvedAttribute(typeof(PreserveSigAttribute).ToTypeReference()); |
---|
492 | static readonly ITypeReference methodImplAttributeTypeRef = typeof(MethodImplAttribute).ToTypeReference(); |
---|
493 | static readonly ITypeReference methodImplOptionsTypeRef = typeof(MethodImplOptions).ToTypeReference(); |
---|
494 | |
---|
495 | static bool HasAnyAttributes(MethodDefinition methodDefinition) |
---|
496 | { |
---|
497 | if (methodDefinition.HasPInvokeInfo) |
---|
498 | return true; |
---|
499 | if ((methodDefinition.ImplAttributes & ~MethodImplAttributes.CodeTypeMask) != 0) |
---|
500 | return true; |
---|
501 | if (methodDefinition.MethodReturnType.HasFieldMarshal) |
---|
502 | return true; |
---|
503 | return methodDefinition.HasCustomAttributes || methodDefinition.MethodReturnType.HasCustomAttributes; |
---|
504 | } |
---|
505 | |
---|
506 | void AddAttributes(MethodDefinition methodDefinition, IList<IUnresolvedAttribute> attributes, IList<IUnresolvedAttribute> returnTypeAttributes) |
---|
507 | { |
---|
508 | MethodImplAttributes implAttributes = methodDefinition.ImplAttributes & ~MethodImplAttributes.CodeTypeMask; |
---|
509 | |
---|
510 | #region DllImportAttribute |
---|
511 | if (methodDefinition.HasPInvokeInfo && methodDefinition.PInvokeInfo != null) { |
---|
512 | PInvokeInfo info = methodDefinition.PInvokeInfo; |
---|
513 | var dllImport = new DefaultUnresolvedAttribute(dllImportAttributeTypeRef, new[] { KnownTypeReference.String }); |
---|
514 | dllImport.PositionalArguments.Add(CreateSimpleConstantValue(KnownTypeReference.String, info.Module.Name)); |
---|
515 | |
---|
516 | if (info.IsBestFitDisabled) |
---|
517 | dllImport.AddNamedFieldArgument("BestFitMapping", falseValue); |
---|
518 | if (info.IsBestFitEnabled) |
---|
519 | dllImport.AddNamedFieldArgument("BestFitMapping", trueValue); |
---|
520 | |
---|
521 | CallingConvention callingConvention; |
---|
522 | switch (info.Attributes & PInvokeAttributes.CallConvMask) { |
---|
523 | case (PInvokeAttributes)0: |
---|
524 | Debug.WriteLine ("P/Invoke calling convention not set on:" + methodDefinition.FullName); |
---|
525 | callingConvention = 0; |
---|
526 | break; |
---|
527 | case PInvokeAttributes.CallConvCdecl: |
---|
528 | callingConvention = CallingConvention.Cdecl; |
---|
529 | break; |
---|
530 | case PInvokeAttributes.CallConvFastcall: |
---|
531 | callingConvention = CallingConvention.FastCall; |
---|
532 | break; |
---|
533 | case PInvokeAttributes.CallConvStdCall: |
---|
534 | callingConvention = CallingConvention.StdCall; |
---|
535 | break; |
---|
536 | case PInvokeAttributes.CallConvThiscall: |
---|
537 | callingConvention = CallingConvention.ThisCall; |
---|
538 | break; |
---|
539 | case PInvokeAttributes.CallConvWinapi: |
---|
540 | callingConvention = CallingConvention.Winapi; |
---|
541 | break; |
---|
542 | default: |
---|
543 | throw new NotSupportedException("unknown calling convention"); |
---|
544 | } |
---|
545 | if (callingConvention != CallingConvention.Winapi) |
---|
546 | dllImport.AddNamedFieldArgument("CallingConvention", CreateSimpleConstantValue(callingConventionTypeRef, (int)callingConvention)); |
---|
547 | |
---|
548 | CharSet charSet = CharSet.None; |
---|
549 | switch (info.Attributes & PInvokeAttributes.CharSetMask) { |
---|
550 | case PInvokeAttributes.CharSetAnsi: |
---|
551 | charSet = CharSet.Ansi; |
---|
552 | break; |
---|
553 | case PInvokeAttributes.CharSetAuto: |
---|
554 | charSet = CharSet.Auto; |
---|
555 | break; |
---|
556 | case PInvokeAttributes.CharSetUnicode: |
---|
557 | charSet = CharSet.Unicode; |
---|
558 | break; |
---|
559 | } |
---|
560 | if (charSet != CharSet.None) |
---|
561 | dllImport.AddNamedFieldArgument("CharSet", CreateSimpleConstantValue(charSetTypeRef, (int)charSet)); |
---|
562 | |
---|
563 | if (!string.IsNullOrEmpty(info.EntryPoint) && info.EntryPoint != methodDefinition.Name) |
---|
564 | dllImport.AddNamedFieldArgument("EntryPoint", CreateSimpleConstantValue(KnownTypeReference.String, info.EntryPoint)); |
---|
565 | |
---|
566 | if (info.IsNoMangle) |
---|
567 | dllImport.AddNamedFieldArgument("ExactSpelling", trueValue); |
---|
568 | |
---|
569 | if ((implAttributes & MethodImplAttributes.PreserveSig) == MethodImplAttributes.PreserveSig) |
---|
570 | implAttributes &= ~MethodImplAttributes.PreserveSig; |
---|
571 | else |
---|
572 | dllImport.AddNamedFieldArgument("PreserveSig", falseValue); |
---|
573 | |
---|
574 | if (info.SupportsLastError) |
---|
575 | dllImport.AddNamedFieldArgument("SetLastError", trueValue); |
---|
576 | |
---|
577 | if (info.IsThrowOnUnmappableCharDisabled) |
---|
578 | dllImport.AddNamedFieldArgument("ThrowOnUnmappableChar", falseValue); |
---|
579 | if (info.IsThrowOnUnmappableCharEnabled) |
---|
580 | dllImport.AddNamedFieldArgument("ThrowOnUnmappableChar", trueValue); |
---|
581 | |
---|
582 | attributes.Add(interningProvider.Intern(dllImport)); |
---|
583 | } |
---|
584 | #endregion |
---|
585 | |
---|
586 | #region PreserveSigAttribute |
---|
587 | if (implAttributes == MethodImplAttributes.PreserveSig) { |
---|
588 | attributes.Add(preserveSigAttribute); |
---|
589 | implAttributes = 0; |
---|
590 | } |
---|
591 | #endregion |
---|
592 | |
---|
593 | #region MethodImplAttribute |
---|
594 | if (implAttributes != 0) { |
---|
595 | var methodImpl = new DefaultUnresolvedAttribute(methodImplAttributeTypeRef, new[] { methodImplOptionsTypeRef }); |
---|
596 | methodImpl.PositionalArguments.Add(CreateSimpleConstantValue(methodImplOptionsTypeRef, (int)implAttributes)); |
---|
597 | attributes.Add(interningProvider.Intern(methodImpl)); |
---|
598 | } |
---|
599 | #endregion |
---|
600 | |
---|
601 | if (methodDefinition.HasCustomAttributes) { |
---|
602 | AddCustomAttributes(methodDefinition.CustomAttributes, attributes); |
---|
603 | } |
---|
604 | if (methodDefinition.HasSecurityDeclarations) { |
---|
605 | AddSecurityAttributes(methodDefinition.SecurityDeclarations, attributes); |
---|
606 | } |
---|
607 | if (methodDefinition.MethodReturnType.HasMarshalInfo) { |
---|
608 | returnTypeAttributes.Add(ConvertMarshalInfo(methodDefinition.MethodReturnType.MarshalInfo)); |
---|
609 | } |
---|
610 | if (methodDefinition.MethodReturnType.HasCustomAttributes) { |
---|
611 | AddCustomAttributes(methodDefinition.MethodReturnType.CustomAttributes, returnTypeAttributes); |
---|
612 | } |
---|
613 | } |
---|
614 | #endregion |
---|
615 | |
---|
616 | #region Type Attributes |
---|
617 | static readonly DefaultUnresolvedAttribute serializableAttribute = new DefaultUnresolvedAttribute(typeof(SerializableAttribute).ToTypeReference()); |
---|
618 | static readonly DefaultUnresolvedAttribute comImportAttribute = new DefaultUnresolvedAttribute(typeof(ComImportAttribute).ToTypeReference()); |
---|
619 | static readonly ITypeReference structLayoutAttributeTypeRef = typeof(StructLayoutAttribute).ToTypeReference(); |
---|
620 | static readonly ITypeReference layoutKindTypeRef = typeof(LayoutKind).ToTypeReference(); |
---|
621 | static readonly ITypeReference charSetTypeRef = typeof(CharSet).ToTypeReference(); |
---|
622 | |
---|
623 | void AddAttributes(TypeDefinition typeDefinition, IUnresolvedTypeDefinition targetEntity) |
---|
624 | { |
---|
625 | // SerializableAttribute |
---|
626 | if (typeDefinition.IsSerializable) |
---|
627 | targetEntity.Attributes.Add(serializableAttribute); |
---|
628 | |
---|
629 | // ComImportAttribute |
---|
630 | if (typeDefinition.IsImport) |
---|
631 | targetEntity.Attributes.Add(comImportAttribute); |
---|
632 | |
---|
633 | #region StructLayoutAttribute |
---|
634 | LayoutKind layoutKind = LayoutKind.Auto; |
---|
635 | switch (typeDefinition.Attributes & TypeAttributes.LayoutMask) { |
---|
636 | case TypeAttributes.SequentialLayout: |
---|
637 | layoutKind = LayoutKind.Sequential; |
---|
638 | break; |
---|
639 | case TypeAttributes.ExplicitLayout: |
---|
640 | layoutKind = LayoutKind.Explicit; |
---|
641 | break; |
---|
642 | } |
---|
643 | CharSet charSet = CharSet.None; |
---|
644 | switch (typeDefinition.Attributes & TypeAttributes.StringFormatMask) { |
---|
645 | case TypeAttributes.AnsiClass: |
---|
646 | charSet = CharSet.Ansi; |
---|
647 | break; |
---|
648 | case TypeAttributes.AutoClass: |
---|
649 | charSet = CharSet.Auto; |
---|
650 | break; |
---|
651 | case TypeAttributes.UnicodeClass: |
---|
652 | charSet = CharSet.Unicode; |
---|
653 | break; |
---|
654 | } |
---|
655 | LayoutKind defaultLayoutKind = (typeDefinition.IsValueType && !typeDefinition.IsEnum) ? LayoutKind.Sequential: LayoutKind.Auto; |
---|
656 | if (layoutKind != defaultLayoutKind || charSet != CharSet.Ansi || typeDefinition.PackingSize > 0 || typeDefinition.ClassSize > 0) { |
---|
657 | DefaultUnresolvedAttribute structLayout = new DefaultUnresolvedAttribute(structLayoutAttributeTypeRef, new[] { layoutKindTypeRef }); |
---|
658 | structLayout.PositionalArguments.Add(CreateSimpleConstantValue(layoutKindTypeRef, (int)layoutKind)); |
---|
659 | if (charSet != CharSet.Ansi) { |
---|
660 | structLayout.AddNamedFieldArgument("CharSet", CreateSimpleConstantValue(charSetTypeRef, (int)charSet)); |
---|
661 | } |
---|
662 | if (typeDefinition.PackingSize > 0) { |
---|
663 | structLayout.AddNamedFieldArgument("Pack", CreateSimpleConstantValue(KnownTypeReference.Int32, (int)typeDefinition.PackingSize)); |
---|
664 | } |
---|
665 | if (typeDefinition.ClassSize > 0) { |
---|
666 | structLayout.AddNamedFieldArgument("Size", CreateSimpleConstantValue(KnownTypeReference.Int32, (int)typeDefinition.ClassSize)); |
---|
667 | } |
---|
668 | targetEntity.Attributes.Add(interningProvider.Intern(structLayout)); |
---|
669 | } |
---|
670 | #endregion |
---|
671 | |
---|
672 | if (typeDefinition.HasCustomAttributes) { |
---|
673 | AddCustomAttributes(typeDefinition.CustomAttributes, targetEntity.Attributes); |
---|
674 | } |
---|
675 | if (typeDefinition.HasSecurityDeclarations) { |
---|
676 | AddSecurityAttributes(typeDefinition.SecurityDeclarations, targetEntity.Attributes); |
---|
677 | } |
---|
678 | } |
---|
679 | #endregion |
---|
680 | |
---|
681 | #region Field Attributes |
---|
682 | static readonly ITypeReference fieldOffsetAttributeTypeRef = typeof(FieldOffsetAttribute).ToTypeReference(); |
---|
683 | static readonly IUnresolvedAttribute nonSerializedAttribute = new DefaultUnresolvedAttribute(typeof(NonSerializedAttribute).ToTypeReference()); |
---|
684 | |
---|
685 | void AddAttributes(FieldDefinition fieldDefinition, IUnresolvedEntity targetEntity) |
---|
686 | { |
---|
687 | // FieldOffsetAttribute |
---|
688 | if (fieldDefinition.HasLayoutInfo) { |
---|
689 | DefaultUnresolvedAttribute fieldOffset = new DefaultUnresolvedAttribute(fieldOffsetAttributeTypeRef, new[] { KnownTypeReference.Int32 }); |
---|
690 | fieldOffset.PositionalArguments.Add(CreateSimpleConstantValue(KnownTypeReference.Int32, fieldDefinition.Offset)); |
---|
691 | targetEntity.Attributes.Add(interningProvider.Intern(fieldOffset)); |
---|
692 | } |
---|
693 | |
---|
694 | // NonSerializedAttribute |
---|
695 | if (fieldDefinition.IsNotSerialized) { |
---|
696 | targetEntity.Attributes.Add(nonSerializedAttribute); |
---|
697 | } |
---|
698 | |
---|
699 | if (fieldDefinition.HasMarshalInfo) { |
---|
700 | targetEntity.Attributes.Add(ConvertMarshalInfo(fieldDefinition.MarshalInfo)); |
---|
701 | } |
---|
702 | |
---|
703 | if (fieldDefinition.HasCustomAttributes) { |
---|
704 | AddCustomAttributes(fieldDefinition.CustomAttributes, targetEntity.Attributes); |
---|
705 | } |
---|
706 | } |
---|
707 | #endregion |
---|
708 | |
---|
709 | #region Event Attributes |
---|
710 | void AddAttributes(EventDefinition eventDefinition, IUnresolvedEntity targetEntity) |
---|
711 | { |
---|
712 | if (eventDefinition.HasCustomAttributes) { |
---|
713 | AddCustomAttributes(eventDefinition.CustomAttributes, targetEntity.Attributes); |
---|
714 | } |
---|
715 | } |
---|
716 | #endregion |
---|
717 | |
---|
718 | #region Property Attributes |
---|
719 | void AddAttributes(PropertyDefinition propertyDefinition, IUnresolvedEntity targetEntity) |
---|
720 | { |
---|
721 | if (propertyDefinition.HasCustomAttributes) { |
---|
722 | AddCustomAttributes(propertyDefinition.CustomAttributes, targetEntity.Attributes); |
---|
723 | } |
---|
724 | } |
---|
725 | #endregion |
---|
726 | |
---|
727 | #region MarshalAsAttribute (ConvertMarshalInfo) |
---|
728 | static readonly ITypeReference marshalAsAttributeTypeRef = typeof(MarshalAsAttribute).ToTypeReference(); |
---|
729 | static readonly ITypeReference unmanagedTypeTypeRef = typeof(UnmanagedType).ToTypeReference(); |
---|
730 | |
---|
731 | IUnresolvedAttribute ConvertMarshalInfo(MarshalInfo marshalInfo) |
---|
732 | { |
---|
733 | DefaultUnresolvedAttribute attr = new DefaultUnresolvedAttribute(marshalAsAttributeTypeRef, new[] { unmanagedTypeTypeRef }); |
---|
734 | attr.PositionalArguments.Add(CreateSimpleConstantValue(unmanagedTypeTypeRef, (int)marshalInfo.NativeType)); |
---|
735 | |
---|
736 | FixedArrayMarshalInfo fami = marshalInfo as FixedArrayMarshalInfo; |
---|
737 | if (fami != null) { |
---|
738 | attr.AddNamedFieldArgument("SizeConst", CreateSimpleConstantValue(KnownTypeReference.Int32, (int)fami.Size)); |
---|
739 | if (fami.ElementType != NativeType.None) |
---|
740 | attr.AddNamedFieldArgument("ArraySubType", CreateSimpleConstantValue(unmanagedTypeTypeRef, (int)fami.ElementType)); |
---|
741 | } |
---|
742 | SafeArrayMarshalInfo sami = marshalInfo as SafeArrayMarshalInfo; |
---|
743 | if (sami != null && sami.ElementType != VariantType.None) { |
---|
744 | attr.AddNamedFieldArgument("SafeArraySubType", CreateSimpleConstantValue(typeof(VarEnum).ToTypeReference(), (int)sami.ElementType)); |
---|
745 | } |
---|
746 | ArrayMarshalInfo ami = marshalInfo as ArrayMarshalInfo; |
---|
747 | if (ami != null) { |
---|
748 | if (ami.ElementType != NativeType.Max) |
---|
749 | attr.AddNamedFieldArgument("ArraySubType", CreateSimpleConstantValue(unmanagedTypeTypeRef, (int)ami.ElementType)); |
---|
750 | if (ami.Size >= 0) |
---|
751 | attr.AddNamedFieldArgument("SizeConst", CreateSimpleConstantValue(KnownTypeReference.Int32, (int)ami.Size)); |
---|
752 | if (ami.SizeParameterMultiplier != 0 && ami.SizeParameterIndex >= 0) |
---|
753 | attr.AddNamedFieldArgument("SizeParamIndex", CreateSimpleConstantValue(KnownTypeReference.Int16, (short)ami.SizeParameterIndex)); |
---|
754 | } |
---|
755 | CustomMarshalInfo cmi = marshalInfo as CustomMarshalInfo; |
---|
756 | if (cmi != null) { |
---|
757 | if (cmi.ManagedType != null) |
---|
758 | attr.AddNamedFieldArgument("MarshalType", CreateSimpleConstantValue(KnownTypeReference.String, cmi.ManagedType.FullName)); |
---|
759 | if (!string.IsNullOrEmpty(cmi.Cookie)) |
---|
760 | attr.AddNamedFieldArgument("MarshalCookie", CreateSimpleConstantValue(KnownTypeReference.String, cmi.Cookie)); |
---|
761 | } |
---|
762 | FixedSysStringMarshalInfo fssmi = marshalInfo as FixedSysStringMarshalInfo; |
---|
763 | if (fssmi != null) { |
---|
764 | attr.AddNamedFieldArgument("SizeConst", CreateSimpleConstantValue(KnownTypeReference.Int32, (int)fssmi.Size)); |
---|
765 | } |
---|
766 | |
---|
767 | return InterningProvider.Intern(attr); |
---|
768 | } |
---|
769 | #endregion |
---|
770 | |
---|
771 | #region Custom Attributes (ReadAttribute) |
---|
772 | void AddCustomAttributes(Mono.Collections.Generic.Collection<CustomAttribute> attributes, IList<IUnresolvedAttribute> targetCollection) |
---|
773 | { |
---|
774 | foreach (var cecilAttribute in attributes) { |
---|
775 | TypeReference type = cecilAttribute.AttributeType; |
---|
776 | if (type.Namespace == "System.Runtime.CompilerServices") { |
---|
777 | if (type.Name == "DynamicAttribute" || type.Name == "ExtensionAttribute" || type.Name == "DecimalConstantAttribute") |
---|
778 | continue; |
---|
779 | } else if (type.Name == "ParamArrayAttribute" && type.Namespace == "System") { |
---|
780 | continue; |
---|
781 | } |
---|
782 | targetCollection.Add(ReadAttribute(cecilAttribute)); |
---|
783 | } |
---|
784 | } |
---|
785 | |
---|
786 | [CLSCompliant(false)] |
---|
787 | public IUnresolvedAttribute ReadAttribute(CustomAttribute attribute) |
---|
788 | { |
---|
789 | if (attribute == null) |
---|
790 | throw new ArgumentNullException("attribute"); |
---|
791 | MethodReference ctor = attribute.Constructor; |
---|
792 | ITypeReference attributeType = ReadTypeReference(attribute.AttributeType); |
---|
793 | IList<ITypeReference> ctorParameterTypes = EmptyList<ITypeReference>.Instance; |
---|
794 | if (ctor.HasParameters) { |
---|
795 | ctorParameterTypes = new ITypeReference[ctor.Parameters.Count]; |
---|
796 | for (int i = 0; i < ctorParameterTypes.Count; i++) { |
---|
797 | ctorParameterTypes[i] = ReadTypeReference(ctor.Parameters[i].ParameterType); |
---|
798 | } |
---|
799 | ctorParameterTypes = interningProvider.InternList(ctorParameterTypes); |
---|
800 | } |
---|
801 | return interningProvider.Intern(new UnresolvedAttributeBlob(attributeType, ctorParameterTypes, attribute.GetBlob())); |
---|
802 | } |
---|
803 | #endregion |
---|
804 | |
---|
805 | #region Security Attributes |
---|
806 | /// <summary> |
---|
807 | /// Reads a security declaration. |
---|
808 | /// </summary> |
---|
809 | [CLSCompliant(false)] |
---|
810 | public IList<IUnresolvedAttribute> ReadSecurityDeclaration(SecurityDeclaration secDecl) |
---|
811 | { |
---|
812 | if (secDecl == null) |
---|
813 | throw new ArgumentNullException("secDecl"); |
---|
814 | var result = new List<IUnresolvedAttribute>(); |
---|
815 | AddSecurityAttributes(secDecl, result); |
---|
816 | return result; |
---|
817 | } |
---|
818 | |
---|
819 | void AddSecurityAttributes(Mono.Collections.Generic.Collection<SecurityDeclaration> securityDeclarations, IList<IUnresolvedAttribute> targetCollection) |
---|
820 | { |
---|
821 | foreach (var secDecl in securityDeclarations) { |
---|
822 | AddSecurityAttributes(secDecl, targetCollection); |
---|
823 | } |
---|
824 | } |
---|
825 | |
---|
826 | void AddSecurityAttributes(SecurityDeclaration secDecl, IList<IUnresolvedAttribute> targetCollection) |
---|
827 | { |
---|
828 | byte[] blob; |
---|
829 | try { |
---|
830 | blob = secDecl.GetBlob(); |
---|
831 | } catch (NotSupportedException) { |
---|
832 | return; // https://github.com/icsharpcode/SharpDevelop/issues/284 |
---|
833 | } |
---|
834 | var blobSecDecl = new UnresolvedSecurityDeclarationBlob((int)secDecl.Action, blob); |
---|
835 | targetCollection.AddRange(blobSecDecl.UnresolvedAttributes); |
---|
836 | } |
---|
837 | #endregion |
---|
838 | #endregion |
---|
839 | |
---|
840 | #region Read Type Definition |
---|
841 | DefaultUnresolvedTypeDefinition CreateTopLevelTypeDefinition(TypeDefinition typeDefinition) |
---|
842 | { |
---|
843 | string name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(typeDefinition.Name); |
---|
844 | var td = new DefaultUnresolvedTypeDefinition(typeDefinition.Namespace, name); |
---|
845 | if (typeDefinition.HasGenericParameters) |
---|
846 | InitTypeParameters(typeDefinition, td.TypeParameters); |
---|
847 | return td; |
---|
848 | } |
---|
849 | |
---|
850 | static void InitTypeParameters(TypeDefinition typeDefinition, IList<IUnresolvedTypeParameter> typeParameters) |
---|
851 | { |
---|
852 | // Type parameters are initialized within the constructor so that the class can be put into the type storage |
---|
853 | // before the rest of the initialization runs - this allows it to be available for early binding as soon as possible. |
---|
854 | for (int i = 0; i < typeDefinition.GenericParameters.Count; i++) { |
---|
855 | if (typeDefinition.GenericParameters[i].Position != i) |
---|
856 | throw new InvalidOperationException("g.Position != i"); |
---|
857 | typeParameters.Add(new DefaultUnresolvedTypeParameter( |
---|
858 | SymbolKind.TypeDefinition, i, typeDefinition.GenericParameters[i].Name)); |
---|
859 | } |
---|
860 | } |
---|
861 | |
---|
862 | void InitTypeParameterConstraints(TypeDefinition typeDefinition, IList<IUnresolvedTypeParameter> typeParameters) |
---|
863 | { |
---|
864 | for (int i = 0; i < typeParameters.Count; i++) { |
---|
865 | var tp = (DefaultUnresolvedTypeParameter)typeParameters[i]; |
---|
866 | AddConstraints(tp, typeDefinition.GenericParameters[i]); |
---|
867 | tp.ApplyInterningProvider(interningProvider); |
---|
868 | } |
---|
869 | } |
---|
870 | |
---|
871 | void InitTypeDefinition(TypeDefinition typeDefinition, DefaultUnresolvedTypeDefinition td) |
---|
872 | { |
---|
873 | td.Kind = GetTypeKind(typeDefinition); |
---|
874 | InitTypeModifiers(typeDefinition, td); |
---|
875 | InitTypeParameterConstraints(typeDefinition, td.TypeParameters); |
---|
876 | |
---|
877 | // nested types can be initialized only after generic parameters were created |
---|
878 | InitNestedTypes(typeDefinition, td, td.NestedTypes); |
---|
879 | AddAttributes(typeDefinition, td); |
---|
880 | td.HasExtensionMethods = HasExtensionAttribute(typeDefinition); |
---|
881 | |
---|
882 | InitBaseTypes(typeDefinition, td.BaseTypes); |
---|
883 | |
---|
884 | td.AddDefaultConstructorIfRequired = (td.Kind == TypeKind.Struct || td.Kind == TypeKind.Enum); |
---|
885 | InitMembers(typeDefinition, td, td.Members); |
---|
886 | td.ApplyInterningProvider(interningProvider); |
---|
887 | td.Freeze(); |
---|
888 | RegisterCecilObject(td, typeDefinition); |
---|
889 | } |
---|
890 | |
---|
891 | void InitBaseTypes(TypeDefinition typeDefinition, IList<ITypeReference> baseTypes) |
---|
892 | { |
---|
893 | // set base classes |
---|
894 | if (typeDefinition.IsEnum) { |
---|
895 | foreach (FieldDefinition enumField in typeDefinition.Fields) { |
---|
896 | if (!enumField.IsStatic) { |
---|
897 | baseTypes.Add(ReadTypeReference(enumField.FieldType)); |
---|
898 | break; |
---|
899 | } |
---|
900 | } |
---|
901 | } else { |
---|
902 | if (typeDefinition.BaseType != null) { |
---|
903 | baseTypes.Add(ReadTypeReference(typeDefinition.BaseType)); |
---|
904 | } |
---|
905 | if (typeDefinition.HasInterfaces) { |
---|
906 | foreach (TypeReference iface in typeDefinition.Interfaces) { |
---|
907 | baseTypes.Add(ReadTypeReference(iface)); |
---|
908 | } |
---|
909 | } |
---|
910 | } |
---|
911 | } |
---|
912 | |
---|
913 | void InitNestedTypes(TypeDefinition typeDefinition, IUnresolvedTypeDefinition declaringTypeDefinition, IList<IUnresolvedTypeDefinition> nestedTypes) |
---|
914 | { |
---|
915 | if (!typeDefinition.HasNestedTypes) |
---|
916 | return; |
---|
917 | foreach (TypeDefinition nestedTypeDef in typeDefinition.NestedTypes) { |
---|
918 | TypeAttributes visibility = nestedTypeDef.Attributes & TypeAttributes.VisibilityMask; |
---|
919 | if (this.IncludeInternalMembers |
---|
920 | || visibility == TypeAttributes.NestedPublic |
---|
921 | || visibility == TypeAttributes.NestedFamily |
---|
922 | || visibility == TypeAttributes.NestedFamORAssem) |
---|
923 | { |
---|
924 | string name = nestedTypeDef.Name; |
---|
925 | int pos = name.LastIndexOf('/'); |
---|
926 | if (pos > 0) |
---|
927 | name = name.Substring(pos + 1); |
---|
928 | name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(name); |
---|
929 | var nestedType = new DefaultUnresolvedTypeDefinition(declaringTypeDefinition, name); |
---|
930 | InitTypeParameters(nestedTypeDef, nestedType.TypeParameters); |
---|
931 | nestedTypes.Add(nestedType); |
---|
932 | InitTypeDefinition(nestedTypeDef, nestedType); |
---|
933 | } |
---|
934 | } |
---|
935 | } |
---|
936 | |
---|
937 | static TypeKind GetTypeKind(TypeDefinition typeDefinition) |
---|
938 | { |
---|
939 | // set classtype |
---|
940 | if (typeDefinition.IsInterface) { |
---|
941 | return TypeKind.Interface; |
---|
942 | } else if (typeDefinition.IsEnum) { |
---|
943 | return TypeKind.Enum; |
---|
944 | } else if (typeDefinition.IsValueType) { |
---|
945 | return TypeKind.Struct; |
---|
946 | } else if (IsDelegate(typeDefinition)) { |
---|
947 | return TypeKind.Delegate; |
---|
948 | } else if (IsModule(typeDefinition)) { |
---|
949 | return TypeKind.Module; |
---|
950 | } else { |
---|
951 | return TypeKind.Class; |
---|
952 | } |
---|
953 | } |
---|
954 | |
---|
955 | static void InitTypeModifiers(TypeDefinition typeDefinition, AbstractUnresolvedEntity td) |
---|
956 | { |
---|
957 | td.IsSealed = typeDefinition.IsSealed; |
---|
958 | td.IsAbstract = typeDefinition.IsAbstract; |
---|
959 | switch (typeDefinition.Attributes & TypeAttributes.VisibilityMask) { |
---|
960 | case TypeAttributes.NotPublic: |
---|
961 | case TypeAttributes.NestedAssembly: |
---|
962 | td.Accessibility = Accessibility.Internal; |
---|
963 | break; |
---|
964 | case TypeAttributes.Public: |
---|
965 | case TypeAttributes.NestedPublic: |
---|
966 | td.Accessibility = Accessibility.Public; |
---|
967 | break; |
---|
968 | case TypeAttributes.NestedPrivate: |
---|
969 | td.Accessibility = Accessibility.Private; |
---|
970 | break; |
---|
971 | case TypeAttributes.NestedFamily: |
---|
972 | td.Accessibility = Accessibility.Protected; |
---|
973 | break; |
---|
974 | case TypeAttributes.NestedFamANDAssem: |
---|
975 | td.Accessibility = Accessibility.ProtectedAndInternal; |
---|
976 | break; |
---|
977 | case TypeAttributes.NestedFamORAssem: |
---|
978 | td.Accessibility = Accessibility.ProtectedOrInternal; |
---|
979 | break; |
---|
980 | } |
---|
981 | } |
---|
982 | |
---|
983 | static bool IsDelegate(TypeDefinition type) |
---|
984 | { |
---|
985 | if (type.BaseType != null && type.BaseType.Namespace == "System") { |
---|
986 | if (type.BaseType.Name == "MulticastDelegate") |
---|
987 | return true; |
---|
988 | if (type.BaseType.Name == "Delegate" && type.Name != "MulticastDelegate") |
---|
989 | return true; |
---|
990 | } |
---|
991 | return false; |
---|
992 | } |
---|
993 | |
---|
994 | static bool IsModule(TypeDefinition type) |
---|
995 | { |
---|
996 | if (!type.HasCustomAttributes) |
---|
997 | return false; |
---|
998 | foreach (var att in type.CustomAttributes) { |
---|
999 | if (att.AttributeType.FullName == "Microsoft.VisualBasic.CompilerServices.StandardModuleAttribute" |
---|
1000 | || att.AttributeType.FullName == "System.Runtime.CompilerServices.CompilerGlobalScopeAttribute") |
---|
1001 | { |
---|
1002 | return true; |
---|
1003 | } |
---|
1004 | } |
---|
1005 | return false; |
---|
1006 | } |
---|
1007 | |
---|
1008 | void InitMembers(TypeDefinition typeDefinition, IUnresolvedTypeDefinition td, IList<IUnresolvedMember> members) |
---|
1009 | { |
---|
1010 | if (typeDefinition.HasMethods) { |
---|
1011 | foreach (MethodDefinition method in typeDefinition.Methods) { |
---|
1012 | if (IsVisible(method.Attributes) && !IsAccessor(method.SemanticsAttributes)) { |
---|
1013 | SymbolKind type = SymbolKind.Method; |
---|
1014 | if (method.IsSpecialName) { |
---|
1015 | if (method.IsConstructor) |
---|
1016 | type = SymbolKind.Constructor; |
---|
1017 | else if (method.Name.StartsWith("op_", StringComparison.Ordinal)) |
---|
1018 | type = SymbolKind.Operator; |
---|
1019 | } |
---|
1020 | members.Add(ReadMethod(method, td, type)); |
---|
1021 | } |
---|
1022 | } |
---|
1023 | } |
---|
1024 | if (typeDefinition.HasFields) { |
---|
1025 | foreach (FieldDefinition field in typeDefinition.Fields) { |
---|
1026 | if (IsVisible(field.Attributes) && !field.IsSpecialName) { |
---|
1027 | members.Add(ReadField(field, td)); |
---|
1028 | } |
---|
1029 | } |
---|
1030 | } |
---|
1031 | if (typeDefinition.HasProperties) { |
---|
1032 | string defaultMemberName = null; |
---|
1033 | var defaultMemberAttribute = typeDefinition.CustomAttributes.FirstOrDefault( |
---|
1034 | a => a.AttributeType.FullName == typeof(System.Reflection.DefaultMemberAttribute).FullName); |
---|
1035 | if (defaultMemberAttribute != null && defaultMemberAttribute.ConstructorArguments.Count == 1) { |
---|
1036 | defaultMemberName = defaultMemberAttribute.ConstructorArguments[0].Value as string; |
---|
1037 | } |
---|
1038 | foreach (PropertyDefinition property in typeDefinition.Properties) { |
---|
1039 | bool getterVisible = property.GetMethod != null && IsVisible(property.GetMethod.Attributes); |
---|
1040 | bool setterVisible = property.SetMethod != null && IsVisible(property.SetMethod.Attributes); |
---|
1041 | if (getterVisible || setterVisible) { |
---|
1042 | SymbolKind type = SymbolKind.Property; |
---|
1043 | if (property.HasParameters) { |
---|
1044 | // Try to detect indexer: |
---|
1045 | if (property.Name == defaultMemberName) { |
---|
1046 | type = SymbolKind.Indexer; // normal indexer |
---|
1047 | } else if (property.Name.EndsWith(".Item", StringComparison.Ordinal) && (property.GetMethod ?? property.SetMethod).HasOverrides) { |
---|
1048 | // explicit interface implementation of indexer |
---|
1049 | type = SymbolKind.Indexer; |
---|
1050 | // We can't really tell parameterized properties and indexers apart in this case without |
---|
1051 | // resolving the interface, so we rely on the "Item" naming convention instead. |
---|
1052 | } |
---|
1053 | } |
---|
1054 | members.Add(ReadProperty(property, td, type)); |
---|
1055 | } |
---|
1056 | } |
---|
1057 | } |
---|
1058 | if (typeDefinition.HasEvents) { |
---|
1059 | foreach (EventDefinition ev in typeDefinition.Events) { |
---|
1060 | if (ev.AddMethod != null && IsVisible(ev.AddMethod.Attributes)) { |
---|
1061 | members.Add(ReadEvent(ev, td)); |
---|
1062 | } |
---|
1063 | } |
---|
1064 | } |
---|
1065 | } |
---|
1066 | |
---|
1067 | static bool IsAccessor(MethodSemanticsAttributes semantics) |
---|
1068 | { |
---|
1069 | return !(semantics == MethodSemanticsAttributes.None || semantics == MethodSemanticsAttributes.Other); |
---|
1070 | } |
---|
1071 | #endregion |
---|
1072 | |
---|
1073 | #region Lazy-Loaded Type Definition |
---|
1074 | /// <summary> |
---|
1075 | /// Given an assembly that was created by the CecilLoader with lazy-loading enabled, |
---|
1076 | /// this method will eagerly load all classes, and free any references to the source Cecil objects. |
---|
1077 | /// |
---|
1078 | /// The intended usage pattern for this method is: |
---|
1079 | /// 1. use lazy-loading to improve the latency when new assemblies have to be loaded |
---|
1080 | /// 2. later, when the CPU is idle, call FinishLazyLoading() to free up the memory consumed by the Cecil objects |
---|
1081 | /// </summary> |
---|
1082 | public static void FinishLazyLoading(IUnresolvedAssembly assembly) |
---|
1083 | { |
---|
1084 | if (assembly == null) |
---|
1085 | throw new ArgumentNullException("assembly"); |
---|
1086 | foreach (var type in assembly.TopLevelTypeDefinitions) { |
---|
1087 | var lctd = type as LazyCecilTypeDefinition; |
---|
1088 | if (lctd != null) |
---|
1089 | lctd.InitAndReleaseReferences(); |
---|
1090 | } |
---|
1091 | } |
---|
1092 | |
---|
1093 | sealed class LazyCecilTypeDefinition : AbstractUnresolvedEntity, IUnresolvedTypeDefinition |
---|
1094 | { |
---|
1095 | // loader + cecilTypeDef, used for lazy-loading; and set to null after lazy loading is complete |
---|
1096 | CecilLoader loader; |
---|
1097 | TypeDefinition cecilTypeDef; |
---|
1098 | |
---|
1099 | readonly string namespaceName; |
---|
1100 | readonly TypeKind kind; |
---|
1101 | readonly IList<IUnresolvedTypeParameter> typeParameters; |
---|
1102 | |
---|
1103 | // lazy-loaded fields |
---|
1104 | IList<ITypeReference> baseTypes; |
---|
1105 | IList<IUnresolvedTypeDefinition> nestedTypes; |
---|
1106 | IList<IUnresolvedMember> members; |
---|
1107 | |
---|
1108 | public LazyCecilTypeDefinition(CecilLoader loader, TypeDefinition typeDefinition) |
---|
1109 | { |
---|
1110 | this.loader = loader; |
---|
1111 | this.cecilTypeDef = typeDefinition; |
---|
1112 | this.SymbolKind = SymbolKind.TypeDefinition; |
---|
1113 | this.namespaceName = typeDefinition.Namespace; |
---|
1114 | this.Name = ReflectionHelper.SplitTypeParameterCountFromReflectionName(typeDefinition.Name); |
---|
1115 | var tps = new List<IUnresolvedTypeParameter>(); |
---|
1116 | InitTypeParameters(typeDefinition, tps); |
---|
1117 | this.typeParameters = FreezableHelper.FreezeList(tps); |
---|
1118 | |
---|
1119 | this.kind = GetTypeKind(typeDefinition); |
---|
1120 | InitTypeModifiers(typeDefinition, this); |
---|
1121 | loader.InitTypeParameterConstraints(typeDefinition, typeParameters); |
---|
1122 | |
---|
1123 | loader.AddAttributes(typeDefinition, this); |
---|
1124 | flags[FlagHasExtensionMethods] = HasExtensionAttribute(typeDefinition); |
---|
1125 | |
---|
1126 | this.ApplyInterningProvider(loader.interningProvider); |
---|
1127 | this.Freeze(); |
---|
1128 | } |
---|
1129 | |
---|
1130 | public override string Namespace { |
---|
1131 | get { return namespaceName; } |
---|
1132 | set { throw new NotSupportedException(); } |
---|
1133 | } |
---|
1134 | |
---|
1135 | public override string ReflectionName { |
---|
1136 | get { return this.FullTypeName.ReflectionName; } |
---|
1137 | } |
---|
1138 | |
---|
1139 | public FullTypeName FullTypeName { |
---|
1140 | get { |
---|
1141 | return new TopLevelTypeName(namespaceName, this.Name, typeParameters.Count); |
---|
1142 | } |
---|
1143 | } |
---|
1144 | |
---|
1145 | public TypeKind Kind { |
---|
1146 | get { return kind; } |
---|
1147 | } |
---|
1148 | |
---|
1149 | public IList<IUnresolvedTypeParameter> TypeParameters { |
---|
1150 | get { return typeParameters; } |
---|
1151 | } |
---|
1152 | |
---|
1153 | public IList<ITypeReference> BaseTypes { |
---|
1154 | get { |
---|
1155 | var result = LazyInit.VolatileRead(ref this.baseTypes); |
---|
1156 | if (result != null) { |
---|
1157 | return result; |
---|
1158 | } else { |
---|
1159 | return LazyInit.GetOrSet(ref this.baseTypes, TryInitBaseTypes()); |
---|
1160 | } |
---|
1161 | } |
---|
1162 | } |
---|
1163 | |
---|
1164 | IList<ITypeReference> TryInitBaseTypes() |
---|
1165 | { |
---|
1166 | var loader = LazyInit.VolatileRead(ref this.loader); |
---|
1167 | var cecilTypeDef = LazyInit.VolatileRead(ref this.cecilTypeDef); |
---|
1168 | if (loader == null || cecilTypeDef == null) { |
---|
1169 | // Cannot initialize because the references to loader/cecilTypeDef |
---|
1170 | // have already been cleared. |
---|
1171 | // This can only happen if the class was loaded by another thread concurrently to the TryInitBaseTypes() call, |
---|
1172 | // so the GetOrSet() call in the property will retrieve the value set by the other thread. |
---|
1173 | return null; |
---|
1174 | } |
---|
1175 | lock (loader.currentModule) { |
---|
1176 | var result = new List<ITypeReference>(); |
---|
1177 | loader.InitBaseTypes(cecilTypeDef, result); |
---|
1178 | return FreezableHelper.FreezeList(result); |
---|
1179 | } |
---|
1180 | } |
---|
1181 | |
---|
1182 | public IList<IUnresolvedTypeDefinition> NestedTypes { |
---|
1183 | get { |
---|
1184 | var result = LazyInit.VolatileRead(ref this.nestedTypes); |
---|
1185 | if (result != null) { |
---|
1186 | return result; |
---|
1187 | } else { |
---|
1188 | return LazyInit.GetOrSet(ref this.nestedTypes, TryInitNestedTypes()); |
---|
1189 | } |
---|
1190 | } |
---|
1191 | } |
---|
1192 | |
---|
1193 | IList<IUnresolvedTypeDefinition> TryInitNestedTypes() |
---|
1194 | { |
---|
1195 | var loader = LazyInit.VolatileRead(ref this.loader); |
---|
1196 | var cecilTypeDef = LazyInit.VolatileRead(ref this.cecilTypeDef); |
---|
1197 | if (loader == null || cecilTypeDef == null) { |
---|
1198 | // Cannot initialize because the references to loader/cecilTypeDef |
---|
1199 | // have already been cleared. |
---|
1200 | // This can only happen if the class was loaded by another thread concurrently to the TryInitNestedTypes() call, |
---|
1201 | // so the GetOrSet() call in the property will retrieve the value set by the other thread. |
---|
1202 | return null; |
---|
1203 | } |
---|
1204 | lock (loader.currentModule) { |
---|
1205 | var result = new List<IUnresolvedTypeDefinition>(); |
---|
1206 | loader.InitNestedTypes(cecilTypeDef, this, result); |
---|
1207 | return FreezableHelper.FreezeList(result); |
---|
1208 | } |
---|
1209 | } |
---|
1210 | |
---|
1211 | public IList<IUnresolvedMember> Members { |
---|
1212 | get { |
---|
1213 | var result = LazyInit.VolatileRead(ref this.members); |
---|
1214 | if (result != null) { |
---|
1215 | return result; |
---|
1216 | } else { |
---|
1217 | return LazyInit.GetOrSet(ref this.members, TryInitMembers()); |
---|
1218 | } |
---|
1219 | } |
---|
1220 | } |
---|
1221 | |
---|
1222 | IList<IUnresolvedMember> TryInitMembers() |
---|
1223 | { |
---|
1224 | var loader = LazyInit.VolatileRead(ref this.loader); |
---|
1225 | var cecilTypeDef = LazyInit.VolatileRead(ref this.cecilTypeDef); |
---|
1226 | if (loader == null || cecilTypeDef == null) { |
---|
1227 | // Cannot initialize because the references to loader/cecilTypeDef |
---|
1228 | // have already been cleared. |
---|
1229 | // This can only happen if the class was loaded by another thread concurrently to the TryInitMembers() call, |
---|
1230 | // so the GetOrSet() call in the property will retrieve the value set by the other thread. |
---|
1231 | return null; |
---|
1232 | } |
---|
1233 | lock (loader.currentModule) { |
---|
1234 | if (this.members != null) |
---|
1235 | return this.members; |
---|
1236 | var result = new List<IUnresolvedMember>(); |
---|
1237 | loader.InitMembers(cecilTypeDef, this, result); |
---|
1238 | return FreezableHelper.FreezeList(result); |
---|
1239 | } |
---|
1240 | } |
---|
1241 | |
---|
1242 | public void InitAndReleaseReferences() |
---|
1243 | { |
---|
1244 | if (LazyInit.VolatileRead(ref this.baseTypes) == null) |
---|
1245 | LazyInit.GetOrSet(ref this.baseTypes, TryInitBaseTypes()); |
---|
1246 | if (LazyInit.VolatileRead(ref this.nestedTypes) == null) |
---|
1247 | LazyInit.GetOrSet(ref this.nestedTypes, TryInitNestedTypes()); |
---|
1248 | if (LazyInit.VolatileRead(ref this.members) == null) |
---|
1249 | LazyInit.GetOrSet(ref this.members, TryInitMembers()); |
---|
1250 | Thread.MemoryBarrier(); // commit lazily-initialized fields to memory before nulling out the references |
---|
1251 | // Allow the GC to collect the cecil type definition |
---|
1252 | loader = null; |
---|
1253 | cecilTypeDef = null; |
---|
1254 | } |
---|
1255 | |
---|
1256 | public IEnumerable<IUnresolvedMethod> Methods { |
---|
1257 | get { return Members.OfType<IUnresolvedMethod>(); } |
---|
1258 | } |
---|
1259 | |
---|
1260 | public IEnumerable<IUnresolvedProperty> Properties { |
---|
1261 | get { return Members.OfType<IUnresolvedProperty>(); } |
---|
1262 | } |
---|
1263 | |
---|
1264 | public IEnumerable<IUnresolvedField> Fields { |
---|
1265 | get { return Members.OfType<IUnresolvedField>(); } |
---|
1266 | } |
---|
1267 | |
---|
1268 | public IEnumerable<IUnresolvedEvent> Events { |
---|
1269 | get { return Members.OfType<IUnresolvedEvent>(); } |
---|
1270 | } |
---|
1271 | |
---|
1272 | public bool AddDefaultConstructorIfRequired { |
---|
1273 | get { return kind == TypeKind.Struct || kind == TypeKind.Enum; } |
---|
1274 | } |
---|
1275 | |
---|
1276 | public bool? HasExtensionMethods { |
---|
1277 | get { return flags[FlagHasExtensionMethods]; } |
---|
1278 | // we always return true or false, never null. |
---|
1279 | // FlagHasNoExtensionMethods is unused in LazyCecilTypeDefinition |
---|
1280 | } |
---|
1281 | |
---|
1282 | public bool IsPartial { |
---|
1283 | get { return false; } |
---|
1284 | } |
---|
1285 | |
---|
1286 | public override object Clone() |
---|
1287 | { |
---|
1288 | throw new NotSupportedException(); |
---|
1289 | } |
---|
1290 | |
---|
1291 | public IType Resolve(ITypeResolveContext context) |
---|
1292 | { |
---|
1293 | if (context == null) |
---|
1294 | throw new ArgumentNullException("context"); |
---|
1295 | if (context.CurrentAssembly == null) |
---|
1296 | throw new ArgumentException("An ITypeDefinition cannot be resolved in a context without a current assembly."); |
---|
1297 | return context.CurrentAssembly.GetTypeDefinition(this.FullTypeName) |
---|
1298 | ?? (IType)new UnknownType(this.Namespace, this.Name, this.TypeParameters.Count); |
---|
1299 | } |
---|
1300 | |
---|
1301 | public ITypeResolveContext CreateResolveContext(ITypeResolveContext parentContext) |
---|
1302 | { |
---|
1303 | return parentContext; |
---|
1304 | } |
---|
1305 | } |
---|
1306 | #endregion |
---|
1307 | |
---|
1308 | #region Read Method |
---|
1309 | [CLSCompliant(false)] |
---|
1310 | public IUnresolvedMethod ReadMethod(MethodDefinition method, IUnresolvedTypeDefinition parentType, SymbolKind methodType = SymbolKind.Method) |
---|
1311 | { |
---|
1312 | return ReadMethod(method, parentType, methodType, null); |
---|
1313 | } |
---|
1314 | |
---|
1315 | IUnresolvedMethod ReadMethod(MethodDefinition method, IUnresolvedTypeDefinition parentType, SymbolKind methodType, IUnresolvedMember accessorOwner) |
---|
1316 | { |
---|
1317 | if (method == null) |
---|
1318 | return null; |
---|
1319 | DefaultUnresolvedMethod m = new DefaultUnresolvedMethod(parentType, method.Name); |
---|
1320 | m.SymbolKind = methodType; |
---|
1321 | m.AccessorOwner = accessorOwner; |
---|
1322 | m.HasBody = method.HasBody; |
---|
1323 | if (method.HasGenericParameters) { |
---|
1324 | for (int i = 0; i < method.GenericParameters.Count; i++) { |
---|
1325 | if (method.GenericParameters[i].Position != i) |
---|
1326 | throw new InvalidOperationException("g.Position != i"); |
---|
1327 | m.TypeParameters.Add(new DefaultUnresolvedTypeParameter( |
---|
1328 | SymbolKind.Method, i, method.GenericParameters[i].Name)); |
---|
1329 | } |
---|
1330 | for (int i = 0; i < method.GenericParameters.Count; i++) { |
---|
1331 | var tp = (DefaultUnresolvedTypeParameter)m.TypeParameters[i]; |
---|
1332 | AddConstraints(tp, method.GenericParameters[i]); |
---|
1333 | tp.ApplyInterningProvider(interningProvider); |
---|
1334 | } |
---|
1335 | } |
---|
1336 | |
---|
1337 | m.ReturnType = ReadTypeReference(method.ReturnType, typeAttributes: method.MethodReturnType); |
---|
1338 | |
---|
1339 | if (HasAnyAttributes(method)) |
---|
1340 | AddAttributes(method, m.Attributes, m.ReturnTypeAttributes); |
---|
1341 | TranslateModifiers(method, m); |
---|
1342 | |
---|
1343 | if (method.HasParameters) { |
---|
1344 | foreach (ParameterDefinition p in method.Parameters) { |
---|
1345 | m.Parameters.Add(ReadParameter(p)); |
---|
1346 | } |
---|
1347 | } |
---|
1348 | |
---|
1349 | // mark as extension method if the attribute is set |
---|
1350 | if (method.IsStatic && HasExtensionAttribute(method)) { |
---|
1351 | m.IsExtensionMethod = true; |
---|
1352 | } |
---|
1353 | |
---|
1354 | int lastDot = method.Name.LastIndexOf('.'); |
---|
1355 | if (lastDot >= 0 && method.HasOverrides) { |
---|
1356 | // To be consistent with the parser-initialized type system, shorten the method name: |
---|
1357 | m.Name = method.Name.Substring(lastDot + 1); |
---|
1358 | m.IsExplicitInterfaceImplementation = true; |
---|
1359 | foreach (var or in method.Overrides) { |
---|
1360 | m.ExplicitInterfaceImplementations.Add(new DefaultMemberReference( |
---|
1361 | accessorOwner != null ? SymbolKind.Accessor : SymbolKind.Method, |
---|
1362 | ReadTypeReference(or.DeclaringType), |
---|
1363 | or.Name, or.GenericParameters.Count, m.Parameters.Select(p => p.Type).ToList())); |
---|
1364 | } |
---|
1365 | } |
---|
1366 | |
---|
1367 | FinishReadMember(m, method); |
---|
1368 | return m; |
---|
1369 | } |
---|
1370 | |
---|
1371 | static bool HasExtensionAttribute(ICustomAttributeProvider provider) |
---|
1372 | { |
---|
1373 | if (provider.HasCustomAttributes) { |
---|
1374 | foreach (var attr in provider.CustomAttributes) { |
---|
1375 | if (attr.AttributeType.Name == "ExtensionAttribute" && attr.AttributeType.Namespace == "System.Runtime.CompilerServices") |
---|
1376 | return true; |
---|
1377 | } |
---|
1378 | } |
---|
1379 | return false; |
---|
1380 | } |
---|
1381 | |
---|
1382 | bool IsVisible(MethodAttributes att) |
---|
1383 | { |
---|
1384 | att &= MethodAttributes.MemberAccessMask; |
---|
1385 | return IncludeInternalMembers |
---|
1386 | || att == MethodAttributes.Public |
---|
1387 | || att == MethodAttributes.Family |
---|
1388 | || att == MethodAttributes.FamORAssem; |
---|
1389 | } |
---|
1390 | |
---|
1391 | static Accessibility GetAccessibility(MethodAttributes attr) |
---|
1392 | { |
---|
1393 | switch (attr & MethodAttributes.MemberAccessMask) { |
---|
1394 | case MethodAttributes.Public: |
---|
1395 | return Accessibility.Public; |
---|
1396 | case MethodAttributes.FamANDAssem: |
---|
1397 | return Accessibility.ProtectedAndInternal; |
---|
1398 | case MethodAttributes.Assembly: |
---|
1399 | return Accessibility.Internal; |
---|
1400 | case MethodAttributes.Family: |
---|
1401 | return Accessibility.Protected; |
---|
1402 | case MethodAttributes.FamORAssem: |
---|
1403 | return Accessibility.ProtectedOrInternal; |
---|
1404 | default: |
---|
1405 | return Accessibility.Private; |
---|
1406 | } |
---|
1407 | } |
---|
1408 | |
---|
1409 | void TranslateModifiers(MethodDefinition method, AbstractUnresolvedMember m) |
---|
1410 | { |
---|
1411 | if (m.DeclaringTypeDefinition.Kind == TypeKind.Interface) { |
---|
1412 | // interface members don't have modifiers, but we want to handle them as "public abstract" |
---|
1413 | m.Accessibility = Accessibility.Public; |
---|
1414 | m.IsAbstract = true; |
---|
1415 | } else { |
---|
1416 | m.Accessibility = GetAccessibility(method.Attributes); |
---|
1417 | if (method.IsAbstract) { |
---|
1418 | m.IsAbstract = true; |
---|
1419 | m.IsOverride = !method.IsNewSlot; |
---|
1420 | } else if (method.IsFinal) { |
---|
1421 | if (!method.IsNewSlot) { |
---|
1422 | m.IsSealed = true; |
---|
1423 | m.IsOverride = true; |
---|
1424 | } |
---|
1425 | } else if (method.IsVirtual) { |
---|
1426 | if (method.IsNewSlot) |
---|
1427 | m.IsVirtual = true; |
---|
1428 | else |
---|
1429 | m.IsOverride = true; |
---|
1430 | } |
---|
1431 | m.IsStatic = method.IsStatic; |
---|
1432 | } |
---|
1433 | } |
---|
1434 | #endregion |
---|
1435 | |
---|
1436 | #region Read Parameter |
---|
1437 | [CLSCompliant(false)] |
---|
1438 | public IUnresolvedParameter ReadParameter(ParameterDefinition parameter) |
---|
1439 | { |
---|
1440 | if (parameter == null) |
---|
1441 | throw new ArgumentNullException("parameter"); |
---|
1442 | var type = ReadTypeReference(parameter.ParameterType, typeAttributes: parameter); |
---|
1443 | var p = new DefaultUnresolvedParameter(type, interningProvider.Intern(parameter.Name)); |
---|
1444 | |
---|
1445 | if (parameter.ParameterType is Mono.Cecil.ByReferenceType) { |
---|
1446 | if (!parameter.IsIn && parameter.IsOut) |
---|
1447 | p.IsOut = true; |
---|
1448 | else |
---|
1449 | p.IsRef = true; |
---|
1450 | } |
---|
1451 | AddAttributes(parameter, p); |
---|
1452 | |
---|
1453 | if (parameter.IsOptional) { |
---|
1454 | p.DefaultValue = CreateSimpleConstantValue(type, parameter.Constant); |
---|
1455 | } |
---|
1456 | |
---|
1457 | if (parameter.ParameterType is Mono.Cecil.ArrayType) { |
---|
1458 | foreach (CustomAttribute att in parameter.CustomAttributes) { |
---|
1459 | if (att.AttributeType.FullName == typeof(ParamArrayAttribute).FullName) { |
---|
1460 | p.IsParams = true; |
---|
1461 | break; |
---|
1462 | } |
---|
1463 | } |
---|
1464 | } |
---|
1465 | |
---|
1466 | return interningProvider.Intern(p); |
---|
1467 | } |
---|
1468 | #endregion |
---|
1469 | |
---|
1470 | #region Read Field |
---|
1471 | bool IsVisible(FieldAttributes att) |
---|
1472 | { |
---|
1473 | att &= FieldAttributes.FieldAccessMask; |
---|
1474 | return IncludeInternalMembers |
---|
1475 | || att == FieldAttributes.Public |
---|
1476 | || att == FieldAttributes.Family |
---|
1477 | || att == FieldAttributes.FamORAssem; |
---|
1478 | } |
---|
1479 | |
---|
1480 | decimal? TryDecodeDecimalConstantAttribute(CustomAttribute attribute) |
---|
1481 | { |
---|
1482 | if (attribute.ConstructorArguments.Count != 5) |
---|
1483 | return null; |
---|
1484 | |
---|
1485 | BlobReader reader = new BlobReader(attribute.GetBlob(), null); |
---|
1486 | if (reader.ReadUInt16() != 0x0001) { |
---|
1487 | Debug.WriteLine("Unknown blob prolog"); |
---|
1488 | return null; |
---|
1489 | } |
---|
1490 | |
---|
1491 | // DecimalConstantAttribute has the arguments (byte scale, byte sign, uint hi, uint mid, uint low) or (byte scale, byte sign, int hi, int mid, int low) |
---|
1492 | // Both of these invoke the Decimal constructor (int lo, int mid, int hi, bool isNegative, byte scale) with explicit argument conversions if required. |
---|
1493 | var ctorArgs = new object[attribute.ConstructorArguments.Count]; |
---|
1494 | for (int i = 0; i < ctorArgs.Length; i++) { |
---|
1495 | switch (attribute.ConstructorArguments[i].Type.FullName) { |
---|
1496 | case "System.Byte": |
---|
1497 | ctorArgs[i] = reader.ReadByte(); |
---|
1498 | break; |
---|
1499 | case "System.Int32": |
---|
1500 | ctorArgs[i] = reader.ReadInt32(); |
---|
1501 | break; |
---|
1502 | case "System.UInt32": |
---|
1503 | ctorArgs[i] = unchecked((int)reader.ReadUInt32()); |
---|
1504 | break; |
---|
1505 | default: |
---|
1506 | return null; |
---|
1507 | } |
---|
1508 | } |
---|
1509 | |
---|
1510 | if (!ctorArgs.Select(a => a.GetType()).SequenceEqual(new[] { typeof(byte), typeof(byte), typeof(int), typeof(int), typeof(int) })) |
---|
1511 | return null; |
---|
1512 | |
---|
1513 | return new decimal((int)ctorArgs[4], (int)ctorArgs[3], (int)ctorArgs[2], (byte)ctorArgs[1] != 0, (byte)ctorArgs[0]); |
---|
1514 | } |
---|
1515 | |
---|
1516 | [CLSCompliant(false)] |
---|
1517 | public IUnresolvedField ReadField(FieldDefinition field, IUnresolvedTypeDefinition parentType) |
---|
1518 | { |
---|
1519 | if (field == null) |
---|
1520 | throw new ArgumentNullException("field"); |
---|
1521 | if (parentType == null) |
---|
1522 | throw new ArgumentNullException("parentType"); |
---|
1523 | |
---|
1524 | DefaultUnresolvedField f = new DefaultUnresolvedField(parentType, field.Name); |
---|
1525 | f.Accessibility = GetAccessibility(field.Attributes); |
---|
1526 | f.IsReadOnly = field.IsInitOnly; |
---|
1527 | f.IsStatic = field.IsStatic; |
---|
1528 | f.ReturnType = ReadTypeReference(field.FieldType, typeAttributes: field); |
---|
1529 | if (field.HasConstant) { |
---|
1530 | f.ConstantValue = CreateSimpleConstantValue(f.ReturnType, field.Constant); |
---|
1531 | } |
---|
1532 | else { |
---|
1533 | var decConstant = field.CustomAttributes.FirstOrDefault(a => a.AttributeType.FullName == "System.Runtime.CompilerServices.DecimalConstantAttribute"); |
---|
1534 | if (decConstant != null) { |
---|
1535 | var constValue = TryDecodeDecimalConstantAttribute(decConstant); |
---|
1536 | if (constValue != null) |
---|
1537 | f.ConstantValue = CreateSimpleConstantValue(f.ReturnType, constValue); |
---|
1538 | } |
---|
1539 | } |
---|
1540 | AddAttributes(field, f); |
---|
1541 | |
---|
1542 | RequiredModifierType modreq = field.FieldType as RequiredModifierType; |
---|
1543 | if (modreq != null && modreq.ModifierType.FullName == typeof(IsVolatile).FullName) { |
---|
1544 | f.IsVolatile = true; |
---|
1545 | } |
---|
1546 | |
---|
1547 | FinishReadMember(f, field); |
---|
1548 | return f; |
---|
1549 | } |
---|
1550 | |
---|
1551 | static Accessibility GetAccessibility(FieldAttributes attr) |
---|
1552 | { |
---|
1553 | switch (attr & FieldAttributes.FieldAccessMask) { |
---|
1554 | case FieldAttributes.Public: |
---|
1555 | return Accessibility.Public; |
---|
1556 | case FieldAttributes.FamANDAssem: |
---|
1557 | return Accessibility.ProtectedAndInternal; |
---|
1558 | case FieldAttributes.Assembly: |
---|
1559 | return Accessibility.Internal; |
---|
1560 | case FieldAttributes.Family: |
---|
1561 | return Accessibility.Protected; |
---|
1562 | case FieldAttributes.FamORAssem: |
---|
1563 | return Accessibility.ProtectedOrInternal; |
---|
1564 | default: |
---|
1565 | return Accessibility.Private; |
---|
1566 | } |
---|
1567 | } |
---|
1568 | #endregion |
---|
1569 | |
---|
1570 | #region Type Parameter Constraints |
---|
1571 | void AddConstraints(DefaultUnresolvedTypeParameter tp, GenericParameter g) |
---|
1572 | { |
---|
1573 | switch (g.Attributes & GenericParameterAttributes.VarianceMask) { |
---|
1574 | case GenericParameterAttributes.Contravariant: |
---|
1575 | tp.Variance = VarianceModifier.Contravariant; |
---|
1576 | break; |
---|
1577 | case GenericParameterAttributes.Covariant: |
---|
1578 | tp.Variance = VarianceModifier.Covariant; |
---|
1579 | break; |
---|
1580 | } |
---|
1581 | |
---|
1582 | tp.HasReferenceTypeConstraint = g.HasReferenceTypeConstraint; |
---|
1583 | tp.HasValueTypeConstraint = g.HasNotNullableValueTypeConstraint; |
---|
1584 | tp.HasDefaultConstructorConstraint = g.HasDefaultConstructorConstraint; |
---|
1585 | |
---|
1586 | if (g.HasConstraints) { |
---|
1587 | foreach (TypeReference constraint in g.Constraints) { |
---|
1588 | tp.Constraints.Add(ReadTypeReference(constraint)); |
---|
1589 | } |
---|
1590 | } |
---|
1591 | } |
---|
1592 | #endregion |
---|
1593 | |
---|
1594 | #region Read Property |
---|
1595 | |
---|
1596 | Accessibility MergePropertyAccessibility (Accessibility left, Accessibility right) |
---|
1597 | { |
---|
1598 | if (left == Accessibility.Public || right == Accessibility.Public) |
---|
1599 | return Accessibility.Public; |
---|
1600 | |
---|
1601 | if (left == Accessibility.ProtectedOrInternal || right == Accessibility.ProtectedOrInternal) |
---|
1602 | return Accessibility.ProtectedOrInternal; |
---|
1603 | |
---|
1604 | if (left == Accessibility.Protected && right == Accessibility.Internal || |
---|
1605 | left == Accessibility.Internal && right == Accessibility.Protected) |
---|
1606 | return Accessibility.ProtectedOrInternal; |
---|
1607 | |
---|
1608 | if (left == Accessibility.Protected || right == Accessibility.Protected) |
---|
1609 | return Accessibility.Protected; |
---|
1610 | |
---|
1611 | if (left == Accessibility.Internal || right == Accessibility.Internal) |
---|
1612 | return Accessibility.Internal; |
---|
1613 | |
---|
1614 | if (left == Accessibility.ProtectedAndInternal || right == Accessibility.ProtectedAndInternal) |
---|
1615 | return Accessibility.ProtectedAndInternal; |
---|
1616 | |
---|
1617 | return left; |
---|
1618 | } |
---|
1619 | |
---|
1620 | [CLSCompliant(false)] |
---|
1621 | public IUnresolvedProperty ReadProperty(PropertyDefinition property, IUnresolvedTypeDefinition parentType, SymbolKind propertyType = SymbolKind.Property) |
---|
1622 | { |
---|
1623 | if (property == null) |
---|
1624 | throw new ArgumentNullException("property"); |
---|
1625 | if (parentType == null) |
---|
1626 | throw new ArgumentNullException("parentType"); |
---|
1627 | DefaultUnresolvedProperty p = new DefaultUnresolvedProperty(parentType, property.Name); |
---|
1628 | p.SymbolKind = propertyType; |
---|
1629 | TranslateModifiers(property.GetMethod ?? property.SetMethod, p); |
---|
1630 | if (property.GetMethod != null && property.SetMethod != null) |
---|
1631 | p.Accessibility = MergePropertyAccessibility (GetAccessibility (property.GetMethod.Attributes), GetAccessibility (property.SetMethod.Attributes)); |
---|
1632 | |
---|
1633 | p.ReturnType = ReadTypeReference(property.PropertyType, typeAttributes: property); |
---|
1634 | |
---|
1635 | p.Getter = ReadMethod(property.GetMethod, parentType, SymbolKind.Accessor, p); |
---|
1636 | p.Setter = ReadMethod(property.SetMethod, parentType, SymbolKind.Accessor, p); |
---|
1637 | |
---|
1638 | if (property.HasParameters) { |
---|
1639 | foreach (ParameterDefinition par in property.Parameters) { |
---|
1640 | p.Parameters.Add(ReadParameter(par)); |
---|
1641 | } |
---|
1642 | } |
---|
1643 | AddAttributes(property, p); |
---|
1644 | |
---|
1645 | var accessor = p.Getter ?? p.Setter; |
---|
1646 | if (accessor != null && accessor.IsExplicitInterfaceImplementation) { |
---|
1647 | p.Name = property.Name.Substring(property.Name.LastIndexOf('.') + 1); |
---|
1648 | p.IsExplicitInterfaceImplementation = true; |
---|
1649 | foreach (var mr in accessor.ExplicitInterfaceImplementations) { |
---|
1650 | p.ExplicitInterfaceImplementations.Add(new AccessorOwnerMemberReference(mr)); |
---|
1651 | } |
---|
1652 | } |
---|
1653 | |
---|
1654 | FinishReadMember(p, property); |
---|
1655 | return p; |
---|
1656 | } |
---|
1657 | #endregion |
---|
1658 | |
---|
1659 | #region Read Event |
---|
1660 | [CLSCompliant(false)] |
---|
1661 | public IUnresolvedEvent ReadEvent(EventDefinition ev, IUnresolvedTypeDefinition parentType) |
---|
1662 | { |
---|
1663 | if (ev == null) |
---|
1664 | throw new ArgumentNullException("ev"); |
---|
1665 | if (parentType == null) |
---|
1666 | throw new ArgumentNullException("parentType"); |
---|
1667 | |
---|
1668 | DefaultUnresolvedEvent e = new DefaultUnresolvedEvent(parentType, ev.Name); |
---|
1669 | TranslateModifiers(ev.AddMethod, e); |
---|
1670 | e.ReturnType = ReadTypeReference(ev.EventType, typeAttributes: ev); |
---|
1671 | |
---|
1672 | e.AddAccessor = ReadMethod(ev.AddMethod, parentType, SymbolKind.Accessor, e); |
---|
1673 | e.RemoveAccessor = ReadMethod(ev.RemoveMethod, parentType, SymbolKind.Accessor, e); |
---|
1674 | e.InvokeAccessor = ReadMethod(ev.InvokeMethod, parentType, SymbolKind.Accessor, e); |
---|
1675 | |
---|
1676 | AddAttributes(ev, e); |
---|
1677 | |
---|
1678 | var accessor = e.AddAccessor ?? e.RemoveAccessor ?? e.InvokeAccessor; |
---|
1679 | if (accessor != null && accessor.IsExplicitInterfaceImplementation) { |
---|
1680 | e.Name = ev.Name.Substring(ev.Name.LastIndexOf('.') + 1); |
---|
1681 | e.IsExplicitInterfaceImplementation = true; |
---|
1682 | foreach (var mr in accessor.ExplicitInterfaceImplementations) { |
---|
1683 | e.ExplicitInterfaceImplementations.Add(new AccessorOwnerMemberReference(mr)); |
---|
1684 | } |
---|
1685 | } |
---|
1686 | |
---|
1687 | FinishReadMember(e, ev); |
---|
1688 | |
---|
1689 | return e; |
---|
1690 | } |
---|
1691 | #endregion |
---|
1692 | |
---|
1693 | #region FinishReadMember / Interning |
---|
1694 | void FinishReadMember(AbstractUnresolvedMember member, MemberReference cecilDefinition) |
---|
1695 | { |
---|
1696 | member.ApplyInterningProvider(interningProvider); |
---|
1697 | member.Freeze(); |
---|
1698 | RegisterCecilObject(member, cecilDefinition); |
---|
1699 | } |
---|
1700 | #endregion |
---|
1701 | |
---|
1702 | #region Type system translation table |
---|
1703 | readonly Dictionary<object, object> typeSystemTranslationTable; |
---|
1704 | |
---|
1705 | void RegisterCecilObject(IUnresolvedEntity typeSystemObject, MemberReference cecilObject) |
---|
1706 | { |
---|
1707 | if (OnEntityLoaded != null) |
---|
1708 | OnEntityLoaded(typeSystemObject, cecilObject); |
---|
1709 | |
---|
1710 | AddToTypeSystemTranslationTable(typeSystemObject, cecilObject); |
---|
1711 | } |
---|
1712 | |
---|
1713 | void AddToTypeSystemTranslationTable(object typeSystemObject, object cecilObject) |
---|
1714 | { |
---|
1715 | if (typeSystemTranslationTable != null) { |
---|
1716 | // When lazy-loading, the dictionary might be shared between multiple cecil-loaders that are used concurrently |
---|
1717 | lock (typeSystemTranslationTable) { |
---|
1718 | typeSystemTranslationTable[typeSystemObject] = cecilObject; |
---|
1719 | } |
---|
1720 | } |
---|
1721 | } |
---|
1722 | |
---|
1723 | T InternalGetCecilObject<T> (object typeSystemObject) where T : class |
---|
1724 | { |
---|
1725 | if (typeSystemObject == null) |
---|
1726 | throw new ArgumentNullException ("typeSystemObject"); |
---|
1727 | if (!HasCecilReferences) |
---|
1728 | throw new NotSupportedException ("This instance contains no cecil references."); |
---|
1729 | object result; |
---|
1730 | lock (typeSystemTranslationTable) { |
---|
1731 | if (!typeSystemTranslationTable.TryGetValue (typeSystemObject, out result)) |
---|
1732 | return null; |
---|
1733 | } |
---|
1734 | return result as T; |
---|
1735 | } |
---|
1736 | |
---|
1737 | [CLSCompliant(false)] |
---|
1738 | public AssemblyDefinition GetCecilObject (IUnresolvedAssembly content) |
---|
1739 | { |
---|
1740 | return InternalGetCecilObject<AssemblyDefinition> (content); |
---|
1741 | } |
---|
1742 | |
---|
1743 | [CLSCompliant(false)] |
---|
1744 | public TypeDefinition GetCecilObject (IUnresolvedTypeDefinition type) |
---|
1745 | { |
---|
1746 | if (type == null) |
---|
1747 | throw new ArgumentNullException ("type"); |
---|
1748 | return InternalGetCecilObject<TypeDefinition> (type); |
---|
1749 | } |
---|
1750 | |
---|
1751 | [CLSCompliant(false)] |
---|
1752 | public MethodDefinition GetCecilObject (IUnresolvedMethod method) |
---|
1753 | { |
---|
1754 | return InternalGetCecilObject<MethodDefinition> (method); |
---|
1755 | } |
---|
1756 | |
---|
1757 | [CLSCompliant(false)] |
---|
1758 | public FieldDefinition GetCecilObject (IUnresolvedField field) |
---|
1759 | { |
---|
1760 | return InternalGetCecilObject<FieldDefinition> (field); |
---|
1761 | } |
---|
1762 | |
---|
1763 | [CLSCompliant(false)] |
---|
1764 | public EventDefinition GetCecilObject (IUnresolvedEvent evt) |
---|
1765 | { |
---|
1766 | return InternalGetCecilObject<EventDefinition> (evt); |
---|
1767 | } |
---|
1768 | |
---|
1769 | [CLSCompliant(false)] |
---|
1770 | public PropertyDefinition GetCecilObject (IUnresolvedProperty property) |
---|
1771 | { |
---|
1772 | return InternalGetCecilObject<PropertyDefinition> (property); |
---|
1773 | } |
---|
1774 | #endregion |
---|
1775 | } |
---|
1776 | } |
---|