LCOV - code coverage report
Current view: top level - discof/gossip - fd_gossvf_tile.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 0 701 0.0 %
Date: 2026-06-28 05:59:11 Functions: 0 75 0.0 %

          Line data    Source code
       1             : #include "fd_gossip_tile.h"
       2             : #include "../../disco/topo/fd_topo.h"
       3             : #include "../../disco/fd_disco_base.h"
       4             : #include "../../disco/keyguard/fd_keyswitch.h"
       5             : #include "../../disco/keyguard/fd_keyload.h"
       6             : #include "../../disco/metrics/fd_metrics.h"
       7             : #include "../../disco/shred/fd_stake_ci.h"
       8             : #include "../../flamenco/gossip/fd_ping_tracker.h"
       9             : #include "../../flamenco/leaders/fd_leaders_base.h"
      10             : #include "../../util/net/fd_net_headers.h"
      11             : #include "../../disco/net/fd_net_tile.h"
      12             : #include "generated/fd_gossvf_tile_seccomp.h"
      13             : 
      14             : #define DEBUG_PEERS (0)
      15             : 
      16           0 : #define IN_KIND_SHRED_VERSION (0)
      17           0 : #define IN_KIND_NET           (1)
      18           0 : #define IN_KIND_EPOCH         (2)
      19           0 : #define IN_KIND_PINGS         (3)
      20           0 : #define IN_KIND_GOSSIP        (4)
      21             : 
      22             : struct peer {
      23             :   fd_pubkey_t pubkey;
      24             : 
      25             :   fd_ip4_port_t gossip_addr;
      26             :   ushort shred_version;
      27             : 
      28             :   struct {
      29             :     ulong prev;
      30             :     ulong next;
      31             :   } map;
      32             : 
      33             :   struct {
      34             :     ulong next;
      35             :   } pool;
      36             : };
      37             : 
      38             : typedef struct peer peer_t;
      39             : 
      40             : struct ping {
      41             :   fd_pubkey_t pubkey;
      42             :   fd_ip4_port_t addr;
      43             : 
      44             :   struct {
      45             :     ulong prev;
      46             :     ulong next;
      47             :   } map;
      48             : 
      49             :   struct {
      50             :     ulong next;
      51             :   } pool;
      52             : };
      53             : 
      54             : typedef struct ping ping_t;
      55             : 
      56             : struct stake {
      57             :   fd_pubkey_t pubkey;
      58             :   ulong       stake;
      59             : 
      60             :   struct {
      61             :     ulong prev;
      62             :     ulong next;
      63             :   } map;
      64             : 
      65             :   struct {
      66             :     ulong next;
      67             :   } pool;
      68             : };
      69             : 
      70             : typedef struct stake stake_t;
      71             : 
      72             : #define POOL_NAME  peer_pool
      73           0 : #define POOL_T     peer_t
      74             : #define POOL_IDX_T ulong
      75           0 : #define POOL_NEXT  pool.next
      76             : #include "../../util/tmpl/fd_pool.c"
      77             : 
      78             : #define MAP_NAME               peer_map
      79           0 : #define MAP_KEY                pubkey
      80             : #define MAP_ELE_T              peer_t
      81             : #define MAP_KEY_T              fd_pubkey_t
      82             : #define MAP_PREV               map.prev
      83           0 : #define MAP_NEXT               map.next
      84           0 : #define MAP_KEY_EQ(k0,k1)      fd_pubkey_eq( k0, k1 )
      85           0 : #define MAP_KEY_HASH(key,seed) (seed^fd_ulong_load_8( (key)->uc ))
      86             : #include "../../util/tmpl/fd_map_chain.c"
      87             : 
      88             : #define POOL_NAME  ping_pool
      89           0 : #define POOL_T     ping_t
      90             : #define POOL_IDX_T ulong
      91           0 : #define POOL_NEXT  pool.next
      92             : #include "../../util/tmpl/fd_pool.c"
      93             : 
      94             : #define MAP_NAME               ping_map
      95           0 : #define MAP_KEY                pubkey
      96             : #define MAP_ELE_T              ping_t
      97             : #define MAP_KEY_T              fd_pubkey_t
      98             : #define MAP_PREV               map.prev
      99           0 : #define MAP_NEXT               map.next
     100           0 : #define MAP_KEY_EQ(k0,k1)      fd_pubkey_eq( k0, k1 )
     101           0 : #define MAP_KEY_HASH(key,seed) (seed^fd_ulong_load_8( (key)->uc ))
     102             : #include "../../util/tmpl/fd_map_chain.c"
     103             : 
     104             : #define POOL_NAME  stake_pool
     105           0 : #define POOL_T     stake_t
     106             : #define POOL_IDX_T ulong
     107           0 : #define POOL_NEXT  pool.next
     108             : #include "../../util/tmpl/fd_pool.c"
     109             : 
     110             : #define MAP_NAME               stake_map
     111           0 : #define MAP_KEY                pubkey
     112             : #define MAP_ELE_T              stake_t
     113             : #define MAP_KEY_T              fd_pubkey_t
     114           0 : #define MAP_PREV               map.prev
     115           0 : #define MAP_NEXT               map.next
     116           0 : #define MAP_KEY_EQ(k0,k1)      fd_pubkey_eq( k0, k1 )
     117           0 : #define MAP_KEY_HASH(key,seed) (seed^fd_ulong_load_8( (key)->uc ))
     118             : #define MAP_OPTIMIZE_RANDOM_ACCESS_REMOVAL 1
     119             : #include "../../util/tmpl/fd_map_chain.c"
     120             : 
     121             : struct fd_gossvf_tile_ctx {
     122             :   long instance_creation_wallclock_nanos;
     123             :   ushort shred_version;
     124             : 
     125             :   int allow_private_address;
     126             : 
     127             :   fd_ip4_port_t gossip_addr;
     128             :   fd_ip4_port_t src_addr;
     129             : 
     130             :   fd_keyswitch_t * keyswitch;
     131             :   fd_pubkey_t identity_pubkey[1];
     132             : 
     133             :   fd_ip4_port_t entrypoints[ 16UL ];
     134             :   ulong         entrypoints_cnt;
     135             : 
     136             : #if DEBUG_PEERS
     137             :   ulong peer_cnt;
     138             :   ulong ping_cnt;
     139             : #endif
     140             : 
     141             :   peer_t * peers;
     142             :   peer_map_t * peer_map;
     143             : 
     144             :   ping_t * pings;
     145             :   ping_map_t * ping_map;
     146             : 
     147             :   struct {
     148             :     ulong         count;
     149             :     stake_t *     pool;
     150             :     stake_map_t * map;
     151             :     uchar         msg_buf[ FD_EPOCH_INFO_MAX_MSG_SZ ];
     152             :   } stake;
     153             : 
     154             :   uchar payload[ FD_NET_MTU ];
     155             :   fd_ip4_port_t peer;
     156             : 
     157             :   fd_gossip_ping_update_t _ping_update[1];
     158             :   fd_gossip_update_message_t _gossip_update[1];
     159             :   fd_gossip_message_t _message[1];
     160             : 
     161             :   double ticks_per_ns;
     162             :   long   last_wallclock;
     163             :   long   last_tickcount;
     164             : 
     165             :   ulong seed;
     166             : 
     167             :   ulong round_robin_idx;
     168             :   ulong round_robin_cnt;
     169             : 
     170             :   fd_sha512_t sha[ 1 ];
     171             : 
     172             :   struct {
     173             :     ulong   depth;
     174             :     ulong   map_cnt;
     175             :     ulong * sync;
     176             :     ulong * ring;
     177             :     ulong * map;
     178             :   } tcache;
     179             : 
     180             :   struct {
     181             :     int         kind;
     182             :     ulong       chunk0;
     183             :     ulong       wmark;
     184             :     fd_wksp_t * mem;
     185             :     ulong       mtu;
     186             :   } in[ 64UL ];
     187             : 
     188             :   fd_net_rx_bounds_t net_in_bounds[ 64UL ];
     189             : 
     190             :   struct {
     191             :     ulong       chunk0;
     192             :     ulong       chunk;
     193             :     ulong       wmark;
     194             :     fd_wksp_t * mem;
     195             :   } out[ 1 ];
     196             : 
     197             :   struct {
     198             :     ulong message_rx[ FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_CNT ];
     199             :     ulong message_rx_bytes[ FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_CNT ];
     200             :     ulong crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_CNT ];
     201             :     ulong crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_CNT ];
     202             :   } metrics;
     203             : };
     204             : 
     205             : typedef struct fd_gossvf_tile_ctx fd_gossvf_tile_ctx_t;
     206             : 
     207             : FD_FN_CONST static inline ulong
     208           0 : scratch_align( void ) {
     209           0 :   return 128UL;
     210           0 : }
     211             : 
     212             : FD_FN_PURE static inline ulong
     213           0 : scratch_footprint( fd_topo_tile_t const * tile ) {
     214           0 :   ulong l = FD_LAYOUT_INIT;
     215           0 :   l = FD_LAYOUT_APPEND( l, alignof( fd_gossvf_tile_ctx_t ), sizeof( fd_gossvf_tile_ctx_t )                                    );
     216           0 :   l = FD_LAYOUT_APPEND( l, peer_pool_align(),               peer_pool_footprint( FD_CONTACT_INFO_TABLE_SIZE )                 );
     217           0 :   l = FD_LAYOUT_APPEND( l, peer_map_align(),                peer_map_footprint( 2UL*FD_CONTACT_INFO_TABLE_SIZE )              );
     218           0 :   l = FD_LAYOUT_APPEND( l, ping_pool_align(),               ping_pool_footprint( FD_PING_TRACKER_MAX )                        );
     219           0 :   l = FD_LAYOUT_APPEND( l, ping_map_align(),                ping_map_footprint( 2UL*FD_PING_TRACKER_MAX )                     );
     220           0 :   l = FD_LAYOUT_APPEND( l, stake_pool_align(),              stake_pool_footprint( MAX_SHRED_DESTS )                           );
     221           0 :   l = FD_LAYOUT_APPEND( l, stake_map_align(),               stake_map_footprint( stake_map_chain_cnt_est( MAX_SHRED_DESTS ) ) );
     222           0 :   l = FD_LAYOUT_APPEND( l, fd_tcache_align(),               fd_tcache_footprint( tile->gossvf.tcache_depth, 0UL )             );
     223           0 :   return FD_LAYOUT_FINI( l, scratch_align() );
     224           0 : }
     225             : 
     226             : static inline void
     227           0 : during_housekeeping( fd_gossvf_tile_ctx_t * ctx ) {
     228           0 :   ctx->last_wallclock = fd_log_wallclock();
     229           0 :   ctx->last_tickcount = fd_tickcount();
     230             : 
     231           0 :   if( FD_UNLIKELY( fd_keyswitch_state_query( ctx->keyswitch )==FD_KEYSWITCH_STATE_SWITCH_PENDING ) ) {
     232           0 :     memcpy( ctx->identity_pubkey->uc, ctx->keyswitch->bytes, 32UL );
     233           0 :     fd_keyswitch_state( ctx->keyswitch, FD_KEYSWITCH_STATE_COMPLETED );
     234           0 :   }
     235           0 : }
     236             : 
     237             : static inline void
     238           0 : metrics_write( fd_gossvf_tile_ctx_t * ctx ) {
     239           0 :   FD_MCNT_ENUM_COPY( GOSSVF, MESSAGE_RX, ctx->metrics.message_rx );
     240           0 :   FD_MCNT_ENUM_COPY( GOSSVF, MESSAGE_RX_BYTES, ctx->metrics.message_rx_bytes );
     241           0 :   FD_MCNT_ENUM_COPY( GOSSVF, CRDS_RX, ctx->metrics.crds_rx );
     242           0 :   FD_MCNT_ENUM_COPY( GOSSVF, CRDS_RX_BYTES, ctx->metrics.crds_rx_bytes );
     243           0 : }
     244             : 
     245             : static int
     246             : before_frag( fd_gossvf_tile_ctx_t * ctx,
     247             :              ulong                  in_idx,
     248             :              ulong                  seq,
     249           0 :              ulong                  sig ) {
     250           0 :   if( FD_UNLIKELY( !ctx->shred_version && ctx->in[ in_idx ].kind!=IN_KIND_SHRED_VERSION ) ) return -1;
     251             : 
     252           0 :   switch( ctx->in[ in_idx ].kind ) {
     253           0 :     case IN_KIND_SHRED_VERSION: return 0;
     254           0 :     case IN_KIND_NET: return (seq % ctx->round_robin_cnt) != ctx->round_robin_idx;
     255           0 :     case IN_KIND_EPOCH: return 0;
     256           0 :     case IN_KIND_PINGS: return 0;
     257           0 :     case IN_KIND_GOSSIP: return sig!=FD_GOSSIP_UPDATE_TAG_CONTACT_INFO &&
     258           0 :                                 sig!=FD_GOSSIP_UPDATE_TAG_CONTACT_INFO_REMOVE;
     259           0 :     default: FD_LOG_ERR(( "unexpected in_kind %d", ctx->in[ in_idx ].kind )); return -1;
     260           0 :   }
     261           0 : }
     262             : 
     263             : static inline void
     264             : during_frag( fd_gossvf_tile_ctx_t * ctx,
     265             :              ulong                  in_idx,
     266             :              ulong                  seq FD_PARAM_UNUSED,
     267             :              ulong                  sig,
     268             :              ulong                  chunk,
     269             :              ulong                  sz,
     270           0 :              ulong                  ctl ) {
     271           0 :   if( FD_UNLIKELY( chunk<ctx->in[ in_idx ].chunk0 || chunk>ctx->in[ in_idx ].wmark || sz>ctx->in[ in_idx ].mtu ) )
     272           0 :     FD_LOG_ERR(( "chunk %lu %lu corrupt, not in range [%lu,%lu,%lu]", chunk, sz, ctx->in[ in_idx ].chunk0, ctx->in[ in_idx ].wmark, ctx->in[ in_idx ].mtu ));
     273             : 
     274           0 :   switch( ctx->in[ in_idx ].kind ) {
     275           0 :     case IN_KIND_SHRED_VERSION: {
     276           0 :       ctx->shred_version = (ushort)sig;
     277           0 :       FD_TEST( ctx->shred_version );
     278           0 :       break;
     279           0 :     }
     280           0 :     case IN_KIND_NET: {
     281           0 :       uchar const * src = fd_net_rx_translate_frag( &ctx->net_in_bounds[ in_idx ], chunk, ctl, sz );
     282           0 :       fd_memcpy( ctx->payload, src, sz );
     283           0 :       break;
     284           0 :     }
     285           0 :     case IN_KIND_EPOCH: {
     286           0 :       fd_epoch_info_msg_t const * msg = fd_chunk_to_laddr( ctx->in[ in_idx ].mem, chunk );
     287           0 :       if( FD_UNLIKELY( msg->staked_vote_cnt>MAX_COMPRESSED_STAKE_WEIGHTS ) )
     288           0 :         FD_LOG_ERR(( "epoch stakes exceed MAX_COMPRESSED_STAKE_WEIGHTS=%lu", MAX_COMPRESSED_STAKE_WEIGHTS ));
     289           0 :       if( FD_UNLIKELY( msg->staked_id_cnt>MAX_SHRED_DESTS ) )
     290           0 :         FD_LOG_ERR(( "epoch id weights exceed MAX_SHRED_DESTS=%lu", MAX_SHRED_DESTS ));
     291             : 
     292           0 :       ulong msg_sz = fd_epoch_info_msg_sz( msg->staked_vote_cnt, msg->staked_id_cnt );
     293           0 :       fd_memcpy( ctx->stake.msg_buf, msg, msg_sz );
     294           0 :       break;
     295           0 :     }
     296           0 :     case IN_KIND_PINGS: {
     297           0 :       fd_memcpy( ctx->_ping_update, fd_chunk_to_laddr( ctx->in[ in_idx ].mem, chunk ), sz );
     298           0 :       break;
     299           0 :     }
     300           0 :     case IN_KIND_GOSSIP:
     301           0 :       FD_TEST( sz==FD_GOSSIP_UPDATE_SZ_CONTACT_INFO || sz==FD_GOSSIP_UPDATE_SZ_CONTACT_INFO_REMOVE );
     302           0 :       fd_memcpy( ctx->_gossip_update, fd_chunk_to_laddr( ctx->in[ in_idx ].mem, chunk ), sz );
     303           0 :       break;
     304           0 :     default:
     305           0 :       FD_LOG_ERR(( "unexpected in_kind %d", ctx->in[ in_idx ].kind ));
     306           0 :   }
     307           0 : }
     308             : 
     309             : static inline void
     310             : handle_epoch( fd_gossvf_tile_ctx_t *      ctx,
     311           0 :               fd_epoch_info_msg_t const * msg ) {
     312           0 :   stake_map_reset( ctx->stake.map );
     313           0 :   stake_pool_reset( ctx->stake.pool );
     314             : 
     315           0 :   fd_stake_weight_t const * id_weights = fd_epoch_info_msg_id_weights( msg );
     316             : 
     317           0 :   for( ulong i=0UL; i<msg->staked_id_cnt; i++ ) {
     318           0 :     stake_t * entry = stake_pool_ele_acquire( ctx->stake.pool );
     319           0 :     entry->pubkey = id_weights[i].key;
     320           0 :     entry->stake  = id_weights[i].stake;
     321           0 :     stake_map_ele_insert( ctx->stake.map, entry, ctx->stake.pool );
     322           0 :   }
     323           0 :   ctx->stake.count = stake_pool_used( ctx->stake.pool );
     324           0 : }
     325             : 
     326             : static int
     327             : verify_prune( fd_gossip_prune_t const * view,
     328           0 :               fd_sha512_t *             sha ) {
     329           0 :   uchar sign_data[ FD_NET_MTU ];
     330             :   /* Agave serializes the prefix as a bincode length-prefixed &[u8]:
     331             :      8-byte LE u64 length (=18) followed by the 18 raw prefix bytes,
     332             :      totaling 26 bytes for the prefix portion. */
     333           0 :   FD_STORE( ulong, sign_data,                             18UL );
     334           0 :   fd_memcpy(       sign_data+8UL,                         "\xffSOLANA_PRUNE_DATA",       18UL );
     335           0 :   fd_memcpy(       sign_data+26UL,                        view->pubkey,                  32UL );
     336           0 :   FD_STORE( ulong, sign_data+58UL,                        view->prunes_len );
     337           0 :   fd_memcpy(       sign_data+66UL,                        view->prunes,                  view->prunes_len*32UL );
     338           0 :   fd_memcpy(       sign_data+66UL+view->prunes_len*32UL,  view->destination, 32UL );
     339           0 :   FD_STORE( ulong, sign_data+98UL+view->prunes_len*32UL,  view->wallclock );
     340             : 
     341           0 :   ulong sign_data_len = 106UL+view->prunes_len*32UL;
     342           0 :   int err_prefix    = fd_ed25519_verify( sign_data,       sign_data_len,       view->signature, view->pubkey, sha );
     343           0 :   int err_no_prefix = fd_ed25519_verify( sign_data+26UL,  sign_data_len-26UL,  view->signature, view->pubkey, sha );
     344             : 
     345           0 :   if( FD_LIKELY( err_prefix==FD_ED25519_SUCCESS || err_no_prefix==FD_ED25519_SUCCESS ) ) return 0;
     346           0 :   else                                                                                   return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PRUNE_SIGNATURE_IDX;
     347           0 : }
     348             : 
     349             : static int
     350             : verify_crds_value( fd_gossip_value_t const * value,
     351             :                    uchar const *             value_bytes,
     352             :                    ulong                     value_bytes_len,
     353           0 :                    fd_sha512_t *             sha ) {
     354           0 :   return fd_ed25519_verify( value_bytes+64UL, /* signable data begins after signature */
     355           0 :                             value_bytes_len-64UL,                /* signable data length */
     356           0 :                             value->signature,
     357           0 :                             value->origin,
     358           0 :                             sha );
     359           0 : }
     360             : 
     361             : static int
     362             : verify_signatures( fd_gossvf_tile_ctx_t * ctx,
     363             :                    fd_gossip_message_t *  view,
     364             :                    uchar const *          payload,
     365             :                    fd_sha512_t *          sha,
     366           0 :                    uchar *                failed ) {
     367           0 :   switch( view->tag ) {
     368           0 :     case FD_GOSSIP_MESSAGE_PULL_REQUEST: {
     369           0 :       if( FD_UNLIKELY( FD_ED25519_SUCCESS!=verify_crds_value( view->pull_request->contact_info, payload+view->pull_request->contact_info->offset, view->pull_request->contact_info->length, sha ) ) ) {
     370           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_SIGNATURE_IDX;
     371           0 :       } else {
     372           0 :         return 0;
     373           0 :       }
     374           0 :     }
     375           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE: {
     376           0 :       ulong i = 0UL;
     377           0 :       while( i<view->pull_response->values_len ) {
     378           0 :         ulong dedup_tag = ctx->seed ^ fd_ulong_load_8_fast( view->pull_response->values[ i ].signature );
     379           0 :         int ha_dup = 0;
     380           0 :         FD_FN_UNUSED ulong tcache_map_idx = 0; /* ignored */
     381           0 :         FD_TCACHE_QUERY( ha_dup, tcache_map_idx, ctx->tcache.map, ctx->tcache.map_cnt, dedup_tag );
     382           0 :         if( FD_UNLIKELY( ha_dup ) ) {
     383           0 :           if( FD_LIKELY( !failed[ i ] ) ) {
     384           0 :             ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_DUPLICATE_IDX ]++;
     385           0 :             ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_DUPLICATE_IDX ] += view->pull_response->values[ i ].length;
     386           0 :           }
     387           0 :           view->pull_response->values_len--;
     388           0 :           view->pull_response->values[ i ] = view->pull_response->values[ view->pull_response->values_len ];
     389           0 :           failed[ i ] = failed[ view->pull_response->values_len ];
     390           0 :           continue;
     391           0 :         }
     392             : 
     393           0 :         int err = verify_crds_value( &view->pull_response->values[ i ], payload+view->pull_response->values[ i ].offset, view->pull_response->values[ i ].length, sha );
     394           0 :         if( FD_UNLIKELY( err!=FD_ED25519_SUCCESS ) ) {
     395           0 :           if( FD_LIKELY( !failed[ i ] ) ) {
     396           0 :             ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_SIGNATURE_IDX ]++;
     397           0 :             ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_SIGNATURE_IDX ] += view->pull_response->values[ i ].length;
     398           0 :           }
     399           0 :           view->pull_response->values_len--;
     400           0 :           view->pull_response->values[ i ] = view->pull_response->values[ view->pull_response->values_len ];
     401           0 :           failed[ i ] = failed[ view->pull_response->values_len ];
     402           0 :           continue;
     403           0 :         }
     404             : 
     405           0 :         i++;
     406           0 :       }
     407             : 
     408           0 :       if( FD_UNLIKELY( !view->pull_response->values_len ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_RESPONSE_NO_VALID_CRDS_IDX;
     409           0 :       return 0;
     410           0 :     }
     411           0 :     case FD_GOSSIP_MESSAGE_PUSH: {
     412           0 :       ulong i = 0UL;
     413           0 :       while( i<view->push->values_len ) {
     414           0 :         int err = verify_crds_value( &view->push->values[ i ], payload+view->push->values[ i ].offset, view->push->values[ i ].length, sha );
     415           0 :         if( FD_UNLIKELY( err!=FD_ED25519_SUCCESS ) ) {
     416           0 :           if( FD_LIKELY( !failed[ i ] ) ) {
     417           0 :             ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_SIGNATURE_IDX ]++;
     418           0 :             ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_SIGNATURE_IDX ] += view->push->values[ i ].length;
     419           0 :           }
     420           0 :           view->push->values_len--;
     421           0 :           view->push->values[ i ] = view->push->values[ view->push->values_len ];
     422           0 :           failed[ i ] = failed[ view->push->values_len ];
     423           0 :           continue;
     424           0 :         }
     425             : 
     426           0 :         i++;
     427           0 :       }
     428             : 
     429           0 :       if( FD_UNLIKELY( !view->push->values_len ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PUSH_NO_VALID_CRDS_IDX;
     430           0 :       return 0;
     431           0 :     }
     432           0 :     case FD_GOSSIP_MESSAGE_PRUNE: return verify_prune( view->prune, sha );
     433           0 :     case FD_GOSSIP_MESSAGE_PING: {
     434           0 :       if( FD_UNLIKELY( FD_ED25519_SUCCESS!=fd_ed25519_verify( view->ping->token, 32UL, view->ping->signature, view->ping->from, sha ) ) ) {
     435           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PING_SIGNATURE_IDX;
     436           0 :       } else {
     437           0 :         return 0;
     438           0 :       }
     439           0 :     }
     440           0 :     case FD_GOSSIP_MESSAGE_PONG: {
     441           0 :       if( FD_UNLIKELY( FD_ED25519_SUCCESS!=fd_ed25519_verify( view->pong->hash, 32UL, view->pong->signature, view->pong->from, sha ) ) ) {
     442           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PONG_SIGNATURE_IDX;
     443           0 :       } else {
     444           0 :         return 0;
     445           0 :       }
     446           0 :     }
     447           0 :     default: FD_LOG_CRIT(( "unexpected message tag %u", view->tag ));
     448           0 :   };
     449           0 : }
     450             : 
     451             : static void
     452             : filter_shred_version_crds( fd_gossvf_tile_ctx_t * ctx,
     453             :                            uint                   tag,
     454             :                            fd_gossip_value_t *    values,
     455             :                            ulong                  values_len,
     456           0 :                            uchar *                failed ) {
     457           0 :   for( ulong i=0UL; i<values_len; i++ ) {
     458           0 :     if( FD_UNLIKELY( failed[ i ] ) ) continue;
     459             : 
     460           0 :     int keep      = 0;
     461           0 :     int no_origin = 0;
     462           0 :     if( values[ i ].tag==FD_GOSSIP_VALUE_CONTACT_INFO ) {
     463           0 :       keep = values[ i ].contact_info->shred_version==ctx->shred_version;
     464           0 :     } else {
     465           0 :       peer_t const * origin = peer_map_ele_query_const( ctx->peer_map, (fd_pubkey_t*)(values[ i ].origin), NULL, ctx->peers );
     466           0 :       no_origin = !origin;
     467           0 :       keep = origin && origin->shred_version==ctx->shred_version;
     468           0 :     }
     469             : 
     470           0 :     if( FD_UNLIKELY( !keep ) ) {
     471           0 :       if( FD_UNLIKELY( tag==FD_GOSSIP_MESSAGE_PULL_RESPONSE ) ) {
     472           0 :         if( FD_LIKELY( no_origin ) ) {
     473           0 :           ctx->metrics.crds_rx[       FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_ORIGIN_NO_CONTACT_INFO_IDX ]++;
     474           0 :           ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_ORIGIN_NO_CONTACT_INFO_IDX ] += values[ i ].length;
     475           0 :         } else {
     476           0 :           ctx->metrics.crds_rx[       FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_ORIGIN_SHRED_VERSION_IDX ]++;
     477           0 :           ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_ORIGIN_SHRED_VERSION_IDX ] += values[ i ].length;
     478           0 :         }
     479           0 :       } else {
     480           0 :         if( FD_LIKELY( no_origin ) ) {
     481           0 :           ctx->metrics.crds_rx[       FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_ORIGIN_NO_CONTACT_INFO_IDX ]++;
     482           0 :           ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_ORIGIN_NO_CONTACT_INFO_IDX ] += values[ i ].length;
     483           0 :         } else {
     484           0 :           ctx->metrics.crds_rx[       FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_ORIGIN_SHRED_VERSION_IDX ]++;
     485           0 :           ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_ORIGIN_SHRED_VERSION_IDX ] += values[ i ].length;
     486           0 :         }
     487           0 :       }
     488           0 :       failed[ i ] = FD_GOSSIP_FAILED_NO_CONTACT_INFO;
     489           0 :     }
     490           0 :   }
     491           0 : }
     492             : 
     493             : static int
     494             : filter_shred_version( fd_gossvf_tile_ctx_t * ctx,
     495             :                       fd_gossip_message_t *  view,
     496           0 :                       uchar *                failed ) {
     497           0 :   switch( view->tag ) {
     498           0 :     case FD_GOSSIP_MESSAGE_PING:
     499           0 :     case FD_GOSSIP_MESSAGE_PONG:
     500           0 :     case FD_GOSSIP_MESSAGE_PRUNE:
     501           0 :       return 0;
     502           0 :     case FD_GOSSIP_MESSAGE_PUSH: {
     503           0 :       filter_shred_version_crds( ctx, view->tag, view->push->values, view->push->values_len, failed );
     504           0 :       if( FD_UNLIKELY( !view->push->values_len ) ) {
     505           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PUSH_NO_VALID_CRDS_IDX;
     506           0 :       } else {
     507           0 :         return 0;
     508           0 :       }
     509           0 :     }
     510           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE: {
     511           0 :       filter_shred_version_crds( ctx, view->tag, view->pull_response->values, view->pull_response->values_len, failed );
     512           0 :       if( FD_UNLIKELY( !view->pull_response->values_len ) ) {
     513           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_RESPONSE_NO_VALID_CRDS_IDX;
     514           0 :       } else {
     515           0 :         return 0;
     516           0 :       }
     517           0 :     }
     518           0 :     case FD_GOSSIP_MESSAGE_PULL_REQUEST:
     519           0 :       FD_TEST( view->pull_request->contact_info->tag==FD_GOSSIP_VALUE_CONTACT_INFO );
     520           0 :       if( FD_UNLIKELY( view->pull_request->contact_info->contact_info->shred_version!=ctx->shred_version ) ) {
     521           0 :         return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_SHRED_VERSION_IDX;
     522           0 :       } else {
     523           0 :         return 0;
     524           0 :       }
     525           0 :     default:
     526           0 :       FD_LOG_CRIT(( "unexpected message tag %u", view->tag ));
     527           0 :   }
     528           0 : }
     529             : 
     530             : static void
     531             : check_duplicate_instance( fd_gossvf_tile_ctx_t *      ctx,
     532           0 :                           fd_gossip_message_t const * view ) {
     533           0 :   ulong values_len;
     534           0 :   fd_gossip_value_t const * values;
     535           0 :   switch( view->tag ) {
     536           0 :     case FD_GOSSIP_MESSAGE_PING:
     537           0 :     case FD_GOSSIP_MESSAGE_PONG:
     538           0 :     case FD_GOSSIP_MESSAGE_PRUNE:
     539           0 :     case FD_GOSSIP_MESSAGE_PULL_REQUEST:
     540           0 :       return;
     541           0 :     case FD_GOSSIP_MESSAGE_PUSH:
     542           0 :       values = view->push->values;
     543           0 :       values_len = view->push->values_len;
     544           0 :       break;
     545           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE:
     546           0 :       values = view->pull_response->values;
     547           0 :       values_len = view->pull_response->values_len;
     548           0 :       break;
     549           0 :     default:
     550           0 :       FD_LOG_CRIT(( "unexpected message tag %u", view->tag ));
     551           0 :   }
     552             : 
     553           0 :   for( ulong i=0UL; i<values_len; i++ ) {
     554           0 :     fd_gossip_value_t const * value = &values[ i ];
     555           0 :     if( FD_UNLIKELY( value->tag!=FD_GOSSIP_VALUE_CONTACT_INFO ) ) continue;
     556             : 
     557           0 :     if( FD_LIKELY( ctx->instance_creation_wallclock_nanos>=FD_MICRO_TO_NANOSEC( value->contact_info->outset ) ) ) continue;
     558           0 :     if( FD_LIKELY( memcmp( ctx->identity_pubkey->uc, value->origin, 32UL ) ) ) continue;
     559             : 
     560           0 :     FD_LOG_ERR(( "duplicate running instances of the same validator node, our timestamp: %ldns their timestamp: %ldns", ctx->instance_creation_wallclock_nanos, FD_MICRO_TO_NANOSEC( value->contact_info->outset ) ));
     561           0 :   }
     562           0 : }
     563             : 
     564             : static inline int
     565             : is_ping_active( fd_gossvf_tile_ctx_t *  ctx,
     566             :                 fd_ip4_port_t           addr,
     567           0 :                 fd_pubkey_t const *     pubkey ) {
     568             :   /* 1. If the node has more than FD_GOSSIP_STAKED_THRESHOLD lamports
     569             :         staked, it is active */
     570           0 :   stake_t const * stake = stake_map_ele_query_const( ctx->stake.map, pubkey, NULL, ctx->stake.pool );
     571           0 :   if( FD_LIKELY( stake && stake->stake>=FD_GOSSIP_STAKED_THRESHOLD ) ) return 1;
     572             : 
     573             :   /* 2. If the node has actively ponged a ping, it is active */
     574           0 :   ping_t * ping = ping_map_ele_query( ctx->ping_map, pubkey, NULL, ctx->pings );
     575           0 :   return ping!=NULL && ping->addr.addr==addr.addr && ping->addr.port==addr.port;
     576           0 : }
     577             : 
     578             : static int
     579             : ping_if_unponged( fd_gossvf_tile_ctx_t * ctx,
     580             :                   fd_ip4_port_t          addr,
     581             :                   uchar const *          origin,
     582           0 :                   fd_stem_context_t *    stem ) {
     583           0 :   if( FD_UNLIKELY( !is_ping_active( ctx, addr, fd_type_pun_const( origin ) ) ) ) {
     584           0 :     fd_gossip_pingreq_t * pingreq = (fd_gossip_pingreq_t*)fd_chunk_to_laddr( ctx->out->mem, ctx->out->chunk );
     585           0 :     fd_memcpy( pingreq->pubkey.uc, origin, 32UL );
     586           0 :     fd_stem_publish( stem, 0UL, fd_gossvf_sig( addr.addr, addr.port, 1 ), ctx->out->chunk, sizeof(fd_gossip_pingreq_t), 0UL, 0UL, 0UL );
     587           0 :     ctx->out->chunk = fd_dcache_compact_next( ctx->out->chunk, sizeof(fd_gossip_pingreq_t), ctx->out->chunk0, ctx->out->wmark );
     588             : 
     589             : #if DEBUG_PEERS
     590             :     char base58[ FD_BASE58_ENCODED_32_SZ ];
     591             :     fd_base58_encode_32( origin, NULL, base58 );
     592             :     FD_LOG_NOTICE(( "pinging %s (" FD_IP4_ADDR_FMT ":%hu) (%lu)", base58, FD_IP4_ADDR_FMT_ARGS( addr.addr ), addr.port, ctx->ping_cnt ));
     593             : #endif
     594           0 :     return 1;
     595           0 :   }
     596           0 :   return 0;
     597           0 : }
     598             : 
     599             : static int
     600             : check_addr( fd_ip4_port_t addr,
     601           0 :             int           allow_private_address ) {
     602           0 :   if( FD_UNLIKELY( !addr.port || !addr.addr || fd_ip4_addr_is_mcast( addr.addr ) ) ) return 0;
     603           0 :   if( FD_UNLIKELY( !allow_private_address && !fd_ip4_addr_is_public( addr.addr ) ) ) return 0;
     604           0 :   return 1;
     605           0 : }
     606             : 
     607             : static int
     608             : verify_addresses( fd_gossvf_tile_ctx_t * ctx,
     609             :                   fd_gossip_message_t *  view,
     610             :                   uchar *                failed,
     611           0 :                   fd_stem_context_t *    stem ) {
     612           0 :   int is_loopback_peer = (ctx->peer.addr==ctx->src_addr.addr && ctx->peer.port==ctx->src_addr.port) ||
     613           0 :                          (ctx->peer.addr==ctx->gossip_addr.addr && ctx->peer.port==ctx->gossip_addr.port) ||
     614           0 :                          (fd_ip4_addr_is_loopback( ctx->peer.addr ) && ctx->peer.port==ctx->src_addr.port);
     615             : 
     616           0 :   ulong values_len;
     617           0 :   fd_gossip_value_t * values;
     618           0 :   switch( view->tag ) {
     619           0 :     case FD_GOSSIP_MESSAGE_PING:
     620           0 :     case FD_GOSSIP_MESSAGE_PONG:
     621           0 :     case FD_GOSSIP_MESSAGE_PRUNE:
     622           0 :       return 0;
     623           0 :     case FD_GOSSIP_MESSAGE_PULL_REQUEST:
     624           0 :       if( FD_UNLIKELY( !check_addr( ctx->peer, ctx->allow_private_address ) ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_INACTIVE_IDX;
     625           0 :       if( FD_UNLIKELY( is_loopback_peer ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_LOOPBACK_IDX;
     626           0 :       if( FD_UNLIKELY( ping_if_unponged( ctx, ctx->peer, view->pull_request->contact_info->origin, stem ) ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_INACTIVE_IDX;
     627           0 :       return 0;
     628           0 :     case FD_GOSSIP_MESSAGE_PUSH:
     629           0 :       if( FD_UNLIKELY( is_loopback_peer ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PUSH_LOOPBACK_IDX;
     630           0 :       values_len = view->push->values_len;
     631           0 :       values = view->push->values;
     632           0 :       break;
     633           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE:
     634           0 :       if( FD_UNLIKELY( is_loopback_peer ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_RESPONSE_LOOPBACK_IDX;
     635           0 :       values_len = view->pull_response->values_len;
     636           0 :       values = view->pull_response->values;
     637           0 :       break;
     638           0 :     default:
     639           0 :       FD_LOG_ERR(( "unexpected view tag %u", view->tag ));
     640           0 :   }
     641             : 
     642           0 :   for( ulong i=0UL; i<values_len; i++ ) {
     643           0 :     fd_gossip_value_t const * value = &values[ i ];
     644           0 :     if( FD_UNLIKELY( failed[ i ] || value->tag!=FD_GOSSIP_VALUE_CONTACT_INFO ) ) continue;
     645             : 
     646             :     /* We currently don't handle IPv6, so setting the address to 0 will
     647             :        cause it to be always dropped. */
     648           0 :     fd_ip4_port_t addr = {
     649           0 :       .addr = value->contact_info->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].is_ipv6 ? 0U : value->contact_info->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].ip4,
     650           0 :       .port = value->contact_info->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].port
     651           0 :     };
     652             : 
     653             :     /* Sanitize sockets: zero out any with a multicast address.
     654             :        Matches Agave, which omits bad sockets from the cache but
     655             :        still accepts the ContactInfo into CRDS. */
     656           0 :     for( ulong j=0UL; j<FD_GOSSIP_CONTACT_INFO_SOCKET_CNT; j++ ) {
     657           0 :       fd_gossip_socket_t * sock = &values[ i ].contact_info->sockets[ j ];
     658           0 :       if( !sock->port || sock->is_ipv6 ) continue;
     659           0 :       if( FD_UNLIKELY( fd_ip4_addr_is_mcast( sock->ip4 ) ) ) {
     660           0 :         sock->ip4  = 0U;
     661           0 :         sock->port = 0;
     662           0 :       }
     663           0 :     }
     664             : 
     665             :     /* Prevent ping loopback */
     666           0 :     if( FD_UNLIKELY( !memcmp( value->origin, ctx->identity_pubkey, 32UL ) ) ) continue;
     667             : 
     668           0 :     int is_self_addr = (addr.addr==ctx->gossip_addr.addr && addr.port==ctx->gossip_addr.port) ||
     669           0 :                        (addr.addr==ctx->src_addr.addr && addr.port==ctx->src_addr.port);
     670           0 :     int is_loopback  = fd_ip4_addr_is_loopback( addr.addr ) && addr.port==ctx->gossip_addr.port;
     671           0 :     int drop         = (is_self_addr | is_loopback) || !check_addr( addr, ctx->allow_private_address ) || ping_if_unponged( ctx, addr, value->origin, stem );
     672             : 
     673           0 :     if( FD_UNLIKELY( drop ) ) {
     674           0 :       if( FD_LIKELY( view->tag==FD_GOSSIP_MESSAGE_PUSH ) ) {
     675           0 :         ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_INACTIVE_IDX ]++;
     676           0 :         ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_INACTIVE_IDX ] += value->length;
     677           0 :       } else {
     678           0 :         ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_INACTIVE_IDX ]++;
     679           0 :         ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_INACTIVE_IDX ] += value->length;
     680           0 :       }
     681             :       /* Mark as failed instead of removing so gossip tile can
     682             :          track the hash in the purged set. */
     683           0 :       failed[ i ] = FD_GOSSIP_FAILED_NO_CONTACT_INFO;
     684           0 :     }
     685           0 :   }
     686             : 
     687           0 :   return 0;
     688           0 : }
     689             : 
     690             : static void
     691             : handle_ping_update( fd_gossvf_tile_ctx_t *    ctx,
     692           0 :                     fd_gossip_ping_update_t * ping_update ) {
     693             : #if DEBUG_PEERS
     694             :     char base58[ FD_BASE58_ENCODED_32_SZ ];
     695             :     fd_base58_encode_32( ping_update->pubkey.uc, NULL, base58 );
     696             : #endif
     697             : 
     698           0 :   if( FD_UNLIKELY( ping_update->remove ) ) {
     699             : #if DEBUG_PEERS
     700             :     ctx->ping_cnt--;
     701             :     FD_LOG_NOTICE(( "removing ping for %s (" FD_IP4_ADDR_FMT ":%hu) (%lu)", base58, FD_IP4_ADDR_FMT_ARGS( ping_update->gossip_addr.addr ), fd_ushort_bswap( ping_update->gossip_addr.port ), ctx->ping_cnt ));
     702             : #endif
     703             : 
     704           0 :     ping_t * ping = ping_map_ele_remove( ctx->ping_map, &ping_update->pubkey, NULL, ctx->pings );
     705           0 :     FD_TEST( ping );
     706           0 :     ping_pool_ele_release( ctx->pings, ping );
     707           0 :   } else {
     708             : #if DEBUG_PEERS
     709             :     ctx->ping_cnt++;
     710             :     FD_LOG_NOTICE(( "adding ping for %s (" FD_IP4_ADDR_FMT ":%hu) (%lu)", base58, FD_IP4_ADDR_FMT_ARGS( ping_update->gossip_addr.addr ), fd_ushort_bswap( ping_update->gossip_addr.port ), ctx->ping_cnt ));
     711             : #endif
     712             : 
     713           0 :     FD_TEST( ping_pool_free( ctx->pings ) );
     714           0 :     FD_TEST( !ping_map_ele_query( ctx->ping_map, &ping_update->pubkey, NULL, ctx->pings ) );
     715           0 :     ping_t * ping = ping_pool_ele_acquire( ctx->pings );
     716           0 :     ping->addr.l = ping_update->gossip_addr.l;
     717           0 :     fd_memcpy( ping->pubkey.uc, ping_update->pubkey.uc, 32UL );
     718           0 :     ping_map_ele_insert( ctx->ping_map, ping, ctx->pings );
     719           0 :   }
     720           0 : }
     721             : 
     722             : static void
     723             : handle_peer_update( fd_gossvf_tile_ctx_t *       ctx,
     724           0 :                     fd_gossip_update_message_t * gossip_update ) {
     725             : #if DEBUG_PEERS
     726             :     char base58[ FD_BASE58_ENCODED_32_SZ ];
     727             :     fd_base58_encode_32( gossip_update->origin, NULL, base58 );
     728             : #endif
     729             : 
     730           0 :   switch( gossip_update->tag ) {
     731           0 :     case FD_GOSSIP_UPDATE_TAG_CONTACT_INFO: {
     732           0 :       peer_t * peer = peer_map_ele_query( ctx->peer_map, fd_type_pun_const( gossip_update->origin ), NULL, ctx->peers );
     733           0 :       if( FD_LIKELY( peer ) ) {
     734             : #if DEBUG_PEERS
     735             :         FD_LOG_NOTICE(( "updating peer %s (" FD_IP4_ADDR_FMT ":%hu) (%lu)", base58, FD_IP4_ADDR_FMT_ARGS( gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].ip4 ), fd_ushort_bswap( gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].port ), ctx->peer_cnt ));
     736             : #endif
     737             : 
     738           0 :         peer->shred_version = gossip_update->contact_info->value->shred_version;
     739           0 :         peer->gossip_addr.addr = gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].is_ipv6 ? 0U : gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].ip4;
     740           0 :         peer->gossip_addr.port = gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].port;
     741           0 :       } else {
     742             : #if DEBUG_PEERS
     743             :         ctx->peer_cnt++;
     744             :         FD_LOG_NOTICE(( "adding peer %s (" FD_IP4_ADDR_FMT ":%hu) (%lu)", base58, FD_IP4_ADDR_FMT_ARGS( gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].ip4 ), fd_ushort_bswap( gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].port ), ctx->peer_cnt ));
     745             : #endif
     746             : 
     747           0 :         FD_TEST( peer_pool_free( ctx->peers ) );
     748           0 :         peer = peer_pool_ele_acquire( ctx->peers );
     749           0 :         peer->shred_version = gossip_update->contact_info->value->shred_version;
     750           0 :         peer->gossip_addr.addr = gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].is_ipv6 ? 0U : gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].ip4;
     751           0 :         peer->gossip_addr.port = gossip_update->contact_info->value->sockets[ FD_GOSSIP_CONTACT_INFO_SOCKET_GOSSIP ].port;
     752           0 :         fd_memcpy( peer->pubkey.uc, gossip_update->origin, 32UL );
     753           0 :         peer_map_ele_insert( ctx->peer_map, peer, ctx->peers );
     754           0 :       }
     755           0 :       break;
     756           0 :     }
     757           0 :     case FD_GOSSIP_UPDATE_TAG_CONTACT_INFO_REMOVE: {
     758             : #if DEBUG_PEERS
     759             :       ctx->peer_cnt--;
     760             :       FD_LOG_NOTICE(( "removing peer %s (%lu)", base58, ctx->peer_cnt ));
     761             : #endif
     762             : 
     763           0 :       peer_t * peer = peer_map_ele_remove( ctx->peer_map, fd_type_pun_const( gossip_update->origin ), NULL, ctx->peers );
     764           0 :       FD_TEST( peer );
     765           0 :       peer_pool_ele_release( ctx->peers, peer );
     766           0 :       break;
     767           0 :     }
     768           0 :     default: FD_LOG_ERR(( "unexpected gossip_update tag %d", gossip_update->tag ));
     769           0 :   }
     770           0 : }
     771             : 
     772             : static int
     773             : handle_net( fd_gossvf_tile_ctx_t * ctx,
     774             :             ulong                  sz,
     775             :             ulong                  tsorig,
     776           0 :             fd_stem_context_t *    stem ) {
     777           0 :   uchar * payload;
     778           0 :   ulong payload_sz;
     779           0 :   fd_ip4_hdr_t * ip4_hdr;
     780           0 :   fd_udp_hdr_t * udp_hdr;
     781           0 :   FD_TEST( fd_ip4_udp_hdr_strip( ctx->payload, sz, &payload, &payload_sz, NULL, &ip4_hdr, &udp_hdr ) );
     782           0 :   ctx->peer.addr = ip4_hdr->saddr;
     783           0 :   ctx->peer.port = udp_hdr->net_sport;
     784             : 
     785           0 :   long now = ctx->last_wallclock + (long)((double)(fd_tickcount()-ctx->last_tickcount)/ctx->ticks_per_ns);
     786             : 
     787           0 :   fd_gossip_message_t * message = ctx->_message;
     788           0 :   int decoded = fd_gossip_message_deserialize( message, payload, payload_sz );
     789           0 :   if( FD_UNLIKELY( !decoded ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_UNPARSEABLE_IDX;
     790             : 
     791           0 :   if( FD_UNLIKELY( message->tag==FD_GOSSIP_MESSAGE_PULL_REQUEST ) ) {
     792           0 :     if( FD_UNLIKELY( message->pull_request->contact_info->tag!=FD_GOSSIP_VALUE_CONTACT_INFO ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_NOT_CONTACT_INFO_IDX;
     793             :     /* Best-effort: identity_pubkey may momentarily be stale during a
     794             :        keyswitch, but letting a loopback pull request through is
     795             :        harmless (we just waste some CPU answering our own request). */
     796           0 :     if( FD_UNLIKELY( !memcmp( message->pull_request->contact_info->origin, ctx->identity_pubkey, 32UL ) ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_LOOPBACK_IDX;
     797           0 :     if( FD_UNLIKELY( message->pull_request->crds_filter->mask_bits>=64U ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_MASK_BITS_IDX;
     798             : 
     799           0 :     ulong clamp_wallclock_lower_millis = (ulong)(FD_NANOSEC_TO_MILLI( now )-15L*1000L);
     800           0 :     ulong clamp_wallclock_upper_millis = (ulong)(FD_NANOSEC_TO_MILLI( now )+15L*1000L);
     801           0 :     if( FD_UNLIKELY( message->pull_request->contact_info->wallclock<clamp_wallclock_lower_millis ||
     802           0 :                      message->pull_request->contact_info->wallclock>clamp_wallclock_upper_millis ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PULL_REQUEST_WALLCLOCK_IDX;
     803           0 :   }
     804             : 
     805           0 :   if( FD_UNLIKELY( message->tag==FD_GOSSIP_MESSAGE_PRUNE ) ) {
     806             :     /* Best-effort: identity_pubkey may momentarily be stale during a
     807             :        keyswitch, so we may drop a prune correctly addressed to our new
     808             :        key.  This just delays pruning until the peer re-prunes or we
     809             :        detect the redundancy via the prune finder. */
     810           0 :     if( FD_UNLIKELY( !!memcmp( message->prune->destination, ctx->identity_pubkey, 32UL ) ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PRUNE_DESTINATION_IDX;
     811             :     /* Agave uses a window of 500ms here, rather than 1s, but it's too
     812             :        narrow in production and causes us to throw away a lot of prunes
     813             :        that are actually valid and useful. */
     814           0 :     if( FD_UNLIKELY( (ulong)(FD_NANOSEC_TO_MILLI( now )-1000L)>message->prune->wallclock ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PRUNE_WALLCLOCK_IDX;
     815           0 :   }
     816             : 
     817           0 :   if( FD_LIKELY( message->tag==FD_GOSSIP_MESSAGE_PUSH ) ) {
     818           0 :     ulong i = 0UL;
     819           0 :     while( i<message->push->values_len ) {
     820           0 :       fd_gossip_value_t const * value = &message->push->values[ i ];
     821           0 :       if( FD_UNLIKELY( value->wallclock<(ulong)(FD_NANOSEC_TO_MILLI( now )-15L*1000L) ||
     822           0 :                        value->wallclock>(ulong)(FD_NANOSEC_TO_MILLI( now )+15L*1000L) ) ) {
     823           0 :         ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_WALLCLOCK_IDX ]++;
     824           0 :         ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PUSH_WALLCLOCK_IDX ] += value->length;
     825           0 :         message->push->values[ i ] = message->push->values[ message->push->values_len-1UL ];
     826           0 :         message->push->values_len--;
     827           0 :         continue;
     828           0 :       }
     829           0 :       i++;
     830           0 :     }
     831             : 
     832           0 :     if( FD_UNLIKELY( !message->push->values_len ) ) return FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_DROPPED_PUSH_NO_VALID_CRDS_IDX;
     833           0 :   }
     834             : 
     835           0 :   uchar failed[ FD_GOSSIP_MESSAGE_MAX_CRDS ] = {0};
     836             : 
     837           0 :   if( FD_UNLIKELY( message->tag==FD_GOSSIP_MESSAGE_PULL_RESPONSE ) ) {
     838           0 :     int has_staked_node = ctx->stake.count>0UL;
     839           0 :     for( ulong i=0UL; i<message->pull_response->values_len; i++ ) {
     840           0 :       fd_gossip_value_t const * value = &message->pull_response->values[ i ];
     841             : 
     842             :       /* Best-effort: identity_pubkey may momentarily be stale during a
     843             :          keyswitch, but the only effect is that our own values fall back
     844             :          to the normal staleness window instead of being unconditionally
     845             :          accepted. */
     846           0 :       uchar is_me = !memcmp( value->origin, ctx->identity_pubkey, 32UL );
     847           0 :       long accept_after_nanos;
     848           0 :       if( FD_UNLIKELY( is_me ) ) {
     849           0 :         accept_after_nanos = 0L;
     850           0 :       } else {
     851           0 :         stake_t const * entry = stake_map_ele_query_const( ctx->stake.map, (fd_pubkey_t const *)value->origin, NULL, ctx->stake.pool );
     852           0 :         ulong origin_stake = entry ? entry->stake : 0UL;
     853           0 :         if( !origin_stake && has_staked_node ) accept_after_nanos = now-15L*1000L*1000L*1000L;
     854           0 :         else                                   accept_after_nanos = now-432000L*400L*1000L*1000L;
     855           0 :       }
     856             : 
     857           0 :       if( FD_UNLIKELY( (ulong)(FD_NANOSEC_TO_MILLI( accept_after_nanos ))>value->wallclock ) ) {
     858           0 :         peer_t const * origin_peer = peer_map_ele_query_const( ctx->peer_map, (fd_pubkey_t const *)value->origin, NULL, ctx->peers );
     859           0 :         if( FD_UNLIKELY( !origin_peer ) ) {
     860           0 :           ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_WALLCLOCK_IDX ]++;
     861           0 :           ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_DROPPED_PULL_RESPONSE_WALLCLOCK_IDX ] += value->length;
     862           0 :           failed[ i ] = FD_GOSSIP_FAILED_WALLCLOCK;
     863           0 :         }
     864           0 :       }
     865           0 :     }
     866           0 :   }
     867             : 
     868           0 :   int result = filter_shred_version( ctx, message, failed );
     869           0 :   if( FD_UNLIKELY( result ) ) return result;
     870             : 
     871           0 :   result = verify_signatures( ctx, message, payload, ctx->sha, failed );
     872           0 :   if( FD_UNLIKELY( result ) ) return result;
     873             : 
     874             :   /* verify_addresses includes validation checks against origin pubkey,
     875             :      so it must come after verify_signatures. */
     876           0 :   result = verify_addresses( ctx, message, failed, stem );
     877           0 :   if( FD_UNLIKELY( result ) ) return result;
     878             : 
     879           0 :   check_duplicate_instance( ctx, message );
     880             : 
     881           0 :   switch( message->tag ) {
     882           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE: {
     883           0 :       for( ulong i=0UL; i<message->pull_response->values_len; i++ ) {
     884           0 :         if( FD_UNLIKELY( failed[ i ]==FD_GOSSIP_FAILED_NO_CONTACT_INFO ) ) continue; /* Don't add to tcache so we can re-receive after learning contact info */
     885           0 :         ulong dedup_tag = ctx->seed ^ fd_ulong_load_8_fast( message->pull_response->values[ i ].signature );
     886           0 :         int ha_dup = 0;
     887           0 :         FD_TCACHE_INSERT( ha_dup, *ctx->tcache.sync, ctx->tcache.ring, ctx->tcache.depth, ctx->tcache.map, ctx->tcache.map_cnt, dedup_tag );
     888           0 :         (void)ha_dup; /* unused */
     889           0 :       }
     890           0 :       break;
     891           0 :     }
     892           0 :     case FD_GOSSIP_MESSAGE_PUSH: {
     893           0 :       for( ulong i=0UL; i<message->push->values_len; i++ ) {
     894           0 :         if( FD_UNLIKELY( failed[ i ] ) ) continue; /* Don't add to tcache so we can re-receive after learning contact info */
     895           0 :         ulong dedup_tag = ctx->seed ^ fd_ulong_load_8_fast( message->push->values[ i ].signature );
     896           0 :         int ha_dup = 0;
     897           0 :         FD_TCACHE_INSERT( ha_dup, *ctx->tcache.sync, ctx->tcache.ring, ctx->tcache.depth, ctx->tcache.map, ctx->tcache.map_cnt, dedup_tag );
     898           0 :         (void)ha_dup; /* unused */
     899           0 :       }
     900           0 :       break;
     901           0 :     }
     902           0 :     default:
     903           0 :       break;
     904           0 :   }
     905             : 
     906           0 :   switch( message->tag ) {
     907           0 :     case FD_GOSSIP_MESSAGE_PULL_REQUEST:  result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PULL_REQUEST_IDX; break;
     908           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE: result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PULL_RESPONSE_IDX; break;
     909           0 :     case FD_GOSSIP_MESSAGE_PUSH:          result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PUSH_IDX; break;
     910           0 :     case FD_GOSSIP_MESSAGE_PRUNE:         result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PRUNE_IDX; break;
     911           0 :     case FD_GOSSIP_MESSAGE_PING:          result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PING_IDX; break;
     912           0 :     case FD_GOSSIP_MESSAGE_PONG:          result = FD_METRICS_ENUM_GOSSVF_MESSAGE_OUTCOME_V_SUCCESS_PONG_IDX; break;
     913           0 :     default: FD_LOG_ERR(( "unexpected message tag %u", message->tag ));
     914           0 :   }
     915             : 
     916           0 :   switch( message->tag ) {
     917           0 :     case FD_GOSSIP_MESSAGE_PULL_RESPONSE:
     918           0 :       for( ulong i=0UL; i<message->pull_response->values_len; i++ ) {
     919           0 :         if( FD_UNLIKELY( failed[ i ] ) ) continue;
     920           0 :         ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_SUCCESS_PULL_RESPONSE_IDX ]++;
     921           0 :         ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_SUCCESS_PULL_RESPONSE_IDX ] += message->pull_response->values[ i ].length;
     922           0 :       }
     923           0 :       break;
     924           0 :     case FD_GOSSIP_MESSAGE_PUSH:
     925           0 :       for( ulong i=0UL; i<message->push->values_len; i++ ) {
     926           0 :         if( FD_UNLIKELY( failed[ i ] ) ) continue;
     927           0 :         ctx->metrics.crds_rx[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_SUCCESS_PUSH_IDX ]++;
     928           0 :         ctx->metrics.crds_rx_bytes[ FD_METRICS_ENUM_GOSSVF_CRDS_OUTCOME_V_SUCCESS_PUSH_IDX ] += message->push->values[ i ].length;
     929           0 :       }
     930           0 :       break;
     931           0 :     default:
     932           0 :       break;
     933           0 :   }
     934             : 
     935           0 :   uchar * dst = fd_chunk_to_laddr( ctx->out->mem, ctx->out->chunk );
     936           0 :   fd_memcpy( dst, message, sizeof(fd_gossip_message_t ) );
     937           0 :   fd_memcpy( dst+sizeof(fd_gossip_message_t), failed, FD_GOSSIP_MESSAGE_MAX_CRDS );
     938           0 :   fd_memcpy( dst+sizeof(fd_gossip_message_t)+FD_GOSSIP_MESSAGE_MAX_CRDS, payload, payload_sz );
     939             : 
     940           0 :   ulong tspub = (ulong)fd_frag_meta_ts_comp( fd_tickcount() );
     941           0 :   ulong out_sz = sizeof(fd_gossip_message_t)+FD_GOSSIP_MESSAGE_MAX_CRDS+payload_sz;
     942           0 :   fd_stem_publish( stem, 0UL, fd_gossvf_sig( ctx->peer.addr, ctx->peer.port, 0 ), ctx->out->chunk, out_sz, 0UL, tsorig, tspub );
     943           0 :   ctx->out->chunk = fd_dcache_compact_next( ctx->out->chunk, out_sz, ctx->out->chunk0, ctx->out->wmark );
     944             : 
     945           0 :   return result;
     946           0 : }
     947             : 
     948             : static inline void
     949             : after_frag( fd_gossvf_tile_ctx_t * ctx,
     950             :             ulong                  in_idx,
     951             :             ulong                  seq,
     952             :             ulong                  sig,
     953             :             ulong                  sz,
     954             :             ulong                  tsorig,
     955             :             ulong                  _tspub,
     956           0 :             fd_stem_context_t *    stem ) {
     957           0 :   (void)seq;
     958           0 :   (void)sig;
     959           0 :   (void)_tspub;
     960             : 
     961           0 :   switch( ctx->in[ in_idx ].kind ) {
     962           0 :     case IN_KIND_SHRED_VERSION: break;
     963           0 :     case IN_KIND_PINGS:  handle_ping_update( ctx, ctx->_ping_update ); break;
     964           0 :     case IN_KIND_GOSSIP: handle_peer_update( ctx, ctx->_gossip_update ); break;
     965           0 :     case IN_KIND_EPOCH: handle_epoch( ctx, (fd_epoch_info_msg_t const *) ctx->stake.msg_buf ); break;
     966           0 :     case IN_KIND_NET: {
     967           0 :       int result = handle_net( ctx, sz, tsorig, stem );
     968           0 :       ctx->metrics.message_rx[ result ]++;
     969           0 :       ctx->metrics.message_rx_bytes[ result ] += sz;
     970           0 :       break;
     971           0 :     }
     972           0 :     default: FD_LOG_ERR(( "unexpected in_kind %d", ctx->in[ in_idx ].kind ));
     973           0 :   }
     974           0 : }
     975             : 
     976             : static void
     977             : privileged_init( fd_topo_t const *      topo,
     978           0 :                  fd_topo_tile_t const * tile ) {
     979           0 :   void * scratch = fd_topo_obj_laddr( topo, tile->tile_obj_id );
     980             : 
     981           0 :   FD_SCRATCH_ALLOC_INIT( l, scratch );
     982           0 :   fd_gossvf_tile_ctx_t * ctx = FD_SCRATCH_ALLOC_APPEND( l, alignof( fd_gossvf_tile_ctx_t ), sizeof( fd_gossvf_tile_ctx_t ) );
     983           0 :   FD_TEST( fd_rng_secure( &ctx->seed, 8U ) );
     984             : 
     985           0 :   if( FD_UNLIKELY( !strcmp( tile->gossvf.identity_key_path, "" ) ) ) FD_LOG_ERR(( "identity_key_path not set" ));
     986             : 
     987           0 :   ctx->identity_pubkey[ 0 ] = *(fd_pubkey_t const *)fd_type_pun_const( fd_keyload_load( tile->gossvf.identity_key_path, /* pubkey only: */ 1 ) );
     988           0 : }
     989             : 
     990             : static void
     991             : unprivileged_init( fd_topo_t const *      topo,
     992           0 :                    fd_topo_tile_t const * tile ) {
     993           0 :   void * scratch = fd_topo_obj_laddr( topo, tile->tile_obj_id );
     994             : 
     995           0 :   FD_SCRATCH_ALLOC_INIT( l, scratch );
     996           0 :   fd_gossvf_tile_ctx_t * ctx = FD_SCRATCH_ALLOC_APPEND( l, alignof( fd_gossvf_tile_ctx_t ), sizeof( fd_gossvf_tile_ctx_t ) );
     997           0 :   void * _peer_pool          = FD_SCRATCH_ALLOC_APPEND( l, peer_pool_align(),               peer_pool_footprint( FD_CONTACT_INFO_TABLE_SIZE )                 );
     998           0 :   void * _peer_map           = FD_SCRATCH_ALLOC_APPEND( l, peer_map_align(),                peer_map_footprint( 2UL*FD_CONTACT_INFO_TABLE_SIZE )              );
     999           0 :   void * _ping_pool          = FD_SCRATCH_ALLOC_APPEND( l, ping_pool_align(),               ping_pool_footprint( FD_PING_TRACKER_MAX )                        );
    1000           0 :   void * _ping_map           = FD_SCRATCH_ALLOC_APPEND( l, ping_map_align(),                ping_map_footprint( 2UL*FD_PING_TRACKER_MAX )                     );
    1001           0 :   void * _stake_pool         = FD_SCRATCH_ALLOC_APPEND( l, stake_pool_align(),              stake_pool_footprint( MAX_SHRED_DESTS )                           );
    1002           0 :   void * _stake_map          = FD_SCRATCH_ALLOC_APPEND( l, stake_map_align(),               stake_map_footprint( stake_map_chain_cnt_est( MAX_SHRED_DESTS ) ) );
    1003           0 :   void * _tcache             = FD_SCRATCH_ALLOC_APPEND( l, fd_tcache_align(),               fd_tcache_footprint( tile->gossvf.tcache_depth, 0UL )             );
    1004             : 
    1005           0 :   ctx->peers = peer_pool_join( peer_pool_new( _peer_pool, FD_CONTACT_INFO_TABLE_SIZE ) );
    1006           0 :   FD_TEST( ctx->peers );
    1007             : 
    1008           0 :   ctx->peer_map = peer_map_join( peer_map_new( _peer_map, 2UL*FD_CONTACT_INFO_TABLE_SIZE, ctx->seed ) );
    1009           0 :   FD_TEST( ctx->peer_map );
    1010             : 
    1011           0 :   ctx->pings = ping_pool_join( ping_pool_new( _ping_pool, FD_PING_TRACKER_MAX ) );
    1012           0 :   FD_TEST( ctx->pings );
    1013             : 
    1014           0 :   ctx->ping_map = ping_map_join( ping_map_new( _ping_map, 2UL*FD_PING_TRACKER_MAX, ctx->seed ) );
    1015           0 :   FD_TEST( ctx->ping_map );
    1016             : 
    1017           0 :   ctx->stake.count = 0UL;
    1018           0 :   ctx->stake.pool  = stake_pool_join( stake_pool_new( _stake_pool, MAX_SHRED_DESTS ) );
    1019           0 :   FD_TEST( ctx->stake.pool );
    1020             : 
    1021           0 :   ctx->stake.map = stake_map_join( stake_map_new( _stake_map, stake_map_chain_cnt_est( MAX_SHRED_DESTS ), ctx->seed ) );
    1022           0 :   FD_TEST( ctx->stake.map );
    1023             : 
    1024           0 :   ctx->round_robin_cnt = fd_topo_tile_name_cnt( topo, tile->name );
    1025           0 :   ctx->round_robin_idx = tile->kind_id;
    1026             : 
    1027           0 :   ctx->allow_private_address = tile->gossvf.allow_private_address;
    1028           0 :   ctx->gossip_addr             = tile->gossvf.gossip_addr;
    1029           0 :   ctx->src_addr                = tile->gossvf.src_addr;
    1030             : 
    1031           0 :   ctx->keyswitch = fd_keyswitch_join( fd_topo_obj_laddr( topo, tile->id_keyswitch_obj_id ) );
    1032           0 :   FD_TEST( ctx->keyswitch );
    1033             : 
    1034           0 :   ctx->shred_version = tile->gossvf.shred_version;
    1035             : 
    1036           0 :   ctx->ticks_per_ns   = fd_tempo_tick_per_ns( NULL );
    1037           0 :   ctx->last_wallclock = fd_log_wallclock();
    1038           0 :   ctx->last_tickcount = fd_tickcount();
    1039             : 
    1040           0 :   FD_TEST( fd_sha512_join( fd_sha512_new( ctx->sha ) ) );
    1041             : 
    1042           0 :   fd_tcache_t * tcache = fd_tcache_join( fd_tcache_new( _tcache, tile->gossvf.tcache_depth, 0UL ) );
    1043           0 :   FD_TEST( tcache );
    1044             : 
    1045           0 :   ctx->tcache.depth   = fd_tcache_depth       ( tcache );
    1046           0 :   ctx->tcache.map_cnt = fd_tcache_map_cnt     ( tcache );
    1047           0 :   ctx->tcache.sync    = fd_tcache_oldest_laddr( tcache );
    1048           0 :   ctx->tcache.ring    = fd_tcache_ring_laddr  ( tcache );
    1049           0 :   ctx->tcache.map     = fd_tcache_map_laddr   ( tcache );
    1050             : 
    1051           0 :   ctx->entrypoints_cnt = tile->gossvf.entrypoints_cnt;
    1052           0 :   for( ulong i=0UL; i<tile->gossvf.entrypoints_cnt; i++ ) {
    1053           0 :     ctx->entrypoints[ i ].l = tile->gossvf.entrypoints[ i ].l;
    1054             : #if DEBUG_PEERS
    1055             :     FD_LOG_NOTICE(( "entrypoint " FD_IP4_ADDR_FMT ":%hu", FD_IP4_ADDR_FMT_ARGS( ctx->entrypoints[ i ].addr ), fd_ushort_bswap( ctx->entrypoints[ i ].port ) ));
    1056             : #endif
    1057           0 :   }
    1058             : 
    1059             :   /* Conversion to MICROs ensures we are comparing apples to apples in
    1060             :      check_duplicate_instance  */
    1061           0 :   ctx->instance_creation_wallclock_nanos = FD_MICRO_TO_NANOSEC( FD_NANOSEC_TO_MICRO( tile->gossvf.boot_timestamp_nanos ) );
    1062             : 
    1063             : #if DEBUG_PEERS
    1064             :   ctx->peer_cnt = 0UL;
    1065             :   ctx->ping_cnt = 0UL;
    1066             : #endif
    1067             : 
    1068           0 :   memset( &ctx->metrics, 0, sizeof( ctx->metrics ) );
    1069             : 
    1070           0 :   FD_TEST( tile->in_cnt<=sizeof(ctx->in)/sizeof(ctx->in[0]) );
    1071           0 :   for( ulong i=0UL; i<tile->in_cnt; i++ ) {
    1072           0 :     fd_topo_link_t const * link = &topo->links[ tile->in_link_id[ i ] ];
    1073           0 :     fd_topo_wksp_t const * link_wksp = &topo->workspaces[ topo->objs[ link->dcache_obj_id ].wksp_id ];
    1074             : 
    1075           0 :     ctx->in[ i ].mem    = link_wksp->wksp;
    1076           0 :     if( FD_LIKELY( link->mtu ) ) {
    1077           0 :       ctx->in[ i ].chunk0 = fd_dcache_compact_chunk0( ctx->in[ i ].mem, link->dcache );
    1078           0 :       ctx->in[ i ].wmark  = fd_dcache_compact_wmark ( ctx->in[ i ].mem, link->dcache, link->mtu );
    1079           0 :     } else {
    1080           0 :       ctx->in[ i ].chunk0 = 0UL;
    1081           0 :       ctx->in[ i ].wmark  = 0UL;
    1082           0 :     }
    1083           0 :     ctx->in[ i ].mtu    = link->mtu;
    1084             : 
    1085           0 :     if(      !strcmp( link->name, "gossip_gossvf" ) ) ctx->in[ i ].kind = IN_KIND_PINGS;
    1086           0 :     else if( !strcmp( link->name, "ipecho_out"    ) ) ctx->in[ i ].kind = IN_KIND_SHRED_VERSION;
    1087           0 :     else if( !strcmp( link->name, "gossip_out"    ) ) ctx->in[ i ].kind = IN_KIND_GOSSIP;
    1088           0 :     else if( !strcmp( link->name, "net_gossvf"    ) ) {
    1089           0 :       ctx->in[ i ].kind = IN_KIND_NET;
    1090           0 :       fd_net_rx_bounds_init( &ctx->net_in_bounds[ i ], link->dcache );
    1091           0 :     }
    1092           0 :     else if( !strcmp( link->name, "replay_epoch" ) ) ctx->in[ i ].kind = IN_KIND_EPOCH;
    1093           0 :     else FD_LOG_ERR(( "unexpected input link name %s", link->name ));
    1094           0 :   }
    1095             : 
    1096           0 :   FD_TEST( tile->out_cnt==1UL );
    1097           0 :   fd_topo_link_t const * gossvf_out = &topo->links[ tile->out_link_id[ 0UL ] ];
    1098           0 :   ctx->out->mem    = topo->workspaces[ topo->objs[ gossvf_out->dcache_obj_id ].wksp_id ].wksp;
    1099           0 :   ctx->out->chunk0 = fd_dcache_compact_chunk0( ctx->out->mem, gossvf_out->dcache );
    1100           0 :   ctx->out->wmark  = fd_dcache_compact_wmark ( ctx->out->mem, gossvf_out->dcache, gossvf_out->mtu );
    1101           0 :   ctx->out->chunk  = ctx->out->chunk0;
    1102             : 
    1103           0 :   ulong scratch_top = FD_SCRATCH_ALLOC_FINI( l, scratch_align() );
    1104           0 :   if( FD_UNLIKELY( scratch_top > (ulong)scratch + scratch_footprint( tile ) ) )
    1105           0 :     FD_LOG_ERR(( "scratch overflow %lu %lu %lu", scratch_top - (ulong)scratch - scratch_footprint( tile ), scratch_top, (ulong)scratch + scratch_footprint( tile ) ));
    1106           0 : }
    1107             : 
    1108             : static ulong
    1109             : populate_allowed_seccomp( fd_topo_t const *      topo,
    1110             :                           fd_topo_tile_t const * tile,
    1111             :                           ulong                  out_cnt,
    1112           0 :                           struct sock_filter *   out ) {
    1113           0 :   (void)topo;
    1114           0 :   (void)tile;
    1115             : 
    1116           0 :   populate_sock_filter_policy_fd_gossvf_tile( out_cnt, out, (uint)fd_log_private_logfile_fd() );
    1117           0 :   return sock_filter_policy_fd_gossvf_tile_instr_cnt;
    1118           0 : }
    1119             : 
    1120             : static ulong
    1121             : populate_allowed_fds( fd_topo_t const *      topo,
    1122             :                       fd_topo_tile_t const * tile,
    1123             :                       ulong                  out_fds_cnt,
    1124           0 :                       int *                  out_fds ) {
    1125           0 :   (void)topo;
    1126           0 :   (void)tile;
    1127             : 
    1128           0 :   if( FD_UNLIKELY( out_fds_cnt<2UL ) ) FD_LOG_ERR(( "out_fds_cnt %lu", out_fds_cnt ));
    1129             : 
    1130           0 :   ulong out_cnt = 0UL;
    1131           0 :   out_fds[ out_cnt++ ] = 2; /* stderr */
    1132           0 :   if( FD_LIKELY( -1!=fd_log_private_logfile_fd() ) )
    1133           0 :     out_fds[ out_cnt++ ] = fd_log_private_logfile_fd(); /* logfile */
    1134           0 :   return out_cnt;
    1135           0 : }
    1136             : 
    1137           0 : #define STEM_BURST (17UL/*FD_GOSSIP_MSG_MAX_CRDS*/+1UL)
    1138             : 
    1139           0 : #define STEM_LAZY  (128L*3000L)
    1140             : 
    1141           0 : #define STEM_CALLBACK_CONTEXT_TYPE  fd_gossvf_tile_ctx_t
    1142           0 : #define STEM_CALLBACK_CONTEXT_ALIGN alignof(fd_gossvf_tile_ctx_t)
    1143             : 
    1144           0 : #define STEM_CALLBACK_DURING_HOUSEKEEPING during_housekeeping
    1145           0 : #define STEM_CALLBACK_METRICS_WRITE       metrics_write
    1146           0 : #define STEM_CALLBACK_BEFORE_FRAG         before_frag
    1147           0 : #define STEM_CALLBACK_DURING_FRAG         during_frag
    1148           0 : #define STEM_CALLBACK_AFTER_FRAG          after_frag
    1149             : 
    1150             : #include "../../disco/stem/fd_stem.c"
    1151             : 
    1152             : fd_topo_run_tile_t fd_tile_gossvf = {
    1153             :   .name                     = "gossvf",
    1154             :   .populate_allowed_seccomp = populate_allowed_seccomp,
    1155             :   .populate_allowed_fds     = populate_allowed_fds,
    1156             :   .scratch_align            = scratch_align,
    1157             :   .scratch_footprint        = scratch_footprint,
    1158             :   .privileged_init          = privileged_init,
    1159             :   .unprivileged_init        = unprivileged_init,
    1160             :   .run                      = stem_run,
    1161             : };

Generated by: LCOV version 1.14