LCOV - code coverage report
Current view: top level - util/cstr - fd_cstr.h (source / functions) Hit Total Coverage
Test: cov.lcov Lines: 67 98 68.4 %
Date: 2025-01-08 12:08:44 Functions: 41 10050 0.4 %

          Line data    Source code
       1             : #ifndef HEADER_fd_src_cstr_fd_cstr_h
       2             : #define HEADER_fd_src_cstr_fd_cstr_h
       3             : 
       4             : /* APIs for manipulating '\0'-terminated character strings ("cstr") */
       5             : 
       6             : #include "../bits/fd_bits.h"
       7             : 
       8             : FD_PROTOTYPES_BEGIN
       9             : 
      10             : /* cstr input *********************************************************/
      11             : 
      12             : /* fd_cstr_to_T converts the cstr pointed at by s into a T and returns
      13             :    its value.  Caller promises s is non-NULL and points at a cstr.
      14             : 
      15             :    Note fd_cstr_to_cstr just returns s.  As such the lifetime of the
      16             :    returned pointer is the lifetime s and ownership model of the
      17             :    underlying s is defined by the application.
      18             : 
      19             :    fd_cstr_to_char just returns the first character of the cstr (if cstr
      20             :    is the empty string, this will be the '\0' character ... otherwise,
      21             :    it will be a normal string character).  As char do not have a
      22             :    consistent interpretation between platforms due to issues with the
      23             :    language standard itself, the value here should just be treated as a
      24             :    character and not an integer.  Use fd_cstr_schar/fd_cstr_uchar if you
      25             :    need to treat a char as an integer.
      26             : 
      27             :    fd_cstr_to_cstr and fd_cstr_to_char exist primarily for type system
      28             :    completeness / facilitate various generic programming practices.
      29             : 
      30             :    The integer converters work in the strtol sense with base 0 (and thus
      31             :    ignore leading whitespace, handle leading signs and assume octal if
      32             :    the body is prefixed with 0, hexadecimal if prefixed with 0x and
      33             :    decimal otherwise). */
      34             : 
      35             : FD_FN_CONST char const * fd_cstr_to_cstr  ( char const * s );
      36             : FD_FN_PURE  char         fd_cstr_to_char  ( char const * s );
      37             : FD_FN_PURE  schar        fd_cstr_to_schar ( char const * s );
      38             : FD_FN_PURE  short        fd_cstr_to_short ( char const * s );
      39             : FD_FN_PURE  int          fd_cstr_to_int   ( char const * s );
      40             : FD_FN_PURE  long         fd_cstr_to_long  ( char const * s );
      41             : FD_FN_PURE  uchar        fd_cstr_to_uchar ( char const * s );
      42             : FD_FN_PURE  ushort       fd_cstr_to_ushort( char const * s );
      43             : FD_FN_PURE  uint         fd_cstr_to_uint  ( char const * s );
      44             : FD_FN_PURE  ulong        fd_cstr_to_ulong ( char const * s );
      45             : FD_FN_PURE  float        fd_cstr_to_float ( char const * s );
      46             : #if FD_HAS_DOUBLE
      47             : FD_FN_PURE  double       fd_cstr_to_double( char const * s );
      48             : #endif
      49             : 
      50             : /* fd_cstr_to_ulong_octal is the same as fd_cstr_to_ulong but assumes s
      51             :    points is octal.  This is mostly used when dealing parsing UNIX style
      52             :    file permissions. */
      53             : 
      54             : FD_FN_PURE ulong fd_cstr_to_ulong_octal( char const * s );
      55             : 
      56             : /* fd_cstr_to_ulong_seq populates seq (which has room for seq max items)
      57             :    with the sequenced specified by the given cstr.  Sequences are a
      58             :    comma separated list of ranges (e.g. "R0,R1,R2").  The ranges
      59             :    themselves can be themselves be individual integers (e.g. "5") or a
      60             :    simple range (e.g. "4-8", includes both endpoints, stop should be at
      61             :    least start), a range with a skip (e.g. "1-10/3" or "1-10:3", stop
      62             :    should be at least start and stride should be positive).  Ignores
      63             :    internal whitespace.  Robust against overflow / wrapping of ranges
      64             :    against ULONG_MAX.  Items may appear in multiple times and sequences
      65             :    can have an arbitrary order.  Caller promises seq is non-NULL if max
      66             :    is non-zero.  Returns 0 on NULL or malformed cstr or empty sequence
      67             :    (seq contents might have been arbitrarily clobbered on a malformed
      68             :    cstr). */
      69             : 
      70             : ulong                                         /* Actual sequence length, if greater than seq_max returned sequence truncated. */
      71             : fd_cstr_to_ulong_seq( char const * cstr,      /* String to parse, NULL returns 0 */
      72             :                       ulong *      seq,       /* Indexed [0,max), elements [0,min(actual sequence length,seq_max)) populated with
      73             :                                                  the leading portion of the seq.  Any remaining elements of seq are untouched. */
      74             :                       ulong        seq_max ); /* Maximum sequence length */
      75             : 
      76             : /* fd_cstr_hash hashes the cstr pointed to by key to a ulong.
      77             :    fd_cstr_hash_append updates the hash value (it will be as though the
      78             :    fd_cstr_hash was called on the string concatenation of the all the
      79             :    keys provided to hash / hash append in order).  Treats key==NULL the
      80             :    same as the empty string "".  Yields identical cross platform results
      81             :    regardless of how the platform treats the sign of char.  Based on one
      82             :    of the djb2 hash variants (public domain).
      83             : 
      84             :    FIXME: This is simple and fast and pretty good practically for string
      85             :    hashing but more robust and faster algos are probably out there. */
      86             : 
      87             : FD_FN_PURE static inline ulong
      88             : fd_cstr_hash_append( ulong        hash,
      89     4980936 :                      char const * key ) {
      90     4980937 :   if( FD_LIKELY( key ) ) {
      91     4980935 :     uchar const * p = (uchar const *)key;
      92   192305367 :     for(;;) {
      93   192305367 :       ulong c = p[0];
      94   192305367 :       if( FD_UNLIKELY( !c ) ) break;
      95   187324424 :       hash = (hash*33UL) ^ c;
      96   187324424 :       p++;
      97   187324424 :     }
      98     4980935 :   }
      99     4980936 :   return hash;
     100     4980936 : }
     101             : 
     102         213 : FD_FN_PURE static inline ulong fd_cstr_hash( char const * key ) { return fd_cstr_hash_append( 5381UL, key ); }
     103             : 
     104             : /* fd_cstr_casecmp is equivalent to strcasecmp but doesn't require
     105             :    FD_HAS_HOSTED (POSIX) support. */
     106             : 
     107             : FD_FN_PURE int
     108             : fd_cstr_casecmp( char const * a,
     109             :                  char const * b );
     110             : 
     111             : /* fd_cstr_nlen is equivalent to strnlen but doesn't require
     112             :    FD_HAS_HOSTED (POSIX) support. */
     113             : 
     114             : FD_FN_PURE ulong
     115             : fd_cstr_nlen( char const * s,
     116             :               ulong        m );
     117             : 
     118             : /* cstr output ********************************************************/
     119             : 
     120             : /* fd_cstr_printf printf a cstr into the sz byte memory region pointed
     121             :    to by buf.  Always returns buf.
     122             : 
     123             :    If buf is non-NULL and sz is non-zero, on return, buf will point to a
     124             :    cstr such that strlen(buf)<sz.  That is, bytes [0,strlen(buf)] will
     125             :    be non-'\0', byte strlen(buf) will be '\0' and bytes (len,sz) will be
     126             :    unchanged.  If more than sz bytes are needed to hold the requested
     127             :    cstr, the cstr will be truncated to its leading bytes such that
     128             :    strlen(buf)==sz-1.  If opt_len is non-NULL, *opt_len will be set to
     129             :    the strlen(buf) on return.
     130             : 
     131             :    buf==NULL and/or sz==0UL are treated as a no-op.  (If opt_len is
     132             :    non-NULL *opt_len wll be 0UL on return ... this is debatable though
     133             :    given the strlen(buf) property above.  Might be better to this case
     134             :    as U.B., or abort if opt_len is requested when buf==NULL and sz==NULL
     135             :    or return ULONG_MAX in opt_len (-1) to indicate ill defined usage or
     136             :    ...) */
     137             : 
     138             : char *
     139             : fd_cstr_printf( char *       buf,
     140             :                 ulong        sz,
     141             :                 ulong *      opt_len,
     142             :                 char const * fmt, ... ) __attribute__((format(printf,4,5)));
     143             : 
     144             : /* fd_cstr_printf_check is the same as fd_cstr_printf except that it
     145             :    returns 1 if the entire cstr, including the NUL terminating
     146             :    character was written to buf and 0 otherwise.
     147             : 
     148             :    If the cstr was truncated, or there was an error in the printf
     149             :    formatting process, 0 will be returned.  Otherwise, on success, 1
     150             :    will be returned.  If zero bytes are written to buf because the
     151             :    format string is empty, the return value will be 1. */
     152             : 
     153             : int
     154             : fd_cstr_printf_check( char *       buf,
     155             :                       ulong        sz,
     156             :                       ulong *      opt_len,
     157             :                       char const * fmt, ... ) __attribute__((format(printf,4,5)));
     158             : 
     159             : /* fd_cstr_init start writing a cstr into buf.  Returns where the first
     160             :    character of the cstr should be written (==buf). */
     161             : 
     162       59907 : static inline char * fd_cstr_init( char * buf ) { return buf; }
     163             : 
     164             : /* fd_cstr_fini finished writing a cstr to buf.  Assumes p is valid
     165             :    (non-NULL and room for the terminating '\0').  At this point, the buf
     166             :    passed to fd_cstr_init will be properly '\0' terminated. */
     167             : 
     168       59949 : static inline void fd_cstr_fini( char * p ) { *p = '\0'; }
     169             : 
     170             : /* fd_cstr_append_char append character c to cstr.  Assumes p is valid
     171             :    (non-NULL and room for at least this char and a final terminating
     172             :    '\0') and c is not '\0' */
     173             : 
     174     4051447 : static inline char * fd_cstr_append_char( char * p, char c ) { *(p++) = c; return p; }
     175             : 
     176             : /* fd_cstr_append_text appends n characters of text pointed to by t to
     177             :    p.  Assumes p is valid (non-NULL and room for at least n characters
     178             :    and a final terminating '\0') and t is valid (points to n consecutive
     179             :    non-'\0' characters).  n is zero is fine. */
     180             : 
     181             : static inline char *
     182             : fd_cstr_append_text( char *       p,
     183             :                      char const * t,
     184      571923 :                      ulong        n ) {
     185      571923 :   fd_memcpy( p, t, n );
     186      571923 :   return p + n;
     187      571923 : }
     188             : 
     189             : /* fd_cstr_append_cstr appends the cstr pointed to by s to p.  Assumes p
     190             :    is valid (non-NULL and room for at least strlen( s ) characters and a
     191             :    final terminating '\0').  s==NULL is treated as a no-op. */
     192             : 
     193             : static inline char *
     194             : fd_cstr_append_cstr( char *       p,
     195       48450 :                      char const * s ) {
     196       48450 :   if( FD_UNLIKELY( !s ) ) return p;
     197       48450 :   ulong n = strlen( s );
     198       48450 :   fd_memcpy( p, s, n );
     199       48450 :   return p + n;
     200       48450 : }
     201             : 
     202             : /* fd_cstr_append_cstr_safe appends up to n chars of the cstr pointed
     203             :    to by to p.  Assumes p is valid (non-NULL and room for at least n
     204             :    characters and a final terminating '\0').  s==NULL is treated as a
     205             :    no-op. */
     206             : 
     207             : static inline char *
     208             : fd_cstr_append_cstr_safe( char *       p,
     209             :                           char const * s,
     210       10995 :                           ulong        n ) {
     211       10995 :   if( FD_UNLIKELY( !s ) ) return p;
     212       10995 :   ulong l = fd_ulong_min( strlen( s ), n );
     213       10995 :   fd_memcpy( p, s, l );
     214       10995 :   return p + l;
     215       10995 : }
     216             : 
     217             : /* fd_cstr_append_printf appends the printf of the fmt string into p.
     218             :    Assumes p is valid (non-NULL and room for printf characters and a
     219             :    final terminating '\0'). */
     220             : 
     221             : char *
     222             : fd_cstr_append_printf( char *       p,
     223             :                        char const * fmt, ... ) __attribute__((format(printf,2,3)));
     224             : 
     225             : /* fd_cstr_append_ulong_as_text pretty prints the ulong into p (and
     226             :    similarly for the other unsigned integer types).  Assumes p is valid
     227             :    (non-NULL and room for at least n characters and a final terminating
     228             :    '\0'), x is small enough to pretty print to n chars (which implies
     229             :    that n is at least 1).  ws is the character to left pad the converted
     230             :    value with.  pfx is prefix character to use (e.g. '+', '-'), '\0'
     231             :    indicates no prefix.  If a prefix is requested, it will be
     232             :    immediately before the most significant converted character. */
     233             : 
     234             : static inline char *
     235             : fd_cstr_append_uint_as_text( char * p,
     236             :                              char   ws,
     237             :                              char   pm,
     238             :                              uint   x,
     239     3429060 :                              ulong  n ) {
     240     3429060 :   char * p0 = p;
     241     3429060 :   p += n;
     242     3429060 :   char * q = p;
     243     6097173 :   do { uint d = x % 10U; x /= 10U; *(--q) = (char)( d + (uint)'0' ); } while( x );
     244     3429060 :   if( pm ) *(--q) = pm;
     245     5334194 :   while( p0<q ) *(p0++) = ws;
     246     3429060 :   return p;
     247     3429060 : }
     248             : 
     249             : static inline char *
     250             : fd_cstr_append_ulong_as_text( char * p,
     251             :                               char   ws,
     252             :                               char   pm,
     253             :                               ulong  x,
     254         786 :                               ulong  n ) {
     255         786 :   char * p0 = p;
     256         786 :   p += n;
     257         786 :   char * q = p;
     258        7128 :   do { ulong d = x % 10UL; x /= 10UL; *(--q) = (char)( d + (ulong)'0' ); } while( x );
     259         786 :   if( pm ) *(--q) = pm;
     260        1038 :   while( p0<q ) *(p0++) = ws;
     261         786 :   return p;
     262         786 : }
     263             : 
     264             : #if FD_HAS_INT128
     265             : 
     266             : static inline char *
     267             : fd_cstr_append_uint128_as_text( char *  p,
     268             :                                 char    ws,
     269             :                                 char    pm,
     270             :                                 uint128 x,
     271           0 :                                 ulong   n ) {
     272           0 :   char * p0 = p;
     273           0 :   p += n;
     274           0 :   char * q = p;
     275           0 :   do { uint128 d = x % (uint128)10UL; x /= (uint128)10UL; *(--q) = (char)( d + (uint128)'0' ); } while( x );
     276           0 :   if( pm ) *(--q) = pm;
     277           0 :   while( p0<q ) *(p0++) = ws;
     278           0 :   return p;
     279           0 : }
     280             : 
     281             : #endif
     282             : 
     283             : static inline char *
     284             : fd_cstr_append_uchar_as_text ( char * p,
     285             :                                char   ws,
     286             :                                char   pm,
     287             :                                uchar  x,
     288          96 :                                ulong  n ) {
     289          96 :   return fd_cstr_append_uint_as_text( p, ws, pm, (uint)x, n );
     290          96 : }
     291             : 
     292             : static inline char *
     293             : fd_cstr_append_ushort_as_text( char * p,
     294             :                                char   ws,
     295             :                                char   pm,
     296             :                                ushort x,
     297         144 :                                ulong  n ) {
     298         144 :   return fd_cstr_append_uint_as_text( p, ws, pm, (uint)x, n );
     299         144 : }
     300             : 
     301             : /* fd_cstr_append_fxp10_as_text same as the above but for the decimal
     302             :    fixed point value:
     303             :      x / 10^f
     304             :    Assumes p is valid (non-NULL and room for at least n characters and a
     305             :    final terminating '\0'), x / 10^f is not too large to fit within n
     306             :    characters (which implies that n is at least f+2).  ws is the
     307             :    character to left pad the converted value with.  pfx is prefix
     308             :    character to use (e.g. '+', '-'), '\0' indicates no prefix.  If a
     309             :    prefix is requested, it will be immediately before the most
     310             :    significant converted character. */
     311             : 
     312             : FD_FN_UNUSED static char * /* Work around -Winline */
     313             : fd_cstr_append_fxp10_as_text( char * p,
     314             :                               char   ws,
     315             :                               char   pm,
     316             :                               ulong  f,
     317             :                               ulong  x,
     318      573444 :                               ulong  n ) {
     319      573444 :   char * p0 = p;
     320      573444 :   p += n;
     321      573444 :   char * q = p;
     322     5725368 :   while( f ) { ulong d = x % 10UL; x /= 10UL; *(--q) = (char)( d + (ulong)'0' ); f--; }
     323      573444 :   *(--q) = '.';
     324     1052664 :   do { ulong d = x % 10UL; x /= 10UL; *(--q) = (char)( d + (ulong)'0' ); } while( x );
     325      573444 :   if( pm ) *(--q) = pm;
     326      678468 :   while( p0<q ) *(p0++) = ws;
     327      573444 :   return p;
     328      573444 : }
     329             : 
     330             : /* fd_cstr_tokenize tokenizes the cstr of the form whose first
     331             :    byte is pointed to by cstr:
     332             : 
     333             :      [WS][TOKEN 0][DELIM][WS][TOKEN 1][DELIM]...[WS][TOKEN N]{[DELIM][WS][NUL],[NUL]}
     334             : 
     335             :    in-place, into:
     336             : 
     337             :      [WS][TOKEN 0][NUL][WS][TOKEN 1][NUL]...[WS][TOKEN tok_cnt-1][NUL]
     338             : 
     339             :    and returns tok_cnt.
     340             : 
     341             :    Further, on return, tok[i] for i in [0,min(tok_cnt,tok_max)) where
     342             :    tok_cnt is the number of tokens in cstr will point to the first
     343             :    byte of each token.  Due to the tokenization, each one of these will
     344             :    be properly '\0' terminated.
     345             : 
     346             :    Above, [WS] is a sequence of zero or more whitespace characters,
     347             :    [TOKEN *] are a sequence of zero or more non-delim and non-NUL
     348             :    characters and delim is assumed to be a non-NUL non-whitespace
     349             :    character (e.g. ',').
     350             : 
     351             :    As such:
     352             :    - The original cstr is clobbered by this call.
     353             :    - tok[*] point to a properly terminated cstr into the original cstr
     354             :      on return.  They thus have the same lifetime issues as the original
     355             :      cstr.
     356             :    - If tok_cnt > tok_max, tok wasn't large enough to hold all the
     357             :      tokens found in the cstr.  Only the first max are available in
     358             :      tok[*] (the entire string was still tokenized though).
     359             :    - Found tokens will not have any leading whitespace.
     360             :    - Found tokens might have internal or trailing whitespace.
     361             :    - Zero length tokens are possible.  E.g. assuming delim==':', the cstr
     362             :      "a: b::d: :f" has the tokens: "a", "b", "", "d", "", "f".
     363             :    - If the final token is zero length, it should use an explicit
     364             :      delimiter.  E.g. assuming delim=='|':
     365             :        "a|b"     has tokens "a", "b"
     366             :        "a|b|"    has tokens "a", "b"
     367             :        "a|b| "   has tokens "a", "b"
     368             :        "a|b||"   has tokens "a", "b", ""
     369             :        "a|b| |"  has tokens "a", "b", ""
     370             :        "a|b| | " has tokens "a", "b", ""
     371             :    - This is also true if the final token is the initial token.  E.g.
     372             :      assuming delim==';':
     373             :        ""    has no tokens
     374             :        " "   has no tokens
     375             :        ";"   has the token ""
     376             :        " ;"  has the token ""
     377             :        " ; " has the token "" */
     378             : 
     379             : ulong
     380             : fd_cstr_tokenize( char ** tok,
     381             :                   ulong   tok_max,
     382             :                   char *  cstr,
     383             :                   char    delim );
     384             : 
     385             : /* fd_cstr_append_utf8 appends the UTF-8 encoding of a Unicode code
     386             :    point into p.  Assumes p is valid (non-NULL and room for 1-4 chars
     387             :    and a final terminating '\0'). */
     388             : 
     389             : static inline char *
     390             : fd_cstr_append_utf8( char * p,
     391           0 :                      uint   rune ) {
     392           0 :   if( FD_LIKELY( rune<=0x7f ) ) {
     393           0 :     *(p++) = (char)rune;
     394           0 :   } else if( rune<=0x7ff ) {
     395           0 :     *(p++) = (char)( 0xc0 |  (rune>>6)       );
     396           0 :     *(p++) = (char)( 0x80 | ((rune   )&0x3f) );
     397           0 :   } else if( rune<=0xffff ) {
     398           0 :     *(p++) = (char)( 0xe0 |  (rune>>12)       );
     399           0 :     *(p++) = (char)( 0x80 | ((rune>> 6)&0x3f) );
     400           0 :     *(p++) = (char)( 0x80 | ((rune    )&0x3f) );
     401           0 :   } else if( rune<=0x10ffff ) {
     402           0 :     *(p++) = (char)( 0xf0 |  (rune>>18)       );
     403           0 :     *(p++) = (char)( 0x80 | ((rune>>12)&0x3f) );
     404           0 :     *(p++) = (char)( 0x80 | ((rune>> 6)&0x3f) );
     405           0 :     *(p++) = (char)( 0x80 |  (rune     &0x3f) );
     406           0 :   } else {
     407             :     /* replacement char */
     408           0 :     *(p++) = (char)0xef;
     409           0 :     *(p++) = (char)0xbf;
     410           0 :     *(p++) = (char)0xbd;
     411           0 :   }
     412           0 :   return p;
     413           0 : }
     414             : 
     415             : FD_PROTOTYPES_END
     416             : 
     417             : #endif /* HEADER_fd_src_cstr_fd_cstr_h */

Generated by: LCOV version 1.14