1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235
|
/**************************************************************************/
/* */
/* OCaml */
/* */
/* KC Sivaramakrishnan, Indian Institute of Technology, Madras */
/* Stephen Dolan, University of Cambridge */
/* */
/* Copyright 2019 Indian Institute of Technology, Madras */
/* Copyright 2019 University of Cambridge */
/* */
/* All rights reserved. This file is distributed under the terms of */
/* the GNU Lesser General Public License version 2.1, with the */
/* special exception on linking described in the file LICENSE. */
/* */
/**************************************************************************/
#ifndef CAML_DOMAIN_H
#define CAML_DOMAIN_H
#ifdef CAML_INTERNALS
#include "camlatomic.h"
#include "config.h"
#include "mlvalues.h"
#include "domain_state.h"
#ifdef ARCH_SIXTYFOUR
#define Max_domains_def 128
#else
#define Max_domains_def 16
#endif
/* Upper limit for the number of domains. Chosen to be arbitrarily large. Used
* for sanity checking [max_domains] value in OCAMLRUNPARAM. */
#define Max_domains_max 4096
/* is the minor heap full or an external interrupt has been triggered */
Caml_inline int caml_check_gc_interrupt(caml_domain_state * dom_st)
{
CAMLalloc_point_here;
uintnat young_limit = atomic_load_relaxed(&dom_st->young_limit);
if ((uintnat)dom_st->young_ptr < young_limit) {
/* Synchronise for the case when [young_limit] was used to interrupt
us. */
atomic_thread_fence(memory_order_acquire);
return 1;
}
return 0;
}
#define Caml_check_gc_interrupt(dom_st) \
(CAMLunlikely(caml_check_gc_interrupt(dom_st)))
asize_t caml_norm_minor_heap_size (intnat);
int caml_reallocate_minor_heap(asize_t);
void caml_update_minor_heap_max(uintnat minor_heap_wsz);
/* is there a STW interrupt queued that needs servicing */
int caml_incoming_interrupts_queued(void);
void caml_poll_gc_work(void);
void caml_handle_gc_interrupt(void);
void caml_process_external_interrupt(void);
void caml_handle_incoming_interrupts(void);
CAMLextern void caml_interrupt_self(void);
void caml_interrupt_all_signal_safe(void);
void caml_reset_young_limit(caml_domain_state *);
void caml_update_young_limit_after_c_call(caml_domain_state *);
CAMLextern void caml_reset_domain_lock(void);
CAMLextern int caml_bt_is_in_blocking_section(void);
CAMLextern int caml_bt_is_self(void);
CAMLextern intnat caml_domain_is_multicore (void);
CAMLextern void caml_bt_enter_ocaml(void);
CAMLextern void caml_bt_exit_ocaml(void);
CAMLextern void caml_acquire_domain_lock(void);
CAMLextern void caml_release_domain_lock(void);
/* These hooks are not modified after other domains are spawned. */
CAMLextern void (*caml_atfork_hook)(void);
CAMLextern void (*caml_domain_initialize_hook)(void);
CAMLextern void (*caml_domain_stop_hook)(void);
CAMLextern void (*caml_domain_external_interrupt_hook)(void);
CAMLextern void caml_init_domains(uintnat max_domains, uintnat minor_heap_wsz);
CAMLextern void caml_init_domain_self(int);
CAMLextern uintnat caml_minor_heap_max_wsz;
CAMLextern atomic_uintnat caml_num_domains_running;
Caml_inline intnat caml_domain_alone(void)
{
return atomic_load_acquire(&caml_num_domains_running) == 1;
}
/* The index of the current domain. It is an integer unique among
currently-running domains, in the interval [0; N-1] where N is the
peak number of domains running simultaneously so far. The index of
a terminated domain may be reused for a new domain.
This function requires the domain lock to be held.
*/
Caml_inline int caml_domain_index(void)
{
return Caml_state->id;
}
#ifdef DEBUG
int caml_domain_is_in_stw(void);
#endif
int caml_domain_terminating(caml_domain_state *);
int caml_domain_is_terminating(void);
int caml_try_run_on_all_domains_with_spin_work(
int sync,
void (*handler)(caml_domain_state*, void*, int, caml_domain_state**),
void* data,
void (*leader_setup)(caml_domain_state*),
/* return nonzero if there may still be useful work to do while spinning */
int (*enter_spin_callback)(caml_domain_state*, void*),
void* enter_spin_data);
int caml_try_run_on_all_domains(
void (*handler)(caml_domain_state*, void*, int, caml_domain_state**),
void*,
void (*leader_setup)(caml_domain_state*));
/* Function naming conventions for STW callbacks and STW critical sections.
A "STW callback" is a callback passed to one of the
[caml_try_run_on_all_domains*] runners, it will
run on all participant domains in parallel.
The "STW critical section" is the runtime interval between the
start of the execution of the STW callback and the last barrier in
the callback. During this interval, mutator code from registered
participants cannot be running in parallel.
Note:
- Some parts of a STW callback are *not* inside the STW critical
section: all the code after the last barrier, or all the callback
if it does not contain a barrier.
- Program initialization can be considered as a STW critical
section as well, when no mutators or domains are running yet.
Some functions must be called within a STW critical section only,
calling then in a less-synchronized context introduces races with
mutators. To avoid these mistakes we use naming conventions as
a barebones effect system.
1. [stw_*] prefix for STW callbacks.
A function that defines a STW callback starts with [stw_] or
[caml_stw_]. It is passed to the [caml_try_run_on_all_domains*]
runner.
Examples:
- [caml_stw_empty_minor_heap] is a STW callback that empties the
minor heap
- [stw_resize_minor_heap_reservation] is a STW callback that
resizes the memory reservation for the minor heap
2. [*_from_stw] suffix for auxiliary functions that may only be
called within a STW critical section.
3. [*_from_stw_single] suffix for auxiliary functions that may only
be called within a STW critical section, and only by a single
domain at a time -- typically the last one entering a barrier.
5. No [stw] in the name for functions that are not called in a STW
callback, in particular functions that themselves start a STW
context by calling a [caml_try_run_on_all_domains*].
We could consider a [*_outside_stw] suffix for functions that must
not be called inside a STW callback, but it is generally not
necessary to enforce this discipline in the function name.
*/
/* Barriers */
/* Get the number of parties expected to arrive into the barrier, i.e. the
number of domains participating in the STW section. In most cases the barrier
is used directly from an STW callback that already has the number of
participating domains at hand, which should be used instead. */
int caml_global_barrier_num_participating(void);
/* Unconditionally arrive at the barrier and wait for all parties,
[caml_global_barrier] below should be used instead. */
void caml_enter_global_barrier(int num_participating);
/* Arrive at the barrier and wait iff there is more than one party */
Caml_inline void caml_global_barrier(int num_participating) {
if (num_participating != 1) caml_enter_global_barrier(num_participating);
}
typedef uintnat barrier_status;
/* Arrive at the barrier; if we are the final party, immediately returns a
nonzero value to be passed to [caml_global_barrier_release_as_final]
later, otherwise blocks and returns zero. */
barrier_status caml_global_barrier_and_check_final(int num_participating);
/* Release the barrier with the given status */
void caml_global_barrier_release_as_final(barrier_status status);
/* Arrive at the global barrier and run the body if we are the final party.
Other threads will not be released from the barrier until the final party
finishes executing the body.
Example usage:
Caml_global_barrier_if_final(num_participating) {
do_something_in_final_domain();
}
Note: this expands to an [if] and [for] header, do not exit the body using
jumps or returns, and do not put an [else] immediately after.
*/
#define Caml_global_barrier_if_final(num_participating) \
/* fast path when alone */ \
int CAML_GENSYM(alone) = (num_participating) == 1; \
barrier_status CAML_GENSYM(b) = 0; \
if (CAML_GENSYM(alone) || \
(CAML_GENSYM(b) \
= caml_global_barrier_and_check_final(num_participating))) \
for (int CAML_GENSYM(continue) = 1; CAML_GENSYM(continue); \
/* release the barrier after the body has executed once */ \
((CAML_GENSYM(alone) ? (void)0 : \
caml_global_barrier_release_as_final(CAML_GENSYM(b))), \
CAML_GENSYM(continue) = 0))
#endif /* CAML_INTERNALS */
#endif /* CAML_DOMAIN_H */
|