2009-12-02 Jb Evain <jbevain@novell.com>
[mcs.git] / mcs / rootcontext.cs
blob287943898b97f16fdfa71fa34f978bac07492364
1 //
2 // rootcontext.cs: keeps track of our tree representation, and assemblies loaded.
3 //
4 // Author: Miguel de Icaza (miguel@ximian.com)
5 // Ravi Pratap (ravi@ximian.com)
6 // Marek Safar (marek.safar@gmail.com)
7 //
8 //
9 // Dual licensed under the terms of the MIT X11 or GNU GPL
11 // Copyright 2001 Ximian, Inc (http://www.ximian.com)
12 // Copyright 2004-2008 Novell, Inc
14 using System;
15 using System.Collections;
16 using System.Reflection;
17 using System.Reflection.Emit;
18 using System.Diagnostics;
20 namespace Mono.CSharp {
22 public enum LanguageVersion
24 ISO_1 = 1,
25 ISO_2 = 2,
26 V_3 = 3,
27 V_4 = 4,
28 Future = 100,
30 Default = LanguageVersion.V_4,
33 public enum MetadataVersion
35 v1,
36 v2,
40 public class RootContext {
43 // COMPILER OPTIONS CLASS
45 public static Target Target;
46 public static Platform Platform;
47 public static string TargetExt;
48 public static bool VerifyClsCompliance = true;
49 public static bool Optimize = true;
50 public static LanguageVersion Version;
52 public static MetadataVersion MetadataCompatibilityVersion;
55 // We keep strongname related info here because
56 // it's also used as complier options from CSC 8.x
58 public static string StrongNameKeyFile;
59 public static string StrongNameKeyContainer;
60 public static bool StrongNameDelaySign;
63 // If set, enable XML documentation generation
65 public static Documentation Documentation;
67 static public string MainClass;
69 //
70 // The default compiler checked state
72 static public bool Checked;
75 // If true, it means that the compiler is executing as
76 // in eval mode so unresolved variables are resolved in
77 // static classes maintained by the eval engine.
79 static public bool EvalMode;
82 // If true, the compiler is operating in statement mode,
83 // this currently turns local variable declaration into
84 // static variables of a class
86 static public bool StatementMode;
89 // Whether to allow Unsafe code
91 static public bool Unsafe;
94 // Whether we are being linked against the standard libraries.
95 // This is only used to tell whether `System.Object' should
96 // have a base class or not.
98 public static bool StdLib;
100 public static bool NeedsEntryPoint {
101 get { return Target == Target.Exe || Target == Target.WinExe; }
105 // COMPILER OPTIONS CLASS END
109 // Contains the parsed tree
111 static ModuleCompiled root;
114 // This hashtable contains all of the #definitions across the source code
115 // it is used by the ConditionalAttribute handler.
117 static ArrayList AllDefines;
120 // This keeps track of the order in which classes were defined
121 // so that we can poulate them in that order.
123 // Order is important, because we need to be able to tell, by
124 // examining the list of methods of the base class, which ones are virtual
125 // or abstract as well as the parent names (to implement new,
126 // override).
128 static ArrayList type_container_resolve_order;
131 // Holds a reference to the Private Implementation Details
132 // class.
134 static ArrayList helper_classes;
136 static TypeBuilder impl_details_class;
139 // Constructor
141 static RootContext ()
143 Reset (true);
146 public static void PartialReset ()
148 Reset (false);
151 public static void Reset (bool full)
153 if (full)
154 root = null;
156 type_container_resolve_order = new ArrayList ();
157 EntryPoint = null;
158 Checked = false;
159 Unsafe = false;
160 StdLib = true;
161 StrongNameKeyFile = null;
162 StrongNameKeyContainer = null;
163 StrongNameDelaySign = false;
164 MainClass = null;
165 Target = Target.Exe;
166 TargetExt = ".exe";
167 Platform = Platform.AnyCPU;
168 Version = LanguageVersion.Default;
169 Documentation = null;
170 impl_details_class = null;
171 helper_classes = null;
173 #if NET_4_0
174 MetadataCompatibilityVersion = MetadataVersion.v4;
175 #else
176 MetadataCompatibilityVersion = MetadataVersion.v2;
177 #endif
180 // Setup default defines
182 AllDefines = new ArrayList ();
183 AddConditional ("__MonoCS__");
186 public static void AddConditional (string p)
188 if (AllDefines.Contains (p))
189 return;
190 AllDefines.Add (p);
193 public static bool IsConditionalDefined (string value)
195 return AllDefines.Contains (value);
198 static public ModuleCompiled ToplevelTypes {
199 get { return root; }
200 set { root = value; }
203 public static void RegisterOrder (TypeContainer tc)
205 type_container_resolve_order.Add (tc);
208 // <remarks>
209 // This function is used to resolve the hierarchy tree.
210 // It processes interfaces, structs and classes in that order.
212 // It creates the TypeBuilder's as it processes the user defined
213 // types.
214 // </remarks>
215 static public void ResolveTree ()
217 root.Resolve ();
220 // Interfaces are processed next, as classes and
221 // structs might inherit from an object or implement
222 // a set of interfaces, we need to be able to tell
223 // them appart by just using the TypeManager.
225 foreach (TypeContainer tc in root.Types)
226 tc.CreateType ();
228 foreach (TypeContainer tc in root.Types)
229 tc.DefineType ();
231 if (root.Delegates != null)
232 foreach (Delegate d in root.Delegates)
233 d.DefineType ();
236 // <summary>
237 // Closes all open types
238 // </summary>
240 // <remarks>
241 // We usually use TypeBuilder types. When we are done
242 // creating the type (which will happen after we have added
243 // methods, fields, etc) we need to "Define" them before we
244 // can save the Assembly
245 // </remarks>
246 static public void CloseTypes ()
249 // We do this in two passes, first we close the structs,
250 // then the classes, because it seems the code needs it this
251 // way. If this is really what is going on, we should probably
252 // make sure that we define the structs in order as well.
254 foreach (TypeContainer tc in type_container_resolve_order){
255 if (tc.Kind == Kind.Struct && tc.Parent == root){
256 tc.CloseType ();
260 foreach (TypeContainer tc in type_container_resolve_order){
261 if (!(tc.Kind == Kind.Struct && tc.Parent == root))
262 tc.CloseType ();
265 if (root.Delegates != null)
266 foreach (Delegate d in root.Delegates)
267 d.CloseType ();
271 // If we have a <PrivateImplementationDetails> class, close it
273 if (helper_classes != null){
274 foreach (TypeBuilder type_builder in helper_classes) {
275 PredefinedAttributes.Get.CompilerGenerated.EmitAttribute (type_builder);
276 type_builder.CreateType ();
280 type_container_resolve_order = null;
281 helper_classes = null;
282 //root = null;
283 TypeManager.CleanUp ();
286 /// <summary>
287 /// Used to register classes that need to be closed after all the
288 /// user defined classes
289 /// </summary>
290 public static void RegisterCompilerGeneratedType (TypeBuilder helper_class)
292 if (helper_classes == null)
293 helper_classes = new ArrayList ();
295 helper_classes.Add (helper_class);
298 static public DeclSpace PopulateCoreType (TypeContainer root, string name)
300 DeclSpace ds = (DeclSpace) root.GetDefinition (name);
301 // Core type was imported
302 if (ds == null)
303 return null;
305 ds.Define ();
306 return ds;
309 static public void BootCorlib_PopulateCoreTypes ()
311 // Clear -nostdlib flag when object type is imported
312 if (PopulateCoreType (root, "System.Object") == null)
313 RootContext.StdLib = true;
315 PopulateCoreType (root, "System.ValueType");
316 PopulateCoreType (root, "System.Attribute");
317 PopulateCoreType (root, "System.Runtime.CompilerServices.IndexerNameAttribute");
320 // <summary>
321 // Populates the structs and classes with fields and methods
322 // </summary>
324 // This is invoked after all interfaces, structs and classes
325 // have been defined through `ResolveTree'
326 static public void PopulateTypes ()
329 if (type_container_resolve_order != null){
330 foreach (TypeContainer tc in type_container_resolve_order)
331 tc.ResolveType ();
332 foreach (TypeContainer tc in type_container_resolve_order) {
333 try {
334 tc.Define ();
335 } catch (Exception e) {
336 throw new InternalErrorException (tc, e);
341 ArrayList delegates = root.Delegates;
342 if (delegates != null){
343 foreach (Delegate d in delegates)
344 d.Define ();
348 // Check for cycles in the struct layout
350 if (type_container_resolve_order != null){
351 Hashtable seen = new Hashtable ();
352 foreach (TypeContainer tc in type_container_resolve_order)
353 TypeManager.CheckStructCycles (tc, seen);
357 static public void EmitCode ()
359 if (type_container_resolve_order != null) {
360 foreach (TypeContainer tc in type_container_resolve_order)
361 tc.EmitType ();
363 if (ToplevelTypes.Compiler.Report.Errors > 0)
364 return;
366 foreach (TypeContainer tc in type_container_resolve_order)
367 tc.VerifyMembers ();
370 if (root.Delegates != null) {
371 foreach (Delegate d in root.Delegates)
372 d.Emit ();
375 CodeGen.Assembly.Emit (root);
376 root.Emit ();
380 // Public Field, used to track which method is the public entry
381 // point.
383 static public Method EntryPoint;
386 // These are used to generate unique names on the structs and fields.
388 static int field_count;
391 // Makes an initialized struct, returns the field builder that
392 // references the data. Thanks go to Sergey Chaban for researching
393 // how to do this. And coming up with a shorter mechanism than I
394 // was able to figure out.
396 // This works but makes an implicit public struct $ArrayType$SIZE and
397 // makes the fields point to it. We could get more control if we did
398 // use instead:
400 // 1. DefineNestedType on the impl_details_class with our struct.
402 // 2. Define the field on the impl_details_class
404 static public FieldBuilder MakeStaticData (byte [] data)
406 FieldBuilder fb;
408 if (impl_details_class == null){
409 impl_details_class = ToplevelTypes.Builder.DefineType (
410 "<PrivateImplementationDetails>",
411 TypeAttributes.NotPublic,
412 TypeManager.object_type);
414 RegisterCompilerGeneratedType (impl_details_class);
417 fb = impl_details_class.DefineInitializedData (
418 "$$field-" + (field_count++), data,
419 FieldAttributes.Static | FieldAttributes.Assembly);
421 return fb;
424 public static void CheckUnsafeOption (Location loc, Report Report)
426 if (!Unsafe) {
427 Report.Error (227, loc,
428 "Unsafe code requires the `unsafe' command line option to be specified");