You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			101 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			101 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 		Handling GC allocated objects in C
 | |
| 
 | |
| As part of an effort to improve our GC, we need to keep track
 | |
| precisely of where objects are stored, so we can incrementally move
 | |
| from the current conservative GC to a more advanced precise and moving GC.
 | |
| Previously, all global C variables were considered GC roots, but this makes
 | |
| the GC less efficient and increases the chances false references are found
 | |
| to GC memory, hence retaining more memory than needed.
 | |
| We need to tell the GC that some object is supposed to be kept alive
 | |
| as if it was referenced in a global variable.
 | |
| 
 | |
| For Mono embedders
 | |
| ------------------
 | |
| 
 | |
| In C#, if you say:
 | |
| class T {
 | |
| 	static object o;
 | |
| }
 | |
| 
 | |
| Any object which is stored in `o' is considered to be alive -- it will
 | |
| not be collected. `o' is a member of the root set for the GC.
 | |
| 
 | |
| However, in C code, this is not the case. If you have:
 | |
| 
 | |
| static MonoObject* o = NULL;
 | |
| 
 | |
| The object in `o' will *NOT* be scanned.
 | |
| 
 | |
| If you need to store an object in a C variable and prevent it from being 
 | |
| collected, you need to acquire a GC handle for it.
 | |
| 
 | |
| 	guint32 handle = mono_gchandle_new (my_object, TRUE);
 | |
| 
 | |
| TRUE means the object will be pinned, so it won't move in memory 
 | |
| when we'll use a moving GC. You can access the MonoObject* referenced by
 | |
| a handle with:
 | |
| 
 | |
| 	MonoObject* obj = mono_gchandle_get_target (handle);
 | |
| 
 | |
| When you don't need the handle anymore you need to call:
 | |
| 
 | |
| 	mono_gchandle_free (handle);
 | |
| 
 | |
| Note that if you assign a new object to the C var, you need to get a new 
 | |
| handle, it's not enough to store a new object in the C var.
 | |
| 
 | |
| So code that looked like this:
 | |
| 
 | |
| 	static MonoObject* o = NULL;
 | |
| 	...
 | |
| 	o = mono_object_new (...);
 | |
| 	/* use o */
 | |
| 	...
 | |
| 	/* when done to allow the GC to collect o */
 | |
| 	o = NULL;
 | |
| 
 | |
| should now be changed to:
 | |
| 
 | |
| 	static guint32 o_handle;
 | |
| 	...
 | |
| 	MonoObject *o = mono_object_new (...);
 | |
| 	o_handle = mono_gchandle_new (o, TRUE);
 | |
| 	/* use o or mono_gchandle_get_target (o_handle) */
 | |
| 	...
 | |
| 	/* when done to allow the GC to collect o */
 | |
| 	mono_gchandle_free (o_handle);
 | |
| 
 | |
| 
 | |
| For Mono runtime developers
 | |
| ---------------------------
 | |
| 
 | |
| There are two kinds of static vars used to store pointers to GC memory 
 | |
| that we need to consider:
 | |
| *) objects
 | |
| *) other memory chunks allocated with GC_MALLOC().
 | |
| 
 | |
| Objects should be dealt with the GC handle support as detailed above.
 | |
| Other items should register the static pointer as an area to be considered
 | |
| part of the root set with the following:
 | |
| 
 | |
| 	static gpointer my_gc_data = NULL;
 | |
| 	...
 | |
| 	MONO_GC_REGISTER_ROOT (my_gc_data);
 | |
| 	my_gc_data = GC_MALLOC (...);
 | |
| 	
 | |
| Note that this registration is not necessary for *LOCAL* variables,
 | |
| as they are stored on the stack. It is only necessary for global variables,
 | |
| as they are not a part of the GC's root set.
 | |
| 
 | |
| Once you have done the MONO_GC_REGISTER_ROOT, the variable is just like
 | |
| a static variable in C#. To keep an object alive, you have the variable reference
 | |
| the GC memory, to remove the reference, set the variable to NULL.
 | |
| 
 | |
| As we prepare the code for a precise GC, GC_MALLOC () will not be used anymore
 | |
| in this way in most cases: we'll have a mechanism to specify exactly where
 | |
| references to GC memory is stored. This mechanism is now available with the new GC,
 | |
| see usages of mono_gc_alloc_fixed (), mono_gc_register_root () and 
 | |
| mono_gc_make_descr_from_bitmap().
 | |
| See also docs/precise-gc for additional info.
 | |
| 
 |