5 // Jb Evain (jbevain@novell.com)
7 // (C) 2007 Novell, Inc.
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 using System.Collections;
31 using System.Collections.Generic;
33 using System.IO.Compression;
38 namespace Mono.Linker {
40 public class AnnotationStore {
42 readonly Dictionary<AssemblyDefinition, AssemblyAction> assembly_actions = new Dictionary<AssemblyDefinition, AssemblyAction> ();
43 readonly Dictionary<MethodDefinition, MethodAction> method_actions = new Dictionary<MethodDefinition, MethodAction> ();
44 readonly HashSet<IMetadataTokenProvider> marked = new HashSet<IMetadataTokenProvider> ();
45 readonly HashSet<IMetadataTokenProvider> processed = new HashSet<IMetadataTokenProvider> ();
46 readonly Dictionary<TypeDefinition, TypePreserve> preserved_types = new Dictionary<TypeDefinition, TypePreserve> ();
47 readonly Dictionary<IMemberDefinition, List<MethodDefinition>> preserved_methods = new Dictionary<IMemberDefinition, List<MethodDefinition>> ();
48 readonly HashSet<IMetadataTokenProvider> public_api = new HashSet<IMetadataTokenProvider> ();
49 readonly Dictionary<MethodDefinition, List<MethodDefinition>> override_methods = new Dictionary<MethodDefinition, List<MethodDefinition>> ();
50 readonly Dictionary<MethodDefinition, List<MethodDefinition>> base_methods = new Dictionary<MethodDefinition, List<MethodDefinition>> ();
51 readonly Dictionary<AssemblyDefinition, ISymbolReader> symbol_readers = new Dictionary<AssemblyDefinition, ISymbolReader> ();
53 readonly Dictionary<object, Dictionary<IMetadataTokenProvider, object>> custom_annotations = new Dictionary<object, Dictionary<IMetadataTokenProvider, object>> ();
55 readonly Stack<object> dependency_stack = new Stack<object> ();
56 System.Xml.XmlWriter writer;
59 public void PrepareDependenciesDump ()
61 System.Xml.XmlWriterSettings settings = new System.Xml.XmlWriterSettings();
62 settings.Indent = true;
63 settings.IndentChars = "\t";
64 var depsFile = File.OpenWrite ("linker-dependencies.xml.gz");
65 zipStream = new GZipStream (depsFile, CompressionMode.Compress);
67 writer = System.Xml.XmlWriter.Create (zipStream, settings);
68 writer.WriteStartDocument ();
69 writer.WriteStartElement ("dependencies");
70 writer.WriteStartAttribute ("version");
71 writer.WriteString ("1.0");
72 writer.WriteEndAttribute ();
75 public AssemblyAction GetAction (AssemblyDefinition assembly)
77 AssemblyAction action;
78 if (assembly_actions.TryGetValue (assembly, out action))
81 throw new NotSupportedException ();
84 public MethodAction GetAction (MethodDefinition method)
87 if (method_actions.TryGetValue (method, out action))
90 return MethodAction.Nothing;
93 public void SetAction (AssemblyDefinition assembly, AssemblyAction action)
95 assembly_actions [assembly] = action;
98 public bool HasAction (AssemblyDefinition assembly)
100 return assembly_actions.ContainsKey (assembly);
103 public void SetAction (MethodDefinition method, MethodAction action)
105 method_actions [method] = action;
108 public void Mark (IMetadataTokenProvider provider)
110 marked.Add (provider);
111 AddDependency (provider);
114 public bool IsMarked (IMetadataTokenProvider provider)
116 return marked.Contains (provider);
119 public void Processed (IMetadataTokenProvider provider)
121 processed.Add (provider);
124 public bool IsProcessed (IMetadataTokenProvider provider)
126 return processed.Contains (provider);
129 public bool IsPreserved (TypeDefinition type)
131 return preserved_types.ContainsKey (type);
134 public void SetPreserve (TypeDefinition type, TypePreserve preserve)
136 preserved_types [type] = preserve;
139 public TypePreserve GetPreserve (TypeDefinition type)
141 TypePreserve preserve;
142 if (preserved_types.TryGetValue (type, out preserve))
145 throw new NotSupportedException ();
148 public void SetPublic (IMetadataTokenProvider provider)
150 public_api.Add (provider);
153 public bool IsPublic (IMetadataTokenProvider provider)
155 return public_api.Contains (provider);
158 public void AddOverride (MethodDefinition @base, MethodDefinition @override)
160 var methods = GetOverrides (@base);
161 if (methods == null) {
162 methods = new List<MethodDefinition> ();
163 override_methods [@base] = methods;
166 methods.Add (@override);
169 public List<MethodDefinition> GetOverrides (MethodDefinition method)
171 List<MethodDefinition> overrides;
172 if (override_methods.TryGetValue (method, out overrides))
178 public void AddBaseMethod (MethodDefinition method, MethodDefinition @base)
180 var methods = GetBaseMethods (method);
181 if (methods == null) {
182 methods = new List<MethodDefinition> ();
183 base_methods [method] = methods;
189 public List<MethodDefinition> GetBaseMethods (MethodDefinition method)
191 List<MethodDefinition> bases;
192 if (base_methods.TryGetValue (method, out bases))
198 public List<MethodDefinition> GetPreservedMethods (TypeDefinition type)
200 return GetPreservedMethods (type as IMemberDefinition);
203 public void AddPreservedMethod (TypeDefinition type, MethodDefinition method)
205 AddPreservedMethod (type as IMemberDefinition, method);
208 public List<MethodDefinition> GetPreservedMethods (MethodDefinition method)
210 return GetPreservedMethods (method as IMemberDefinition);
213 public void AddPreservedMethod (MethodDefinition key, MethodDefinition method)
215 AddPreservedMethod (key as IMemberDefinition, method);
218 List<MethodDefinition> GetPreservedMethods (IMemberDefinition definition)
220 List<MethodDefinition> preserved;
221 if (preserved_methods.TryGetValue (definition, out preserved))
227 void AddPreservedMethod (IMemberDefinition definition, MethodDefinition method)
229 var methods = GetPreservedMethods (definition);
230 if (methods == null) {
231 methods = new List<MethodDefinition> ();
232 preserved_methods [definition] = methods;
235 methods.Add (method);
238 public void AddSymbolReader (AssemblyDefinition assembly, ISymbolReader symbolReader)
240 symbol_readers [assembly] = symbolReader;
243 public void CloseSymbolReader (AssemblyDefinition assembly)
245 ISymbolReader symbolReader;
246 if (!symbol_readers.TryGetValue (assembly, out symbolReader))
249 symbol_readers.Remove (assembly);
250 symbolReader.Dispose ();
253 public Dictionary<IMetadataTokenProvider, object> GetCustomAnnotations (object key)
255 Dictionary<IMetadataTokenProvider, object> slots;
256 if (custom_annotations.TryGetValue (key, out slots))
259 slots = new Dictionary<IMetadataTokenProvider, object> ();
260 custom_annotations.Add (key, slots);
264 public void AddDependency (object o)
269 KeyValuePair<object, object> pair = new KeyValuePair<object, object> (dependency_stack.Count > 0 ? dependency_stack.Peek () : null, o);
270 writer.WriteStartElement ("edge");
271 writer.WriteAttributeString ("b", TokenString (pair.Key));
272 writer.WriteAttributeString ("e", TokenString (pair.Value));
273 writer.WriteEndElement ();
276 public void Push (object o)
281 if (dependency_stack.Count > 0)
283 dependency_stack.Push (o);
291 dependency_stack.Pop ();
294 string TokenString (object o)
299 if (o is IMetadataTokenProvider)
300 return (o as IMetadataTokenProvider).MetadataToken.TokenType + ":" + o;
305 public void SaveDependencies ()
310 writer.WriteEndElement ();
311 writer.WriteEndDocument ();
317 zipStream.Dispose ();