Line data Source code
1 : #ifndef _GNU_SOURCE /* GCC seems to do this this is on the command line somehow when using g++ */
2 : #define _GNU_SOURCE
3 : #endif
4 :
5 : #include <ctype.h>
6 : #include <errno.h>
7 : #include <pthread.h>
8 : #include <unistd.h>
9 : #include <sched.h>
10 : #include <syscall.h>
11 : #include <sys/resource.h>
12 : #include <sys/mman.h>
13 : #include <sys/prctl.h>
14 :
15 : #include "../sanitize/fd_sanitize.h"
16 : #include "fd_tile_private.h"
17 :
18 : /* Operating system shims ********************************************/
19 :
20 : struct fd_tile_private_cpu_config {
21 : int prio;
22 : };
23 :
24 : typedef struct fd_tile_private_cpu_config fd_tile_private_cpu_config_t;
25 :
26 : /* Configure the CPU optimally */
27 :
28 : static inline void
29 : fd_tile_private_cpu_config( fd_tile_private_cpu_config_t * save,
30 2430 : ulong cpu_idx ) {
31 :
32 : /* If a floating tile, leave scheduler priority unchanged from however
33 : the thread group launcher configured it. */
34 :
35 2430 : if( cpu_idx==65535UL ) {
36 2283 : save->prio = INT_MIN;
37 2283 : return;
38 2283 : }
39 :
40 : /* Otherwise, configure high scheduler priority */
41 :
42 147 : errno = 0;
43 147 : int prio = getpriority( PRIO_PROCESS, (id_t)0 );
44 147 : if( prio==-1 && errno ) {
45 0 : FD_LOG_WARNING(( "fd_tile: getpriority failed (%i-%s)\n\t"
46 0 : "Unable to determine initial tile priority so not configuring the tile\n\t"
47 0 : "for high scheduler priority. Attempting to continue but this thread\n\t"
48 0 : "group's performance and stability might be compromised. Probably should\n\t"
49 0 : "configure 'ulimit -e 39' (or 40 and this might require adjusting\n\t"
50 0 : "/etc/security/limits.conf as superuser to nice -19 or -20 for this user)\n\t"
51 0 : "to eliminate this warning. Also consider starting this thread group\n\t"
52 0 : "with 'nice --19'.",
53 0 : errno, fd_io_strerror( errno ) ));
54 0 : save->prio = INT_MIN;
55 0 : }
56 :
57 147 : if( FD_UNLIKELY( prio!=-19 ) && FD_UNLIKELY( setpriority( PRIO_PROCESS, (id_t)0, -19 ) ) ) {
58 147 : FD_LOG_WARNING(( "fd_tile: setpriority failed (%i-%s)\n\t"
59 147 : "Unable to configure this tile for high scheduler priority. Attempting\n\t"
60 147 : "to continue but this thread group's performance and stability might be\n\t"
61 147 : "compromised. Probably should configure 'ulimit -e 39' (or 40 and this\n\t"
62 147 : "might require adjusting /etc/security/limits.conf to nice -19 or -20\n\t"
63 147 : "for this user) to eliminate this warning. Also consider starting this\n\t"
64 147 : "thread group with 'nice --19'.",
65 147 : errno, fd_io_strerror( errno ) ));
66 147 : save->prio = INT_MIN;
67 147 : return;
68 147 : }
69 :
70 0 : save->prio = prio;
71 0 : }
72 :
73 : /* Restore the CPU to the given state */
74 :
75 : static inline void
76 1254 : fd_tile_private_cpu_restore( fd_tile_private_cpu_config_t * save ) {
77 1254 : int prio = save->prio;
78 1254 : if( FD_LIKELY( prio!=INT_MIN ) && FD_UNLIKELY( prio!=-19 ) && FD_UNLIKELY( setpriority( PRIO_PROCESS, (id_t)0, prio ) ) )
79 0 : FD_LOG_WARNING(( "fd_tile: setpriority failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
80 1254 : }
81 :
82 : void *
83 : fd_tile_private_stack_new( int optimize,
84 36 : ulong cpu_idx ) { /* Ignored if optimize is not requested */
85 :
86 36 : uchar * stack = NULL;
87 :
88 36 : if( optimize ) { /* Create a NUMA and TLB optimized stack for a tile running on cpu cpu_idx */
89 :
90 36 : stack = (uchar *)
91 36 : fd_shmem_acquire( FD_SHMEM_HUGE_PAGE_SZ, (FD_TILE_PRIVATE_STACK_SZ/FD_SHMEM_HUGE_PAGE_SZ)+2UL, cpu_idx ); /* logs details */
92 :
93 36 : if( FD_LIKELY( stack ) ) { /* Make space for guard lo and guard hi */
94 :
95 36 : fd_shmem_release( stack, FD_SHMEM_HUGE_PAGE_SZ, 1UL );
96 :
97 36 : stack += FD_SHMEM_HUGE_PAGE_SZ;
98 :
99 36 : fd_shmem_release( stack + FD_TILE_PRIVATE_STACK_SZ, FD_SHMEM_HUGE_PAGE_SZ, 1UL );
100 :
101 36 : } else {
102 :
103 0 : ulong numa_idx = fd_shmem_numa_idx( cpu_idx );
104 0 : static ulong warn = 0UL;
105 0 : if( FD_LIKELY( !(warn & (1UL<<numa_idx) ) ) ) {
106 0 : FD_LOG_WARNING(( "fd_tile: fd_shmem_acquire failed\n\t"
107 0 : "There are probably not enough huge pages allocated by the OS on numa\n\t"
108 0 : "node %lu. Falling back on normal page backed stack for tile on cpu %lu\n\t"
109 0 : "and attempting to continue. Run:\n\t"
110 0 : "\techo [CNT] > /sys/devices/system/node/node%lu/hugepages/hugepages-2048kB/nr_hugepages\n\t"
111 0 : "(probably as superuser) or equivalent where [CNT] is a sufficient number\n\t"
112 0 : "huge pages to reserve on this numa node system wide and/or adjust\n\t"
113 0 : "/etc/security/limits.conf to permit this user to lock a sufficient\n\t"
114 0 : "amount of memory to eliminate this warning.",
115 0 : numa_idx, cpu_idx, numa_idx ));
116 0 : warn |= 1UL<<numa_idx;
117 0 : }
118 :
119 0 : }
120 :
121 36 : }
122 :
123 36 : if( !stack ) { /* Request for a non-optimized stack (or optimized stack creation failed above and we are falling back) */
124 :
125 0 : ulong mmap_sz = FD_TILE_PRIVATE_STACK_SZ + 2UL*FD_SHMEM_NORMAL_PAGE_SZ;
126 0 : stack = (uchar *)mmap( NULL, mmap_sz, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, (off_t)0 );
127 :
128 0 : if( FD_LIKELY( ((void *)stack)!=MAP_FAILED ) ) { /* Make space for guard lo and guard hi */
129 :
130 0 : if( FD_UNLIKELY( munmap( stack, FD_SHMEM_NORMAL_PAGE_SZ ) ) )
131 0 : FD_LOG_WARNING(( "fd_tile: munmap failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
132 :
133 0 : stack += FD_SHMEM_NORMAL_PAGE_SZ;
134 :
135 0 : if( FD_UNLIKELY( munmap( stack + FD_TILE_PRIVATE_STACK_SZ, FD_SHMEM_NORMAL_PAGE_SZ ) ) )
136 0 : FD_LOG_WARNING(( "fd_tile: munmap failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
137 :
138 0 : } else {
139 :
140 0 : FD_LOG_WARNING(( "fd_tile: mmap(NULL,%lu KiB,PROT_READ|PROT_WRITE,MAP_PRIVATE|MAP_ANONYMOUS,-1,0) failed (%i-%s)\n\t"
141 0 : "Falling back on pthreads created stack and attempting to continue.",
142 0 : mmap_sz >> 10, errno, fd_io_strerror( errno ) ));
143 0 : return NULL;
144 :
145 0 : }
146 :
147 0 : }
148 :
149 : /* Create the guard regions in the extra space */
150 :
151 36 : void * guard_lo = (void *)(stack - FD_SHMEM_NORMAL_PAGE_SZ );
152 36 : if( FD_UNLIKELY( mmap( guard_lo, FD_SHMEM_NORMAL_PAGE_SZ, PROT_NONE,
153 36 : MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED, -1, (off_t)0 )!=guard_lo ) )
154 0 : FD_LOG_WARNING(( "fd_tile: mmap failed (%i-%s)\n\tAttempting to continue without stack guard lo.",
155 36 : errno, fd_io_strerror( errno ) ));
156 :
157 36 : void * guard_hi = (void *)(stack + FD_TILE_PRIVATE_STACK_SZ);
158 36 : if( FD_UNLIKELY( mmap( guard_hi, FD_SHMEM_NORMAL_PAGE_SZ, PROT_NONE,
159 36 : MAP_PRIVATE | MAP_ANONYMOUS | MAP_FIXED, -1, (off_t)0 )!=guard_hi ) )
160 0 : FD_LOG_WARNING(( "fd_tile: mmap failed (%i-%s)\n\tAttempting to continue without stack guard hi.",
161 36 : errno, fd_io_strerror( errno ) ));
162 :
163 36 : return stack;
164 36 : }
165 :
166 : static void
167 36 : fd_tile_private_stack_delete( void * _stack ) {
168 36 : if( FD_UNLIKELY( !_stack ) ) return;
169 :
170 36 : uchar * stack = (uchar *)_stack;
171 36 : uchar * guard_lo = stack - FD_SHMEM_NORMAL_PAGE_SZ;
172 36 : uchar * guard_hi = stack + FD_TILE_PRIVATE_STACK_SZ;
173 :
174 36 : if( FD_UNLIKELY( munmap( guard_hi, FD_SHMEM_NORMAL_PAGE_SZ ) ) )
175 0 : FD_LOG_WARNING(( "fd_tile: munmap failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
176 :
177 36 : if( FD_UNLIKELY( munmap( guard_lo, FD_SHMEM_NORMAL_PAGE_SZ ) ) )
178 0 : FD_LOG_WARNING(( "fd_tile: munmap failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
179 :
180 : /* Note that fd_shmem_release is just a wrapper around munmap such
181 : that this covers both the optimized and non-optimized cases */
182 :
183 36 : if( FD_UNLIKELY( munmap( stack, FD_TILE_PRIVATE_STACK_SZ ) ) )
184 0 : FD_LOG_WARNING(( "fd_tile: munmap failed (%i-%s); attempting to continue", errno, fd_io_strerror( errno ) ));
185 36 : }
186 :
187 : /* Tile side APIs ****************************************************/
188 :
189 : static ulong fd_tile_private_id0; /* Zeroed at app start, initialized by the boot manager */
190 : static ulong fd_tile_private_id1; /* " */
191 : static ulong fd_tile_private_cnt; /* " */
192 :
193 78 : ulong fd_tile_id0( void ) { return fd_tile_private_id0; }
194 36 : ulong fd_tile_id1( void ) { return fd_tile_private_id1; }
195 219 : ulong fd_tile_cnt( void ) { return fd_tile_private_cnt; }
196 :
197 : static FD_TL ulong fd_tile_private_id; /* Zeroed at app/thread start, initialized by the boot / tile manager */
198 : static FD_TL ulong fd_tile_private_idx; /* " */
199 : /**/ FD_TL ulong fd_tile_private_stack0; /* " */
200 : /**/ FD_TL ulong fd_tile_private_stack1; /* " */
201 :
202 54 : ulong fd_tile_id ( void ) { return fd_tile_private_id; }
203 309 : ulong fd_tile_idx( void ) { return fd_tile_private_idx; }
204 :
205 : static ushort fd_tile_private_cpu_id[ FD_TILE_MAX ]; /* Zeroed at app start, initialized by boot */
206 :
207 : ulong
208 54 : fd_tile_cpu_id( ulong tile_idx ) {
209 54 : if( FD_UNLIKELY( tile_idx>=fd_tile_private_cnt ) ) return ULONG_MAX;
210 54 : ulong cpu_idx = (ulong)fd_tile_private_cpu_id[ tile_idx ];
211 54 : return fd_ulong_if( cpu_idx<65535UL, cpu_idx, ULONG_MAX-1UL );
212 54 : }
213 :
214 : /* This is used for the OS services to communicate information with the
215 : tile managers */
216 :
217 5 : #define FD_TILE_PRIVATE_STATE_BOOT (0) /* Tile is booting */
218 189 : #define FD_TILE_PRIVATE_STATE_IDLE (1) /* Tile is idle */
219 105 : #define FD_TILE_PRIVATE_STATE_EXEC (2) /* Tile is executing a task */
220 36 : #define FD_TILE_PRIVATE_STATE_HALT (3) /* Tile is halting */
221 :
222 : struct __attribute__((aligned(128))) fd_tile_private { /* Double cache line aligned to avoid aclpf false sharing */
223 : ulong id;
224 : ulong idx;
225 : int state; /* FD_TILE_PRIVATE_STATE_* */
226 : int argc;
227 : char ** argv;
228 : fd_tile_task_t task;
229 : char const * fail;
230 : int ret;
231 : };
232 :
233 : typedef struct fd_tile_private fd_tile_private_t;
234 :
235 : struct fd_tile_private_manager_args {
236 : ulong id;
237 : ulong idx;
238 : ulong cpu_idx;
239 : void * stack; /* NULL if pthread created, non-NULL if user created */
240 : ulong stack_sz;
241 : fd_tile_private_t * tile;
242 : };
243 :
244 : typedef struct fd_tile_private_manager_args fd_tile_private_manager_args_t;
245 :
246 : static void *
247 36 : fd_tile_private_manager( void * _args ) {
248 36 : fd_tile_private_manager_args_t * args = (fd_tile_private_manager_args_t *)_args;
249 :
250 : # if !__GLIBC__
251 : if( args->cpu_idx<65535UL ) {
252 : FD_CPUSET_DECL( cpu_set );
253 : fd_cpuset_insert( cpu_set, args->cpu_idx );
254 : int err = fd_cpuset_setaffinity( (pid_t)0, cpu_set );
255 : if( FD_UNLIKELY( err ) )
256 : FD_LOG_WARNING(( "fd_tile: fd_cpuset_setaffinity_failed (%i-%s)\n\t"
257 : "Unable to set the thread affinity for tile %lu to cpu %lu. Attempting to\n\t"
258 : "continue without explicitly specifying this tile's cpu affinity but it\n\t"
259 : "is likely this thread group's performance and stability are compromised\n\t"
260 : "(possibly catastrophically so). Update --tile-cpus to specify a set of\n\t"
261 : "allowed cpus that have been reserved for this thread group on this host\n\t"
262 : "to eliminate this warning.", err, fd_io_strerror( err ), args->idx, args->cpu_idx ));
263 : }
264 : # endif /* !__GLIBC__ */
265 :
266 36 : ulong id = args->id;
267 36 : ulong idx = args->idx;
268 36 : void * stack = args->stack;
269 36 : ulong stack_sz = args->stack_sz;
270 :
271 36 : char thread_name[ 20 ];
272 36 : FD_TEST( fd_cstr_printf_check( thread_name, sizeof( thread_name ), NULL, "tile:%lu", idx ) );
273 36 : if( FD_UNLIKELY( prctl( PR_SET_NAME, thread_name, 0, 0, 0 ) ) ) FD_LOG_ERR(( "prctl(PR_SET_NAME) failed (%i-%s)", errno, fd_io_strerror( errno ) ));
274 :
275 36 : if( FD_UNLIKELY( !( (id ==fd_log_thread_id() ) &
276 36 : (idx==(id-fd_tile_private_id0) ) &
277 36 : ((fd_tile_private_id0<id) & (id<fd_tile_private_id1) ) &
278 36 : (fd_tile_private_cnt==(fd_tile_private_id1-fd_tile_private_id0)) ) ) )
279 0 : FD_LOG_ERR(( "fd_tile: internal error (unexpected thread identifiers)" ));
280 :
281 36 : fd_tile_private_t tile[1];
282 36 : FD_VOLATILE( tile->id ) = id;
283 36 : FD_VOLATILE( tile->idx ) = idx;
284 36 : FD_VOLATILE( tile->state ) = FD_TILE_PRIVATE_STATE_BOOT;
285 36 : FD_VOLATILE( tile->argc ) = 0;
286 36 : FD_VOLATILE( tile->argv ) = NULL;
287 36 : FD_VOLATILE( tile->task ) = NULL;
288 36 : FD_VOLATILE( tile->fail ) = NULL;
289 36 : FD_VOLATILE( tile->ret ) = 0;
290 :
291 : /* state is BOOT ... configure the tile, transition to IDLE and then
292 : start polling for tasks */
293 :
294 36 : fd_tile_private_id = id;
295 36 : fd_tile_private_idx = idx;
296 :
297 36 : if( FD_LIKELY( stack ) ) { /* User provided stack */
298 36 : fd_tile_private_stack0 = (ulong)stack;
299 36 : fd_tile_private_stack1 = (ulong)stack + stack_sz;
300 :
301 : /* Prevent another fork() from smashing the stack */
302 36 : if( FD_UNLIKELY( madvise( stack, FD_TILE_PRIVATE_STACK_SZ, MADV_DONTFORK ) ) ) {
303 0 : FD_LOG_ERR(( "madvise(stack,MADV_DONTFORK) failed (%i-%s)", errno, fd_io_strerror( errno ) ));
304 0 : }
305 :
306 36 : } else { /* Pthread provided stack */
307 0 : fd_log_private_stack_discover( stack_sz, &fd_tile_private_stack0, &fd_tile_private_stack1 ); /* logs details */
308 0 : if( FD_UNLIKELY( !fd_tile_private_stack0 ) )
309 0 : FD_LOG_WARNING(( "stack diagnostics not available on this tile; attempting to continue" ));
310 0 : }
311 :
312 36 : fd_tile_private_cpu_config_t dummy[1];
313 36 : fd_tile_private_cpu_config( dummy, args->cpu_idx );
314 :
315 36 : ulong app_id = fd_log_app_id();
316 36 : FD_LOG_INFO(( "fd_tile: boot tile %lu success (thread %lu:%lu in thread group %lu:%lu/%lu)",
317 36 : idx, app_id, id, app_id, fd_tile_private_id0, fd_tile_private_cnt ));
318 :
319 36 : FD_COMPILER_MFENCE();
320 36 : FD_VOLATILE( tile->state ) = FD_TILE_PRIVATE_STATE_IDLE;
321 36 : FD_VOLATILE( args->tile ) = tile;
322 :
323 220938448 : for(;;) {
324 :
325 : /* We are awake ... see what we should do next */
326 :
327 220938448 : int state = FD_VOLATILE_CONST( tile->state );
328 220938448 : if( FD_UNLIKELY( state!=FD_TILE_PRIVATE_STATE_EXEC ) ) {
329 156296653 : if( FD_UNLIKELY( state!=FD_TILE_PRIVATE_STATE_IDLE ) ) break;
330 : /* state is IDLE ... try again */
331 156296617 : FD_SPIN_PAUSE();
332 156296617 : continue;
333 156296653 : }
334 :
335 : /* state is EXEC ... the run assigned task and then
336 : transition to IDLE when done */
337 : /* FIXME: MORE SOPHISTICATED HANDLING OF EXCEPTIONS */
338 :
339 64641795 : int argc = FD_VOLATILE_CONST( tile->argc );
340 64641795 : char ** argv = FD_VOLATILE_CONST( tile->argv );
341 64641795 : fd_tile_task_t task = FD_VOLATILE_CONST( tile->task );
342 64641795 : try {
343 64641795 : FD_VOLATILE( tile->ret ) = task( argc, argv );
344 64641795 : FD_VOLATILE( tile->fail ) = NULL;
345 64641795 : } catch( ... ) {
346 0 : FD_VOLATILE( tile->fail ) = "uncaught exception";
347 0 : }
348 :
349 64641795 : FD_COMPILER_MFENCE();
350 105 : FD_VOLATILE( tile->state ) = FD_TILE_PRIVATE_STATE_IDLE;
351 105 : }
352 :
353 : /* state is HALT, clean up and then reset back to BOOT */
354 :
355 5 : FD_LOG_INFO(( "fd_tile: halting tile %lu", idx ));
356 :
357 5 : FD_COMPILER_MFENCE();
358 5 : FD_VOLATILE( tile->state ) = FD_TILE_PRIVATE_STATE_BOOT;
359 5 : return stack;
360 36 : }
361 :
362 : /* Dispatch side APIs ************************************************/
363 :
364 : static struct __attribute__((aligned(128))) { /* Each on its own cache line pair to limit false sharing in parallel dispatch */
365 : fd_tile_private_t * lock; /* Non-NULL if tile idx is available for dispatch, ==tile otherwise */
366 : fd_tile_private_t * tile;
367 : pthread_t pthread;
368 : } fd_tile_private[ FD_TILE_MAX ];
369 :
370 : /* FIXME: ATOMIC_XCHG BASED INSTEAD? */
371 : static inline fd_tile_private_t *
372 108 : fd_tile_private_trylock( ulong tile_idx ) {
373 108 : fd_tile_private_t * volatile * vtile = (fd_tile_private_t * volatile *)&fd_tile_private[ tile_idx ].lock;
374 108 : fd_tile_private_t * tile = *vtile;
375 108 : if( FD_LIKELY( tile ) && FD_LIKELY( FD_ATOMIC_CAS( vtile, tile, NULL )==tile ) ) return tile;
376 3 : return NULL;
377 108 : }
378 :
379 : static inline fd_tile_private_t *
380 36 : fd_tile_private_lock( ulong tile_idx ) {
381 36 : fd_tile_private_t * volatile * vtile = (fd_tile_private_t * volatile *)&fd_tile_private[ tile_idx ].lock;
382 36 : fd_tile_private_t * tile;
383 36 : for(;;) {
384 36 : tile = *vtile;
385 36 : if( FD_LIKELY( tile ) && FD_LIKELY( FD_ATOMIC_CAS( vtile, tile, NULL )==tile ) ) break;
386 0 : FD_SPIN_PAUSE();
387 0 : }
388 36 : return tile;
389 36 : }
390 :
391 : static inline void
392 : fd_tile_private_unlock( ulong tile_idx,
393 141 : fd_tile_private_t * tile ) {
394 141 : FD_VOLATILE( fd_tile_private[ tile_idx ].lock ) = tile;
395 141 : }
396 :
397 : fd_tile_exec_t *
398 : fd_tile_exec_new( ulong idx,
399 : fd_tile_task_t task,
400 : int argc,
401 141 : char ** argv ) {
402 141 : if( FD_UNLIKELY( (idx==fd_tile_private_idx) | (!idx) ) ) return NULL; /* Can't dispatch to self or to tile 0 */
403 :
404 108 : fd_tile_private_t * tile = fd_tile_private_trylock( idx );
405 108 : if( FD_UNLIKELY( !tile ) ) return NULL;
406 :
407 : /* Exec holds the lock and tile state is idle here */
408 105 : FD_VOLATILE( tile->argc ) = argc;
409 105 : FD_VOLATILE( tile->argv ) = argv;
410 105 : FD_VOLATILE( tile->task ) = task;
411 105 : FD_COMPILER_MFENCE();
412 105 : FD_VOLATILE( tile->state ) = FD_TILE_PRIVATE_STATE_EXEC;
413 105 : return (fd_tile_exec_t *)tile;
414 108 : }
415 :
416 : char const *
417 : fd_tile_exec_delete( fd_tile_exec_t * exec,
418 105 : int * opt_ret ) {
419 105 : fd_tile_private_t * tile = (fd_tile_private_t *)exec;
420 105 : ulong tile_idx = tile->idx;
421 :
422 105 : int state;
423 1772673 : for(;;) {
424 1772673 : state = FD_VOLATILE_CONST( tile->state );
425 1772673 : if( FD_LIKELY( state==FD_TILE_PRIVATE_STATE_IDLE ) ) break;
426 1772568 : FD_SPIN_PAUSE();
427 1772568 : }
428 : /* state is IDLE at this point */
429 105 : char const * fail = FD_VOLATILE_CONST( tile->fail );
430 105 : if( FD_LIKELY( (!fail) & (!!opt_ret) ) ) *opt_ret = FD_VOLATILE_CONST( tile->ret );
431 105 : fd_tile_private_unlock( tile_idx, tile );
432 105 : return fail;
433 105 : }
434 :
435 108 : fd_tile_exec_t * fd_tile_exec( ulong tile_idx ) { return (fd_tile_exec_t *)fd_tile_private[ tile_idx ].tile; }
436 :
437 0 : ulong fd_tile_exec_id ( fd_tile_exec_t const * exec ) { return ((fd_tile_private_t const *)exec)->id; }
438 12 : ulong fd_tile_exec_idx ( fd_tile_exec_t const * exec ) { return ((fd_tile_private_t const *)exec)->idx; }
439 12 : fd_tile_task_t fd_tile_exec_task( fd_tile_exec_t const * exec ) { return ((fd_tile_private_t const *)exec)->task; }
440 12 : int fd_tile_exec_argc( fd_tile_exec_t const * exec ) { return ((fd_tile_private_t const *)exec)->argc; }
441 12 : char ** fd_tile_exec_argv( fd_tile_exec_t const * exec ) { return ((fd_tile_private_t const *)exec)->argv; }
442 :
443 : int
444 12 : fd_tile_exec_done( fd_tile_exec_t const * exec ) {
445 12 : fd_tile_private_t const * tile = (fd_tile_private_t const *)exec;
446 12 : return FD_VOLATILE_CONST( tile->state )==FD_TILE_PRIVATE_STATE_IDLE;
447 12 : }
448 :
449 : /* Boot/halt APIs ****************************************************/
450 :
451 : /* Parse a list of cpu tiles */
452 :
453 : FD_STATIC_ASSERT( FD_TILE_MAX<65535, update_tile_to_cpu_type );
454 :
455 : ulong
456 : fd_tile_private_cpus_parse( char const * cstr,
457 2394 : ushort * tile_to_cpu ) {
458 2394 : if( !cstr ) return 0UL;
459 111 : ulong cnt = 0UL;
460 :
461 111 : FD_CPUSET_DECL( assigned_set );
462 :
463 111 : char const * p = cstr;
464 225 : for(;;) {
465 :
466 225 : while( fd_isspace( (int)p[0] ) ) p++; /* Munch whitespace */
467 :
468 225 : if( p[0]=='f' ) { /* These tiles have been requested to float on the original core set */
469 0 : p++;
470 :
471 0 : ulong float_cnt;
472 :
473 0 : while( fd_isspace( (int)p[0] ) ) p++; /* Munch whitespace */
474 0 : if ( p[0]==',' ) float_cnt = 1UL, p++;
475 0 : else if( p[0]=='\0' ) float_cnt = 1UL;
476 0 : else if( !fd_isdigit( (int)p[0] ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (malformed count)" ));
477 0 : else {
478 0 : float_cnt = fd_cstr_to_ulong( p );
479 0 : if( FD_UNLIKELY( !float_cnt ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (bad count)" ));
480 0 : p++; while( fd_isdigit( (int)p[0] ) ) p++; /* FIXME: USE STRTOUL ENDPTR FOR CORRECT HANDLING OF NON-BASE-10 */
481 0 : while( fd_isspace( (int)p[0] ) ) p++; /* Munch whitespace */
482 0 : if( FD_UNLIKELY( !( p[0]==',' || p[0]=='\0' ) ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (bad count delimiter)" ));
483 0 : if( p[0]==',' ) p++;
484 0 : }
485 :
486 : /* float_cnt is at least 1 at this point */
487 0 : do {
488 0 : if( FD_UNLIKELY( cnt>=FD_TILE_MAX ) ) FD_LOG_ERR(( "fd_tile: too many --tile-cpus" ));
489 0 : tile_to_cpu[ cnt++ ] = (ushort)65535;
490 0 : } while( --float_cnt );
491 :
492 0 : continue;
493 0 : }
494 :
495 225 : if( !fd_isdigit( (int)p[0] ) ) {
496 111 : if( FD_UNLIKELY( p[0]!='\0' ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (range lo not a cpu)" ));
497 111 : break;
498 111 : }
499 114 : ulong cpu0 = fd_cstr_to_ulong( p );
500 114 : ulong cpu1 = cpu0;
501 114 : ulong stride = 1UL;
502 156 : p++; while( fd_isdigit( (int)p[0] ) ) p++; /* FIXME: USE STRTOUL ENDPTR FOR CORRECT HANDLING OF NON-BASE-10 */
503 114 : while( fd_isspace( (int)p[0] ) ) p++;
504 114 : if( p[0]=='-' ) {
505 6 : p++;
506 6 : while( fd_isspace( (int)p[0] ) ) p++;
507 6 : if( FD_UNLIKELY( !fd_isdigit( (int)p[0] ) ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (range hi not a cpu)" ));
508 6 : cpu1 = fd_cstr_to_ulong( p );
509 6 : p++; while( fd_isdigit( (int)p[0] ) ) p++; /* FIXME: USE STRTOUL ENDPTR FOR CORRECT HANDLING OF NON-BASE-10 */
510 6 : while( fd_isspace( (int)p[0] ) ) p++;
511 6 : if( p[0]=='/' || p[0]==':' ) {
512 3 : p++;
513 3 : while( fd_isspace( (int)p[0] ) ) p++;
514 3 : if( FD_UNLIKELY( !fd_isdigit( (int)p[0] ) ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (stride not an int)" ));
515 3 : stride = fd_cstr_to_ulong( p );
516 3 : p++; while( fd_isdigit( (int)p[0] ) ) p++; /* FIXME: USE STRTOUL ENDPTR FOR CORRECT HANDLING OF NON-BASE-10 */
517 3 : }
518 6 : }
519 114 : while( fd_isspace( (int)p[0] ) ) p++;
520 114 : if( FD_UNLIKELY( !( p[0]==',' || p[0]=='\0' ) ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (bad range delimiter)" ));
521 114 : if( p[0]==',' ) p++;
522 114 : cpu1++;
523 114 : if( FD_UNLIKELY( cpu1<=cpu0 ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (invalid range)" ));
524 114 : if( FD_UNLIKELY( !stride ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (invalid stride)" ));
525 :
526 261 : for( ulong cpu=cpu0; cpu<cpu1; cpu+=stride ) {
527 147 : if( FD_UNLIKELY( cnt>=FD_TILE_MAX ) ) FD_LOG_ERR(( "fd_tile: too many --tile-cpus" ));
528 147 : if( FD_UNLIKELY( fd_cpuset_test( assigned_set, cpu ) ) ) FD_LOG_ERR(( "fd_tile: malformed --tile-cpus (repeated cpu)" ));
529 147 : tile_to_cpu[ cnt++ ] = (ushort)cpu;
530 147 : fd_cpuset_insert( assigned_set, cpu );
531 147 : }
532 114 : }
533 :
534 111 : return cnt;
535 111 : }
536 :
537 : static fd_tile_private_cpu_config_t fd_tile_private_cpu_config_save[1];
538 :
539 : void
540 : fd_tile_private_map_boot( ushort * tile_to_cpu,
541 2394 : ulong tile_cnt ) {
542 2394 : fd_tile_private_id0 = fd_log_thread_id();
543 2394 : fd_tile_private_id1 = fd_tile_private_id0 + tile_cnt;
544 2394 : fd_tile_private_cnt = tile_cnt;
545 :
546 2394 : ulong app_id = fd_log_app_id();
547 2394 : ulong host_id = fd_log_host_id();
548 2394 : FD_LOG_INFO(( "fd_tile: booting thread group %lu:%lu/%lu", app_id, fd_tile_private_id0, fd_tile_private_cnt ));
549 :
550 : /* We create the tiles [1,tile_cnt) first so that any floating tiles
551 : in this inherit the appropriate scheduler priorities and affinities
552 : from the thread group launcher. */
553 :
554 2430 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ ) {
555 :
556 36 : ulong cpu_idx = (ulong)tile_to_cpu[ tile_idx ];
557 36 : int fixed = (cpu_idx<65535UL);
558 :
559 36 : if( fixed ) FD_LOG_INFO(( "fd tile: booting tile %lu on cpu %lu:%lu", tile_idx, host_id, cpu_idx ));
560 0 : else FD_LOG_INFO(( "fd tile: booting tile %lu on cpu %lu:float", tile_idx, host_id ));
561 :
562 36 : pthread_attr_t attr[1];
563 36 : int err = pthread_attr_init( attr );
564 36 : if( FD_UNLIKELY( err ) ) FD_LOG_ERR(( "fd_tile: pthread_attr_init failed (%i-%s) for tile %lu.\n\t",
565 36 : err, fd_io_strerror( err ), tile_idx ));
566 :
567 : /* Set affinity ahead of time. This is a GNU-specific extension
568 : that is not available on musl. On musl, we just skip this
569 : step as we call sched_setaffinity(2) later on regardless. */
570 :
571 36 : # if __GLIBC__
572 36 : if( fixed ) {
573 36 : FD_CPUSET_DECL( cpu_set );
574 36 : fd_cpuset_insert( cpu_set, cpu_idx );
575 36 : err = pthread_attr_setaffinity_np( attr, fd_cpuset_footprint(), (cpu_set_t const *)fd_type_pun_const( cpu_set ) );
576 36 : if( FD_UNLIKELY( err ) ) FD_LOG_WARNING(( "fd_tile: pthread_attr_setaffinity_failed (%i-%s)\n\t"
577 36 : "Unable to set the thread affinity for tile %lu on cpu %lu. Attempting to\n\t"
578 36 : "continue without explicitly specifying this cpu's thread affinity but it\n\t"
579 36 : "is likely this thread group's performance and stability are compromised\n\t"
580 36 : "(possibly catastrophically so). Update --tile-cpus to specify a set of\n\t"
581 36 : "allowed cpus that have been reserved for this thread group on this host\n\t"
582 36 : "to eliminate this warning.",
583 36 : err, fd_io_strerror( err ), tile_idx, cpu_idx ));
584 36 : }
585 36 : # endif /* __GLIBC__ */
586 :
587 : /* Create an optimized stack with guard regions if the build target
588 : is x86 (e.g. supports huge pages necessary to optimize TLB usage)
589 : and the tile is assigned to a particular CPU (e.g. bind the stack
590 : memory to the NUMA node closest to the cpu).
591 :
592 : Otherwise (or if an optimized stack could not be created), create
593 : vanilla pthread-style stack with guard regions. We DIY here
594 : because pthreads seems to be missing an API to determine the
595 : extents of the stacks it creates and we need to know the stack
596 : extents for run-time stack diagnostics. Though we can use
597 : fd_log_private_stack_discover to determine stack extents after
598 : the thread is started, it is faster, more flexible, more reliable
599 : and more portable to use a user specified stack when possible.
600 :
601 : If neither can be done, we will let pthreads create the tile's
602 : stack and try to discover the stack extents after the thread is
603 : started. */
604 :
605 36 : int optimize = FD_HAS_X86 & fixed;
606 :
607 36 : void * stack = fd_tile_private_stack_new( optimize, cpu_idx );
608 36 : if( FD_LIKELY( stack ) ) {
609 36 : err = pthread_attr_setstack( attr, stack, FD_TILE_PRIVATE_STACK_SZ );
610 36 : if( FD_UNLIKELY( err ) ) {
611 0 : FD_LOG_WARNING(( "fd_tile: pthread_attr_setstack failed (%i-%s)\n\t", err, fd_io_strerror( err ) ));
612 0 : fd_tile_private_stack_delete( stack );
613 0 : stack = NULL;
614 0 : }
615 36 : }
616 :
617 36 : if( FD_UNLIKELY( !stack ) ) FD_LOG_WARNING(( "fd_tile: Unable to create a stack for tile %lu.\n\t"
618 36 : "Attempting to continue with the default stack but it is likely this\n\t"
619 36 : "thread group's performance and stability is compromised (possibly\n\t"
620 36 : "catastrophically so).",
621 36 : tile_idx ));
622 :
623 36 : ulong stack_sz;
624 36 : err = pthread_attr_getstacksize( attr, &stack_sz );
625 36 : if( FD_UNLIKELY( err ) ) FD_LOG_ERR(( "fd_tile: pthread_attr_getstacksize failed (%i-%s) for tile %lu.\n\t",
626 36 : err, fd_io_strerror( err ), tile_idx ));
627 :
628 36 : FD_VOLATILE( fd_tile_private[ tile_idx ].lock ) = NULL;
629 :
630 36 : fd_tile_private_manager_args_t args[1];
631 :
632 36 : FD_VOLATILE( args->id ) = fd_tile_private_id0 + tile_idx;
633 36 : FD_VOLATILE( args->idx ) = tile_idx;
634 36 : FD_VOLATILE( args->cpu_idx ) = cpu_idx;
635 36 : FD_VOLATILE( args->stack ) = stack;
636 36 : FD_VOLATILE( args->stack_sz ) = stack_sz;
637 36 : FD_VOLATILE( args->tile ) = NULL;
638 :
639 36 : FD_COMPILER_MFENCE();
640 :
641 36 : err = pthread_create( &fd_tile_private[tile_idx].pthread, attr, fd_tile_private_manager, args );
642 36 : if( FD_UNLIKELY( err ) ) {
643 0 : if( fixed ) FD_LOG_ERR(( "fd_tile: pthread_create failed (%i-%s)\n\t"
644 0 : "Unable to start up the tile %lu on cpu %lu. Likely causes for this include\n\t"
645 0 : "this cpu is restricted from the user or does not exist on this host.\n\t"
646 0 : "Update --tile-cpus to specify a set of allowed cpus that have been reserved\n\t"
647 0 : "for this thread group on this host.",
648 0 : err, fd_io_strerror( err ), tile_idx, cpu_idx ));
649 0 : FD_LOG_ERR(( "fd_tile: pthread_create failed (%i-%s)\n\tUnable to start up the tile %lu (floating).",
650 0 : err, fd_io_strerror( err ), tile_idx ));
651 0 : }
652 :
653 : /* Wait for the tile to be ready to exec */
654 :
655 36 : fd_tile_private_t * tile;
656 37723 : for(;;) {
657 37723 : tile = FD_VOLATILE_CONST( args->tile );
658 37723 : if( FD_LIKELY( tile ) ) break;
659 37687 : FD_YIELD();
660 37687 : }
661 36 : FD_VOLATILE( fd_tile_private[ tile_idx ].tile ) = tile;
662 36 : FD_VOLATILE( fd_tile_private[ tile_idx ].lock ) = tile;
663 :
664 : /* Tile is running, args is safe to reuse */
665 :
666 36 : err = pthread_attr_destroy( attr );
667 36 : if( FD_UNLIKELY( err ) )
668 0 : FD_LOG_WARNING(( "fd_tile: pthread_attr_destroy failed (%i-%s) for tile %lu; attempting to continue",
669 36 : err, fd_io_strerror( err ), tile_idx ));
670 36 : }
671 :
672 : /* And now we "boot" tile 0 */
673 :
674 2394 : ulong cpu_idx = (ulong)tile_to_cpu[ 0UL ];
675 2394 : int fixed = (cpu_idx<65535UL);
676 2394 : if( fixed ) FD_LOG_INFO(( "fd tile: booting tile %lu on cpu %lu:%lu", 0UL, host_id, cpu_idx ));
677 2283 : else FD_LOG_INFO(( "fd tile: booting tile %lu on cpu %lu:float", 0UL, host_id ));
678 :
679 2394 : if( fixed ) {
680 :
681 111 : int good_taskset;
682 111 : FD_CPUSET_DECL( cpu_set );
683 111 : if( FD_UNLIKELY( fd_cpuset_getaffinity( (pid_t)0, cpu_set ) ) ) {
684 0 : FD_LOG_WARNING(( "fd_tile: fd_cpuset_getaffinity failed (%i-%s) for tile 0 on cpu %lu",
685 0 : errno, fd_io_strerror( errno ), cpu_idx ));
686 0 : good_taskset = 0;
687 111 : } else {
688 111 : ulong cnt = fd_cpuset_cnt( cpu_set );
689 111 : ulong idx = fd_cpuset_first( cpu_set );
690 111 : good_taskset = (cnt==1UL) & (idx==cpu_idx);
691 111 : }
692 :
693 111 : if( FD_UNLIKELY( !good_taskset ) ) {
694 9 : FD_LOG_WARNING(( "fd_tile: --tile-cpus for tile 0 may not match initial kernel affinity\n\t"
695 9 : "Tile 0 might not be fully optimized because of kernel first touch.\n\t"
696 9 : "Overriding fd_log_cpu_id(), fd_log_cpu(), fd_log_thread() on tile 0 to\n\t"
697 9 : "match --tile-cpus and attempting to continue. Launch this thread\n\t"
698 9 : "group via 'taskset -c %lu' or equivalent to eliminate this warning.", cpu_idx ));
699 9 : fd_cpuset_null( cpu_set );
700 9 : fd_cpuset_insert( cpu_set, cpu_idx );
701 9 : if( FD_UNLIKELY( fd_cpuset_setaffinity( (pid_t)0, cpu_set ) ) )
702 0 : FD_LOG_WARNING(( "fd_tile: fd_cpuset_setaffinity_failed (%i-%s)\n\t"
703 9 : "Unable to set the thread affinity for tile 0 on cpu %lu. Attempting to\n\t"
704 9 : "continue without explicitly specifying this cpu's thread affinity but it\n\t"
705 9 : "is likely this thread group's performance and stability are compromised\n\t"
706 9 : "(possibly catastrophically so). Update --tile-cpus to specify a set of\n\t"
707 9 : "allowed cpus that have been reserved for this thread group on this host\n\t"
708 9 : "to eliminate this warning.",
709 9 : errno, fd_io_strerror( errno ), cpu_idx ));
710 9 : fd_log_private_cpu_id_set( cpu_idx );
711 9 : fd_log_cpu_set ( NULL );
712 9 : fd_log_thread_set( NULL );
713 9 : }
714 111 : }
715 :
716 : /* Tile 0 "pthread_create" */
717 2394 : fd_tile_private[0].pthread = pthread_self();
718 : /* FIXME: ON X86, DETECT IF TILE 0 STACK ISN'T HUGE PAGE AND WARN AS NECESSARY? */
719 :
720 : /* Tile 0 "thread manager init" */
721 2394 : fd_tile_private_id = fd_tile_private_id0;
722 2394 : fd_tile_private_idx = 0UL;
723 :
724 2394 : # if !FD_HAS_ASAN
725 2394 : fd_log_private_stack_discover( fd_log_private_main_stack_sz(),
726 2394 : &fd_tile_private_stack0, &fd_tile_private_stack1 ); /* logs details */
727 2394 : if( FD_UNLIKELY( !fd_tile_private_stack0 ) )
728 0 : FD_LOG_WARNING(( "stack diagnostics not available on this tile; attempting to continue" ));
729 2394 : # endif /* FD_HAS_ASAN */
730 :
731 2394 : fd_tile_private_cpu_config( fd_tile_private_cpu_config_save, cpu_idx );
732 2394 : fd_tile_private[0].lock = NULL; /* Can't dispatch to tile 0 */
733 2394 : fd_tile_private[0].tile = NULL; /* " */
734 :
735 2394 : FD_LOG_INFO(( "fd_tile: boot tile %lu success (thread %lu:%lu in thread group %lu:%lu/%lu)",
736 2394 : fd_tile_private_idx, app_id, fd_tile_private_id, app_id, fd_tile_private_id0, fd_tile_private_cnt ));
737 :
738 2394 : fd_memcpy( fd_tile_private_cpu_id, tile_to_cpu, fd_tile_private_cnt*sizeof(ushort) );
739 :
740 2394 : FD_LOG_INFO(( "fd_tile: boot success" ));
741 2394 : }
742 :
743 : void
744 2394 : fd_tile_private_boot_str( char const * cpus ) {
745 2394 : ushort tile_to_cpu[ FD_TILE_MAX ];
746 2394 : ulong tile_cnt = fd_tile_private_cpus_parse( cpus, tile_to_cpu );
747 :
748 2394 : if( FD_UNLIKELY( !tile_cnt ) ) {
749 2283 : FD_LOG_INFO(( "fd_tile: no cpus specified; treating thread group as single tile running on O/S assigned cpu(s)" ));
750 2283 : tile_to_cpu[0] = (ushort)65535;
751 2283 : tile_cnt = 1UL;
752 2283 : }
753 :
754 2394 : fd_tile_private_map_boot( tile_to_cpu, tile_cnt );
755 2394 : }
756 :
757 : void
758 : fd_tile_private_boot( int * pargc,
759 2394 : char *** pargv ) {
760 : /* Extract the tile configuration from the command line */
761 :
762 2394 : char const * cpus = fd_env_strip_cmdline_cstr( pargc, pargv, "--tile-cpus", "FD_TILE_CPUS", NULL );
763 :
764 2394 : if( !cpus ) FD_LOG_INFO(( "fd_tile: --tile-cpus not specified" ));
765 111 : else FD_LOG_INFO(( "fd_tile: --tile-cpus \"%s\"", cpus ));
766 :
767 2394 : fd_tile_private_boot_str( cpus );
768 2394 : }
769 :
770 : void
771 1254 : fd_tile_private_halt( void ) {
772 1254 : FD_LOG_INFO(( "fd_tile: halt" ));
773 :
774 1254 : fd_memset( fd_tile_private_cpu_id, 0, fd_tile_private_cnt*sizeof(ushort) );
775 :
776 1254 : ulong tile_cnt = fd_tile_private_cnt;
777 :
778 1254 : fd_tile_private_t * tile[ FD_TILE_MAX ]; /* FIXME: ALLOCA TO TILE_CNT? */
779 :
780 1254 : FD_LOG_INFO(( "fd_tile: disabling dispatch" ));
781 1290 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ ) tile[ tile_idx ] = fd_tile_private_lock( tile_idx );
782 : /* All tile to tile dispatches will fail at this point */
783 :
784 1254 : FD_LOG_INFO(( "fd_tile: waiting for all tasks to complete" ));
785 1290 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ )
786 36 : while( FD_VOLATILE_CONST( tile[ tile_idx ]->state )!=FD_TILE_PRIVATE_STATE_IDLE ) FD_YIELD();
787 : /* All halt transitions will be valid at this point */
788 :
789 1254 : FD_LOG_INFO(( "fd_tile: signaling all tiles to halt" ));
790 1290 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ ) FD_VOLATILE( tile[ tile_idx ]->state ) = FD_TILE_PRIVATE_STATE_HALT;
791 : /* All tiles are halting at this point. tile[*] is no longer safe */
792 :
793 1254 : FD_LOG_INFO(( "fd_tile: waiting for all tiles to halt" ));
794 1290 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ ) {
795 36 : void * stack;
796 36 : int err = pthread_join( fd_tile_private[ tile_idx ].pthread, &stack );
797 36 : if( FD_UNLIKELY( err ) ) FD_LOG_ERR(( "fd_tile: pthread_join failed (%i-%s)", err, fd_io_strerror( err ) ));
798 36 : fd_tile_private_stack_delete( stack );
799 36 : FD_LOG_INFO(( "fd_tile: halt tile %lu success", tile_idx ));
800 36 : }
801 :
802 : /* All tiles but this one are halted at this point */
803 :
804 1254 : fd_tile_private_cpu_restore( fd_tile_private_cpu_config_save );
805 :
806 1254 : FD_LOG_INFO(( "fd_tile: halt tile 0 success" ));
807 :
808 1254 : FD_LOG_INFO(( "fd_tile: cleaning up" ));
809 :
810 1290 : for( ulong tile_idx=1UL; tile_idx<tile_cnt; tile_idx++ ) fd_tile_private_unlock( tile_idx, NULL );
811 :
812 1254 : fd_memset( fd_tile_private_cpu_config_save, 0, sizeof(fd_tile_private_cpu_config_t) );
813 :
814 1254 : fd_tile_private_stack1 = 0UL;
815 1254 : fd_tile_private_stack0 = 0UL;
816 1254 : fd_tile_private_idx = 0UL;
817 1254 : fd_tile_private_id = 0UL;
818 :
819 1254 : fd_tile_private_cnt = 0UL;
820 1254 : fd_tile_private_id1 = 0UL;
821 1254 : fd_tile_private_id0 = 0UL;
822 :
823 1254 : FD_LOG_INFO(( "fd_tile: halt success" ));
824 1254 : }
|