LCOV - code coverage report
Current view: top level - util/pod - fd_pod_ctl.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 415 532 78.0 %
Date: 2025-01-08 12:08:44 Functions: 6 6 100.0 %

          Line data    Source code
       1             : #include "../fd_util.h"
       2             : 
       3             : #if FD_HAS_HOSTED
       4             : 
       5             : #include <stdio.h>
       6             : #include <stdlib.h>
       7             : #include <ctype.h>
       8             : #include <sys/types.h>
       9             : #include <sys/stat.h>
      10             : #include <fcntl.h>
      11             : #include <unistd.h>
      12             : 
      13             : FD_IMPORT_CSTR( fd_pod_ctl_help, "src/util/pod/fd_pod_ctl_help" );
      14             : 
      15             : static int
      16         132 : supported_val_type( int val_type ) {
      17         132 :   return (val_type==FD_POD_VAL_TYPE_CSTR  ) | (val_type==FD_POD_VAL_TYPE_CHAR  )
      18         132 :        | (val_type==FD_POD_VAL_TYPE_SCHAR ) | (val_type==FD_POD_VAL_TYPE_SHORT )
      19         132 :        | (val_type==FD_POD_VAL_TYPE_INT   ) | (val_type==FD_POD_VAL_TYPE_LONG  )
      20         132 :        | (val_type==FD_POD_VAL_TYPE_UCHAR ) | (val_type==FD_POD_VAL_TYPE_USHORT)
      21         132 :        | (val_type==FD_POD_VAL_TYPE_UINT  ) | (val_type==FD_POD_VAL_TYPE_ULONG )
      22         132 :        | (val_type==FD_POD_VAL_TYPE_FLOAT )
      23         132 : #      if FD_HAS_DOUBLE
      24         132 :        | (val_type==FD_POD_VAL_TYPE_DOUBLE)
      25         132 : #      endif
      26         132 :   ;
      27         132 : }
      28             : 
      29             : static ulong
      30             : insert_val( uchar *      pod,
      31             :             char const * path,
      32             :             int          val_type,
      33         108 :             char const * val ) {
      34         108 :   ulong off;
      35         108 :   switch( val_type ) {
      36          15 :   case FD_POD_VAL_TYPE_CSTR:   off = fd_pod_insert_cstr  ( pod, path, fd_cstr_to_cstr  ( val ) ); break;
      37           9 :   case FD_POD_VAL_TYPE_CHAR:   off = fd_pod_insert_char  ( pod, path, fd_cstr_to_char  ( val ) ); break;
      38           9 :   case FD_POD_VAL_TYPE_SCHAR:  off = fd_pod_insert_schar ( pod, path, fd_cstr_to_schar ( val ) ); break;
      39           9 :   case FD_POD_VAL_TYPE_SHORT:  off = fd_pod_insert_short ( pod, path, fd_cstr_to_short ( val ) ); break;
      40           9 :   case FD_POD_VAL_TYPE_INT:    off = fd_pod_insert_int   ( pod, path, fd_cstr_to_int   ( val ) ); break;
      41           9 :   case FD_POD_VAL_TYPE_LONG:   off = fd_pod_insert_long  ( pod, path, fd_cstr_to_long  ( val ) ); break;
      42           9 :   case FD_POD_VAL_TYPE_UCHAR:  off = fd_pod_insert_uchar ( pod, path, fd_cstr_to_uchar ( val ) ); break;
      43           9 :   case FD_POD_VAL_TYPE_USHORT: off = fd_pod_insert_ushort( pod, path, fd_cstr_to_ushort( val ) ); break;
      44           9 :   case FD_POD_VAL_TYPE_UINT:   off = fd_pod_insert_uint  ( pod, path, fd_cstr_to_uint  ( val ) ); break;
      45           9 :   case FD_POD_VAL_TYPE_ULONG:  off = fd_pod_insert_ulong ( pod, path, fd_cstr_to_ulong ( val ) ); break;
      46          12 :   case FD_POD_VAL_TYPE_FLOAT:  off = fd_pod_insert_float ( pod, path, fd_cstr_to_float ( val ) ); break;
      47           0 : # if FD_HAS_DOUBLE
      48           0 :   case FD_POD_VAL_TYPE_DOUBLE: off = fd_pod_insert_double( pod, path, fd_cstr_to_double( val ) ); break;
      49           0 : # endif
      50           0 :   default: FD_LOG_ERR(( "never get here" ));
      51         108 :   }
      52         108 :   return off;
      53         108 : }
      54             : 
      55             : static inline int
      56         135 : issingleprint( int c ) {
      57         135 :   return isalnum( c ) | ispunct( c ) | (c==' ');
      58         135 : }
      59             : 
      60             : static void
      61          81 : printf_path( fd_pod_info_t const * info ) {
      62          81 :   if( FD_UNLIKELY( !info ) ) return;
      63             : 
      64          81 :   fd_pod_info_t const * node = info;
      65          81 :   ulong                 sz   = 0UL;
      66         120 :   do {
      67         120 :     ulong key_sz = node->key_sz;
      68         120 :     if( FD_UNLIKELY( !key_sz ) ) return;
      69         120 :     sz   += key_sz;
      70         120 :     node  = node->parent;
      71         120 :   } while( node );
      72             : 
      73          81 :   char * buf = malloc( sz );
      74          81 :   if( !buf ) return;
      75             : 
      76          81 :   char * p      = buf + sz;
      77          81 :   int    subpod = 0;
      78          81 :   node = info;
      79         120 :   do {
      80         120 :     ulong key_sz = node->key_sz;
      81         120 :     p -= key_sz;
      82         120 :     strcpy( p, node->key );
      83         120 :     if( subpod ) p[ key_sz-1UL ] = '.';
      84         120 :     subpod = 1;
      85         120 :     node = node->parent;
      86         120 :   } while( node );
      87             : 
      88          81 :   printf( "%s", buf );
      89          81 :   free( buf );
      90          81 : }
      91             : 
      92             : static void
      93          96 : printf_val( fd_pod_info_t const * info ) {
      94          96 :   switch( info->val_type ) {
      95             : 
      96          39 :   case FD_POD_VAL_TYPE_SUBPOD: {
      97          39 :     uchar * subpod = (uchar *)info->val;
      98          39 :     printf( "max %lu bytes, used %lu bytes, kcnt %lu keys", fd_pod_max( subpod ), fd_pod_used( subpod ), fd_pod_cnt( subpod ) );
      99          39 :     break;
     100           0 :   }
     101             : 
     102           0 :   default:
     103           6 :   case FD_POD_VAL_TYPE_BUF: {
     104           6 :     uchar const * buf = (uchar const *)info->val;
     105           6 :     ulong         sz  = info->val_sz;
     106           6 :     printf( "sz %lu", sz );
     107         138 :     for( ulong off=0UL; off<sz; off++ ) {
     108         132 :       ulong col = off & 15UL;
     109             :       /* FIXME: USER SPECIFIED INDENT AND CONFIGURE OFF WIDTH BASED ON SZ */
     110         132 :       if( FD_UNLIKELY( col==0UL ) ) printf( "\n\t\t%04lx: ", off );
     111         132 :       if( FD_UNLIKELY( col==8UL ) ) putc( ' ', stdout );
     112         132 :       printf( "%02x ", (uint)buf[ off ] );
     113         132 :       if( FD_UNLIKELY( (col==15UL) | ((off+1UL)==sz) ) ) { /* End of row */
     114             :         /* Output whitespace to align 2nd column */
     115          72 :         for( ulong rem=48UL-3UL*col; rem; rem-- ) putc( ' ', stdout );
     116             :         /* Output single character friendly bytes from row in 2nd column */
     117           9 :         char const * p = (char const *)(buf + (off & ~15UL));
     118         141 :         for( ulong rem=col+1UL; rem; rem-- ) { int c = (int)*(p++); putc( issingleprint( c ) ? c : '.', stdout ); }
     119           9 :       }
     120         132 :     }
     121           6 :     break;
     122           0 :   }
     123             : 
     124          15 :   case FD_POD_VAL_TYPE_CSTR: {
     125          15 :     if( !info->val_sz ) printf( "(null)" );
     126          15 :     else                printf( "\"%s\"", (char const *)info->val );
     127          15 :     break;
     128           0 :   }
     129             : 
     130           3 :   case FD_POD_VAL_TYPE_CHAR: {
     131           3 :     int c = (int)*(char *)info->val;
     132           3 :     if( issingleprint( c ) ) printf( "'%c'", c );
     133           0 :     else                     printf( "0x%02x", (uint)(uchar)c );
     134           3 :     break;
     135           0 :   }
     136             : 
     137           3 :   case FD_POD_VAL_TYPE_UCHAR:  { uint  u = (uint)*(uchar *)info->val; printf( "%u", u ); break; }
     138           3 :   case FD_POD_VAL_TYPE_USHORT:
     139           6 :   case FD_POD_VAL_TYPE_UINT:
     140           9 :   case FD_POD_VAL_TYPE_ULONG:  { ulong u; fd_ulong_svw_dec( info->val, &u ); printf( "%lu", u ); break; }
     141             : 
     142           6 :   case FD_POD_VAL_TYPE_SCHAR:  { int   i = (int) *(schar *)info->val; printf( "%i", i ); break; }
     143           3 :   case FD_POD_VAL_TYPE_SHORT:
     144           6 :   case FD_POD_VAL_TYPE_INT: 
     145           9 :   case FD_POD_VAL_TYPE_LONG:   { ulong u; fd_ulong_svw_dec( info->val, &u ); printf( "%li", fd_long_zz_dec( u ) ); break; }
     146             : 
     147           0 : # if FD_HAS_INT128
     148           0 :   case FD_POD_VAL_TYPE_INT128: {
     149           0 :     union { ulong w[2]; uint128 u; } tmp;
     150           0 :     fd_ulong_svw_dec( fd_ulong_svw_dec( (uchar const *)info->val, tmp.w ), tmp.w+1 );
     151           0 :     tmp.u = (uint128)fd_int128_zz_dec( tmp.u ); /* FIXME: INT128 decimal pretty printer */
     152           0 :     printf( "0x%016lx%016lx", (ulong)(tmp.u>>64), (ulong)tmp.u );
     153           0 :     break;
     154           6 :   }
     155             : 
     156           0 :   case FD_POD_VAL_TYPE_UINT128: {
     157           0 :     union { ulong w[2]; uint128 u; } tmp;
     158           0 :     fd_ulong_svw_dec( fd_ulong_svw_dec( (uchar const *)info->val, tmp.w ), tmp.w+1 );
     159             :     /* FIXME: UINT128 decimal pretty printer */
     160           0 :     printf( "0x%016lx%016lx", (ulong)(tmp.u>>64), (ulong)tmp.u );
     161           0 :     break;
     162           6 :   }
     163           0 : # endif
     164             : 
     165           6 :   case FD_POD_VAL_TYPE_FLOAT:  { float  f = *(float  *)info->val; printf( "%.21e", (double)f ); break; }
     166           0 : # if FD_HAS_DOUBLE
     167           0 :   case FD_POD_VAL_TYPE_DOUBLE: { double f = *(double *)info->val; printf( "%.21e", f );         break; }
     168          96 : # endif
     169             : 
     170          96 :   }
     171          96 : }
     172             : 
     173             : int
     174             : main( int     argc,
     175         204 :       char ** argv ) {
     176         204 :   fd_boot( &argc, &argv );
     177             : 
     178        1017 : # define SHIFT(n) argv+=(n),argc-=(n)
     179             : 
     180         204 :   if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "no arguments" ));
     181         204 :   char const * bin = argv[0];
     182         204 :   SHIFT(1); 
     183             : 
     184         204 :   ulong tag = 1UL;
     185             : 
     186         204 :   int cnt = 0;
     187         534 :   while( argc ) {
     188         486 :     char const * cmd = argv[0];
     189         486 :     SHIFT(1);
     190             : 
     191         486 :     if( !strcmp( cmd, "help" ) ) {
     192             : 
     193           3 :       fputs( fd_pod_ctl_help, stdout );
     194             : 
     195           3 :       FD_LOG_NOTICE(( "%i: %s: success", cnt, cmd ));
     196             : 
     197         483 :     } else if( !strcmp( cmd, "tag" ) ) {
     198             : 
     199           6 :       if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     200             : 
     201           3 :       tag = fd_cstr_to_ulong( argv[0] );
     202             : 
     203           3 :       FD_LOG_NOTICE(( "%i: %s %lu: success", cnt, cmd, tag ));
     204           3 :       SHIFT(1);
     205             : 
     206         477 :     } else if( !strcmp( cmd, "new" ) ) {
     207             : 
     208          15 :       if( FD_UNLIKELY( argc<2 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     209             : 
     210           9 :       char const * name =                   argv[0];
     211           9 :       ulong        max  = fd_cstr_to_ulong( argv[1] ); if( !max ) max = 4096UL;
     212             : 
     213           9 :       ulong align     = fd_pod_align();
     214           9 :       ulong footprint = fd_pod_footprint( max );
     215             : 
     216           9 :       if( FD_UNLIKELY( !footprint ) )
     217           3 :         FD_LOG_ERR(( "%i: %s: bad max (%lu)\n\tDo %s help for help", cnt, cmd, max, bin ));
     218             : 
     219           6 :       fd_wksp_t * wksp = fd_wksp_attach( name );
     220           6 :       if( FD_UNLIKELY( !wksp ) )
     221           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_attach( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, name, bin ));
     222             : 
     223           3 :       ulong gaddr = fd_wksp_alloc( wksp, align, footprint, tag );
     224           3 :       if( FD_UNLIKELY( !gaddr ) ) {
     225           0 :         fd_wksp_detach( wksp );
     226           0 :         FD_LOG_ERR(( "%i: %s: fd_wksp_alloc( \"%s\", %lu, %lu, %lu ) failed\n\tDo %s help for help",
     227           0 :                      cnt, cmd, name, align, footprint, tag, bin ));
     228           0 :       }
     229             : 
     230           3 :       void * shmem = fd_wksp_laddr( wksp, gaddr );
     231           3 :       if( FD_UNLIKELY( !shmem ) ) { /* should be impossible given fd_wksp_alloc success */
     232           0 :         fd_wksp_free( wksp, gaddr );
     233           0 :         fd_wksp_detach( wksp );
     234           0 :         FD_LOG_ERR(( "%i: %s: fd_wksp_laddr( \"%s\", %lu ) failed\n\tDo %s help for help", cnt, cmd, name, gaddr, bin ));
     235           0 :       }
     236             : 
     237           3 :       if( FD_UNLIKELY( !fd_pod_new( shmem, max ) ) ) {;
     238           0 :         fd_wksp_free( wksp, gaddr );
     239           0 :         fd_wksp_detach( wksp );
     240           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_new( \"%s:%lu\", %lu ) failed\n\tDo %s help for help", cnt, cmd, name, gaddr, max, bin ));
     241           0 :       }
     242             : 
     243           3 :       char cstr[ FD_WKSP_CSTR_MAX ];
     244           3 :       if( FD_UNLIKELY( !fd_wksp_cstr( wksp, gaddr, cstr ) ) ) {
     245           0 :         fd_wksp_free( wksp, gaddr );
     246           0 :         fd_wksp_detach( wksp );
     247           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_cstr( \"%s:%lu\" ) failed\n\tDo %s help for help", cnt, cmd, name, gaddr, bin ));
     248           0 :       }
     249             : 
     250           3 :       printf( "%s\n", cstr );
     251             : 
     252           3 :       fd_wksp_detach( wksp );
     253             : 
     254           3 :       FD_LOG_NOTICE(( "%i: %s %s %lu: success", cnt, cmd, name, max ));
     255           3 :       SHIFT(2);
     256             : 
     257         462 :     } else if( !strcmp( cmd, "delete" ) ) {
     258             : 
     259          12 :       if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     260             : 
     261           9 :       char const * cstr = argv[0];
     262             : 
     263           9 :       void * shmem = fd_wksp_map( cstr );
     264           9 :       if( FD_UNLIKELY( !shmem ) )
     265           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     266             : 
     267           6 :       if( FD_UNLIKELY( !fd_pod_delete( shmem ) ) ) {
     268           0 :         fd_wksp_unmap( shmem );
     269           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_delete( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     270           0 :       }
     271             : 
     272           6 :       fd_wksp_free_laddr( shmem );
     273           6 :       fd_wksp_unmap( shmem );
     274             : 
     275           6 :       FD_LOG_NOTICE(( "%i: %s %s: success", cnt, cmd, cstr ));
     276           6 :       SHIFT(1);
     277             : 
     278         450 :     } else if( !strcmp( cmd, "reset" ) ) {
     279             : 
     280           9 :       if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     281             : 
     282           6 :       char const * cstr = argv[0];
     283             : 
     284           6 :       void * shmem = fd_wksp_map( cstr );
     285           6 :       if( FD_UNLIKELY( !shmem ) )
     286           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     287             : 
     288           3 :       uchar * pod = fd_pod_join( shmem );
     289           3 :       if( FD_UNLIKELY( !pod ) ) {
     290           0 :         fd_wksp_unmap( shmem );
     291           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     292           0 :       }
     293             : 
     294           3 :       fd_pod_reset( pod );
     295             : 
     296           3 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     297             : 
     298           3 :       FD_LOG_NOTICE(( "%i: %s %s: success", cnt, cmd, cstr ));
     299           3 :       SHIFT(1);
     300             : 
     301         441 :     } else if( !strcmp( cmd, "list" ) ) {
     302             : 
     303           9 :       if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     304             : 
     305           6 :       char const * cstr = argv[0];
     306             : 
     307           6 :       void * shmem = fd_wksp_map( cstr );
     308           6 :       if( FD_UNLIKELY( !shmem ) )
     309           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     310             : 
     311           3 :       uchar * pod = fd_pod_join( shmem );
     312           3 :       if( FD_UNLIKELY( !pod ) ) {
     313           0 :         fd_wksp_unmap( shmem );
     314           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     315           0 :       }
     316             : 
     317           3 :       fd_pod_info_t * info;
     318           3 :       ulong info_cnt = fd_pod_cnt_recursive( pod );
     319           3 :       if( FD_UNLIKELY( !info_cnt ) ) info = NULL;
     320           3 :       else { 
     321           3 :         info = (fd_pod_info_t *)aligned_alloc( alignof(fd_pod_info_t), info_cnt*sizeof(fd_pod_info_t) );
     322           3 :         if( FD_UNLIKELY( !info ) ) {
     323           0 :           fd_wksp_unmap( fd_pod_leave( pod ) );
     324           0 :           FD_LOG_ERR(( "%i: %s: aligned_alloc failed\n\tDo %s help for help", cnt, cmd, bin ));
     325           0 :         }
     326           3 :         if( FD_UNLIKELY( !fd_pod_list_recursive( pod, info ) ) ) {
     327           0 :           free( info );
     328           0 :           fd_wksp_unmap( fd_pod_leave( pod ) );
     329           0 :           FD_LOG_ERR(( "%i: %s: fd_pod_list_recursive( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     330           0 :         }
     331           3 :       }
     332             : 
     333           3 :       printf( "pod %s\n", cstr );
     334           3 :       printf( "\tmax   %20lu bytes  used  %20lu bytes  avail %20lu bytes\n",
     335           3 :               fd_pod_max( pod ), fd_pod_used ( pod ), fd_pod_avail( pod ) );
     336           3 :       printf( "\tkcnt  %20lu keys   icnt  %20lu paths\n", fd_pod_cnt( pod ), info_cnt );
     337          84 :       for( ulong info_idx=0UL; info_idx<info_cnt; info_idx++ ) {
     338          81 :         fd_pod_info_t * node = &info[ info_idx ];
     339          81 :         char type[ FD_POD_VAL_TYPE_CSTR_MAX ]; fd_pod_val_type_to_cstr( node->val_type, type );
     340          81 :         printf( "\t%s %s ", cstr, type );
     341          81 :         printf_path( node );
     342          81 :         printf( " " );
     343          81 :         printf_val(  node );
     344          81 :         printf( "\n" );
     345          81 :       }
     346             : 
     347           3 :       if( FD_LIKELY( info ) ) free( info );
     348           3 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     349             : 
     350           3 :       FD_LOG_NOTICE(( "%i: %s %s: success", cnt, cmd, cstr ));
     351           3 :       SHIFT(1);
     352             : 
     353         432 :     } else if( !strcmp( cmd, "insert" ) ) {
     354             : 
     355          54 :       if( FD_UNLIKELY( argc<4 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     356             : 
     357          42 :       char const * cstr = argv[0];
     358          42 :       char const * type = argv[1];
     359          42 :       char const * path = argv[2];
     360          42 :       char const * val  = argv[3];
     361             : 
     362          42 :       int val_type = fd_cstr_to_pod_val_type( type );
     363          42 :       if( FD_UNLIKELY( !supported_val_type( val_type ) ) )
     364           3 :         FD_LOG_ERR(( "%i: %s: unsupported type %s\n\tDo %s help for help", cnt, cmd, type, bin ));
     365             : 
     366          39 :       void * shmem = fd_wksp_map( cstr );
     367          39 :       if( FD_UNLIKELY( !shmem ) )
     368           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     369             : 
     370          36 :       uchar * pod = fd_pod_join( shmem );
     371          36 :       if( FD_UNLIKELY( !pod ) ) {
     372           0 :         fd_wksp_unmap( shmem );
     373           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     374           0 :       }
     375             : 
     376          36 :       ulong off = insert_val( pod, path, val_type, val );
     377             : 
     378          36 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     379             : 
     380          36 :       if( FD_UNLIKELY( !off ) )
     381           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_insert_%s( \"%s\", \"%s\", \"%s\" ) failed\n\tDo %s help for help",
     382          36 :                      cnt, cmd, type, cstr, path, val, bin ));
     383             : 
     384          36 :       FD_LOG_NOTICE(( "%i: %s %s %s %s %s: success", cnt, cmd, cstr, type, path, val ));
     385          36 :       SHIFT(4);
     386             : 
     387         378 :     } else if( !strcmp( cmd, "insert-file" ) ) {
     388             : 
     389          27 :       if( FD_UNLIKELY( argc<3 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     390             : 
     391          18 :       char const * cstr = argv[0];
     392          18 :       char const * path = argv[1];
     393          18 :       char const * file = argv[2];
     394             : 
     395          18 :       void * shmem = fd_wksp_map( cstr );
     396          18 :       if( FD_UNLIKELY( !shmem ) )
     397           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     398             : 
     399          15 :       uchar * pod = fd_pod_join( shmem );
     400          15 :       if( FD_UNLIKELY( !pod ) ) {
     401           0 :         fd_wksp_unmap( shmem );
     402           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     403           0 :       }
     404             : 
     405          15 :       int fd = open( file, O_RDONLY );
     406          15 :       if( FD_UNLIKELY( fd == -1 ) )
     407           3 :         FD_LOG_ERR(( "%i: %s: open( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, file, bin ));
     408             : 
     409          12 :       struct stat st;
     410          12 :       if( FD_UNLIKELY( fstat( fd, &st ) == -1 ) )
     411           0 :         FD_LOG_ERR(( "%i: %s: fstat( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, file, bin ));
     412          12 :       ulong buf_sz = (ulong)st.st_size;
     413             : 
     414          12 :       ulong off = fd_pod_alloc( pod, path, FD_POD_VAL_TYPE_BUF, buf_sz );
     415          12 :       if( FD_UNLIKELY( !off ) )
     416           6 :         FD_LOG_ERR(( "%i: %s: fd_pod_alloc( \"%s\", \"%s\", FD_POD_VAL_TYPE_BUF, %lu ) failed\n\tDo %s help for help",
     417          12 :                      cnt, cmd, cstr, path, buf_sz, bin ));
     418             : 
     419           6 :       if( FD_UNLIKELY( read( fd, pod + off, buf_sz )!=(long)buf_sz ) ) {
     420           0 :         if( FD_UNLIKELY( fd_pod_remove( pod, path ) ) )
     421           0 :           FD_LOG_WARNING(( "%i: %s: fd_pod_remove( \"%s\", \"%s\" ) failed; pod likely corrupt", cnt, cmd, cstr, path ));
     422           0 :         FD_LOG_ERR(( "%i: %s: read( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, file, bin ));
     423           0 :       }
     424             : 
     425           6 :       if( FD_UNLIKELY( close( fd ) ) )
     426           0 :         FD_LOG_WARNING(( "%i: %s: close( \"%s\" ) failed; attempting to continue", cnt, cmd, file ));
     427             : 
     428           6 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     429             : 
     430           6 :       FD_LOG_NOTICE(( "%i: %s %s %s %s: success", cnt, cmd, cstr, path, file ));
     431           6 :       SHIFT(3);
     432             : 
     433         351 :     } else if( !strcmp( cmd, "remove" ) ) {
     434             : 
     435          57 :       if( FD_UNLIKELY( argc<2 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     436             : 
     437          51 :       char const * cstr = argv[0];
     438          51 :       char const * path = argv[1];
     439             : 
     440          51 :       void * shmem = fd_wksp_map( cstr );
     441          51 :       if( FD_UNLIKELY( !shmem ) )
     442           0 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     443             : 
     444          51 :       uchar * pod = fd_pod_join( shmem );
     445          51 :       if( FD_UNLIKELY( !pod ) ) {
     446           0 :         fd_wksp_unmap( shmem );
     447           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     448           0 :       }
     449             : 
     450          51 :       int err = fd_pod_remove( pod, path );
     451             : 
     452          51 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     453             : 
     454          51 :       if( FD_UNLIKELY( err ) )
     455          12 :         FD_LOG_ERR(( "%i: %s: fd_pod_remove( \"%s\", \"%s\" ) failed (%i-%s)\n\tDo %s help for help",
     456          51 :                      cnt, cmd, cstr, path, err, fd_pod_strerror( err ), bin ));
     457             : 
     458          39 :       FD_LOG_NOTICE(( "%i: %s %s %s: success", cnt, cmd, cstr, path ));
     459          39 :       SHIFT(2);
     460             : 
     461         294 :     } else if( !strcmp( cmd, "update" ) ) {
     462             : 
     463          54 :       if( FD_UNLIKELY( argc<4 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     464             : 
     465          42 :       char const * cstr = argv[0];
     466          42 :       char const * type = argv[1];
     467          42 :       char const * path = argv[2];
     468          42 :       char const * val  = argv[3];
     469             : 
     470          42 :       int val_type = fd_cstr_to_pod_val_type( type );
     471          42 :       if( FD_UNLIKELY( !supported_val_type( val_type ) ) )
     472           3 :         FD_LOG_ERR(( "%i: %s: unsupported type %s\n\tDo %s help for help", cnt, cmd, type, bin ));
     473             : 
     474          39 :       void * shmem = fd_wksp_map( cstr );
     475          39 :       if( FD_UNLIKELY( !shmem ) )
     476           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     477             : 
     478          36 :       uchar * pod = fd_pod_join( shmem );
     479          36 :       if( FD_UNLIKELY( !pod ) ) {
     480           0 :         fd_wksp_unmap( shmem );
     481           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     482           0 :       }
     483             : 
     484          36 :       fd_pod_info_t info[1];
     485          36 :       int err = fd_pod_query( pod, path, info );
     486          36 :       if( FD_UNLIKELY( !!err ) ) {
     487           0 :         fd_wksp_unmap( fd_pod_leave( pod ) );
     488           0 :         FD_LOG_ERR(( "%i: %s: no path %s to type (%i-%s) in pod %s (%i-%s)\n\tDo %s help for help",
     489           0 :                      cnt, cmd, path, val_type, type, cstr, err, fd_pod_strerror( err ), bin ));
     490           0 :       }
     491             : 
     492          36 :       if( FD_UNLIKELY( info->val_type!=val_type ) ) {
     493           3 :         fd_wksp_unmap( fd_pod_leave( pod ) );
     494           3 :         char buf[ FD_POD_VAL_TYPE_CSTR_MAX ];
     495           3 :         FD_LOG_ERR(( "%i: %s: type (%i-%s) at %s %s does not match requested type (%i-%s)\n\tDo %s help for help",
     496           3 :                      cnt, cmd, info->val_type, fd_pod_val_type_to_cstr( info->val_type, buf ),
     497           3 :                      cstr, path, val_type, type, bin ));
     498           3 :       }
     499             : 
     500          33 :       err = fd_pod_remove( pod, path );
     501          33 :       if( FD_UNLIKELY( err ) ) {
     502           0 :         fd_wksp_unmap( fd_pod_leave( pod ) );
     503           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_remove( \"%s\", \"%s\" ) failed (%i-%s)\n\tDo %s help for help",
     504           0 :                      cnt, cmd, cstr, path, err, fd_pod_strerror( err ), bin ));
     505           0 :       }
     506             : 
     507          33 :       ulong off = insert_val( pod, path, val_type, val );
     508             : 
     509          33 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     510             : 
     511          33 :       if( FD_UNLIKELY( !off ) )
     512           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_insert_%s( \"%s\", \"%s\", \"%s\" ) failed\n\tDo %s help for help",
     513          33 :                      cnt, cmd, type, cstr, path, val, bin ));
     514             : 
     515          33 :       FD_LOG_NOTICE(( "%i: %s %s %s %s %s: success", cnt, cmd, cstr, type, path, val ));
     516          33 :       SHIFT(4);
     517             : 
     518         240 :     } else if( !strcmp( cmd, "set" ) ) {
     519             : 
     520          60 :       if( FD_UNLIKELY( argc<4 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     521             : 
     522          48 :       char const * cstr = argv[0];
     523          48 :       char const * type = argv[1];
     524          48 :       char const * path = argv[2];
     525          48 :       char const * val  = argv[3];
     526             : 
     527          48 :       int val_type = fd_cstr_to_pod_val_type( type );
     528          48 :       if( FD_UNLIKELY( !supported_val_type( val_type ) ) )
     529           3 :         FD_LOG_ERR(( "%i: %s: unsupported type %s\n\tDo %s help for help", cnt, cmd, type, bin ));
     530             : 
     531          45 :       void * shmem = fd_wksp_map( cstr );
     532          45 :       if( FD_UNLIKELY( !shmem ) )
     533           3 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     534             : 
     535          42 :       uchar * pod = fd_pod_join( shmem );
     536          42 :       if( FD_UNLIKELY( !pod ) ) {
     537           0 :         fd_wksp_unmap( shmem );
     538           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     539           0 :       }
     540             : 
     541          42 :       fd_pod_info_t info[1];
     542          42 :       int err = fd_pod_query( pod, path, info );
     543          42 :       if( FD_LIKELY( !err ) ) {
     544             : 
     545          36 :         if( FD_UNLIKELY( info->val_type!=val_type ) ) {
     546           3 :           fd_wksp_unmap( fd_pod_leave( pod ) );
     547           3 :           char buf[ FD_POD_VAL_TYPE_CSTR_MAX ];
     548           3 :           FD_LOG_ERR(( "%i: %s: type (%i-%s) at %s %s does not match requested type (%i-%s)\n\tDo %s help for help",
     549           3 :                        cnt, cmd, info->val_type, fd_pod_val_type_to_cstr( info->val_type, buf ),
     550           3 :                        cstr, path, val_type, type, bin ));
     551           3 :         }
     552             : 
     553          33 :         err = fd_pod_remove( pod, path );
     554          33 :         if( FD_UNLIKELY( err ) ) {
     555           0 :           fd_wksp_unmap( fd_pod_leave( pod ) );
     556           0 :           FD_LOG_ERR(( "%i: %s: fd_pod_remove( \"%s\", \"%s\" ) failed (%i-%s)\n\tDo %s help for help",
     557           0 :                        cnt, cmd, cstr, path, err, fd_pod_strerror( err ), bin ));
     558           0 :         }
     559             : 
     560          33 :       } else if( FD_UNLIKELY( err!=FD_POD_ERR_RESOLVE ) ) {
     561             : 
     562           0 :         fd_wksp_unmap( fd_pod_leave( pod ) );
     563           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_query( \"%s\", \"%s\" ) failed (%i-%s)\n\tDo %s help for help",
     564           0 :                      cnt, cmd, cstr, path, err, fd_pod_strerror( err ), bin ));
     565             : 
     566           0 :       }
     567             : 
     568          39 :       ulong off = insert_val( pod, path, val_type, val );
     569             : 
     570          39 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     571             : 
     572          39 :       if( FD_UNLIKELY( !off ) )
     573           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_insert_%s( \"%s\", \"%s\", \"%s\" ) failed\n\tDo %s help for help",
     574          39 :                      cnt, cmd, type, cstr, path, val, bin ));
     575             : 
     576          39 :       FD_LOG_NOTICE(( "%i: %s %s %s %s %s: success", cnt, cmd, cstr, type, path, val ));
     577          39 :       SHIFT(4);
     578             : 
     579             : 
     580         180 :     } else if( !strcmp( cmd, "compact" ) ) {
     581             : 
     582           0 :       if( FD_UNLIKELY( argc<2 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     583             : 
     584           0 :       char const * cstr =                 argv[0];
     585           0 :       int          full = fd_cstr_to_int( argv[1] );
     586             : 
     587           0 :       void * shmem = fd_wksp_map( cstr );
     588           0 :       if( FD_UNLIKELY( !shmem ) )
     589           0 :         FD_LOG_ERR(( "%i: %s: fd_wksp_map( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     590             : 
     591           0 :       uchar * pod = fd_pod_join( shmem );
     592           0 :       if( FD_UNLIKELY( !pod ) ) {
     593           0 :         fd_wksp_unmap( shmem );
     594           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_join( \"%s\" ) failed\n\tDo %s help for help", cnt, cmd, cstr, bin ));
     595           0 :       }
     596             : 
     597           0 :       ulong new_max = fd_pod_compact( pod, full );
     598             : 
     599           0 :       fd_wksp_unmap( fd_pod_leave( pod ) );
     600             : 
     601           0 :       if( FD_UNLIKELY( !new_max ) )
     602           0 :         FD_LOG_ERR(( "%i: %s: fd_pod_compact( \"%s\", %i ) failed\n\tDo %s help for help", cnt, cmd, cstr, full, bin ));
     603             : 
     604           0 :       FD_LOG_NOTICE(( "%i: %s %s %i: success", cnt, cmd, cstr, full ));
     605           0 :       SHIFT(2);
     606             : 
     607         180 :     } else if( !strcmp( cmd, "query-root" ) ) {
     608             : 
     609          51 :       if( FD_UNLIKELY( argc<2 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     610             : 
     611          45 :       char const * what = argv[0];
     612          45 :       char const * cstr = argv[1];
     613             : 
     614          45 :       void *  shmem = NULL;
     615          45 :       uchar * pod   = NULL;
     616          45 :       int     err   = FD_POD_ERR_INVAL;
     617             : 
     618          45 :       shmem = fd_wksp_map( cstr );
     619          45 :       if( FD_LIKELY( shmem ) ) {
     620          24 :         pod = fd_pod_join( shmem );
     621          24 :         if( FD_LIKELY( pod ) ) err = 0;
     622          24 :       }
     623             : 
     624          45 :       if(      !strcmp( what, "test"       ) ) printf( "%i\n",  err );
     625          39 :       else if( !strcmp( what, "max"        ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_max          ( pod ) : 0UL );
     626          33 :       else if( !strcmp( what, "used"       ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_used         ( pod ) : 0UL );
     627          27 :       else if( !strcmp( what, "avail"      ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_avail        ( pod ) : 0UL );
     628          21 :       else if( !strcmp( what, "cnt"        ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_cnt          ( pod ) : 0UL );
     629          15 :       else if( !strcmp( what, "recursive"  ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_cnt_recursive( pod ) : 0UL );
     630           9 :       else if( !strcmp( what, "subpod-cnt" ) ) printf( "%lu\n", FD_LIKELY(!err) ? fd_pod_cnt_subpod   ( pod ) : 0UL );
     631           3 :       else                                     FD_LOG_ERR(( "unknown query %s", what ));
     632             : 
     633          42 :       if( FD_LIKELY( pod   ) ) fd_pod_leave( pod );
     634          42 :       if( FD_LIKELY( shmem ) ) fd_wksp_unmap( shmem );
     635          42 :       FD_LOG_NOTICE(( "%i: %s %s %s: success", cnt, cmd, what, cstr ));
     636          42 :       SHIFT(2);
     637             : 
     638         129 :     } else if( !strcmp( cmd, "query" ) ) {
     639             : 
     640         126 :       if( FD_UNLIKELY( argc<3 ) ) FD_LOG_ERR(( "%i: %s: too few arguments\n\tDo %s help for help", cnt, cmd, bin ));
     641             : 
     642         117 :       char const * what = argv[0];
     643         117 :       char const * cstr = argv[1];
     644         117 :       char const * path = argv[2];
     645             : 
     646         117 :       void *        shmem = NULL;
     647         117 :       uchar *       pod   = NULL;
     648         117 :       int           err   = FD_POD_ERR_INVAL;
     649         117 :       fd_pod_info_t info[1];
     650         117 :       char          type[ FD_POD_VAL_TYPE_CSTR_MAX ];
     651         117 :       int           is_subpod = 0;
     652             : 
     653         117 :       shmem = fd_wksp_map( cstr );
     654         117 :       if( FD_LIKELY( shmem ) ) {
     655         114 :         pod = fd_pod_join( shmem );
     656         114 :         if( FD_LIKELY( pod ) ) {
     657         114 :           err = fd_pod_query( pod, path, info );
     658         114 :           if( FD_LIKELY( !err ) ) {
     659          75 :             is_subpod = (info->val_type==FD_POD_VAL_TYPE_SUBPOD);
     660          75 :             if( FD_UNLIKELY( !fd_pod_val_type_to_cstr( info->val_type, type ) ) ) { /* only possible if corruption */
     661           0 :               err = FD_POD_ERR_INVAL;
     662           0 :             }
     663          75 :           }
     664         114 :         }
     665         114 :       }
     666             : 
     667         117 :       if(      !strcmp( what, "test"       ) ) printf( "%i\n",  err );
     668         102 :       else if( !strcmp( what, "type"       ) ) printf( "%s\n",  FD_LIKELY( !err )      ? type : "void" );
     669          90 :       else if( !strcmp( what, "val"        ) ) {
     670           9 :         if( FD_UNLIKELY( err ) ) printf( "void\n" );
     671           6 :         else {
     672           6 :           printf_val( info );
     673           6 :           printf( "\n" );
     674           6 :         }
     675           9 :       }
     676          81 :       else if( !strcmp( what, "max"        ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_max          ( info->val ) : 0UL );
     677          72 :       else if( !strcmp( what, "used"       ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_used         ( info->val ) : 0UL );
     678          63 :       else if( !strcmp( what, "avail"      ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_avail        ( info->val ) : 0UL );
     679          54 :       else if( !strcmp( what, "cnt"        ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_cnt          ( info->val ) : 0UL );
     680          45 :       else if( !strcmp( what, "recursive"  ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_cnt_recursive( info->val ) : 0UL );
     681          36 :       else if( !strcmp( what, "subpod-cnt" ) ) printf( "%lu\n", FD_LIKELY( is_subpod ) ? fd_pod_cnt_subpod   ( info->val ) : 0UL );
     682          27 :       else if( !strcmp( what, "gaddr" ) ) {
     683          12 :         char buf[ FD_WKSP_CSTR_MAX ];
     684          12 :         printf( "%s\n", (FD_LIKELY( !err ) && FD_LIKELY( fd_wksp_cstr_laddr( info->val, buf ) )) ? buf : "null" );
     685          12 :       }
     686          15 :       else if( !strcmp( what, "full"         ) ) {
     687          12 :         if( FD_UNLIKELY( err ) ) printf( "%s void %s void\n", cstr, path );
     688           9 :         else {
     689           9 :           printf( "%s %s %s ", cstr, type, path );
     690           9 :           printf_val( info );
     691           9 :           printf( "\n" );
     692           9 :         }
     693          12 :       }
     694           3 :       else                                       FD_LOG_ERR(( "unknown query %s", what ));
     695             : 
     696         114 :       if( FD_LIKELY( pod   ) ) fd_pod_leave( pod );
     697         114 :       if( FD_LIKELY( shmem ) ) fd_wksp_unmap( shmem );
     698         114 :       FD_LOG_NOTICE(( "%i: %s %s %s %s: success", cnt, cmd, what, cstr, path ));
     699         114 :       SHIFT(3);
     700             : 
     701         114 :     } else {
     702             : 
     703           3 :       FD_LOG_ERR(( "%i: %s: unknown command\n\t"
     704           3 :                    "Do %s help for help", cnt, cmd, bin ));
     705             : 
     706           3 :     }
     707         330 :     cnt++;
     708         330 :   }
     709             : 
     710          48 :   if( FD_UNLIKELY( cnt<1 ) ) FD_LOG_NOTICE(( "processed %i commands\n\tDo %s help for help", cnt, bin ));
     711          45 :   else                       FD_LOG_NOTICE(( "processed %i commands", cnt ));
     712             : 
     713          48 : # undef SHIFT
     714          48 :   fd_halt();
     715          48 :   return 0;
     716         204 : }
     717             : 
     718             : #else
     719             : 
     720             : int
     721             : main( int     argc,
     722             :       char ** argv ) {
     723             :   fd_boot( &argc, &argv );
     724             :   if( FD_UNLIKELY( argc<1 ) ) FD_LOG_ERR(( "No arguments" ));
     725             :   if( FD_UNLIKELY( argc>1 ) ) FD_LOG_ERR(( "fd_pod_ctl not supported on this platform" ));
     726             :   FD_LOG_NOTICE(( "processed 0 commands" ));
     727             :   fd_halt();
     728             :   return 0;
     729             : }
     730             : 
     731             : #endif
     732             : 

Generated by: LCOV version 1.14