Fix potential crash for Encoder.Convert (#20522)
[mono-project.git] / mono / sgen / sgen-client.h
blob53a1a43f45de1d2dee76d47e21993b589bad5552
1 /**
2 * \file
3 * SGen client interface.
5 * Copyright (C) 2014 Xamarin Inc
7 * Licensed under the MIT license. See LICENSE file in the project root for full license information.
8 */
10 #include "mono/sgen/sgen-pointer-queue.h"
13 * Init whatever needs initing. This is called relatively early in SGen initialization.
14 * Must initialized the small ID for the current thread.
16 void sgen_client_init (void);
19 * The slow path for getting an object's size. We're passing in the vtable because we've
20 * already fetched it.
22 mword sgen_client_slow_object_get_size (GCVTable vtable, GCObject* o);
25 * Fill the given range with a dummy object. If the range is too short to be filled with an
26 * object, null it. Return `TRUE` if the range was filled with an object, `FALSE` if it was
27 * nulled.
29 gboolean sgen_client_array_fill_range (char *start, size_t size);
32 * This is called if the nursery clearing policy at `clear-at-gc`, which is usually only
33 * used for debugging. If `size` is large enough for the memory to have been filled with a
34 * dummy, object, zero its header. Note that there might not actually be a header there.
36 void sgen_client_zero_array_fill_header (void *p, size_t size);
39 * Return whether the given object is an array fill dummy object.
41 gboolean sgen_client_object_is_array_fill (GCObject *o);
44 * Return whether the given finalizable object's finalizer is critical, i.e., needs to run
45 * after all non-critical finalizers have run.
47 gboolean sgen_client_object_has_critical_finalizer (GCObject *obj);
50 * Called after an object is enqueued for finalization. This is a very low-level callback.
51 * It should almost certainly be a NOP.
53 * FIXME: Can we merge this with `sgen_client_object_has_critical_finalizer()`?
55 void sgen_client_object_queued_for_finalization (GCObject *obj);
58 * Run the given object's finalizer.
60 void sgen_client_run_finalize (GCObject *obj);
63 * Is called after a collection if there are objects to finalize. The world is still
64 * stopped. This will usually notify the finalizer thread that it needs to run.
66 void sgen_client_finalize_notify (void);
69 * Returns TRUE if no ephemerons have been marked. Will be called again if it returned
70 * FALSE. If ephemerons are not supported, just return TRUE.
72 gboolean sgen_client_mark_ephemerons (ScanCopyContext ctx)
73 MONO_PERMIT (need (sgen_gc_locked));
76 * Clear ephemeron pairs with unreachable keys.
77 * We pass the copy func so we can figure out if an array was promoted or not.
79 void sgen_client_clear_unreachable_ephemerons (ScanCopyContext ctx)
80 MONO_PERMIT (need (sgen_gc_locked));
83 * May return NULL. Must be an aligned pointer.
85 gpointer sgen_client_default_metadata (void);
86 gpointer sgen_client_metadata_for_object (GCObject *obj);
89 * No action required.
91 void sgen_client_gchandle_created (int handle_type, GCObject *obj, guint32 handle);
92 void sgen_client_gchandle_destroyed (int handle_type, guint32 handle);
93 void sgen_client_ensure_weak_gchandles_accessible (void);
96 * This is called for objects that are larger than one card. If it's possible to scan only
97 * parts of the object based on which cards are marked, do so and return TRUE. Otherwise,
98 * return FALSE.
100 gboolean sgen_client_cardtable_scan_object (GCObject *obj, guint8 *cards, ScanCopyContext ctx);
103 * Called after nursery objects have been pinned. No action is necessary.
105 void sgen_client_nursery_objects_pinned (void **definitely_pinned, int count);
108 * Called at a semi-random point during minor collections. No action is necessary.
110 void sgen_client_collecting_minor_report_roots (SgenPointerQueue *fin_ready_queue, SgenPointerQueue *critical_fin_queue);
113 * Called at semi-random points during major collections. No action is necessary.
115 void sgen_client_collecting_major_report_roots (SgenPointerQueue *fin_ready_queue, SgenPointerQueue *critical_fin_queue);
118 * Called after a LOS object has been pinned. No action is necessary.
120 void sgen_client_pinned_los_object (GCObject *obj);
123 * Called for each cemented obj
125 void sgen_client_pinned_cemented_object (GCObject *obj);
128 * Called for each major heap obj pinned
130 void sgen_client_pinned_major_heap_object (GCObject *obj);
132 void sgen_client_pinning_start (void);
133 void sgen_client_pinning_end (void);
136 * Called for every degraded allocation. No action is necessary.
138 void sgen_client_degraded_allocation (void);
141 * Called whenever the amount of memory allocated for the managed heap changes. No action
142 * is necessary.
144 void sgen_client_total_allocated_heap_changed (size_t allocated_heap_size);
147 * If the client has registered any internal memory types, this must return a string
148 * describing the given type. Only used for debugging.
150 const char* sgen_client_description_for_internal_mem_type (int type);
153 * Only used for debugging. `sgen_client_vtable_get_namespace()` may return NULL.
155 gboolean sgen_client_vtable_is_inited (GCVTable vtable);
156 const char* sgen_client_vtable_get_namespace (GCVTable vtable);
157 const char* sgen_client_vtable_get_name (GCVTable vtable);
160 * Called before starting collections. The world is already stopped. No action is
161 * necessary.
163 void sgen_client_pre_collection_checks (void);
166 * Must set the thread's thread info to `info`. If the thread's small ID was not already
167 * initialized in `sgen_client_init()` (for the main thread, usually), it must be done here.
169 * `stack_bottom_fallback` is the value passed through via `sgen_thread_attach()`.
171 void sgen_client_thread_attach (SgenThreadInfo* info);
173 void sgen_client_thread_detach_with_lock (SgenThreadInfo *p);
176 * Called on each worker thread when it starts up. Must initialize the thread's small ID.
178 void sgen_client_thread_register_worker (void);
181 * The least this function needs to do is scan all registers and thread stacks. To do this
182 * conservatively, use `sgen_conservatively_pin_objects_from()`.
184 void sgen_client_scan_thread_data (void *start_nursery, void *end_nursery, gboolean precise, ScanCopyContext ctx);
187 * Stop and restart the world, i.e., all threads that interact with the managed heap. For
188 * single-threaded programs this is a nop.
190 void sgen_client_stop_world (int generation, gboolean serial_collection)
191 MONO_PERMIT (need (sgen_gc_locked));
192 void sgen_client_restart_world (int generation, gboolean serial_collection, gint64 *stw_time)
193 MONO_PERMIT (need (sgen_gc_locked));
196 * Must return FALSE. The bridge is not supported outside of Mono.
198 gboolean sgen_client_bridge_need_processing (void);
201 * None of these should ever be called.
203 void sgen_client_bridge_reset_data (void);
204 void sgen_client_bridge_processing_stw_step (void);
205 void sgen_client_bridge_wait_for_processing (void);
206 void sgen_client_bridge_processing_finish (int generation);
207 gboolean sgen_client_bridge_is_bridge_object (GCObject *obj);
208 void sgen_client_bridge_register_finalized_object (GCObject *object);
211 * No action is necessary.
213 void sgen_client_mark_togglerefs (char *start, char *end, ScanCopyContext ctx);
214 void sgen_client_clear_togglerefs (char *start, char *end, ScanCopyContext ctx);
215 void sgen_foreach_toggleref_root (void (*callback)(MonoObject*, gpointer), gpointer data);
218 * Called to handle `MONO_GC_PARAMS` and `MONO_GC_DEBUG` options. The `handle` functions
219 * must return TRUE if they have recognized and processed the option, FALSE otherwise.
221 gboolean sgen_client_handle_gc_param (const char *opt);
222 void sgen_client_print_gc_params_usage (void);
223 gboolean sgen_client_handle_gc_debug (const char *opt);
224 void sgen_client_print_gc_debug_usage (void);
227 * Called to obtain an identifier for the current location, such as a method pointer. This
228 * is used for logging the provenances of allocations with the heavy binary protocol.
230 gpointer sgen_client_get_provenance (void);
233 * Called by the debugging infrastructure to describe pointers that have an invalid vtable.
234 * Should usually print to `stdout`.
236 void sgen_client_describe_invalid_pointer (GCObject *ptr);
239 * Return the weak bitmap for a class
241 gsize *sgen_client_get_weak_bitmap (GCVTable vt, int *nbits);
244 * Scheduled @cv to be invoked later in the background.
246 * This function is idepotent WRT background execution. Meaning that calling it multiple times with the same funciton pointer before any bg execution happens will only call @cb once.
248 void sgen_client_schedule_background_job (void (*cb)(void));
251 * These client binary protocol functions are called from the respective binary protocol
252 * functions. No action is necessary. We suggest implementing them as inline functions in
253 * the client header file so that no overhead is incurred if they don't actually do
254 * anything.
257 #define TYPE_INT int
258 #define TYPE_LONGLONG long long
259 #define TYPE_SIZE size_t
260 #define TYPE_POINTER gpointer
261 #define TYPE_BOOL gboolean
263 #define BEGIN_PROTOCOL_ENTRY0(method) \
264 void sgen_client_ ## method (void);
265 #define BEGIN_PROTOCOL_ENTRY_HEAVY0(method) \
266 void sgen_client_ ## method (void);
267 #define BEGIN_PROTOCOL_ENTRY1(method,t1,f1) \
268 void sgen_client_ ## method (t1 f1);
269 #define BEGIN_PROTOCOL_ENTRY_HEAVY1(method,t1,f1) \
270 void sgen_client_ ## method (t1 f1);
271 #define BEGIN_PROTOCOL_ENTRY2(method,t1,f1,t2,f2) \
272 void sgen_client_ ## method (t1 f1, t2 f2);
273 #define BEGIN_PROTOCOL_ENTRY_HEAVY2(method,t1,f1,t2,f2) \
274 void sgen_client_ ## method (t1 f1, t2 f2);
275 #define BEGIN_PROTOCOL_ENTRY3(method,t1,f1,t2,f2,t3,f3) \
276 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3);
277 #define BEGIN_PROTOCOL_ENTRY_HEAVY3(method,t1,f1,t2,f2,t3,f3) \
278 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3);
279 #define BEGIN_PROTOCOL_ENTRY4(method,t1,f1,t2,f2,t3,f3,t4,f4) \
280 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4);
281 #define BEGIN_PROTOCOL_ENTRY_HEAVY4(method,t1,f1,t2,f2,t3,f3,t4,f4) \
282 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4);
283 #define BEGIN_PROTOCOL_ENTRY5(method,t1,f1,t2,f2,t3,f3,t4,f4,t5,f5) \
284 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4, t5 f5);
285 #define BEGIN_PROTOCOL_ENTRY_HEAVY5(method,t1,f1,t2,f2,t3,f3,t4,f4,t5,f5) \
286 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4, t5 f5);
287 #define BEGIN_PROTOCOL_ENTRY6(method,t1,f1,t2,f2,t3,f3,t4,f4,t5,f5,t6,f6) \
288 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4, t5 f5, t6 f6);
289 #define BEGIN_PROTOCOL_ENTRY_HEAVY6(method,t1,f1,t2,f2,t3,f3,t4,f4,t5,f5,t6,f6) \
290 void sgen_client_ ## method (t1 f1, t2 f2, t3 f3, t4 f4, t5 f5, t6 f6);
292 #define DEFAULT_PRINT()
293 #define CUSTOM_PRINT(_)
295 #define IS_ALWAYS_MATCH(_)
296 #define MATCH_INDEX(_)
297 #define IS_VTABLE_MATCH(_)
299 #define END_PROTOCOL_ENTRY
300 #define END_PROTOCOL_ENTRY_FLUSH
301 #define END_PROTOCOL_ENTRY_HEAVY
303 #include "sgen-protocol-def.h"
305 #undef TYPE_INT
306 #undef TYPE_LONGLONG
307 #undef TYPE_SIZE
308 #undef TYPE_POINTER
309 #undef TYPE_BOOL
311 #ifdef SGEN_WITHOUT_MONO
313 * Get the current thread's thread info. This will only be called on managed threads.
315 SgenThreadInfo* mono_thread_info_current (void);
318 * Get the current thread's small ID. This will be called on managed and worker threads.
320 int mono_thread_info_get_small_id (void);
322 #endif