-
Notifications
You must be signed in to change notification settings - Fork 77
/
Copy pathdomain.h
232 lines (186 loc) · 9 KB
/
domain.h
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
/**************************************************************************/
/* */
/* 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 __cplusplus
extern "C" {
#endif
#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_spawn_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;
}
#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*, void*),
/* 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*, void*));
/* 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 betweeen 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 */
#ifdef __cplusplus
}
#endif
#endif /* CAML_DOMAIN_H */