Line data Source code
1 : #include "../tiles.h"
2 :
3 : #include "generated/fd_pack_tile_seccomp.h"
4 :
5 : #include "../../util/pod/fd_pod_format.h"
6 : #include "../../discof/replay/fd_replay_tile.h" // layering violation
7 : #include "../fd_txn_m.h"
8 : #include "../keyguard/fd_keyload.h"
9 : #include "../keyguard/fd_keyswitch.h"
10 : #include "../keyguard/fd_keyguard.h"
11 : #include "../keyguard/fd_keyguard_client.h"
12 : #include "../metrics/fd_metrics.h"
13 : #include "../pack/fd_pack.h"
14 : #include "../pack/fd_pack_cost.h"
15 : #include "../pack/fd_pack_pacing.h"
16 :
17 : #include <string.h>
18 :
19 : /* fd_pack is responsible for taking verified transactions, and
20 : arranging them into "microblocks" (groups) of transactions to
21 : be executed serially. It can try to do clever things so that
22 : multiple microblocks can execute in parallel, if they don't
23 : write to the same accounts. */
24 :
25 0 : #define IN_KIND_RESOLV (0UL)
26 0 : #define IN_KIND_POH (1UL)
27 0 : #define IN_KIND_EXECLE (2UL)
28 0 : #define IN_KIND_SIGN (3UL)
29 0 : #define IN_KIND_REPLAY (4UL)
30 0 : #define IN_KIND_EXECUTED_TXN (5UL)
31 :
32 : /* Pace microblocks, but only slightly. This helps keep performance
33 : more stable. This limit is 2,000 microblocks/second/execle. At
34 : MAX_TXN_PER_MICROBLOCK transactions/microblock, that's
35 : 2000*MAX_TXN_PER_MICROBLOCK txn/sec/execle. */
36 0 : #define MICROBLOCK_DURATION_NS (0L)
37 :
38 : /* There are 151 accepted blockhashes, but those don't include skips.
39 : This check is neither precise nor accurate, but just good enough.
40 : The execle tile does the final check. We give a little margin for a
41 : few percent skip rate. */
42 0 : #define TRANSACTION_LIFETIME_SLOTS 160UL
43 :
44 : /* Time is normally a long, but pack expects a ulong. Add -LONG_MIN to
45 : the time values so that LONG_MIN maps to 0, LONG_MAX maps to
46 : ULONG_MAX, and everything in between maps linearly with a slope of 1.
47 : Just subtracting LONG_MIN results in signed integer overflow, which
48 : is U.B. */
49 : #define TIME_OFFSET 0x8000000000000000UL
50 : FD_STATIC_ASSERT( (ulong)LONG_MIN+TIME_OFFSET==0UL, time_offset );
51 : FD_STATIC_ASSERT( (ulong)LONG_MAX+TIME_OFFSET==ULONG_MAX, time_offset );
52 :
53 : /* 1.6 M cost units, enough for 1 max size transaction */
54 : const ulong CUS_PER_MICROBLOCK = 1600000UL;
55 :
56 : #define SMALL_MICROBLOCKS 1
57 :
58 : #if SMALL_MICROBLOCKS
59 : const float VOTE_FRACTION = 1.0f; /* schedule all available votes first */
60 0 : #define EFFECTIVE_TXN_PER_MICROBLOCK 1UL
61 : #else
62 : const float VOTE_FRACTION = 0.75f; /* TODO: Is this the right value? */
63 : #define EFFECTIVE_TXN_PER_MICROBLOCK MAX_TXN_PER_MICROBLOCK
64 : #endif
65 :
66 : #if !SMALL_MICROBLOCKS
67 : /* There's overhead associated with each microblock the execle tile
68 : tries to execute it, so the optimal strategy is not to produce a
69 : microblock with a single transaction as soon as we receive it.
70 : Basically, if we have less than MAX_TXN_PER_MICROBLOCK transactions,
71 : we want to wait a little to see if we receive additional transactions
72 : before we schedule a microblock. We can model the optimum amount of
73 : time to wait, but the equation is complicated enough that we want to
74 : compute it before compile time. wait_duration[i] for i in
75 : [0, MAX_TXN_PER_MICROBLOCK] gives the time in nanoseconds pack should
76 : wait after receiving its most recent transaction before scheduling if
77 : it has i transactions available. Unsurprisingly,
78 : wait_duration[MAX_TXN_PER_MICROBLOCK] is 0. wait_duration[0] is
79 : ULONG_MAX, so we'll always wait if we have 0 transactions. */
80 : FD_IMPORT( wait_duration, "src/disco/pack/pack_delay.bin", ulong, 6, "" );
81 : #endif
82 :
83 :
84 :
85 : #if FD_PACK_USE_EXTRA_STORAGE
86 : /* When we are done being leader for a slot and we are leader in the
87 : very next slot, it can still take some time to transition. This is
88 : because the bank has to be finalized, a hash calculated, and various
89 : other things done in the replay stage to create the new child bank.
90 :
91 : During that time, pack cannot send transactions to execles so it
92 : needs to be able to buffer. Typically, these so called "leader
93 : transitions" are short (<15 millis), so a low value here would
94 : suffice. However, in some cases when there is memory pressure on the
95 : NUMA node or when the operating system context switches relevant
96 : threads out, it can take significantly longer.
97 :
98 : To prevent drops in these cases and because we assume execles are
99 : fast enough to drain this buffer once we do become leader, we set
100 : this buffer size to be quite large. */
101 :
102 : #define DEQUE_NAME extra_txn_deq
103 : #define DEQUE_T fd_txn_e_t
104 : #define DEQUE_MAX (128UL*1024UL)
105 : #include "../../../../util/tmpl/fd_deque.c"
106 :
107 : #endif
108 :
109 : /* Sync with src/app/shared/fd_config.c */
110 0 : #define FD_PACK_STRATEGY_PERF 0
111 0 : #define FD_PACK_STRATEGY_BALANCED 1
112 :
113 : static char const * const schedule_strategy_strings[2] = { "PRF", "BAL" };
114 :
115 :
116 : typedef struct {
117 : fd_acct_addr_t commission_pubkey[1];
118 : ulong commission;
119 : } block_builder_info_t;
120 :
121 : typedef struct {
122 : long time;
123 : ulong sched_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_CNT ];
124 : } fd_pack_sched_results_snap_t;
125 :
126 : typedef struct {
127 : fd_wksp_t * mem;
128 : ulong chunk0;
129 : ulong wmark;
130 : } fd_pack_in_ctx_t;
131 :
132 : typedef struct {
133 : fd_pack_t * pack;
134 : fd_txn_e_t * cur_spot;
135 : int is_bundle; /* is the current transaction a bundle */
136 :
137 : uchar executed_txn_sig[ 64UL ];
138 : uchar txn_committed;
139 :
140 : /* One of the FD_PACK_STRATEGY_* values defined above */
141 : int strategy;
142 :
143 : /* The value passed to fd_pack_new, etc. */
144 : ulong max_pending_transactions;
145 :
146 : /* The leader slot we are currently packing for, or ULONG_MAX if we
147 : are not the leader. */
148 : ulong leader_slot;
149 : void const * leader_bank;
150 : ulong leader_bank_idx;
151 :
152 : fd_became_leader_t _became_leader[1];
153 :
154 : /* The number of microblocks we have packed for the current leader
155 : slot. Will always be <= slot_max_microblocks. We must track
156 : this so that when we are done we can tell the PoH tile how many
157 : microblocks to expect in the slot. */
158 : ulong slot_microblock_cnt;
159 :
160 : /* Counter which increments when we've finished packing for a slot */
161 : uint pack_idx;
162 :
163 : ulong pack_txn_cnt; /* total num transactions packed since startup */
164 :
165 : /* The maximum number of microblocks that can be packed in this slot.
166 : Provided by the PoH tile when we become leader.*/
167 : ulong slot_max_microblocks;
168 :
169 : /* Cap (in bytes) of the amount of transaction data we produce in each
170 : block to avoid hitting the shred limits. See where this is set for
171 : more explanation. */
172 : ulong slot_max_data;
173 : int larger_shred_limits_per_block;
174 :
175 : /* Consensus critical slot cost limits. */
176 : struct {
177 : ulong slot_max_cost;
178 : ulong slot_max_vote_cost;
179 : ulong slot_max_write_cost_per_acct;
180 : } limits;
181 :
182 : /* If drain_execle is non-zero, then the pack tile must wait until all
183 : execle are idle before scheduling any more microblocks. This is
184 : primarily helpful in irregular leader transitions, e.g. while being
185 : leader for slot N, we switch forks to a slot M (!=N+1) in which we
186 : are also leader. We don't want to execute microblocks for
187 : different slots concurrently. */
188 : int drain_execle;
189 :
190 : /* Updated during housekeeping and used only for checking if the
191 : leader slot has ended. Might be off by one housekeeping duration,
192 : but that should be small relative to a slot duration. */
193 : long approx_wallclock_ns;
194 :
195 : /* approx_tickcount is updated in during_housekeeping() with
196 : fd_tickcount() and will match approx_wallclock_ns. This is done
197 : because we need to include an accurate nanosecond timestamp in
198 : every fd_txn_p_t but don't want to have to call the expensive
199 : fd_log_wallclock() in in the critical path. We can use
200 : fd_tempo_tick_per_ns() to convert from ticks to nanoseconds over
201 : small periods of time. */
202 : long approx_tickcount;
203 :
204 : fd_rng_t * rng;
205 :
206 : /* The end wallclock time of the leader slot we are currently packing
207 : for, if we are currently packing for a slot.*/
208 : long slot_end_ns;
209 :
210 : /* The current dynamic upper bound on total microblocks for this slot.
211 : Monotonically decreasing over the slot lifetime. */
212 : ulong slot_dynamic_max_microblocks;
213 :
214 : /* Set by during_housekeeping when the dynamic bound drops below
215 : slot_max_microblocks. Consumed by after_credit which publishes
216 : the updated bound to POH over the pack_poh link. */
217 : int pending_reduce_mb_bound;
218 :
219 : /* pacer and ticks_per_ns are used for pacing CUs through the slot,
220 : i.e. deciding when to schedule a microblock given the number of CUs
221 : that have been consumed so far. pacer is an opaque pacing object,
222 : which is initialized when the pack tile is packing a slot.
223 : ticks_per_ns is the cached value from tempo. */
224 : fd_pack_pacing_t pacer[1];
225 : double ticks_per_ns;
226 :
227 : /* last_successful_insert stores the tickcount of the last
228 : successful transaction insert. */
229 : long last_successful_insert;
230 :
231 : /* highest_observed_slot stores the highest slot number we've seen
232 : from any transaction coming from the resolv tile. When this
233 : increases, we expire old transactions. */
234 : ulong highest_observed_slot;
235 :
236 : /* microblock_duration_ns, and wait_duration
237 : respectively scaled to be in ticks instead of nanoseconds */
238 : ulong microblock_duration_ticks;
239 : #if !SMALL_MICROBLOCKS
240 : ulong wait_duration_ticks[ MAX_TXN_PER_MICROBLOCK+1UL ];
241 : #endif
242 :
243 : #if FD_PACK_USE_EXTRA_STORAGE
244 : /* In addition to the available transactions that pack knows about, we
245 : also store a larger ring buffer for handling cases when pack is
246 : full. This is an fd_deque. */
247 : fd_txn_e_t * extra_txn_deq;
248 : int insert_to_extra; /* whether the last insert was into pack or the extra deq */
249 : #endif
250 :
251 : fd_pack_in_ctx_t in[ 32 ];
252 : int in_kind[ 32 ];
253 :
254 : ulong execle_cnt;
255 : ulong execle_idle_bitset; /* bit i is 1 if we've observed *execle_current[i]==execle_expect[i] */
256 : int poll_cursor; /* in [0, execle_cnt), the next execle to poll */
257 : int use_consumed_cus;
258 : long skip_cnt;
259 : ulong * execle_current[ FD_PACK_MAX_EXECLE_TILES ];
260 : ulong execle_expect[ FD_PACK_MAX_EXECLE_TILES ];
261 : /* execle_ready_at[x] means don't check execle x until tickcount is at
262 : least execle_ready_at[x]. */
263 : long execle_ready_at[ FD_PACK_MAX_EXECLE_TILES ];
264 :
265 : fd_wksp_t * execle_out_mem;
266 : ulong execle_out_chunk0;
267 : ulong execle_out_wmark;
268 : ulong execle_out_chunk;
269 :
270 : fd_wksp_t * poh_out_mem;
271 : ulong poh_out_chunk0;
272 : ulong poh_out_wmark;
273 : ulong poh_out_chunk;
274 :
275 : ulong insert_result[ FD_PACK_INSERT_RETVAL_CNT ];
276 : fd_histf_t schedule_duration[ 1 ];
277 : fd_histf_t no_sched_duration[ 1 ];
278 : fd_histf_t insert_duration [ 1 ];
279 : fd_histf_t complete_duration[ 1 ];
280 :
281 : struct {
282 : uint metric_state;
283 : long metric_state_begin;
284 : long metric_timing[ 16 ];
285 : };
286 :
287 : /* last_sched_metrics is a snapshot of the schedule outcome counters
288 : during the last schedule which included at least one successful
289 : outcome. */
290 : fd_pack_sched_results_snap_t last_sched_metrics[ 1 ];
291 :
292 : /* last_sched_metrics is a snapshot of the schedule outcome counters
293 : at the last start-of-leader-block event. */
294 : fd_pack_sched_results_snap_t start_block_sched_metrics[ 1 ];
295 :
296 : struct {
297 : ulong id;
298 : ulong txn_cnt;
299 : ulong txn_received;
300 : ulong min_blockhash_slot;
301 : fd_txn_e_t * _txn[ FD_PACK_MAX_TXN_PER_BUNDLE ];
302 : fd_txn_e_t * const * bundle; /* points to _txn when non-NULL */
303 : } current_bundle[1];
304 :
305 : block_builder_info_t blk_engine_cfg[1];
306 :
307 : struct {
308 : int enabled;
309 : int ib_inserted; /* in this slot */
310 : fd_acct_addr_t vote_pubkey[1];
311 : fd_acct_addr_t identity_pubkey[1];
312 : fd_bundle_crank_gen_t gen[1];
313 : fd_acct_addr_t tip_receiver_owner[1];
314 : ulong epoch;
315 : fd_bundle_crank_tip_payment_config_t prev_config[1]; /* as of start of slot, then updated */
316 : uchar recent_blockhash[32];
317 : fd_ed25519_sig_t last_sig[1];
318 :
319 : fd_keyswitch_t * keyswitch;
320 : fd_keyguard_client_t keyguard_client[1];
321 :
322 : ulong metrics[4];
323 : } crank[1];
324 :
325 :
326 : /* Used between during_frag and after_frag */
327 : ulong pending_rebate_sz;
328 : union{ fd_pack_rebate_t rebate[1]; uchar footprint[USHORT_MAX]; } rebate[1];
329 : } fd_pack_ctx_t;
330 :
331 0 : #define BUNDLE_META_SZ 40UL
332 : FD_STATIC_ASSERT( sizeof(block_builder_info_t)==BUNDLE_META_SZ, blk_engine_cfg );
333 :
334 0 : #define FD_PACK_METRIC_STATE_TRANSACTIONS 0
335 0 : #define FD_PACK_METRIC_STATE_EXECLES 1
336 0 : #define FD_PACK_METRIC_STATE_LEADER 2
337 0 : #define FD_PACK_METRIC_STATE_MICROBLOCKS 3
338 :
339 : /* Updates one component of the metric state. If the state has changed,
340 : records the change. */
341 : static inline void
342 : update_metric_state( fd_pack_ctx_t * ctx,
343 : long effective_as_of,
344 : int type,
345 0 : int status ) {
346 0 : uint current_state = fd_uint_insert_bit( ctx->metric_state, type, status );
347 0 : if( FD_UNLIKELY( current_state!=ctx->metric_state ) ) {
348 0 : ctx->metric_timing[ ctx->metric_state ] += effective_as_of - ctx->metric_state_begin;
349 0 : ctx->metric_state_begin = effective_as_of;
350 0 : ctx->metric_state = current_state;
351 0 : }
352 0 : }
353 :
354 : static inline void
355 0 : remove_ib( fd_pack_ctx_t * ctx ) {
356 : /* It's likely the initializer bundle is long scheduled, but we want to
357 : try deleting it just in case. */
358 0 : if( FD_UNLIKELY( ctx->crank->enabled & ctx->crank->ib_inserted ) ) {
359 0 : ulong deleted = fd_pack_delete_transaction( ctx->pack, (fd_ed25519_sig_t const *)ctx->crank->last_sig );
360 0 : FD_MCNT_INC( PACK, TXN_DELETED, deleted );
361 0 : }
362 0 : ctx->crank->ib_inserted = 0;
363 0 : }
364 :
365 :
366 : FD_FN_CONST static inline ulong
367 0 : scratch_align( void ) {
368 0 : return 4096UL;
369 0 : }
370 :
371 : FD_FN_PURE static inline ulong
372 0 : scratch_footprint( fd_topo_tile_t const * tile ) {
373 0 : fd_pack_limits_t limits[1] = {{
374 0 : .max_cost_per_block = tile->pack.larger_max_cost_per_block ? LARGER_MAX_COST_PER_BLOCK : FD_PACK_MAX_COST_PER_BLOCK_UPPER_BOUND,
375 0 : .max_vote_cost_per_block = FD_PACK_MAX_VOTE_COST_PER_BLOCK_UPPER_BOUND,
376 0 : .max_write_cost_per_acct = FD_PACK_MAX_WRITE_COST_PER_ACCT_UPPER_BOUND,
377 0 : .max_data_bytes_per_block = tile->pack.larger_shred_limits_per_block ? LARGER_MAX_DATA_PER_BLOCK : FD_PACK_MAX_DATA_PER_BLOCK,
378 0 : .max_txn_per_microblock = EFFECTIVE_TXN_PER_MICROBLOCK,
379 0 : .max_microblocks_per_block = (ulong)UINT_MAX, /* Limit not known yet */
380 0 : .max_allocated_data_per_block = FD_PACK_MAX_ALLOCATED_DATA_PER_BLOCK,
381 0 : }};
382 :
383 0 : ulong l = FD_LAYOUT_INIT;
384 0 : l = FD_LAYOUT_APPEND( l, alignof( fd_pack_ctx_t ), sizeof( fd_pack_ctx_t ) );
385 0 : l = FD_LAYOUT_APPEND( l, fd_rng_align(), fd_rng_footprint() );
386 0 : l = FD_LAYOUT_APPEND( l, fd_pack_align(), fd_pack_footprint( tile->pack.max_pending_transactions,
387 0 : BUNDLE_META_SZ,
388 0 : tile->pack.execle_tile_count,
389 0 : limits ) );
390 : #if FD_PACK_USE_EXTRA_STORAGE
391 : l = FD_LAYOUT_APPEND( l, extra_txn_deq_align(), extra_txn_deq_footprint() );
392 : #endif
393 0 : return FD_LAYOUT_FINI( l, scratch_align() );
394 0 : }
395 :
396 : static inline void
397 : log_end_block_metrics( fd_pack_ctx_t * ctx,
398 : long now,
399 : char const * reason,
400 0 : ulong cus_consumed_in_block ) {
401 0 : #define DELTA( m ) (fd_metrics_tl[ MIDX(COUNTER, PACK, TXN_SCHEDULED_##m) ] - ctx->last_sched_metrics->sched_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_##m##_IDX ])
402 0 : #define AVAIL( m ) (fd_metrics_tl[ MIDX(GAUGE, PACK, TXN_AVAILABLE_##m) ])
403 0 : FD_LOG_INFO(( "pack_end_block(slot=%lu,%s,%lx,ticks_since_last_schedule=%ld,reasons=%lu,%lu,%lu,%lu,%lu,%lu,%lu,%lu;remaining=%lu+%lu+%lu+%lu;smallest=%lu;cus=%lu->%lu)",
404 0 : ctx->leader_slot, reason, ctx->execle_idle_bitset, now-ctx->last_sched_metrics->time,
405 0 : DELTA( TAKEN ), DELTA( CU_LIMIT ), DELTA( FAST_PATH ), DELTA( BYTE_LIMIT ), DELTA( ALLOC_LIMIT ), DELTA( WRITE_COST ), DELTA( SLOW_PATH ), DELTA( DEFER_SKIP ),
406 0 : AVAIL(REGULAR), AVAIL(VOTES), AVAIL(BUNDLES), AVAIL(CONFLICTING),
407 0 : (fd_metrics_tl[ MIDX(GAUGE, PACK, TXN_PENDING_SMALLEST_CU) ]),
408 0 : (cus_consumed_in_block),
409 0 : (fd_metrics_tl[ MIDX(GAUGE, PACK, BLOCK_CU_CONSUMED) ])
410 0 : ));
411 0 : #undef AVAIL
412 0 : #undef DELTA
413 0 : }
414 :
415 : static inline void
416 0 : get_done_packing( fd_pack_ctx_t * ctx, fd_done_packing_t * done_packing, int reason ) {
417 0 : done_packing->microblocks_in_slot = ctx->slot_microblock_cnt;
418 0 : done_packing->end_slot_reason = reason;
419 0 : fd_pack_get_block_limits( ctx->pack, done_packing->limits_usage, done_packing->limits );
420 :
421 0 : #define DELTA( mem, m ) (fd_metrics_tl[ MIDX(COUNTER, PACK, TXN_SCHEDULED_##m) ] - ctx->mem->sched_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_##m##_IDX ])
422 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_TAKEN_IDX ] = DELTA( start_block_sched_metrics, TAKEN );
423 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_CU_LIMIT_IDX ] = DELTA( start_block_sched_metrics, CU_LIMIT );
424 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_FAST_PATH_IDX ] = DELTA( start_block_sched_metrics, FAST_PATH );
425 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_BYTE_LIMIT_IDX ] = DELTA( start_block_sched_metrics, BYTE_LIMIT );
426 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_ALLOC_LIMIT_IDX ] = DELTA( start_block_sched_metrics, ALLOC_LIMIT );
427 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_WRITE_COST_IDX ] = DELTA( start_block_sched_metrics, WRITE_COST );
428 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_SLOW_PATH_IDX ] = DELTA( start_block_sched_metrics, SLOW_PATH );
429 0 : done_packing->block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_DEFER_SKIP_IDX ] = DELTA( start_block_sched_metrics, DEFER_SKIP );
430 :
431 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_TAKEN_IDX ] = DELTA( last_sched_metrics, TAKEN );
432 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_CU_LIMIT_IDX ] = DELTA( last_sched_metrics, CU_LIMIT );
433 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_FAST_PATH_IDX ] = DELTA( last_sched_metrics, FAST_PATH );
434 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_BYTE_LIMIT_IDX ] = DELTA( last_sched_metrics, BYTE_LIMIT );
435 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_ALLOC_LIMIT_IDX ] = DELTA( last_sched_metrics, ALLOC_LIMIT );
436 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_WRITE_COST_IDX ] = DELTA( last_sched_metrics, WRITE_COST );
437 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_SLOW_PATH_IDX ] = DELTA( last_sched_metrics, SLOW_PATH );
438 0 : done_packing->end_block_results[ FD_METRICS_ENUM_PACK_TXN_SCHEDULE_V_DEFER_SKIP_IDX ] = DELTA( last_sched_metrics, DEFER_SKIP );
439 0 : #undef DELTA
440 :
441 0 : fd_pack_get_pending_smallest( ctx->pack, done_packing->pending_smallest, done_packing->pending_votes_smallest );
442 0 : }
443 :
444 : static inline void
445 0 : metrics_write( fd_pack_ctx_t * ctx ) {
446 0 : FD_MCNT_ENUM_COPY( PACK, TXN_INSERTED, ctx->insert_result );
447 0 : FD_MCNT_ENUM_COPY( PACK, STATE_DURATION_NANOS, ((ulong*)ctx->metric_timing) );
448 0 : FD_MCNT_ENUM_COPY( PACK, BUNDLE_CRANK_RESULT, ctx->crank->metrics );
449 0 : FD_MHIST_COPY( PACK, SCHEDULE_MICROBLOCK_DURATION_SECONDS, ctx->schedule_duration );
450 0 : FD_MHIST_COPY( PACK, NO_SCHEDULE_MICROBLOCK_DURATION_SECONDS, ctx->no_sched_duration );
451 0 : FD_MHIST_COPY( PACK, INSERT_TRANSACTION_DURATION_SECONDS, ctx->insert_duration );
452 0 : FD_MHIST_COPY( PACK, COMPLETE_MICROBLOCK_DURATION_SECONDS, ctx->complete_duration );
453 :
454 0 : fd_pack_metrics_write( ctx->pack );
455 0 : }
456 :
457 : /* compute_dynamic_max_microblocks: Computes the upper bound on total
458 : microblocks based on remaining time and bank count.
459 :
460 : The basic idea here is that if there is 1ms left in the slot, we
461 : don't expect to schedule 130k microblocks. We can reduce our
462 : remaining budget which allows POH to advance hashing a bit further
463 : and avoid having to do a lot of hashing after the slot ends.
464 :
465 : The fastest we can execute a transaction is about 1us. With n
466 : execle tiles, that means we can execute at most n txn/us. If we have
467 : k ms left in the block, only reserve up to k*n*1000 microblocks. */
468 :
469 : static inline ulong
470 0 : compute_dynamic_max_microblocks( fd_pack_ctx_t * ctx ) {
471 0 : long now = ctx->approx_wallclock_ns + (long)((double)(fd_tickcount() - ctx->approx_tickcount) / ctx->ticks_per_ns);
472 0 : long end = ctx->slot_end_ns;
473 :
474 : /* If the slot has ended, don't reserve any more microblocks. */
475 0 : if( FD_UNLIKELY( now>=end ) ) return ctx->slot_microblock_cnt;
476 :
477 : /* remaining_ns * n / 1000 = (remaining_ns/1e6 ms) * n * 1000.
478 :
479 : Overflow: remaining_ns is at most ~4e8, n at most 64, so
480 : remaining_ns * n is at most ~2.6e10 << 1.8e19. */
481 :
482 0 : ulong remaining_ns = (ulong)(end - now);
483 0 : ulong n = ctx->execle_cnt;
484 0 : ulong cnt = ctx->slot_microblock_cnt;
485 0 : ulong R = ctx->slot_max_microblocks - cnt;
486 0 : ulong can_execute = remaining_ns * n / 1000UL;
487 :
488 0 : return cnt + fd_ulong_min( R, can_execute );
489 0 : }
490 :
491 : static inline void
492 0 : during_housekeeping( fd_pack_ctx_t * ctx ) {
493 0 : ctx->approx_wallclock_ns = fd_log_wallclock();
494 0 : ctx->approx_tickcount = fd_tickcount();
495 :
496 0 : if( FD_UNLIKELY( ctx->crank->enabled && fd_keyswitch_state_query( ctx->crank->keyswitch )==FD_KEYSWITCH_STATE_SWITCH_PENDING ) ) {
497 0 : fd_memcpy( ctx->crank->identity_pubkey, ctx->crank->keyswitch->bytes, 32UL );
498 0 : fd_keyswitch_state( ctx->crank->keyswitch, FD_KEYSWITCH_STATE_COMPLETED );
499 0 : }
500 :
501 0 : if( FD_LIKELY( ctx->leader_slot!=ULONG_MAX ) ) {
502 0 : ulong raw = compute_dynamic_max_microblocks( ctx );
503 0 : ulong prev = ctx->slot_dynamic_max_microblocks;
504 :
505 : /* Enforce monotonically decreasing. This ensures pack's bound is
506 : always smaller than POH's. */
507 0 : ctx->slot_dynamic_max_microblocks = fd_ulong_min( raw, prev );
508 :
509 : /* If the bound decreased, we must update pack's internal scheduling
510 : limit. Otherwise, pack could schedule a microblock that exceeds
511 : the remaining capacity from the tile's and poh's perspectives
512 : (e.g. pack might produce a 5-transaction microblock when only 4
513 : microblocks worth of space remain). */
514 0 : if( FD_UNLIKELY( ctx->slot_dynamic_max_microblocks < prev ) ) {
515 0 : fd_pack_limits_t limits[1];
516 0 : limits->max_cost_per_block = ctx->limits.slot_max_cost;
517 0 : limits->max_data_bytes_per_block = ctx->slot_max_data;
518 0 : limits->max_microblocks_per_block = ctx->slot_dynamic_max_microblocks;
519 0 : limits->max_vote_cost_per_block = ctx->limits.slot_max_vote_cost;
520 0 : limits->max_write_cost_per_acct = ctx->limits.slot_max_write_cost_per_acct;
521 0 : limits->max_txn_per_microblock = ULONG_MAX; /* unused */
522 0 : limits->max_allocated_data_per_block = FD_PACK_MAX_ALLOCATED_DATA_PER_BLOCK;
523 0 : fd_pack_set_block_limits( ctx->pack, limits );
524 :
525 0 : ctx->pending_reduce_mb_bound = 1; /* publish bound decrease */
526 0 : }
527 0 : }
528 0 : }
529 :
530 : static inline void
531 : before_credit( fd_pack_ctx_t * ctx,
532 : fd_stem_context_t * stem,
533 0 : int * charge_busy ) {
534 0 : (void)stem;
535 :
536 0 : if( FD_UNLIKELY( (ctx->cur_spot!=NULL) & !ctx->is_bundle ) ) {
537 0 : *charge_busy = 1;
538 :
539 : /* If we were overrun while processing a frag from an in, then
540 : cur_spot is left dangling and not cleaned up, so clean it up here
541 : (by returning the slot to the pool of free slots). If the last
542 : transaction was a bundle, then we don't want to return it. When
543 : we try to process the first transaction in the next bundle, we'll
544 : see we never got the full bundle and cancel the whole last
545 : bundle, returning all the storage to the pool. */
546 : #if FD_PACK_USE_EXTRA_STORAGE
547 : if( FD_LIKELY( !ctx->insert_to_extra ) ) fd_pack_insert_txn_cancel( ctx->pack, ctx->cur_spot );
548 : else extra_txn_deq_remove_tail( ctx->extra_txn_deq );
549 : #else
550 0 : fd_pack_insert_txn_cancel( ctx->pack, ctx->cur_spot );
551 0 : #endif
552 0 : ctx->cur_spot = NULL;
553 0 : }
554 0 : }
555 :
556 : #if FD_PACK_USE_EXTRA_STORAGE
557 : /* insert_from_extra: helper method to pop the transaction at the head
558 : off the extra txn deque and insert it into pack. Requires that
559 : ctx->extra_txn_deq is non-empty, but it's okay to call it if pack is
560 : full. Returns the result of fd_pack_insert_txn_fini. */
561 : static inline int
562 : insert_from_extra( fd_pack_ctx_t * ctx ) {
563 : fd_txn_e_t * spot = fd_pack_insert_txn_init( ctx->pack );
564 : fd_txn_e_t const * insert = extra_txn_deq_peek_head( ctx->extra_txn_deq );
565 : fd_txn_t const * insert_txn = TXN(insert->txnp);
566 : fd_memcpy( spot->txnp->payload, insert->txnp->payload, insert->txnp->payload_sz );
567 : fd_memcpy( TXN(spot->txnp), insert_txn, fd_txn_footprint( insert_txn->instr_cnt, insert_txn->addr_table_lookup_cnt ) );
568 : fd_memcpy( spot->alt_accts, insert->alt_accts, insert_txn->addr_table_adtl_cnt*sizeof(fd_acct_addr_t) );
569 : spot->txnp->payload_sz = insert->txnp->payload_sz;
570 : spot->txnp->source_tpu = insert->txnp->source_tpu;
571 : spot->txnp->source_ipv4 = insert->txnp->source_ipv4;
572 : spot->txnp->scheduler_arrival_time_nanos = insert->txnp->scheduler_arrival_time_nanos;
573 : extra_txn_deq_remove_head( ctx->extra_txn_deq );
574 :
575 : ulong blockhash_slot = insert->txnp->blockhash_slot;
576 :
577 : ulong deleted;
578 : long insert_duration = -fd_tickcount();
579 : int result = fd_pack_insert_txn_fini( ctx->pack, spot, blockhash_slot, &deleted );
580 : insert_duration += fd_tickcount();
581 :
582 : FD_MCNT_INC( PACK, TXN_DELETED, deleted );
583 : ctx->insert_result[ result + FD_PACK_INSERT_RETVAL_OFF ]++;
584 : fd_histf_sample( ctx->insert_duration, (ulong)insert_duration );
585 : FD_MCNT_INC( PACK, TXN_EXTRA_RETRIEVED, 1UL );
586 : return result;
587 : }
588 : #endif
589 :
590 : static inline void
591 : after_credit( fd_pack_ctx_t * ctx,
592 : fd_stem_context_t * stem,
593 : int * opt_poll_in FD_PARAM_UNUSED,
594 0 : int * charge_busy ) {
595 0 : if( FD_UNLIKELY( (ctx->skip_cnt--)>0L ) ) return; /* It would take ages for this to hit LONG_MIN */
596 :
597 0 : long now = fd_tickcount();
598 :
599 0 : int pacing_execle_cnt = (int)fd_pack_pacing_enabled_bank_cnt( ctx->pacer, now );
600 :
601 0 : ulong execle_cnt = ctx->execle_cnt;
602 :
603 :
604 : /* If any execle are busy, check one of the busy ones see if it is
605 : still busy. */
606 0 : if( FD_LIKELY( ctx->execle_idle_bitset!=fd_ulong_mask_lsb( (int)execle_cnt ) ) ) {
607 0 : int poll_cursor = ctx->poll_cursor;
608 0 : ulong busy_bitset = (~ctx->execle_idle_bitset) & fd_ulong_mask_lsb( (int)execle_cnt );
609 :
610 : /* Suppose execle_cnt is 4 and idle_bitset looks something like this
611 : (pretending it's a uchar):
612 : 0000 1001
613 : ^ busy cursor is 1
614 : Then busy_bitset is
615 : 0000 0110
616 : Rotate it right by 2 bits
617 : 1000 0001
618 : Find lsb returns 0, so busy cursor remains 2, and we poll
619 : execle 2.
620 :
621 : If instead idle_bitset were
622 : 0000 1110
623 : ^
624 : The rotated version would be
625 : 0100 0000
626 : Find lsb will return 6, so busy cursor would be set to 0, and
627 : we'd poll execle 0, which is the right one. */
628 0 : poll_cursor++;
629 0 : poll_cursor = (poll_cursor + fd_ulong_find_lsb( fd_ulong_rotate_right( busy_bitset, (poll_cursor&63) ) )) & 63;
630 :
631 0 : if( FD_UNLIKELY(
632 : /* if microblock duration is 0, bypass the execle_ready_at check
633 : to avoid a potential cache miss. Can't use an ifdef here
634 : because FD_UNLIKELY is a macro, but the compiler should
635 : eliminate the check easily. */
636 0 : ( (MICROBLOCK_DURATION_NS==0L) || (ctx->execle_ready_at[poll_cursor]<now) ) &&
637 0 : (fd_fseq_query( ctx->execle_current[poll_cursor] )==ctx->execle_expect[poll_cursor]) ) ) {
638 0 : *charge_busy = 1;
639 0 : ctx->execle_idle_bitset |= 1UL<<poll_cursor;
640 :
641 0 : long complete_duration = -fd_tickcount();
642 0 : int completed = fd_pack_microblock_complete( ctx->pack, (ulong)poll_cursor );
643 0 : complete_duration += fd_tickcount();
644 0 : if( FD_LIKELY( completed ) ) fd_histf_sample( ctx->complete_duration, (ulong)complete_duration );
645 0 : }
646 :
647 0 : ctx->poll_cursor = poll_cursor;
648 0 : }
649 :
650 :
651 : /* If we time out on our slot, then stop being leader. This can only
652 : happen in the first after_credit after a housekeeping. */
653 0 : if( FD_UNLIKELY( ctx->approx_wallclock_ns>=ctx->slot_end_ns && ctx->leader_slot!=ULONG_MAX ) ) {
654 0 : *charge_busy = 1;
655 :
656 0 : fd_done_packing_t * done_packing = fd_chunk_to_laddr( ctx->poh_out_mem, ctx->poh_out_chunk );
657 0 : get_done_packing( ctx, done_packing, FD_PACK_END_SLOT_REASON_TIME ); /* needs to be called before fd_pack_end_block */
658 0 : fd_pack_end_block( ctx->pack );
659 0 : fd_pack_get_top_writers( ctx->pack, done_packing->limits_usage->top_writers ); /* needs to be called after fd_pack_end_block */
660 :
661 0 : fd_stem_publish( stem, 1UL, fd_disco_execle_sig( ctx->leader_slot, ctx->pack_idx ), ctx->poh_out_chunk, sizeof(fd_done_packing_t), 0UL, 0UL, fd_frag_meta_ts_comp( fd_tickcount() ) );
662 0 : ctx->poh_out_chunk = fd_dcache_compact_next( ctx->poh_out_chunk, sizeof(fd_done_packing_t), ctx->poh_out_chunk0, ctx->poh_out_wmark );
663 0 : ctx->pack_idx++;
664 :
665 0 : log_end_block_metrics( ctx, now, "time", done_packing->limits_usage->block_cost );
666 0 : ctx->drain_execle = 1;
667 0 : ctx->leader_slot = ULONG_MAX;
668 0 : ctx->slot_microblock_cnt = 0UL;
669 0 : remove_ib( ctx );
670 :
671 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_LEADER, 0 );
672 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_EXECLES, 0 );
673 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_MICROBLOCKS, 0 );
674 0 : return;
675 0 : }
676 :
677 : /* Am I leader? If not, see about inserting at most one transaction
678 : from extra storage. It's important not to insert too many
679 : transactions here, or we won't end up servicing dedup_pack enough.
680 : If extra storage is empty or pack is full, do nothing. */
681 0 : if( FD_UNLIKELY( ctx->leader_slot==ULONG_MAX ) ) {
682 : #if FD_PACK_USE_EXTRA_STORAGE
683 : if( FD_UNLIKELY( !extra_txn_deq_empty( ctx->extra_txn_deq ) &&
684 : fd_pack_avail_txn_cnt( ctx->pack )<ctx->max_pending_transactions ) ) {
685 : *charge_busy = 1;
686 :
687 : int result = insert_from_extra( ctx );
688 : if( FD_LIKELY( result>=0 ) ) ctx->last_successful_insert = now;
689 : }
690 : #endif
691 0 : return;
692 0 : }
693 :
694 : /* Am I in drain mode? If so, check if I can exit it */
695 0 : if( FD_UNLIKELY( ctx->drain_execle ) ) {
696 0 : if( FD_LIKELY( ctx->execle_idle_bitset==fd_ulong_mask_lsb( (int)execle_cnt ) ) ) {
697 0 : ctx->drain_execle = 0;
698 :
699 : /* Pack notifies poh when execle are drained so that poh can
700 : relinquish pack's ownership over the slot execle (by decrementing
701 : its Arc). We do this by sending a FD_PACK_MSG_DONE_DRAINING
702 : sig over the pack_poh mcache.
703 :
704 : TODO: This is only needed for Frankendancer, not Firedancer,
705 : which manages bank lifetime different. */
706 0 : fd_stem_publish( stem, 1UL, FD_PACK_MSG_DONE_DRAINING, 0UL, 0UL, 0UL, 0UL, fd_frag_meta_ts_comp( fd_tickcount() ) );
707 0 : *charge_busy = 1;
708 0 : return;
709 0 : } else {
710 0 : return;
711 0 : }
712 0 : }
713 :
714 0 : if( FD_UNLIKELY( ctx->pending_reduce_mb_bound ) ) {
715 0 : ctx->pending_reduce_mb_bound = 0;
716 0 : ulong * dst = fd_chunk_to_laddr( ctx->poh_out_mem, ctx->poh_out_chunk );
717 0 : *dst = ctx->slot_dynamic_max_microblocks;
718 0 : fd_stem_publish( stem, 1UL, FD_PACK_MSG_REDUCE_MB_BOUND, ctx->poh_out_chunk, sizeof(ulong), 0UL, 0UL, fd_frag_meta_ts_comp( fd_tickcount() ) );
719 0 : ctx->poh_out_chunk = fd_dcache_compact_next( ctx->poh_out_chunk, sizeof(ulong), ctx->poh_out_chunk0, ctx->poh_out_wmark );
720 0 : *charge_busy = 1;
721 0 : return;
722 0 : }
723 :
724 : /* Have I sent the max allowed microblocks? Nothing to do. */
725 0 : if( FD_UNLIKELY( ctx->slot_microblock_cnt>=ctx->slot_dynamic_max_microblocks ) ) return;
726 :
727 : /* Do I have enough transactions and/or have I waited enough time? */
728 : #if !SMALL_MICROBLOCKS
729 : if( FD_UNLIKELY( (ulong)(now-ctx->last_successful_insert) <
730 : ctx->wait_duration_ticks[ fd_ulong_min( fd_pack_avail_txn_cnt( ctx->pack ), MAX_TXN_PER_MICROBLOCK ) ] ) ) {
731 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_TRANSACTIONS, 0 );
732 : return;
733 : }
734 : #endif
735 :
736 0 : int any_ready = 0;
737 0 : int any_scheduled = 0;
738 :
739 0 : *charge_busy = 1;
740 :
741 0 : if( FD_LIKELY( ctx->crank->enabled ) ) {
742 0 : block_builder_info_t const * top_meta = fd_pack_peek_bundle_meta( ctx->pack );
743 0 : if( FD_UNLIKELY( top_meta ) ) {
744 : /* Have bundles, in a reasonable state to crank. */
745 :
746 0 : fd_txn_e_t * _bundle[ 1UL ];
747 0 : fd_txn_e_t * const * bundle = fd_pack_insert_bundle_init( ctx->pack, _bundle, 1UL );
748 :
749 0 : ulong txn_sz = fd_bundle_crank_generate( ctx->crank->gen, ctx->crank->prev_config, top_meta->commission_pubkey,
750 0 : ctx->crank->identity_pubkey, ctx->crank->tip_receiver_owner, ctx->crank->epoch, top_meta->commission,
751 0 : bundle[0]->txnp->payload, TXN( bundle[0]->txnp ) );
752 :
753 0 : if( FD_LIKELY( txn_sz==0UL ) ) { /* Everything in good shape! */
754 0 : fd_pack_insert_bundle_cancel( ctx->pack, bundle, 1UL );
755 0 : fd_pack_set_initializer_bundles_ready( ctx->pack );
756 0 : ctx->crank->metrics[ 0 ]++; /* BUNDLE_CRANK_RESULT_NOT_NEEDED */
757 0 : }
758 0 : else if( FD_LIKELY( txn_sz<ULONG_MAX ) ) {
759 0 : bundle[0]->txnp->payload_sz = (ushort)txn_sz;
760 0 : bundle[0]->txnp->source_tpu = FD_TXN_M_TPU_SOURCE_BUNDLE;
761 0 : bundle[0]->txnp->source_ipv4 = 0; /* not applicable */
762 0 : bundle[0]->txnp->scheduler_arrival_time_nanos = ctx->approx_wallclock_ns + (long)((double)(fd_tickcount() - ctx->approx_tickcount) / ctx->ticks_per_ns);
763 0 : memcpy( bundle[0]->txnp->payload+TXN(bundle[0]->txnp)->recent_blockhash_off, ctx->crank->recent_blockhash, 32UL );
764 :
765 0 : fd_keyguard_client_sign( ctx->crank->keyguard_client, bundle[0]->txnp->payload+1UL,
766 0 : bundle[0]->txnp->payload+65UL, txn_sz-65UL, FD_KEYGUARD_SIGN_TYPE_ED25519 );
767 :
768 0 : memcpy( ctx->crank->last_sig, bundle[0]->txnp->payload+1UL, 64UL );
769 :
770 0 : ctx->crank->ib_inserted = 1;
771 0 : ulong deleted;
772 0 : int retval = fd_pack_insert_bundle_fini( ctx->pack, bundle, 1UL, ctx->leader_slot-1UL, 1, NULL, &deleted );
773 0 : FD_MCNT_INC( PACK, TXN_DELETED, deleted );
774 0 : ctx->insert_result[ retval + FD_PACK_INSERT_RETVAL_OFF ]++;
775 0 : if( FD_UNLIKELY( retval<0 ) ) {
776 0 : ctx->crank->metrics[ 3 ]++; /* BUNDLE_CRANK_RESULT_INSERTION_FAILED */
777 0 : FD_LOG_WARNING(( "inserting initializer bundle returned %i", retval ));
778 0 : } else {
779 : /* Update the cached copy of the on-chain state. This seems a
780 : little dangerous, since we're updating it as if the bundle
781 : succeeded without knowing if that's true, but here's why
782 : it's safe:
783 :
784 : From now until we get the rebate call for this initializer
785 : bundle (which lets us know if it succeeded or failed), pack
786 : will be in [Pending] state, which means peek_bundle_meta
787 : will return NULL, so we won't read this state.
788 :
789 : Then, if the initializer bundle failed, we'll go into
790 : [Failed] IB state until the end of the block, which will
791 : cause top_meta to remain NULL so we don't read these values
792 : again.
793 :
794 : Otherwise, the initializer bundle succeeded, which means
795 : that these are the right values to use. */
796 0 : fd_bundle_crank_apply( ctx->crank->gen, ctx->crank->prev_config, top_meta->commission_pubkey,
797 0 : ctx->crank->tip_receiver_owner, ctx->crank->epoch, top_meta->commission );
798 0 : ctx->crank->metrics[ 1 ]++; /* BUNDLE_CRANK_RESULT_INSERTED */
799 0 : }
800 0 : } else {
801 : /* Already logged a warning in this case */
802 0 : fd_pack_insert_bundle_cancel( ctx->pack, bundle, 1UL );
803 0 : ctx->crank->metrics[ 2 ]++; /* BUNDLE_CRANK_RESULT_CREATION_FAILED' */
804 0 : }
805 0 : }
806 0 : }
807 :
808 : /* Try to schedule the next microblock. */
809 0 : if( FD_LIKELY( ctx->execle_idle_bitset ) ) { /* Optimize for schedule */
810 0 : any_ready = 1;
811 :
812 0 : int i = fd_ulong_find_lsb( ctx->execle_idle_bitset );
813 :
814 0 : int flags;
815 :
816 0 : switch( ctx->strategy ) {
817 0 : default:
818 0 : case FD_PACK_STRATEGY_PERF:
819 0 : flags = FD_PACK_SCHEDULE_VOTE | FD_PACK_SCHEDULE_BUNDLE | FD_PACK_SCHEDULE_TXN;
820 0 : break;
821 0 : case FD_PACK_STRATEGY_BALANCED:
822 : /* We want to exempt votes from pacing, so we always allow
823 : scheduling votes. It doesn't really make much sense to pace
824 : bundles, because they get scheduled in FIFO order. However,
825 : we keep pacing for normal transactions. For example, if
826 : pacing_execle_cnt is 0, then pack won't schedule normal
827 : transactions to any execle tile. */
828 0 : flags = FD_PACK_SCHEDULE_VOTE | fd_int_if( i==0, FD_PACK_SCHEDULE_BUNDLE, 0 )
829 0 : | fd_int_if( i<pacing_execle_cnt, FD_PACK_SCHEDULE_TXN, 0 );
830 0 : break;
831 0 : }
832 :
833 0 : fd_txn_e_t * microblock_dst = fd_chunk_to_laddr( ctx->execle_out_mem, ctx->execle_out_chunk );
834 0 : long schedule_duration = -fd_tickcount();
835 0 : ulong schedule_cnt = fd_pack_schedule_next_microblock( ctx->pack, CUS_PER_MICROBLOCK, VOTE_FRACTION, (ulong)i, flags, microblock_dst );
836 0 : schedule_duration += fd_tickcount();
837 0 : fd_histf_sample( (schedule_cnt>0UL) ? ctx->schedule_duration : ctx->no_sched_duration, (ulong)schedule_duration );
838 :
839 0 : if( FD_LIKELY( schedule_cnt ) ) {
840 0 : any_scheduled = 1;
841 0 : long now2 = fd_tickcount();
842 0 : ulong tsorig = (ulong)fd_frag_meta_ts_comp( now ); /* A bound on when we observed execle was idle */
843 0 : ulong tspub = (ulong)fd_frag_meta_ts_comp( now2 );
844 0 : ulong chunk = ctx->execle_out_chunk;
845 0 : ulong msg_sz = schedule_cnt*sizeof(fd_txn_e_t);
846 0 : fd_microblock_execle_trailer_t * trailer = (fd_microblock_execle_trailer_t*)(microblock_dst+schedule_cnt);
847 0 : trailer->bank = ctx->leader_bank;
848 0 : trailer->bank_idx = ctx->leader_bank_idx;
849 0 : trailer->microblock_idx = ctx->slot_microblock_cnt;
850 0 : trailer->pack_idx = ctx->pack_idx;
851 0 : trailer->pack_txn_idx = ctx->pack_txn_cnt;
852 0 : trailer->is_bundle = !!(microblock_dst->txnp->flags & FD_TXN_P_FLAGS_BUNDLE);
853 :
854 : /* When sending MAX_TXN_PER_MICROBLOCK transactions as fd_txn_e_t
855 : to execle, there must be room for the trailer at the end. */
856 0 : FD_STATIC_ASSERT( MAX_TXN_PER_MICROBLOCK*sizeof(fd_txn_e_t)+sizeof(fd_microblock_execle_trailer_t)<=MAX_MICROBLOCK_SZ, pack_execle_mtu );
857 :
858 0 : ulong sig = fd_disco_poh_sig( ctx->leader_slot, POH_PKT_TYPE_MICROBLOCK, (ulong)i );
859 0 : fd_stem_publish( stem, 0UL, sig, chunk, msg_sz+sizeof(fd_microblock_execle_trailer_t), 0UL, tsorig, tspub );
860 0 : ctx->execle_expect[ i ] = stem->seqs[0]-1UL;
861 0 : ctx->execle_ready_at[i] = now2 + (long)ctx->microblock_duration_ticks;
862 0 : ctx->execle_out_chunk = fd_dcache_compact_next( ctx->execle_out_chunk, msg_sz+sizeof(fd_microblock_execle_trailer_t), ctx->execle_out_chunk0, ctx->execle_out_wmark );
863 0 : ctx->slot_microblock_cnt += fd_ulong_if( trailer->is_bundle, schedule_cnt, 1UL );
864 0 : ctx->pack_idx += fd_uint_if( trailer->is_bundle, (uint)schedule_cnt, 1U );
865 0 : ctx->pack_txn_cnt += schedule_cnt;
866 :
867 0 : ctx->execle_idle_bitset = fd_ulong_pop_lsb( ctx->execle_idle_bitset );
868 0 : ctx->skip_cnt = (long)schedule_cnt * fd_long_if( ctx->use_consumed_cus, (long)execle_cnt/2L, 1L );
869 0 : fd_pack_pacing_update_consumed_cus( ctx->pacer, fd_pack_current_block_cost( ctx->pack ), now2 );
870 :
871 0 : ctx->last_sched_metrics->time = now2;
872 0 : fd_pack_get_sched_metrics( ctx->pack, ctx->last_sched_metrics->sched_results );
873 :
874 : /* If we're using CU rebates, then we have one in for each execle
875 : in addition to the two normal ones. We want to skip schedule
876 : attempts for (execle_cnt + 1) link polls after a successful
877 : schedule attempt. */
878 0 : fd_long_store_if( ctx->use_consumed_cus, &(ctx->skip_cnt), (long)(ctx->execle_cnt + 1) );
879 0 : }
880 0 : }
881 :
882 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_EXECLES, any_ready );
883 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_MICROBLOCKS, any_scheduled );
884 0 : now = fd_tickcount();
885 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_TRANSACTIONS, fd_pack_avail_txn_cnt( ctx->pack )>0 );
886 :
887 : #if FD_PACK_USE_EXTRA_STORAGE
888 : if( FD_UNLIKELY( !extra_txn_deq_empty( ctx->extra_txn_deq ) ) ) {
889 : /* Don't start pulling from the extra storage until the available
890 : transaction count drops below half. */
891 : ulong avail_space = (ulong)fd_long_max( 0L, (long)(ctx->max_pending_transactions>>1)-(long)fd_pack_avail_txn_cnt( ctx->pack ) );
892 : ulong qty_to_insert = fd_ulong_min( 10UL, fd_ulong_min( extra_txn_deq_cnt( ctx->extra_txn_deq ), avail_space ) );
893 : int any_successes = 0;
894 : for( ulong i=0UL; i<qty_to_insert; i++ ) any_successes |= (0<=insert_from_extra( ctx ));
895 : if( FD_LIKELY( any_successes ) ) ctx->last_successful_insert = now;
896 : }
897 : #endif
898 :
899 : /* Did we send the maximum allowed microblocks? Then end the slot. */
900 0 : if( FD_UNLIKELY( ctx->slot_microblock_cnt==ctx->slot_dynamic_max_microblocks )) {
901 0 : *charge_busy = 1;
902 :
903 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_LEADER, 0 );
904 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_EXECLES, 0 );
905 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_MICROBLOCKS, 0 );
906 : /* The pack object also does this accounting and increases this
907 : metric, but we end the slot early so won't see it unless we also
908 : increment it here. */
909 0 : FD_MCNT_INC( PACK, MICROBLOCK_PER_BLOCK_LIMIT_REACHED, 1UL );
910 :
911 0 : fd_done_packing_t * done_packing = fd_chunk_to_laddr( ctx->poh_out_mem, ctx->poh_out_chunk );
912 0 : get_done_packing( ctx, done_packing, FD_PACK_END_SLOT_REASON_MICROBLOCK );
913 0 : fd_pack_end_block( ctx->pack );
914 0 : fd_pack_get_top_writers( ctx->pack, done_packing->limits_usage->top_writers );
915 :
916 0 : fd_stem_publish( stem, 1UL, fd_disco_execle_sig( ctx->leader_slot, ctx->pack_idx ), ctx->poh_out_chunk, sizeof(fd_done_packing_t), 0UL, 0UL, fd_frag_meta_ts_comp( fd_tickcount() ) );
917 0 : ctx->poh_out_chunk = fd_dcache_compact_next( ctx->poh_out_chunk, sizeof(fd_done_packing_t), ctx->poh_out_chunk0, ctx->poh_out_wmark );
918 0 : ctx->pack_idx++;
919 :
920 0 : log_end_block_metrics( ctx, now, "microblock", done_packing->limits_usage->block_cost );
921 0 : ctx->drain_execle = 1;
922 0 : ctx->leader_slot = ULONG_MAX;
923 0 : ctx->slot_microblock_cnt = 0UL;
924 0 : remove_ib( ctx );
925 :
926 0 : return;
927 0 : }
928 0 : }
929 :
930 :
931 : /* At this point, we have started receiving frag seq with details in
932 : mline at time now. Speculatively process it here. */
933 :
934 : static inline void
935 : during_frag( fd_pack_ctx_t * ctx,
936 : ulong in_idx,
937 : ulong seq FD_PARAM_UNUSED,
938 : ulong sig,
939 : ulong chunk,
940 : ulong sz,
941 0 : ulong ctl FD_PARAM_UNUSED ) {
942 :
943 0 : uchar const * dcache_entry = fd_chunk_to_laddr_const( ctx->in[ in_idx ].mem, chunk );
944 :
945 0 : switch( ctx->in_kind[ in_idx ] ) {
946 0 : case IN_KIND_REPLAY: {
947 0 : if( FD_LIKELY( sig==REPLAY_SIG_TXN_EXECUTED ) ) {
948 0 : fd_replay_txn_executed_t const * txn_executed = fd_type_pun_const( dcache_entry );
949 0 : ctx->txn_committed = !!txn_executed->is_committable;
950 0 : if( FD_UNLIKELY( !txn_executed->is_committable ) ) return;
951 0 : memcpy( ctx->executed_txn_sig, fd_txn_get_signatures( TXN(txn_executed->txn), txn_executed->txn->payload ), FD_TXN_SIGNATURE_SZ );
952 0 : return;
953 0 : }
954 0 : if( FD_LIKELY( sig!=REPLAY_SIG_BECAME_LEADER ) ) return;
955 :
956 : /* There was a leader transition. Handle it. */
957 0 : if( FD_UNLIKELY( chunk<ctx->in[ in_idx ].chunk0 || chunk>ctx->in[ in_idx ].wmark || sz!=sizeof(fd_became_leader_t) ) )
958 0 : FD_LOG_ERR(( "chunk %lu %lu corrupt, not in range [%lu,%lu]", chunk, sz, ctx->in[ in_idx ].chunk0, ctx->in[ in_idx ].wmark ));
959 :
960 0 : fd_memcpy( ctx->_became_leader, dcache_entry, sizeof(fd_became_leader_t) );
961 0 : return;
962 0 : }
963 0 : case IN_KIND_POH: {
964 : /* Not interested in stamped microblocks, only leader updates. */
965 0 : if( fd_disco_poh_sig_pkt_type( sig )!=POH_PKT_TYPE_BECAME_LEADER ) return;
966 :
967 : /* There was a leader transition. Handle it. */
968 0 : if( FD_UNLIKELY( chunk<ctx->in[ in_idx ].chunk0 || chunk>ctx->in[ in_idx ].wmark || sz!=sizeof(fd_became_leader_t) ) )
969 0 : FD_LOG_ERR(( "chunk %lu %lu corrupt, not in range [%lu,%lu]", chunk, sz, ctx->in[ in_idx ].chunk0, ctx->in[ in_idx ].wmark ));
970 :
971 0 : fd_memcpy( ctx->_became_leader, dcache_entry, sizeof(fd_became_leader_t) );
972 0 : return;
973 0 : }
974 0 : case IN_KIND_EXECLE: {
975 0 : FD_TEST( ctx->use_consumed_cus );
976 : /* For a previous slot */
977 0 : if( FD_UNLIKELY( sig!=ctx->leader_slot ) ) return;
978 :
979 0 : if( FD_UNLIKELY( chunk<ctx->in[ in_idx ].chunk0 || chunk>ctx->in[ in_idx ].wmark || sz<FD_PACK_REBATE_MIN_SZ
980 0 : || sz>FD_PACK_REBATE_MAX_SZ ) )
981 0 : FD_LOG_ERR(( "chunk %lu %lu corrupt, not in range [%lu,%lu]", chunk, sz, ctx->in[ in_idx ].chunk0, ctx->in[ in_idx ].wmark ));
982 :
983 0 : ctx->pending_rebate_sz = sz;
984 0 : fd_memcpy( ctx->rebate, dcache_entry, sz );
985 0 : return;
986 0 : }
987 0 : case IN_KIND_RESOLV: {
988 0 : if( FD_UNLIKELY( chunk<ctx->in[ in_idx ].chunk0 || chunk>ctx->in[ in_idx ].wmark || sz>FD_TPU_RESOLVED_MTU ) )
989 0 : FD_LOG_ERR(( "chunk %lu %lu corrupt, not in range [%lu,%lu]", chunk, sz, ctx->in[ in_idx ].chunk0, ctx->in[ in_idx ].wmark ));
990 :
991 0 : fd_txn_m_t * txnm = (fd_txn_m_t *)dcache_entry;
992 0 : ulong payload_sz = txnm->payload_sz;
993 0 : ulong txn_t_sz = txnm->txn_t_sz;
994 0 : uint source_ipv4 = txnm->source_ipv4;
995 0 : uchar source_tpu = txnm->source_tpu;
996 0 : FD_TEST( payload_sz<=FD_TPU_MTU );
997 0 : FD_TEST( txn_t_sz <=FD_TXN_MAX_SZ );
998 0 : fd_txn_t * txn = fd_txn_m_txn_t( txnm );
999 :
1000 0 : ulong addr_table_sz = 32UL*txn->addr_table_adtl_cnt;
1001 0 : FD_TEST( addr_table_sz<=32UL*FD_TXN_ACCT_ADDR_MAX );
1002 :
1003 0 : if( FD_UNLIKELY( (ctx->leader_slot==ULONG_MAX) & (sig>ctx->highest_observed_slot) ) ) {
1004 : /* Using the resolv tile's knowledge of the current slot is a bit
1005 : of a hack, since we don't get any info if there are no
1006 : transactions and we're not leader. We're actually in exactly
1007 : the case where that's okay though. The point of calling
1008 : expire_before long before we become leader is so that we don't
1009 : drop new but low-fee-paying transactions when pack is clogged
1010 : with expired but high-fee-paying transactions. That can only
1011 : happen if we are getting transactions. */
1012 0 : ctx->highest_observed_slot = sig;
1013 0 : ulong exp_cnt = fd_pack_expire_before( ctx->pack, fd_ulong_max( ctx->highest_observed_slot, TRANSACTION_LIFETIME_SLOTS )-TRANSACTION_LIFETIME_SLOTS );
1014 0 : FD_MCNT_INC( PACK, TXN_EXPIRED, exp_cnt );
1015 0 : }
1016 :
1017 :
1018 0 : ulong bundle_id = txnm->block_engine.bundle_id;
1019 0 : if( FD_UNLIKELY( bundle_id ) ) {
1020 0 : ctx->is_bundle = 1;
1021 0 : if( FD_LIKELY( bundle_id!=ctx->current_bundle->id ) ) {
1022 0 : if( FD_UNLIKELY( ctx->current_bundle->bundle ) ) {
1023 0 : FD_MCNT_INC( PACK, TXN_PARTIAL_BUNDLE, ctx->current_bundle->txn_received );
1024 0 : fd_pack_insert_bundle_cancel( ctx->pack, ctx->current_bundle->bundle, ctx->current_bundle->txn_cnt );
1025 0 : }
1026 0 : ctx->current_bundle->id = bundle_id;
1027 0 : ctx->current_bundle->txn_cnt = txnm->block_engine.bundle_txn_cnt;
1028 0 : ctx->current_bundle->min_blockhash_slot = ULONG_MAX;
1029 0 : ctx->current_bundle->txn_received = 0UL;
1030 :
1031 0 : if( FD_UNLIKELY( ctx->current_bundle->txn_cnt==0UL ) ) {
1032 0 : FD_MCNT_INC( PACK, TXN_PARTIAL_BUNDLE, 1UL );
1033 0 : ctx->current_bundle->id = 0UL;
1034 0 : return;
1035 0 : }
1036 0 : ctx->blk_engine_cfg->commission = txnm->block_engine.commission;
1037 0 : memcpy( ctx->blk_engine_cfg->commission_pubkey->b, txnm->block_engine.commission_pubkey, 32UL );
1038 :
1039 0 : ctx->current_bundle->bundle = fd_pack_insert_bundle_init( ctx->pack, ctx->current_bundle->_txn, ctx->current_bundle->txn_cnt );
1040 0 : }
1041 0 : ctx->cur_spot = ctx->current_bundle->bundle[ ctx->current_bundle->txn_received ];
1042 0 : ctx->current_bundle->min_blockhash_slot = fd_ulong_min( ctx->current_bundle->min_blockhash_slot, sig );
1043 0 : } else {
1044 0 : ctx->is_bundle = 0;
1045 : #if FD_PACK_USE_EXTRA_STORAGE
1046 : if( FD_LIKELY( ctx->leader_slot!=ULONG_MAX || fd_pack_avail_txn_cnt( ctx->pack )<ctx->max_pending_transactions ) ) {
1047 : ctx->cur_spot = fd_pack_insert_txn_init( ctx->pack );
1048 : ctx->insert_to_extra = 0;
1049 : } else {
1050 : if( FD_UNLIKELY( extra_txn_deq_full( ctx->extra_txn_deq ) ) ) {
1051 : extra_txn_deq_remove_head( ctx->extra_txn_deq );
1052 : FD_MCNT_INC( PACK, TXN_EXTRA_DROPPED, 1UL );
1053 : }
1054 : ctx->cur_spot = extra_txn_deq_peek_tail( extra_txn_deq_insert_tail( ctx->extra_txn_deq ) );
1055 : /* We want to store the current time in cur_spot so that we can
1056 : track its expiration better. We just stash it in the CU
1057 : fields, since those aren't important right now. */
1058 : ctx->cur_spot->txnp->blockhash_slot = sig;
1059 : ctx->insert_to_extra = 1;
1060 : FD_MCNT_INC( PACK, TXN_EXTRA_INSERTED, 1UL );
1061 : }
1062 : #else
1063 0 : ctx->cur_spot = fd_pack_insert_txn_init( ctx->pack );
1064 0 : #endif
1065 0 : }
1066 :
1067 : /* We get transactions from the resolv tile.
1068 : The transactions should have been parsed and verified. */
1069 0 : FD_MCNT_INC( PACK, TXN_NORMAL_RX, 1UL );
1070 :
1071 0 : fd_memcpy( ctx->cur_spot->txnp->payload, fd_txn_m_payload( txnm ), payload_sz );
1072 0 : fd_memcpy( TXN(ctx->cur_spot->txnp), txn, txn_t_sz );
1073 0 : fd_memcpy( ctx->cur_spot->alt_accts, fd_txn_m_alut( txnm ), addr_table_sz );
1074 0 : ctx->cur_spot->txnp->scheduler_arrival_time_nanos = ctx->approx_wallclock_ns + (long)((double)(fd_tickcount() - ctx->approx_tickcount) / ctx->ticks_per_ns);
1075 0 : ctx->cur_spot->txnp->payload_sz = payload_sz;
1076 0 : ctx->cur_spot->txnp->source_ipv4 = source_ipv4;
1077 0 : ctx->cur_spot->txnp->source_tpu = source_tpu;
1078 :
1079 0 : break;
1080 0 : }
1081 0 : case IN_KIND_EXECUTED_TXN: {
1082 0 : FD_TEST( sz==64UL );
1083 0 : fd_memcpy( ctx->executed_txn_sig, dcache_entry, sz );
1084 0 : break;
1085 0 : }
1086 0 : }
1087 0 : }
1088 :
1089 :
1090 : /* After the transaction has been fully received, and we know we were
1091 : not overrun while reading it, insert it into pack. */
1092 :
1093 : static inline void
1094 : after_frag( fd_pack_ctx_t * ctx,
1095 : ulong in_idx,
1096 : ulong seq,
1097 : ulong sig,
1098 : ulong sz,
1099 : ulong tsorig,
1100 : ulong tspub,
1101 0 : fd_stem_context_t * stem ) {
1102 0 : (void)seq;
1103 0 : (void)sz;
1104 0 : (void)tsorig;
1105 0 : (void)tspub;
1106 0 : (void)stem;
1107 :
1108 0 : long now = fd_tickcount();
1109 :
1110 0 : ulong leader_slot = ULONG_MAX;
1111 0 : switch( ctx->in_kind[ in_idx ] ) {
1112 0 : case IN_KIND_REPLAY:
1113 0 : if( FD_LIKELY( sig==REPLAY_SIG_TXN_EXECUTED && ctx->txn_committed ) ) {
1114 0 : ulong deleted = fd_pack_delete_transaction( ctx->pack, fd_type_pun( ctx->executed_txn_sig ) );
1115 0 : FD_MCNT_INC( PACK, TXN_ALREADY_EXECUTED, deleted );
1116 0 : }
1117 0 : if( FD_UNLIKELY( sig!=REPLAY_SIG_BECAME_LEADER ) ) return;
1118 0 : leader_slot = ctx->_became_leader->slot;
1119 :
1120 0 : ctx->start_block_sched_metrics->time = now;
1121 0 : fd_pack_get_sched_metrics( ctx->pack, ctx->start_block_sched_metrics->sched_results );
1122 0 : break;
1123 0 : case IN_KIND_POH:
1124 0 : if( fd_disco_poh_sig_pkt_type( sig )!=POH_PKT_TYPE_BECAME_LEADER ) return;
1125 0 : leader_slot = fd_disco_poh_sig_slot( sig );
1126 0 : break;
1127 0 : default:
1128 0 : break;
1129 0 : }
1130 :
1131 0 : switch( ctx->in_kind[ in_idx ] ) {
1132 0 : case IN_KIND_REPLAY:
1133 0 : case IN_KIND_POH: {
1134 0 : long now_ticks = fd_tickcount();
1135 0 : long now_ns = fd_log_wallclock();
1136 :
1137 0 : if( FD_UNLIKELY( ctx->leader_slot!=ULONG_MAX ) ) {
1138 0 : fd_done_packing_t * done_packing = fd_chunk_to_laddr( ctx->poh_out_mem, ctx->poh_out_chunk );
1139 0 : get_done_packing( ctx, done_packing, FD_PACK_END_SLOT_REASON_LEADER_SWITCH );
1140 0 : fd_pack_end_block( ctx->pack );
1141 0 : fd_pack_get_top_writers( ctx->pack, done_packing->limits_usage->top_writers );
1142 :
1143 0 : fd_stem_publish( stem, 1UL, fd_disco_execle_sig( ctx->leader_slot, ctx->pack_idx ), ctx->poh_out_chunk, sizeof(fd_done_packing_t), 0UL, 0UL, fd_frag_meta_ts_comp( fd_tickcount() ) );
1144 0 : ctx->poh_out_chunk = fd_dcache_compact_next( ctx->poh_out_chunk, sizeof(fd_done_packing_t), ctx->poh_out_chunk0, ctx->poh_out_wmark );
1145 0 : ctx->pack_idx++;
1146 :
1147 0 : FD_LOG_WARNING(( "switching to slot %lu while packing for slot %lu. Draining execle tiles.", leader_slot, ctx->leader_slot ));
1148 0 : log_end_block_metrics( ctx, now_ticks, "switch", done_packing->limits_usage->block_cost );
1149 0 : ctx->drain_execle = 1;
1150 0 : ctx->leader_slot = ULONG_MAX;
1151 0 : ctx->slot_microblock_cnt = 0UL;
1152 0 : remove_ib( ctx );
1153 0 : }
1154 0 : ctx->leader_slot = leader_slot;
1155 :
1156 0 : ulong exp_cnt = fd_pack_expire_before( ctx->pack, fd_ulong_max( ctx->leader_slot, TRANSACTION_LIFETIME_SLOTS )-TRANSACTION_LIFETIME_SLOTS );
1157 0 : FD_MCNT_INC( PACK, TXN_EXPIRED, exp_cnt );
1158 :
1159 0 : ctx->leader_bank = ctx->_became_leader->bank;
1160 0 : ctx->leader_bank_idx = ctx->_became_leader->bank_idx;
1161 0 : ctx->slot_max_microblocks = ctx->_became_leader->max_microblocks_in_slot;
1162 : /* Reserve some space in the block for ticks */
1163 0 : ctx->slot_max_data = (ctx->larger_shred_limits_per_block ? LARGER_MAX_DATA_PER_BLOCK : FD_PACK_MAX_DATA_PER_BLOCK)
1164 0 : - 48UL*(ctx->_became_leader->ticks_per_slot+ctx->_became_leader->total_skipped_ticks);
1165 :
1166 0 : ctx->limits.slot_max_cost = ctx->_became_leader->limits.slot_max_cost;
1167 0 : ctx->limits.slot_max_vote_cost = ctx->_became_leader->limits.slot_max_vote_cost;
1168 0 : ctx->limits.slot_max_write_cost_per_acct = ctx->_became_leader->limits.slot_max_write_cost_per_acct;
1169 :
1170 : /* ticks_per_ns is probably relatively stable over 400ms, but not
1171 : over several hours, so we need to compute the slot duration in
1172 : milliseconds first and then convert to ticks. This doesn't need
1173 : to be super accurate, but we don't want it to vary wildly. */
1174 0 : long end_ticks = now_ticks + (long)((double)fd_long_max( ctx->_became_leader->slot_end_ns - now_ns, 1L )*ctx->ticks_per_ns);
1175 : /* We may still get overrun, but then we'll never use this and just
1176 : reinitialize it the next time when we actually become leader. */
1177 0 : fd_pack_pacing_init( ctx->pacer, now_ticks, end_ticks, (float)ctx->ticks_per_ns, ctx->limits.slot_max_cost );
1178 :
1179 0 : if( FD_UNLIKELY( ctx->crank->enabled ) ) {
1180 : /* If we get overrun, we'll just never use these values, but the
1181 : old values aren't really useful either. */
1182 0 : ctx->crank->epoch = ctx->_became_leader->epoch;
1183 0 : *(ctx->crank->prev_config) = *(ctx->_became_leader->bundle->config);
1184 0 : memcpy( ctx->crank->recent_blockhash, ctx->_became_leader->bundle->last_blockhash, 32UL );
1185 0 : memcpy( ctx->crank->tip_receiver_owner, ctx->_became_leader->bundle->tip_receiver_owner, 32UL );
1186 0 : }
1187 :
1188 0 : FD_LOG_INFO(( "pack_became_leader(slot=%lu,ends_at=%ld)", ctx->leader_slot, ctx->_became_leader->slot_end_ns ));
1189 :
1190 0 : update_metric_state( ctx, fd_tickcount(), FD_PACK_METRIC_STATE_LEADER, 1 );
1191 :
1192 0 : ctx->slot_end_ns = ctx->_became_leader->slot_end_ns;
1193 0 : ctx->slot_dynamic_max_microblocks = ctx->slot_max_microblocks;
1194 0 : ctx->pending_reduce_mb_bound = 0;
1195 0 : fd_pack_limits_t limits[ 1 ];
1196 0 : limits->max_cost_per_block = ctx->limits.slot_max_cost;
1197 0 : limits->max_data_bytes_per_block = ctx->slot_max_data;
1198 0 : limits->max_microblocks_per_block = ctx->slot_max_microblocks;
1199 0 : limits->max_vote_cost_per_block = ctx->limits.slot_max_vote_cost;
1200 0 : limits->max_write_cost_per_acct = ctx->limits.slot_max_write_cost_per_acct;
1201 0 : limits->max_txn_per_microblock = ULONG_MAX; /* unused */
1202 0 : limits->max_allocated_data_per_block = FD_PACK_MAX_ALLOCATED_DATA_PER_BLOCK;
1203 0 : fd_pack_set_block_limits( ctx->pack, limits );
1204 0 : fd_pack_pacing_update_consumed_cus( ctx->pacer, fd_pack_current_block_cost( ctx->pack ), now );
1205 :
1206 0 : break;
1207 0 : }
1208 0 : case IN_KIND_EXECLE: {
1209 : /* For a previous slot */
1210 0 : if( FD_UNLIKELY( sig!=ctx->leader_slot ) ) return;
1211 :
1212 0 : fd_pack_rebate_cus( ctx->pack, ctx->rebate->rebate );
1213 0 : ctx->pending_rebate_sz = 0UL;
1214 0 : fd_pack_pacing_update_consumed_cus( ctx->pacer, fd_pack_current_block_cost( ctx->pack ), now );
1215 0 : break;
1216 0 : }
1217 0 : case IN_KIND_RESOLV: {
1218 : /* Normal transaction case */
1219 : #if FD_PACK_USE_EXTRA_STORAGE
1220 : if( FD_LIKELY( !ctx->insert_to_extra ) ) {
1221 : #else
1222 0 : if( 1 ) {
1223 0 : #endif
1224 0 : if( FD_UNLIKELY( ctx->is_bundle ) ) {
1225 0 : if( FD_UNLIKELY( ctx->current_bundle->txn_cnt==0UL ) ) return;
1226 0 : if( FD_UNLIKELY( ++(ctx->current_bundle->txn_received)==ctx->current_bundle->txn_cnt ) ) {
1227 0 : ulong deleted;
1228 0 : long insert_duration = -fd_tickcount();
1229 0 : int result = fd_pack_insert_bundle_fini( ctx->pack, ctx->current_bundle->bundle, ctx->current_bundle->txn_cnt, ctx->current_bundle->min_blockhash_slot, 0, ctx->blk_engine_cfg, &deleted );
1230 0 : insert_duration += fd_tickcount();
1231 0 : FD_MCNT_INC( PACK, TXN_DELETED, deleted );
1232 0 : ctx->insert_result[ result + FD_PACK_INSERT_RETVAL_OFF ] += ctx->current_bundle->txn_received;
1233 0 : fd_histf_sample( ctx->insert_duration, (ulong)insert_duration );
1234 0 : ctx->current_bundle->bundle = NULL;
1235 0 : }
1236 0 : } else {
1237 0 : ulong blockhash_slot = sig;
1238 0 : ulong deleted;
1239 0 : long insert_duration = -fd_tickcount();
1240 0 : int result = fd_pack_insert_txn_fini( ctx->pack, ctx->cur_spot, blockhash_slot, &deleted );
1241 0 : insert_duration += fd_tickcount();
1242 0 : FD_MCNT_INC( PACK, TXN_DELETED, deleted );
1243 0 : ctx->insert_result[ result + FD_PACK_INSERT_RETVAL_OFF ]++;
1244 0 : fd_histf_sample( ctx->insert_duration, (ulong)insert_duration );
1245 0 : if( FD_LIKELY( result>=0 ) ) ctx->last_successful_insert = now;
1246 0 : }
1247 0 : }
1248 :
1249 0 : ctx->cur_spot = NULL;
1250 0 : break;
1251 0 : }
1252 0 : case IN_KIND_EXECUTED_TXN: {
1253 0 : ulong deleted = fd_pack_delete_transaction( ctx->pack, fd_type_pun( ctx->executed_txn_sig ) );
1254 0 : FD_MCNT_INC( PACK, TXN_ALREADY_EXECUTED, deleted );
1255 0 : break;
1256 0 : }
1257 0 : }
1258 :
1259 0 : update_metric_state( ctx, now, FD_PACK_METRIC_STATE_TRANSACTIONS, fd_pack_avail_txn_cnt( ctx->pack )>0 );
1260 0 : }
1261 :
1262 : static void
1263 : privileged_init( fd_topo_t const * topo,
1264 0 : fd_topo_tile_t const * tile ) {
1265 0 : if( FD_LIKELY( !tile->pack.bundle.enabled ) ) return;
1266 0 : if( FD_UNLIKELY( !tile->pack.bundle.vote_account_path[0] ) ) {
1267 0 : FD_LOG_WARNING(( "Disabling bundle crank because no vote account was specified" ));
1268 0 : return;
1269 0 : }
1270 :
1271 0 : void * scratch = fd_topo_obj_laddr( topo, tile->tile_obj_id );
1272 :
1273 0 : FD_SCRATCH_ALLOC_INIT( l, scratch );
1274 0 : fd_pack_ctx_t * ctx = FD_SCRATCH_ALLOC_APPEND( l, alignof( fd_pack_ctx_t ), sizeof( fd_pack_ctx_t ) );
1275 :
1276 0 : if( FD_UNLIKELY( !strcmp( tile->pack.bundle.identity_key_path, "" ) ) )
1277 0 : FD_LOG_ERR(( "identity_key_path not set" ));
1278 :
1279 0 : const uchar * identity_key = fd_keyload_load( tile->pack.bundle.identity_key_path, /* pubkey only: */ 1 );
1280 0 : fd_memcpy( ctx->crank->identity_pubkey->b, identity_key, 32UL );
1281 :
1282 0 : if( FD_UNLIKELY( !fd_base58_decode_32( tile->pack.bundle.vote_account_path, ctx->crank->vote_pubkey->b ) ) ) {
1283 0 : const uchar * vote_key = fd_keyload_load( tile->pack.bundle.vote_account_path, /* pubkey only: */ 1 );
1284 0 : fd_memcpy( ctx->crank->vote_pubkey->b, vote_key, 32UL );
1285 0 : }
1286 0 : }
1287 :
1288 : static void
1289 : unprivileged_init( fd_topo_t const * topo,
1290 0 : fd_topo_tile_t const * tile ) {
1291 0 : void * scratch = fd_topo_obj_laddr( topo, tile->tile_obj_id );
1292 :
1293 0 : if( FD_UNLIKELY( tile->pack.max_pending_transactions >= USHORT_MAX-10UL ) ) FD_LOG_ERR(( "pack tile supports up to %lu pending transactions", USHORT_MAX-11UL ));
1294 :
1295 0 : fd_pack_limits_t limits_upper[1] = {{
1296 0 : .max_cost_per_block = tile->pack.larger_max_cost_per_block ? LARGER_MAX_COST_PER_BLOCK : FD_PACK_MAX_COST_PER_BLOCK_UPPER_BOUND,
1297 0 : .max_vote_cost_per_block = FD_PACK_MAX_VOTE_COST_PER_BLOCK_UPPER_BOUND,
1298 0 : .max_write_cost_per_acct = FD_PACK_MAX_WRITE_COST_PER_ACCT_UPPER_BOUND,
1299 0 : .max_data_bytes_per_block = tile->pack.larger_shred_limits_per_block ? LARGER_MAX_DATA_PER_BLOCK : FD_PACK_MAX_DATA_PER_BLOCK,
1300 0 : .max_txn_per_microblock = EFFECTIVE_TXN_PER_MICROBLOCK,
1301 0 : .max_microblocks_per_block = (ulong)UINT_MAX, /* Limit not known yet */
1302 0 : .max_allocated_data_per_block = FD_PACK_MAX_ALLOCATED_DATA_PER_BLOCK,
1303 0 : }};
1304 :
1305 0 : ulong pack_footprint = fd_pack_footprint( tile->pack.max_pending_transactions, BUNDLE_META_SZ, tile->pack.execle_tile_count, limits_upper );
1306 :
1307 0 : FD_SCRATCH_ALLOC_INIT( l, scratch );
1308 0 : fd_pack_ctx_t * ctx = FD_SCRATCH_ALLOC_APPEND( l, alignof( fd_pack_ctx_t ), sizeof( fd_pack_ctx_t ) );
1309 0 : fd_rng_t * rng = fd_rng_join( fd_rng_new( FD_SCRATCH_ALLOC_APPEND( l, fd_rng_align(), fd_rng_footprint() ), 0U, 0UL ) );
1310 0 : if( FD_UNLIKELY( !rng ) ) FD_LOG_ERR(( "fd_rng_new failed" ));
1311 :
1312 0 : fd_pack_limits_t limits_lower[1] = {{
1313 0 : .max_cost_per_block = tile->pack.larger_max_cost_per_block ? LARGER_MAX_COST_PER_BLOCK : FD_PACK_MAX_COST_PER_BLOCK_LOWER_BOUND,
1314 0 : .max_vote_cost_per_block = FD_PACK_MAX_VOTE_COST_PER_BLOCK_LOWER_BOUND,
1315 0 : .max_write_cost_per_acct = FD_PACK_MAX_WRITE_COST_PER_ACCT_LOWER_BOUND,
1316 0 : .max_data_bytes_per_block = tile->pack.larger_shred_limits_per_block ? LARGER_MAX_DATA_PER_BLOCK : FD_PACK_MAX_DATA_PER_BLOCK,
1317 0 : .max_txn_per_microblock = EFFECTIVE_TXN_PER_MICROBLOCK,
1318 0 : .max_microblocks_per_block = (ulong)UINT_MAX, /* Limit not known yet */
1319 0 : .max_allocated_data_per_block = FD_PACK_MAX_ALLOCATED_DATA_PER_BLOCK,
1320 0 : }};
1321 :
1322 0 : ctx->pack = fd_pack_join( fd_pack_new( FD_SCRATCH_ALLOC_APPEND( l, fd_pack_align(), pack_footprint ),
1323 0 : tile->pack.max_pending_transactions, BUNDLE_META_SZ, tile->pack.execle_tile_count,
1324 0 : limits_lower,
1325 0 : fd_type_pun_const( tile->pack.acct_blocklist ), tile->pack.acct_blocklist_cnt,
1326 0 : rng ) );
1327 0 : if( FD_UNLIKELY( !ctx->pack ) ) FD_LOG_ERR(( "fd_pack_new failed" ));
1328 :
1329 0 : if( FD_UNLIKELY( tile->in_cnt>32UL ) ) FD_LOG_ERR(( "Too many input links (%lu>32) to pack tile", tile->in_cnt ));
1330 :
1331 0 : FD_TEST( tile->in_cnt<sizeof( ctx->in_kind )/sizeof( ctx->in_kind[0] ) );
1332 0 : for( ulong i=0UL; i<tile->in_cnt; i++ ) {
1333 0 : fd_topo_link_t const * link = &topo->links[ tile->in_link_id[ i ] ];
1334 :
1335 0 : if( FD_LIKELY( !strcmp( link->name, "resolv_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_RESOLV;
1336 0 : else if( FD_LIKELY( !strcmp( link->name, "resolh_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_RESOLV;
1337 0 : else if( FD_LIKELY( !strcmp( link->name, "poh_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_POH;
1338 0 : else if( FD_LIKELY( !strcmp( link->name, "pohh_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_POH;
1339 0 : else if( FD_LIKELY( !strcmp( link->name, "bank_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_EXECLE;
1340 0 : else if( FD_LIKELY( !strcmp( link->name, "execle_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_EXECLE;
1341 0 : else if( FD_LIKELY( !strcmp( link->name, "sign_pack" ) ) ) ctx->in_kind[ i ] = IN_KIND_SIGN;
1342 0 : else if( FD_LIKELY( !strcmp( link->name, "replay_out" ) ) ) ctx->in_kind[ i ] = IN_KIND_REPLAY;
1343 0 : else if( FD_LIKELY( !strcmp( link->name, "executed_txn" ) ) ) ctx->in_kind[ i ] = IN_KIND_EXECUTED_TXN;
1344 0 : else FD_LOG_ERR(( "pack tile has unexpected input link %lu %s", i, link->name ));
1345 0 : }
1346 :
1347 0 : ulong execle_cnt = 0UL;
1348 0 : for( ulong i=0UL; i<topo->tile_cnt; i++ ) {
1349 0 : fd_topo_tile_t const * consumer_tile = &topo->tiles[ i ];
1350 0 : if( FD_UNLIKELY( strcmp( consumer_tile->name, "execle" ) && strcmp( consumer_tile->name, "replay" ) ) ) continue;
1351 0 : for( ulong j=0UL; j<consumer_tile->in_cnt; j++ ) {
1352 0 : if( FD_UNLIKELY( consumer_tile->in_link_id[ j ]==tile->out_link_id[ 0 ] ) ) execle_cnt++;
1353 0 : }
1354 0 : }
1355 :
1356 : // if( FD_UNLIKELY( !execle_cnt ) ) FD_LOG_ERR(( "pack tile connects to no execle tiles" ));
1357 0 : if( FD_UNLIKELY( execle_cnt>FD_PACK_MAX_EXECLE_TILES ) ) FD_LOG_ERR(( "pack tile connects to too many execle tiles" ));
1358 : // if( FD_UNLIKELY( execle_cnt!=tile->pack.execle_tile_count ) ) FD_LOG_ERR(( "pack tile connects to %lu execle tiles, but tile->pack.execle_tile_count is %lu", execle_cnt, tile->pack.execle_tile_count ));
1359 :
1360 0 : FD_TEST( (tile->pack.schedule_strategy>=0) & (tile->pack.schedule_strategy<=FD_PACK_STRATEGY_BALANCED) );
1361 :
1362 0 : ctx->crank->enabled = tile->pack.bundle.enabled;
1363 0 : if( FD_UNLIKELY( tile->pack.bundle.enabled ) ) {
1364 0 : if( FD_UNLIKELY( !fd_bundle_crank_gen_init( ctx->crank->gen, (fd_acct_addr_t const *)tile->pack.bundle.tip_distribution_program_addr,
1365 0 : (fd_acct_addr_t const *)tile->pack.bundle.tip_payment_program_addr,
1366 0 : (fd_acct_addr_t const *)ctx->crank->vote_pubkey->b,
1367 0 : (fd_acct_addr_t const *)tile->pack.bundle.tip_distribution_authority,
1368 0 : schedule_strategy_strings[ tile->pack.schedule_strategy ],
1369 0 : tile->pack.bundle.commission_bps ) ) ) {
1370 0 : FD_LOG_ERR(( "constructing bundle generator failed" ));
1371 0 : }
1372 :
1373 0 : ulong sign_in_idx = fd_topo_find_tile_in_link ( topo, tile, "sign_pack", tile->kind_id );
1374 0 : ulong sign_out_idx = fd_topo_find_tile_out_link( topo, tile, "pack_sign", tile->kind_id );
1375 0 : FD_TEST( sign_in_idx!=ULONG_MAX );
1376 0 : fd_topo_link_t const * sign_in = &topo->links[ tile->in_link_id[ sign_in_idx ] ];
1377 0 : fd_topo_link_t const * sign_out = &topo->links[ tile->out_link_id[ sign_out_idx ] ];
1378 0 : if( FD_UNLIKELY( !fd_keyguard_client_join( fd_keyguard_client_new( ctx->crank->keyguard_client,
1379 0 : sign_out->mcache,
1380 0 : sign_out->dcache,
1381 0 : sign_in->mcache,
1382 0 : sign_in->dcache,
1383 0 : sign_out->mtu ) ) ) ) {
1384 0 : FD_LOG_ERR(( "failed to construct keyguard" ));
1385 0 : }
1386 : /* Initialize enough of the prev config that it produces a
1387 : transaction */
1388 0 : ctx->crank->prev_config->discriminator = 0x82ccfa1ee0aa0c9bUL;
1389 0 : ctx->crank->prev_config->tip_receiver->b[1] = 1;
1390 0 : ctx->crank->prev_config->block_builder->b[2] = 1;
1391 :
1392 0 : memset( ctx->crank->tip_receiver_owner, '\0', 32UL );
1393 0 : memset( ctx->crank->recent_blockhash, '\0', 32UL );
1394 0 : memset( ctx->crank->last_sig, '\0', 64UL );
1395 0 : ctx->crank->ib_inserted = 0;
1396 0 : ctx->crank->epoch = 0UL;
1397 0 : ctx->crank->keyswitch = fd_keyswitch_join( fd_topo_obj_laddr( topo, tile->id_keyswitch_obj_id ) );
1398 0 : FD_TEST( ctx->crank->keyswitch );
1399 0 : } else {
1400 0 : memset( ctx->crank, '\0', sizeof(ctx->crank) );
1401 0 : }
1402 :
1403 :
1404 : #if FD_PACK_USE_EXTRA_STORAGE
1405 : ctx->extra_txn_deq = extra_txn_deq_join( extra_txn_deq_new( FD_SCRATCH_ALLOC_APPEND( l, extra_txn_deq_align(),
1406 : extra_txn_deq_footprint() ) ) );
1407 : #endif
1408 :
1409 0 : ctx->cur_spot = NULL;
1410 0 : ctx->is_bundle = 0;
1411 0 : ctx->strategy = tile->pack.schedule_strategy;
1412 0 : ctx->max_pending_transactions = tile->pack.max_pending_transactions;
1413 0 : ctx->leader_slot = ULONG_MAX;
1414 0 : ctx->leader_bank = NULL;
1415 0 : ctx->leader_bank_idx = ULONG_MAX;
1416 0 : ctx->pack_idx = 0UL;
1417 0 : ctx->slot_microblock_cnt = 0UL;
1418 0 : ctx->pack_txn_cnt = 0UL;
1419 0 : ctx->slot_max_microblocks = 0UL;
1420 0 : ctx->slot_dynamic_max_microblocks = 0UL;
1421 0 : ctx->pending_reduce_mb_bound = 0;
1422 0 : ctx->slot_max_data = 0UL;
1423 0 : ctx->larger_shred_limits_per_block = tile->pack.larger_shred_limits_per_block;
1424 0 : ctx->drain_execle = 0;
1425 0 : ctx->approx_wallclock_ns = fd_log_wallclock();
1426 0 : ctx->approx_tickcount = fd_tickcount();
1427 0 : ctx->rng = rng;
1428 0 : ctx->ticks_per_ns = fd_tempo_tick_per_ns( NULL );
1429 0 : ctx->last_successful_insert = 0L;
1430 0 : ctx->highest_observed_slot = 0UL;
1431 0 : ctx->microblock_duration_ticks = (ulong)(fd_tempo_tick_per_ns( NULL )*(double)MICROBLOCK_DURATION_NS + 0.5);
1432 : #if FD_PACK_USE_EXTRA_STORAGE
1433 : ctx->insert_to_extra = 0;
1434 : #endif
1435 0 : ctx->use_consumed_cus = tile->pack.use_consumed_cus;
1436 0 : ctx->crank->enabled = tile->pack.bundle.enabled;
1437 :
1438 : #if !SMALL_MICROBLOCKS
1439 : ctx->wait_duration_ticks[ 0 ] = ULONG_MAX;
1440 : for( ulong i=1UL; i<MAX_TXN_PER_MICROBLOCK+1UL; i++ ) {
1441 : ctx->wait_duration_ticks[ i ]=(ulong)(fd_tempo_tick_per_ns( NULL )*(double)wait_duration[ i ] + 0.5);
1442 : }
1443 : #endif
1444 :
1445 0 : ctx->limits.slot_max_cost = limits_lower->max_cost_per_block;
1446 0 : ctx->limits.slot_max_vote_cost = limits_lower->max_vote_cost_per_block;
1447 0 : ctx->limits.slot_max_write_cost_per_acct = limits_lower->max_write_cost_per_acct;
1448 :
1449 0 : ctx->execle_cnt = tile->pack.execle_tile_count;
1450 0 : ctx->poll_cursor = 0;
1451 0 : ctx->skip_cnt = 0L;
1452 0 : ctx->execle_idle_bitset = fd_ulong_mask_lsb( (int)tile->pack.execle_tile_count );
1453 0 : for( ulong i=0UL; i<tile->pack.execle_tile_count; i++ ) {
1454 0 : ulong busy_obj_id = fd_pod_queryf_ulong( topo->props, ULONG_MAX, "execle_busy.%lu", i );
1455 0 : FD_TEST( busy_obj_id!=ULONG_MAX );
1456 0 : ctx->execle_current[ i ] = fd_fseq_join( fd_topo_obj_laddr( topo, busy_obj_id ) );
1457 0 : ctx->execle_expect[ i ] = ULONG_MAX;
1458 0 : if( FD_UNLIKELY( !ctx->execle_current[ i ] ) ) FD_LOG_ERR(( "execle tile %lu has no busy flag", i ));
1459 0 : ctx->execle_ready_at[ i ] = 0L;
1460 0 : FD_TEST( ULONG_MAX==fd_fseq_query( ctx->execle_current[ i ] ) );
1461 0 : }
1462 :
1463 0 : for( ulong i=0UL; i<tile->in_cnt; i++ ) {
1464 0 : fd_topo_link_t const * link = &topo->links[ tile->in_link_id[ i ] ];
1465 0 : fd_topo_wksp_t const * link_wksp = &topo->workspaces[ topo->objs[ link->dcache_obj_id ].wksp_id ];
1466 :
1467 0 : ctx->in[ i ].mem = link_wksp->wksp;
1468 0 : ctx->in[ i ].chunk0 = fd_dcache_compact_chunk0( ctx->in[ i ].mem, link->dcache );
1469 0 : ctx->in[ i ].wmark = fd_dcache_compact_wmark ( ctx->in[ i ].mem, link->dcache, link->mtu );
1470 0 : }
1471 :
1472 0 : ctx->execle_out_mem = topo->workspaces[ topo->objs[ topo->links[ tile->out_link_id[ 0 ] ].dcache_obj_id ].wksp_id ].wksp;
1473 0 : ctx->execle_out_chunk0 = fd_dcache_compact_chunk0( ctx->execle_out_mem, topo->links[ tile->out_link_id[ 0 ] ].dcache );
1474 0 : ctx->execle_out_wmark = fd_dcache_compact_wmark ( ctx->execle_out_mem, topo->links[ tile->out_link_id[ 0 ] ].dcache, topo->links[ tile->out_link_id[ 0 ] ].mtu );
1475 0 : ctx->execle_out_chunk = ctx->execle_out_chunk0;
1476 :
1477 0 : ctx->poh_out_mem = topo->workspaces[ topo->objs[ topo->links[ tile->out_link_id[ 1 ] ].dcache_obj_id ].wksp_id ].wksp;
1478 0 : ctx->poh_out_chunk0 = fd_dcache_compact_chunk0( ctx->poh_out_mem, topo->links[ tile->out_link_id[ 1 ] ].dcache );
1479 0 : ctx->poh_out_wmark = fd_dcache_compact_wmark ( ctx->poh_out_mem, topo->links[ tile->out_link_id[ 1 ] ].dcache, topo->links[ tile->out_link_id[ 1 ] ].mtu );
1480 0 : ctx->poh_out_chunk = ctx->poh_out_chunk0;
1481 :
1482 : /* Initialize metrics storage */
1483 0 : memset( ctx->insert_result, '\0', FD_PACK_INSERT_RETVAL_CNT * sizeof(ulong) );
1484 0 : fd_histf_join( fd_histf_new( ctx->schedule_duration, FD_MHIST_SECONDS_MIN( PACK, SCHEDULE_MICROBLOCK_DURATION_SECONDS ),
1485 0 : FD_MHIST_SECONDS_MAX( PACK, SCHEDULE_MICROBLOCK_DURATION_SECONDS ) ) );
1486 0 : fd_histf_join( fd_histf_new( ctx->no_sched_duration, FD_MHIST_SECONDS_MIN( PACK, NO_SCHEDULE_MICROBLOCK_DURATION_SECONDS ),
1487 0 : FD_MHIST_SECONDS_MAX( PACK, NO_SCHEDULE_MICROBLOCK_DURATION_SECONDS ) ) );
1488 0 : fd_histf_join( fd_histf_new( ctx->insert_duration, FD_MHIST_SECONDS_MIN( PACK, INSERT_TRANSACTION_DURATION_SECONDS ),
1489 0 : FD_MHIST_SECONDS_MAX( PACK, INSERT_TRANSACTION_DURATION_SECONDS ) ) );
1490 0 : fd_histf_join( fd_histf_new( ctx->complete_duration, FD_MHIST_SECONDS_MIN( PACK, COMPLETE_MICROBLOCK_DURATION_SECONDS ),
1491 0 : FD_MHIST_SECONDS_MAX( PACK, COMPLETE_MICROBLOCK_DURATION_SECONDS ) ) );
1492 0 : ctx->metric_state = 0;
1493 0 : ctx->metric_state_begin = fd_tickcount();
1494 0 : memset( ctx->metric_timing, '\0', 16*sizeof(long) );
1495 0 : memset( ctx->current_bundle, '\0', sizeof(ctx->current_bundle) );
1496 0 : memset( ctx->blk_engine_cfg, '\0', sizeof(ctx->blk_engine_cfg) );
1497 0 : memset( ctx->last_sched_metrics, '\0', sizeof(ctx->last_sched_metrics) );
1498 0 : memset( ctx->start_block_sched_metrics, '\0', sizeof(ctx->start_block_sched_metrics) );
1499 0 : memset( ctx->crank->metrics, '\0', sizeof(ctx->crank->metrics) );
1500 :
1501 0 : FD_LOG_INFO(( "packing microblocks of at most %lu transactions to %lu execle tiles using strategy %i", EFFECTIVE_TXN_PER_MICROBLOCK, tile->pack.execle_tile_count, ctx->strategy ));
1502 :
1503 0 : ulong scratch_top = FD_SCRATCH_ALLOC_FINI( l, scratch_align() );
1504 0 : if( FD_UNLIKELY( scratch_top > (ulong)scratch + scratch_footprint( tile ) ) )
1505 0 : FD_LOG_ERR(( "scratch overflow %lu %lu %lu", scratch_top - (ulong)scratch - scratch_footprint( tile ), scratch_top, (ulong)scratch + scratch_footprint( tile ) ));
1506 0 : }
1507 :
1508 : static ulong
1509 : populate_allowed_seccomp( fd_topo_t const * topo,
1510 : fd_topo_tile_t const * tile,
1511 : ulong out_cnt,
1512 0 : struct sock_filter * out ) {
1513 0 : (void)topo;
1514 0 : (void)tile;
1515 :
1516 0 : populate_sock_filter_policy_fd_pack_tile( out_cnt, out, (uint)fd_log_private_logfile_fd() );
1517 0 : return sock_filter_policy_fd_pack_tile_instr_cnt;
1518 0 : }
1519 :
1520 : static ulong
1521 : populate_allowed_fds( fd_topo_t const * topo,
1522 : fd_topo_tile_t const * tile,
1523 : ulong out_fds_cnt,
1524 0 : int * out_fds ) {
1525 0 : (void)topo;
1526 0 : (void)tile;
1527 :
1528 0 : if( FD_UNLIKELY( out_fds_cnt<2UL ) ) FD_LOG_ERR(( "out_fds_cnt %lu", out_fds_cnt ));
1529 :
1530 0 : ulong out_cnt = 0UL;
1531 0 : out_fds[ out_cnt++ ] = 2; /* stderr */
1532 0 : if( FD_LIKELY( -1!=fd_log_private_logfile_fd() ) )
1533 0 : out_fds[ out_cnt++ ] = fd_log_private_logfile_fd(); /* logfile */
1534 0 : return out_cnt;
1535 0 : }
1536 :
1537 : /* Pack can publish a frag in the following scenarios:
1538 :
1539 : after_credit:
1540 : A. TIMED_OUT. Sets ctx->leader_slot=ULONG_MAX. return.
1541 : B. DONE_DRAINING. *after* DONE_PACKING. return.
1542 : C. REDUCE_MB_BOUND. return.
1543 : D. SCHEDULE_MB. *doesn't* return.
1544 : E. EXHAUST_MICROBLOCKS. Sets ctx->leader_slot=ULONG_MAX. return.
1545 : after_frag:
1546 : F. LEADER_SWITCH. Requires ctx->leader_slot!=ULONG_MAX
1547 :
1548 : It isn't possible to get a burst of 3, but a burst of 2 is possible
1549 : in these situations.
1550 :
1551 : C -> F
1552 : D -> F
1553 : D -> E
1554 : */
1555 0 : #define STEM_BURST (2UL)
1556 :
1557 : /* We want lazy (measured in ns) to be small enough that the producer
1558 : and the consumer never have to wait for credits. For most tango
1559 : links, we use a default worst case speed coming from 100 Gbps
1560 : Ethernet. That's not very suitable for microblocks that go from
1561 : pack to bank. Instead we manually estimate the very aggressive
1562 : 1000ns per microblock, and then reduce it further (in line with the
1563 : default lazy value computation) to ensure the random value chosen
1564 : based on this won't lead to credit return stalls. */
1565 0 : #define STEM_LAZY (128L*3000L)
1566 :
1567 0 : #define STEM_CALLBACK_CONTEXT_TYPE fd_pack_ctx_t
1568 0 : #define STEM_CALLBACK_CONTEXT_ALIGN alignof(fd_pack_ctx_t)
1569 :
1570 0 : #define STEM_CALLBACK_DURING_HOUSEKEEPING during_housekeeping
1571 0 : #define STEM_CALLBACK_BEFORE_CREDIT before_credit
1572 0 : #define STEM_CALLBACK_AFTER_CREDIT after_credit
1573 0 : #define STEM_CALLBACK_DURING_FRAG during_frag
1574 0 : #define STEM_CALLBACK_AFTER_FRAG after_frag
1575 0 : #define STEM_CALLBACK_METRICS_WRITE metrics_write
1576 :
1577 : #include "../stem/fd_stem.c"
1578 :
1579 : fd_topo_run_tile_t fd_tile_pack = {
1580 : .name = "pack",
1581 : .populate_allowed_seccomp = populate_allowed_seccomp,
1582 : .populate_allowed_fds = populate_allowed_fds,
1583 : .scratch_align = scratch_align,
1584 : .scratch_footprint = scratch_footprint,
1585 : .privileged_init = privileged_init,
1586 : .unprivileged_init = unprivileged_init,
1587 : .run = stem_run,
1588 : };
|