LCOV - code coverage report
Current view: top level - util/net - fd_pcapng.c (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 255 510 50.0 %
Date: 2025-01-08 12:08:44 Functions: 11 15 73.3 %

          Line data    Source code
       1             : #include "fd_pcapng_private.h"
       2             : #include "../fd_util.h"
       3             : 
       4             : /* Capture related ****************************************************/
       5             : 
       6             : #include <errno.h>
       7             : #if defined(__linux__) || defined(__FreeBSD__)
       8             : #include <net/if.h>
       9             : #endif
      10             : 
      11             : void
      12           0 : fd_pcapng_shb_defaults( fd_pcapng_shb_opts_t * opt ) {
      13           0 : # if FD_HAS_X86
      14           0 :   opt->hardware = "x86_64";
      15           0 : # endif
      16             : 
      17           0 : # if defined(__linux__)
      18           0 :   opt->os       = "Linux";
      19           0 : # endif
      20             : 
      21           0 :   opt->userappl = "Firedancer";
      22           0 : }
      23             : 
      24             : int
      25             : fd_pcapng_idb_defaults( fd_pcapng_idb_opts_t * opt,
      26           0 :                         uint                   if_idx ) {
      27           0 : # if defined(__linux__) || defined(__FreeBSD__)
      28           0 :   static FD_TL char _name[ IF_NAMESIZE ];
      29           0 :   char * name = if_indextoname( if_idx, _name );
      30           0 :   if( FD_UNLIKELY( !name ) ) return 0;
      31           0 :   FD_STATIC_ASSERT( 16>=IF_NAMESIZE, ifname_sz );
      32           0 :   memcpy( opt->name, _name, 16UL );
      33             : # else
      34             :   (void)if_idx;
      35             : # endif
      36             : 
      37           0 :   opt->tsresol = FD_PCAPNG_TSRESOL_NS;
      38             : 
      39             :   /* TODO get ip4_addr, mac_addr, hardware from rtnetlink */
      40             : 
      41           0 :   return 1;
      42           0 : }
      43             : 
      44             : #if FD_HAS_HOSTED
      45             : 
      46             : #include <stdio.h>
      47             : 
      48             : /* Parsers ************************************************************/
      49             : 
      50             : /* FIXME Option parsing spec violation
      51             : 
      52             :      https://www.ietf.org/archive/id/draft-ietf-opsawg-pcapng-00.html#name-options
      53             : 
      54             :      > Code that reads pcapng files MUST NOT assume an option list will
      55             :      have an opt_endofopt option at the end; it MUST also check for the
      56             :      end of the block, and SHOULD treat blocks where the option list has
      57             :      no opt_endofopt option as if the option list had an opt_endofopt
      58             :      option at the end.
      59             : 
      60             :      This parser currently does not handle missing opt_endofopt */
      61             : 
      62             : FD_FN_CONST ulong
      63           3 : fd_pcapng_iter_align( void ) {
      64           3 :   return alignof(fd_pcapng_iter_t);
      65           3 : }
      66             : 
      67             : FD_FN_CONST ulong
      68           3 : fd_pcapng_iter_footprint( void ) {
      69           3 :   return sizeof(fd_pcapng_iter_t);
      70           3 : }
      71             : 
      72             : static char const *
      73             : fd_pcapng_iter_strerror( int    error,
      74           0 :                          FILE * file ) {
      75           0 :   static FD_TL char err_cstr_buf[ 1024UL ];
      76           0 :   char * err_cstr = fd_cstr_init( err_cstr_buf );
      77           0 :   if( error==EPROTO ) {
      78           0 :     return fd_cstr_printf( err_cstr, sizeof(err_cstr_buf), NULL, "parse error at %#lx", (ulong)ftell(file) );
      79           0 :   } else if( error==-1 && !feof( file ) ) {
      80           0 :     return "end of section";
      81           0 :   } else {
      82           0 :     return fd_cstr_printf( err_cstr, sizeof(err_cstr_buf), NULL, "%i-%s", error, fd_io_strerror( error ) );
      83           0 :   }
      84           0 : }
      85             : 
      86             : static int
      87             : fd_pcapng_peek_block( FILE *                  stream,
      88             :                       fd_pcapng_block_hdr_t * _hdr,
      89          21 :                       long *                  end_ptr ) {
      90             : 
      91             :   /* Remember offset of block */
      92          21 :   long pos = ftell( stream );
      93          21 :   if( FD_UNLIKELY( pos<0L ) )
      94           0 :     return ferror( stream );
      95          21 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)pos, 4U ) ) ) {
      96           0 :     FD_LOG_DEBUG(( "pcapng: misaligned stream at %#lx", (ulong)pos ));
      97           0 :     return EPROTO;
      98           0 :   }
      99             : 
     100             :   /* Read header */
     101          21 :   fd_pcapng_block_hdr_t hdr;
     102          21 :   if( FD_UNLIKELY( 1UL!=fread( &hdr, sizeof(fd_pcapng_block_hdr_t), 1, stream ) ) ) {
     103           3 :     if( FD_LIKELY( feof( stream ) ) ) return -1; /* eof */
     104           0 :     else                              return ferror( stream );
     105           3 :   }
     106             : 
     107             :   /* Coherence check length field */
     108          18 :   if( FD_UNLIKELY( (hdr.block_sz <   12U) /* header and footer are mandatory */
     109          18 :                  | (hdr.block_sz >32768U) /* way too large */
     110          18 :                  | (!fd_ulong_is_aligned( hdr.block_sz, 4U )) ) ) {
     111           0 :     FD_LOG_DEBUG(( "pcapng: block with invalid size %#x at %#lx", hdr.block_sz, (ulong)pos ));
     112           0 :     return EPROTO;
     113           0 :   }
     114             : 
     115             :   /* Seek to block footer */
     116          18 :   if( FD_UNLIKELY( 0!=fseek( stream, (long)(hdr.block_sz - 12U), SEEK_CUR ) ) )
     117           0 :     return errno;
     118             : 
     119             :   /* Read footer */
     120          18 :   uint block_sz;
     121          18 :   if( FD_UNLIKELY( 1UL!=fread( &block_sz, sizeof(uint), 1, stream ) ) )
     122           0 :     return ferror( stream );
     123             : 
     124             :   /* Restore cursor */
     125          18 :   if( FD_UNLIKELY( 0!=fseek( stream, pos, SEEK_SET ) ) )
     126           0 :     return errno;
     127             : 
     128             :   /* Check that header and footer match */
     129          18 :   if( FD_UNLIKELY( hdr.block_sz != block_sz ) ) {
     130           0 :     FD_LOG_DEBUG(( "pcapng: block size in header and footer don't match at %#lx", (ulong)pos ));
     131           0 :     return EPROTO;
     132           0 :   }
     133             : 
     134          18 :   *_hdr = hdr;
     135          18 :   if( end_ptr ) *end_ptr = pos + (long)fd_uint_align_up( hdr.block_sz, 4U );
     136             : 
     137          18 :   return 0; /* success */
     138          18 : }
     139             : 
     140             : static int
     141             : fd_pcapng_read_option( FILE *               stream,
     142          15 :                        fd_pcapng_option_t * opt ) {
     143             : 
     144          15 :   struct __attribute__((packed)) {
     145          15 :     ushort type;
     146          15 :     ushort sz;
     147          15 :   } opt_hdr;
     148             : 
     149          15 :   if( FD_UNLIKELY( 1UL!=fread( &opt_hdr, 4UL, 1UL, stream ) ) )
     150           0 :     return ferror( stream );
     151             : 
     152          15 :   uint end_off = fd_uint_align_up( opt_hdr.sz, 4U );
     153          15 :   uint read_sz = fd_uint_min( end_off, opt->sz );
     154             : 
     155          15 :   if( read_sz ) {
     156           3 :     if( FD_UNLIKELY( 1UL!=fread( opt->value, read_sz, 1UL, stream ) ) )
     157           0 :       return ferror( stream );
     158           3 :     end_off -= read_sz;
     159           3 :   }
     160             : 
     161          15 :   if( FD_UNLIKELY( 0!=fseek( stream, end_off, SEEK_CUR ) ) )
     162           0 :     return errno;
     163             : 
     164          15 :   return 0; /* success */
     165          15 : }
     166             : 
     167             : fd_pcapng_iter_t *
     168             : fd_pcapng_iter_new( void * mem,
     169           3 :                     void * _file ) {
     170             : 
     171           3 :   if( FD_UNLIKELY( !mem ) ) {
     172           0 :     FD_LOG_WARNING(( "NULL mem" ));
     173           0 :     return NULL;
     174           0 :   }
     175           3 :   if( FD_UNLIKELY( !fd_ulong_is_aligned( (ulong)mem, alignof(fd_pcapng_iter_t) ) ) ) {
     176           0 :     FD_LOG_WARNING(( "unaligned mem" ));
     177           0 :     return NULL;
     178           0 :   }
     179           3 :   if( FD_UNLIKELY( !_file ) ) {
     180           0 :     FD_LOG_WARNING(( "NULL file" ));
     181           0 :     return NULL;
     182           0 :   }
     183             : 
     184           3 :   FILE * file = (FILE *)_file;
     185             : 
     186           3 :   memset( mem, 0, sizeof(fd_pcapng_iter_t) );
     187           3 :   fd_pcapng_iter_t * iter = (fd_pcapng_iter_t *)mem;
     188           3 :   iter->stream = (FILE *)file;
     189             : 
     190             :   /* File starts with a Section Header Block */
     191             : 
     192           3 :   fd_pcapng_block_hdr_t shb_hdr;
     193           3 :   int err = fd_pcapng_peek_block( file, &shb_hdr, NULL );
     194           3 :   if( FD_UNLIKELY( err ) ) {
     195           0 :     FD_LOG_WARNING(( "pcapng: SHB read failed (%s)", fd_pcapng_iter_strerror( err, file ) ));
     196           0 :     return NULL;
     197           0 :   }
     198           3 :   if( FD_UNLIKELY( shb_hdr.block_type!=FD_PCAPNG_BLOCK_TYPE_SHB
     199           3 :                 || shb_hdr.block_sz  < sizeof(fd_pcapng_shb_t)  ) ) {
     200           0 :     FD_LOG_WARNING(( "pcapng: not a valid Section Header Block" ));
     201           0 :     return NULL;
     202           0 :   }
     203             : 
     204             :   /* Read Section Header Block */
     205             : 
     206           3 :   fd_pcapng_shb_t shb;
     207           3 :   if( FD_UNLIKELY( 1UL!=fread( &shb, sizeof(fd_pcapng_shb_t), 1UL, file )
     208           3 :                 || 0  !=fseek( file, (long)shb_hdr.block_sz - (long)sizeof(fd_pcapng_shb_t), SEEK_CUR ) ) ) {
     209           0 :     FD_LOG_WARNING(( "pcapng: SHB read failed (%s)", fd_pcapng_iter_strerror( err, file ) ));
     210           0 :     return NULL;
     211           0 :   }
     212             : 
     213           3 :   if( FD_UNLIKELY( (shb.version_major!=1) | (shb.version_minor!=0) ) ) {
     214           0 :     FD_LOG_WARNING(( "pcapng: unsupported file format version %u.%u",
     215           0 :                      shb.version_major, shb.version_minor ));
     216           0 :     return NULL;
     217           0 :   }
     218             : 
     219           3 :   return iter;
     220           3 : }
     221             : 
     222             : void *
     223           0 : fd_pcapng_iter_delete( fd_pcapng_iter_t * iter ) {
     224           0 :   void * mem = (void *)iter;
     225           0 :   memset( mem, 0, sizeof(fd_pcapng_iter_t) );
     226           0 :   return mem;
     227           0 : }
     228             : 
     229             : fd_pcapng_frame_t *
     230          15 : fd_pcapng_iter_next( fd_pcapng_iter_t * iter ) {
     231             : 
     232          15 :   static FD_TL fd_pcapng_frame_t pkt;
     233             : 
     234             :   /* Clear fields */
     235          15 :   pkt.ts      = 0L;
     236          15 :   pkt.type    = 0U;
     237          15 :   pkt.data_sz = 0U;
     238          15 :   pkt.orig_sz = 0U;
     239          15 :   pkt.if_idx  = 0U;
     240             : 
     241          15 :   FILE * stream = iter->stream;
     242             : 
     243             :   /* Attempt a number of times to find a frame of known type.
     244             :      Abort if there are too many unknown frames. */
     245          18 :   for( uint attempt=0U; attempt<256U; attempt++ ) {
     246             : 
     247          18 :     fd_pcapng_block_hdr_t hdr;
     248          18 :     long                  end;
     249          18 :     if( FD_UNLIKELY( 0!=(iter->error = fd_pcapng_peek_block( stream, &hdr, &end )) ) ) {
     250           3 :       if( FD_UNLIKELY( iter->error != -1 ) )
     251           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     252           3 :       return NULL;
     253           3 :     }
     254             : 
     255          15 :     switch( hdr.block_type ) {
     256           0 :     case FD_PCAPNG_BLOCK_TYPE_SHB: {
     257           0 :       iter->error = -1; /* eof */
     258           0 :       return NULL;
     259           0 :     }
     260           3 :     case FD_PCAPNG_BLOCK_TYPE_IDB: {
     261             :       /* Read IDB */
     262           3 :       if( FD_UNLIKELY( hdr.block_sz<sizeof(fd_pcapng_idb_t) ) ) {
     263           0 :         iter->error = EPROTO;
     264           0 :         FD_LOG_WARNING(( "pcapng: invalid IDB block size (%#x)", hdr.block_sz ));
     265           0 :         return NULL;
     266           0 :       }
     267           3 :       fd_pcapng_idb_t idb;
     268           3 :       if( FD_UNLIKELY( 1UL!=fread( &idb, sizeof(fd_pcapng_idb_t), 1UL, stream ) ) ) {
     269           0 :         iter->error = ferror( stream );
     270           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     271           0 :         return NULL;
     272           0 :       }
     273             : 
     274             :       /* Add interface to list */
     275           3 :       if( FD_UNLIKELY( iter->iface_cnt>=FD_PCAPNG_IFACE_CNT ) ) {
     276           0 :         iter->error = EPROTO;
     277           0 :         FD_LOG_WARNING(( "pcapng: too many interfaces (max %d)", FD_PCAPNG_IFACE_CNT ));
     278           0 :         return NULL;
     279           0 :       }
     280             : 
     281           3 :       fd_pcapng_idb_desc_t * iface = &iter->iface[ iter->iface_cnt++ ];
     282           3 :       memset( iface, 0, sizeof(fd_pcapng_idb_desc_t) );
     283           3 :       iface->link_type = idb.link_type;
     284             : 
     285             :       /* Read options */
     286           3 :       for( uint j=0; j<FD_PCAPNG_MAX_OPT_CNT; j++ ) {
     287           3 :         uchar opt_buf[ 128UL ] __attribute__((aligned(32UL)));
     288           3 :         fd_pcapng_option_t opt = { .sz=sizeof(opt_buf), .value=&opt_buf };
     289           3 :         if( FD_UNLIKELY( 0!=(iter->error = fd_pcapng_read_option( stream, &opt )) ) ) {
     290           0 :           FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     291           0 :           return NULL;
     292           0 :         }
     293           3 :         if( !opt.type ) break;
     294           0 :         switch( opt.type ) {
     295           0 :         case FD_PCAPNG_OPT_COMMENT:
     296           0 :           FD_LOG_HEXDUMP_DEBUG(( "IDB comment", opt_buf, opt.sz ));
     297           0 :           break;
     298           0 :         case FD_PCAPNG_IDB_OPT_NAME:
     299           0 :           fd_cstr_fini( fd_cstr_append_cstr_safe( fd_cstr_init( iface->opts.name     ), (char const *)opt_buf, sizeof(opt_buf) ) );
     300           0 :           break;
     301           0 :         case FD_PCAPNG_IDB_OPT_HARDWARE:
     302           0 :           fd_cstr_fini( fd_cstr_append_cstr_safe( fd_cstr_init( iface->opts.hardware ), (char const *)opt_buf, sizeof(opt_buf) ) );
     303           0 :           break;
     304           0 :         case FD_PCAPNG_IDB_OPT_IPV4_ADDR:
     305           0 :           if( FD_UNLIKELY( opt.sz!=4U ) )
     306           0 :             continue;
     307           0 :           memcpy( iface->opts.ip4_addr, opt_buf, 4UL );
     308           0 :           break;
     309           0 :         case FD_PCAPNG_IDB_OPT_MAC_ADDR:
     310           0 :           if( FD_UNLIKELY( opt.sz!=6U ) )
     311           0 :             continue;
     312           0 :           memcpy( iface->opts.mac_addr, opt_buf, 6UL );
     313           0 :           break;
     314           0 :         case FD_PCAPNG_IDB_OPT_TSRESOL:
     315           0 :           if( FD_UNLIKELY( opt.sz!=1U ) )
     316           0 :             continue;
     317           0 :           iface->opts.tsresol = opt_buf[ 0 ];
     318           0 :           break;
     319           0 :         default:
     320           0 :           FD_LOG_DEBUG(( "Ignoring unknown IDB option type %#x", opt.type ));
     321           0 :           break;
     322           0 :         }
     323           0 :       }
     324             : 
     325             :       /* Seek to end of block */
     326           3 :       if( FD_UNLIKELY( 0!=fseek( stream, end, SEEK_SET ) ) ) {
     327           0 :         iter->error = errno;
     328           0 :         FD_LOG_WARNING(( "pcapng: seek failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     329           0 :         return NULL;
     330           0 :       }
     331             : 
     332             :       /* Next */
     333           3 :       break;
     334           3 :     }
     335           3 :     case FD_PCAPNG_BLOCK_TYPE_SPB: {
     336             :       /* Read SPB */
     337           0 :       if( FD_UNLIKELY( hdr.block_sz<sizeof(fd_pcapng_spb_t)
     338           0 :                     || hdr.block_sz>FD_PCAPNG_FRAME_SZ ) ) {
     339           0 :         iter->error = EPROTO;
     340           0 :         FD_LOG_WARNING(( "pcapng: invalid SPB block size (%#x)", hdr.block_sz ));
     341           0 :         return NULL;
     342           0 :       }
     343             : 
     344           0 :       uint hdr_sz  = sizeof(fd_pcapng_spb_t);
     345           0 :       uint data_sz = hdr.block_sz - hdr_sz;
     346             : 
     347           0 :       fd_pcapng_spb_t spb;
     348           0 :       if( FD_UNLIKELY( 1UL!=fread( &spb,      hdr_sz,  1UL, stream ) ) ) {
     349           0 :         iter->error = ferror( stream );
     350           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     351           0 :         return NULL;
     352           0 :       }
     353           0 :       if( FD_UNLIKELY( 1UL!=fread( &pkt.data, data_sz, 1UL, stream ) ) ) {
     354           0 :         iter->error = ferror( stream );
     355           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     356           0 :         return NULL;
     357           0 :       }
     358             : 
     359             :       /* Seek to end of block */
     360           0 :       if( FD_UNLIKELY( 0!=fseek( stream, end, SEEK_SET ) ) ) {
     361           0 :         iter->error = errno;
     362           0 :         FD_LOG_WARNING(( "pcapng: seek failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     363           0 :         return NULL;
     364           0 :       }
     365             : 
     366           0 :       pkt.type    = FD_PCAPNG_FRAME_SIMPLE;
     367           0 :       pkt.data_sz = (ushort)data_sz;
     368           0 :       pkt.orig_sz = (ushort)spb.orig_len;
     369           0 :       return &pkt;
     370           0 :     }
     371           9 :     case FD_PCAPNG_BLOCK_TYPE_EPB: {
     372             :       /* Read EPB */
     373           9 :       if( FD_UNLIKELY( hdr.block_sz<sizeof(fd_pcapng_epb_t)
     374           9 :                     || hdr.block_sz>FD_PCAPNG_FRAME_SZ ) ) {
     375           0 :         iter->error = EPROTO;
     376           0 :         FD_LOG_WARNING(( "pcapng: invalid EPB block size (%#x)", hdr.block_sz ));
     377           0 :         return NULL;
     378           0 :       }
     379             : 
     380           9 :       fd_pcapng_epb_t epb;
     381           9 :       if( FD_UNLIKELY( 1UL!=fread( &epb, sizeof(fd_pcapng_epb_t), 1UL, stream ) ) ) {
     382           0 :         iter->error = ferror( stream );
     383           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     384           0 :         return NULL;
     385           0 :       }
     386           9 :       if( FD_UNLIKELY( epb.cap_len>FD_PCAPNG_FRAME_SZ ) ) {
     387           0 :         iter->error = EPROTO;
     388           0 :         FD_LOG_WARNING(( "pcapng: oversize EPB data (%#x)", epb.cap_len ));
     389           0 :         return NULL;
     390           0 :       }
     391           9 :       if( FD_UNLIKELY( 1UL!=fread( &pkt.data, epb.cap_len, 1UL, stream ) ) ) {
     392           0 :         iter->error = ferror( stream );
     393           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     394           0 :         return NULL;
     395           0 :       }
     396             : 
     397             :       /* Read options */
     398           9 :       for( uint j=0; j<FD_PCAPNG_MAX_OPT_CNT; j++ ) {
     399           9 :         uchar opt_buf[ 128UL ] __attribute__((aligned(32UL)));
     400           9 :         fd_pcapng_option_t opt = { .sz=sizeof(opt_buf), .value=&opt_buf };
     401           9 :         if( FD_UNLIKELY( 0!=(iter->error = fd_pcapng_read_option( stream, &opt )) ) ) {
     402           0 :           FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     403           0 :           return NULL;
     404           0 :         }
     405           9 :         if( !opt.type ) break;
     406           0 :         switch( opt.type ) {
     407           0 :         case FD_PCAPNG_OPT_COMMENT:
     408           0 :           FD_LOG_HEXDUMP_DEBUG(( "Packet comment", opt_buf, opt.sz ));
     409           0 :           break;
     410           0 :         default:
     411           0 :           FD_LOG_DEBUG(( "Ignoring unknown EPB option type %#x", opt.type ));
     412           0 :           break;
     413           0 :         }
     414           0 :       }
     415             : 
     416           9 :       if( FD_LIKELY( epb.if_idx < iter->iface_cnt ) ) {
     417           9 :         ulong raw = ( ((ulong)epb.ts_hi << 32UL) | (ulong)epb.ts_lo );
     418             :         /* FIXME support more timestamp resolutions */
     419           9 :         if( iter->iface[ epb.if_idx ].opts.tsresol == FD_PCAPNG_TSRESOL_NS ) {
     420           0 :           pkt.ts = (long)raw;
     421           0 :         }
     422           9 :       }
     423             : 
     424             :       /* Seek to end of block */
     425           9 :       if( FD_UNLIKELY( 0!=fseek( stream, end, SEEK_SET ) ) ) {
     426           0 :         iter->error = errno;
     427           0 :         FD_LOG_WARNING(( "pcapng: seek failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     428           0 :         return NULL;
     429           0 :       }
     430             : 
     431           9 :       pkt.type    = FD_PCAPNG_FRAME_ENHANCED;
     432           9 :       pkt.data_sz = (ushort)epb.cap_len;
     433           9 :       pkt.orig_sz = (ushort)epb.orig_len;
     434           9 :       pkt.if_idx  = epb.if_idx;
     435           9 :       return &pkt;
     436           9 :     }
     437           3 :     case FD_PCAPNG_BLOCK_TYPE_DSB: {
     438             :       /* Read DSB */
     439           3 :       if( FD_UNLIKELY( hdr.block_sz<sizeof(fd_pcapng_dsb_t)
     440           3 :                     || hdr.block_sz>FD_PCAPNG_FRAME_SZ ) ) {
     441           0 :         iter->error = EPROTO;
     442           0 :         FD_LOG_WARNING(( "pcapng: invalid DSB block size (%#x)", hdr.block_sz ));
     443           0 :         return NULL;
     444           0 :       }
     445             : 
     446           3 :       fd_pcapng_dsb_t dsb;
     447           3 :       if( FD_UNLIKELY( 1UL!=fread( &dsb, sizeof(fd_pcapng_dsb_t), 1UL, stream ) ) ) {
     448           0 :         iter->error = ferror( stream );
     449           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     450           0 :         return NULL;
     451           0 :       }
     452           3 :       if( FD_UNLIKELY( dsb.secret_sz>FD_PCAPNG_FRAME_SZ ) ) {
     453           0 :         iter->error = EPROTO;
     454           0 :         FD_LOG_WARNING(( "pcapng: oversize DSB data (%#x)", dsb.secret_sz ));
     455           0 :         return NULL;
     456           0 :       }
     457           3 :       if( FD_UNLIKELY( 1UL!=fread( &pkt.data, dsb.secret_sz, 1UL, stream ) ) ) {
     458           0 :         iter->error = ferror( stream );
     459           0 :         FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     460           0 :         return NULL;
     461           0 :       }
     462             : 
     463             :       /* Read options */
     464           3 :       for( uint j=0; j<FD_PCAPNG_MAX_OPT_CNT; j++ ) {
     465           3 :         uchar opt_buf[ 128UL ] __attribute__((aligned(32UL)));
     466           3 :         fd_pcapng_option_t opt = { .sz=sizeof(opt_buf), .value=&opt_buf };
     467           3 :         if( FD_UNLIKELY( 0!=(iter->error = fd_pcapng_read_option( stream, &opt )) ) ) {
     468           0 :           FD_LOG_WARNING(( "pcapng: read failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     469           0 :           return NULL;
     470           0 :         }
     471           3 :         if( !opt.type ) break;
     472           0 :         switch( opt.type ) {
     473           0 :         case FD_PCAPNG_OPT_COMMENT:
     474           0 :           FD_LOG_HEXDUMP_DEBUG(( "Decryption secrets comment", opt_buf, opt.sz ));
     475           0 :           break;
     476           0 :         default:
     477           0 :           FD_LOG_DEBUG(( "Ignoring unknown DSB option type %#x", opt.type ));
     478           0 :           break;
     479           0 :         }
     480           0 :       }
     481             : 
     482           3 :       if( dsb.secret_type!=FD_PCAPNG_SECRET_TYPE_TLS ) {
     483           0 :         FD_LOG_DEBUG(( "Ignoring secret (type %#x)", dsb.secret_type ));
     484           0 :         break;
     485           0 :       }
     486             : 
     487             :       /* Seek to end of block */
     488           3 :       if( FD_UNLIKELY( 0!=fseek( stream, end, SEEK_SET ) ) ) {
     489           0 :         iter->error = errno;
     490           0 :         FD_LOG_WARNING(( "pcapng: seek failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     491           0 :         return NULL;
     492           0 :       }
     493             : 
     494           3 :       pkt.type    = FD_PCAPNG_FRAME_TLSKEYS;
     495           3 :       pkt.data_sz = dsb.secret_sz;
     496           3 :       return &pkt;
     497           3 :     }
     498           0 :     default:
     499           0 :       FD_LOG_DEBUG(( "pcapng: skipping unknown block (type=%#x)", hdr.block_type ));
     500           0 :       if( FD_UNLIKELY( 0!=fseek( stream, hdr.block_sz, SEEK_CUR ) ) ) {
     501           0 :         iter->error = errno;
     502           0 :         FD_LOG_WARNING(( "pcapng: seek failed (%s)", fd_pcapng_iter_strerror( iter->error, stream ) ));
     503           0 :         return NULL;
     504           0 :       }
     505          15 :     }
     506             : 
     507             :     /* Read block that is not interesting to user, continue to next */
     508          15 :   }
     509             : 
     510             :   /* Found no blocks that are interesting to user */
     511           0 :   iter->error = EPROTO;
     512           0 :   FD_LOG_WARNING(( "pcapng: aborting, too many non-packet frames" ));
     513           0 :   return NULL;
     514          15 : }
     515             : 
     516             : FD_FN_PURE int
     517           3 : fd_pcapng_iter_err( fd_pcapng_iter_t const * iter ) {
     518           3 :   return iter->error;
     519           3 : }
     520             : 
     521             : /* fwrite-style funcs *************************************************/
     522             : 
     523             : /* What follows are a bunch of serialization / writer functions.  They
     524             :    maintain the following properties:
     525             : 
     526             :      - file handle is 4 byte aligned
     527             :      - buf is the write buffer up to
     528             :      - cursor is the next free byte in buffer (or next byte after end of
     529             :        buf is space exhausted)
     530             :      - Invariant: cursor <= FD_PCAPNG_BLOCK_SZ
     531             :      - fwrite is called once per func and write size is 4 byte aligned
     532             :        and no larger than FD_PCAPNG_BLOCK_SZ */
     533             : 
     534             : /* FD_PCAPNG_FWRITE_OPT writes an option in the context of an fwrite-
     535             :    style function.  Assumes that given length is <=65532.
     536             : 
     537             :    Args:
     538             :      ushort t (option type)
     539             :      ushort l (option length)
     540             :      void * v (ptr to option data) */
     541             : 
     542             : #define FD_PCAPNG_FWRITE_OPT(t,l,v)                                    \
     543          57 :   do {                                                                 \
     544          57 :     ulong _sz       = (ushort)( l );                                   \
     545          57 :     ulong _sz_align = (ushort)fd_ulong_align_up( _sz, 4UL );           \
     546          57 :     if( FD_UNLIKELY( cursor+4UL+_sz_align > FD_PCAPNG_BLOCK_SZ ) ) {   \
     547           0 :       FD_LOG_WARNING(( "oversz pcapng block" ));                       \
     548           0 :       return 0UL;                                                      \
     549           0 :     }                                                                  \
     550          57 :     *(ushort *)( buf+cursor ) = ( (ushort)(t) ); cursor+=2UL;          \
     551          57 :     *(ushort *)( buf+cursor ) = ( (ushort)_sz ); cursor+=2UL;          \
     552          57 :     if( _sz ) fd_memcpy( buf+cursor, (v), _sz );                       \
     553          57 :     fd_memset( buf+cursor+_sz, 0, _sz_align-_sz );                     \
     554          57 :     cursor+=_sz_align;                                                 \
     555          57 :   } while(0);
     556             : 
     557             : #define FD_PCAPNG_FWRITE_NULLOPT()                                     \
     558          15 :   do {                                                                 \
     559          15 :     if( FD_UNLIKELY( cursor+4UL > FD_PCAPNG_BLOCK_SZ ) ) {             \
     560           0 :       FD_LOG_WARNING(( "oversz pcapng block" ));                       \
     561           0 :       return 0UL;                                                      \
     562           0 :     }                                                                  \
     563          15 :     fd_memset( buf+cursor, 0, 4UL );                                   \
     564          15 :     cursor+=4UL;                                                       \
     565          15 :   } while(0);
     566             : 
     567             : 
     568             : /* FD_PCAPNG_FWRITE_BLOCK_TERM terminates a block buffer being
     569             :    serialized in the context of an fwrite-style function. */
     570             : 
     571             : #define FD_PCAPNG_FWRITE_BLOCK_TERM()                                  \
     572          15 :   do {                                                                 \
     573          15 :     if( FD_UNLIKELY( cursor+4UL > FD_PCAPNG_BLOCK_SZ ) ) {             \
     574           0 :       FD_LOG_WARNING(( "oversz pcapng block" ));                       \
     575           0 :       return 0UL;                                                      \
     576           0 :     }                                                                  \
     577          15 :     block->block_sz         = (uint)(cursor+4UL);                      \
     578          15 :     *(uint *)( buf+cursor ) = (uint)(cursor+4UL);                      \
     579          15 :     cursor+=4UL;                                                       \
     580          15 :   } while(0);
     581             : 
     582             : ulong
     583             : fd_pcapng_fwrite_shb( fd_pcapng_shb_opts_t const * opt,
     584           9 :                       void *                       file ) {
     585             : 
     586           9 :   uchar buf[ FD_PCAPNG_BLOCK_SZ ];
     587             : 
     588           9 :   fd_pcapng_shb_t * block = (fd_pcapng_shb_t *)buf;
     589             : 
     590           9 :   ulong cursor = sizeof(fd_pcapng_shb_t);
     591           9 :   *block = (fd_pcapng_shb_t) {
     592           9 :     .block_type       = FD_PCAPNG_BLOCK_TYPE_SHB,
     593             :     /* block_sz set later */
     594           9 :     .byte_order_magic = FD_PCAPNG_BYTE_ORDER_MAGIC,
     595           9 :     .version_major    = (ushort)1,
     596           9 :     .version_minor    = (ushort)0,
     597           9 :     .section_sz       = ULONG_MAX
     598           9 :   };
     599             : 
     600           9 :   if( opt ) {
     601           9 :     if( opt->hardware ) FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_SHB_OPT_HARDWARE, strlen( opt->hardware ), opt->hardware );
     602           9 :     if( opt->os       ) FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_SHB_OPT_OS,       strlen( opt->os       ), opt->os       );
     603           9 :     if( opt->userappl ) FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_SHB_OPT_USERAPPL, strlen( opt->userappl ), opt->userappl );
     604           9 :   }
     605           9 :   FD_PCAPNG_FWRITE_NULLOPT();
     606             : 
     607           9 :   FD_PCAPNG_FWRITE_BLOCK_TERM();
     608             : 
     609           9 :   return fwrite( buf, cursor, 1UL, (FILE *)file );
     610           9 : }
     611             : 
     612             : ulong
     613             : fd_pcapng_fwrite_idb( uint                         link_type,
     614             :                       fd_pcapng_idb_opts_t const * opt,
     615           6 :                       void *                       file ) {
     616             : 
     617           6 :   uchar buf[ FD_PCAPNG_BLOCK_SZ ];
     618             : 
     619           6 :   fd_pcapng_idb_t * block = (fd_pcapng_idb_t *)buf;
     620             : 
     621           6 :   ulong cursor = sizeof(fd_pcapng_idb_t);
     622           6 :   *block = (fd_pcapng_idb_t) {
     623           6 :     .block_type       = FD_PCAPNG_BLOCK_TYPE_IDB,
     624             :     /* block_sz set later */
     625           6 :     .link_type        = (ushort)link_type,
     626           6 :     .snap_len         = 0U, /* FIXME should appropriately set snap_len
     627             :                                But this is not trivial.  Needs balancing
     628             :                                between buffer space available for meta
     629             :                                and payload. (meta is variable length) */
     630           6 :   };
     631             : 
     632           6 :   uchar tsresol = FD_PCAPNG_TSRESOL_NS;
     633           6 :   FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_IDB_OPT_TSRESOL, 1UL, &tsresol );
     634             : 
     635           6 :   if( opt ) {
     636             : 
     637           6 :     if( opt->name[0] )
     638           6 :       FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_IDB_OPT_NAME,      fd_cstr_nlen( opt->name, 16UL ),     opt->name     );
     639           6 :     if( fd_uint_load_4( opt->ip4_addr ) )
     640           6 :       FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_IDB_OPT_IPV4_ADDR, 4UL,                                 opt->ip4_addr );
     641           6 :     if( fd_ulong_load_6( opt->mac_addr ) )
     642           6 :       FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_IDB_OPT_MAC_ADDR,  6UL,                                 opt->mac_addr );
     643             : 
     644           6 :     if( opt->hardware[0] )
     645           6 :       FD_PCAPNG_FWRITE_OPT( FD_PCAPNG_IDB_OPT_HARDWARE,  fd_cstr_nlen( opt->hardware, 64UL ), opt->hardware );
     646             : 
     647           6 :   }
     648           6 :   FD_PCAPNG_FWRITE_NULLOPT();
     649             : 
     650           6 :   FD_PCAPNG_FWRITE_BLOCK_TERM();
     651             : 
     652           6 :   return fwrite( buf, cursor, 1UL, (FILE *)file );
     653           6 : }
     654             : 
     655             : ulong
     656             : fd_pcapng_fwrite_pkt( long         ts,
     657             :                       void const * payload,
     658             :                       ulong        payload_sz,
     659          12 :                       void *       _file ) {
     660             : 
     661          12 :   FILE * file = (FILE *)_file;
     662          12 :   FD_TEST( fd_ulong_is_aligned( (ulong)ftell( file ), 4UL ) );
     663             : 
     664          12 :   ulong cursor = sizeof(fd_pcapng_epb_t);
     665          12 :   fd_pcapng_epb_t block = {
     666          12 :     .block_type = FD_PCAPNG_BLOCK_TYPE_EPB,
     667             :     /* block_sz set later */
     668          12 :     .if_idx     = 0U,
     669          12 :     .ts_hi      = (uint)( (ulong)ts >> 32UL ),
     670          12 :     .ts_lo      = (uint)( (ulong)ts         ),
     671          12 :     .cap_len    = (uint)payload_sz,
     672          12 :     .orig_len   = (uint)payload_sz
     673          12 :   };
     674             : 
     675          12 :   ulong payload_sz_align = fd_ulong_align_up( payload_sz, 4UL );
     676          12 :   uchar pad[8UL]={0};
     677          12 :   ulong pad_sz = payload_sz_align-payload_sz;
     678          12 :   cursor+=payload_sz_align;
     679             : 
     680             :   /* Empty option list */
     681          12 :   cursor+=4UL;
     682             : 
     683             :   /* Trailer */
     684          12 :   block.block_sz = (uint)cursor+4U;
     685             : 
     686             :   /* write header */
     687          12 :   if( FD_UNLIKELY( 1UL!=fwrite( &block,  sizeof(fd_pcapng_epb_t), 1UL, file ) ) )
     688           0 :     return 0UL;
     689             :   /* copy payload */
     690          12 :   if( FD_UNLIKELY( 1UL!=fwrite( payload, payload_sz,              1UL, file ) ) )
     691           0 :     return 0UL;
     692             :   /* align */
     693          12 :   if( pad_sz )
     694           9 :     if( FD_UNLIKELY( 1UL!=fwrite( pad, pad_sz, 1UL, file ) ) )
     695           0 :       return 0UL;
     696             :   /* empty options */
     697          12 :   if( FD_UNLIKELY( 1UL!=fwrite( pad, 4UL,    1UL, file ) ) )
     698           0 :     return 0UL;
     699             :   /* write length trailer */
     700          12 :   if( FD_UNLIKELY( 1UL!=fwrite( &block.block_sz, 4UL, 1UL, file ) ) )
     701           0 :     return 0UL;
     702             : 
     703          12 :   return 1UL;
     704          12 : }
     705             : 
     706             : ulong
     707             : fd_pcapng_fwrite_tls_key_log( uchar const * log,
     708             :                               uint          log_sz,
     709           6 :                               void *        _file ) {
     710             : 
     711           6 :   FILE * file = (FILE *)_file;
     712           6 :   FD_TEST( fd_ulong_is_aligned( (ulong)ftell( file ), 4UL ) );
     713             : 
     714           6 :   ulong cursor = sizeof(fd_pcapng_dsb_t);
     715           6 :   fd_pcapng_dsb_t block = {
     716           6 :     .block_type  = FD_PCAPNG_BLOCK_TYPE_DSB,
     717             :     /* block_sz set later */
     718           6 :     .secret_type = FD_PCAPNG_SECRET_TYPE_TLS,
     719           6 :     .secret_sz   = log_sz
     720           6 :   };
     721             : 
     722           6 :   uint log_sz_align = fd_uint_align_up( log_sz, 4UL );
     723           6 :   uchar pad[8] = {0};
     724           6 :   ulong pad_sz = log_sz_align-log_sz;
     725           6 :   cursor+=log_sz_align;
     726             : 
     727             :   /* end of options block */
     728           6 :   cursor+=4UL;
     729             : 
     730             :   /* derive size ahead of time */
     731           6 :   block.block_sz = (uint)cursor + 4U;
     732             : 
     733             :   /* write header */
     734           6 :   if( FD_UNLIKELY( 1UL!=fwrite( &block, sizeof(fd_pcapng_dsb_t), 1UL, file ) ) )
     735           0 :     return 0UL;
     736             :   /* copy log */
     737           6 :   if( FD_UNLIKELY( 1UL!=fwrite( log, log_sz, 1UL, file ) ) )
     738           0 :     return 0UL;
     739             :   /* align */
     740           6 :   if( pad_sz )
     741           6 :     if( FD_UNLIKELY( 1UL!=fwrite( pad, pad_sz, 1UL, file ) ) )
     742           0 :       return 0UL;
     743             :   /* empty options */
     744           6 :   if( FD_UNLIKELY( 1UL!=fwrite( pad, 4UL,    1UL, file ) ) )
     745           0 :     return 0UL;
     746             :   /* write length trailer */
     747           6 :   if( FD_UNLIKELY( 1UL!=fwrite( &block.block_sz, sizeof(uint), 1, file ) ) )
     748           0 :     return 0UL;
     749             : 
     750           6 :   return 1UL;
     751           6 : }
     752             : 
     753             : #endif /* FD_HAS_HOSTED */

Generated by: LCOV version 1.14