diff options
author | Vito Caputo <vcaputo@pengaru.com> | 2023-06-11 17:00:43 -0700 |
---|---|---|
committer | Vito Caputo <vcaputo@pengaru.com> | 2023-06-12 16:16:12 -0700 |
commit | 3398e156c561981ce30f4847ac56458c875f8316 (patch) | |
tree | f45b8a8aae6b332f2e3dee45ee3f6015cbd1fc26 /src | |
parent | 7f7272b5b87d3baf8051a6856507b4b2618765c9 (diff) |
til_str: add preliminary growable string type
Preparation for eliminating open_memstream() usage...
Diffstat (limited to 'src')
-rw-r--r-- | src/Makefile.am | 2 | ||||
-rw-r--r-- | src/til_str.c | 180 | ||||
-rw-r--r-- | src/til_str.h | 14 |
3 files changed, 195 insertions, 1 deletions
diff --git a/src/Makefile.am b/src/Makefile.am index b5f4e62..5da7150 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -1,7 +1,7 @@ SUBDIRS = libs modules noinst_LTLIBRARIES = libtil.la -libtil_la_SOURCES = til.c til.h til_args.c til_args.h til_fb.c til_fb.h til_jenkins.c til_jenkins.h til_limits.h til_module_context.c til_module_context.h til_settings.h til_settings.c til_setup.c til_setup.h til_stream.c til_stream.h til_tap.h til_threads.c til_threads.h til_util.c til_util.h +libtil_la_SOURCES = til.c til.h til_args.c til_args.h til_fb.c til_fb.h til_jenkins.c til_jenkins.h til_limits.h til_module_context.c til_module_context.h til_settings.h til_settings.c til_setup.c til_setup.h til_str.c til_str.h til_stream.c til_stream.h til_tap.h til_threads.c til_threads.h til_util.c til_util.h libtil_la_CPPFLAGS = -I@top_srcdir@/src libtil_la_LIBADD = modules/blinds/libblinds.la modules/checkers/libcheckers.la modules/compose/libcompose.la modules/drizzle/libdrizzle.la modules/flui2d/libflui2d.la modules/julia/libjulia.la modules/meta2d/libmeta2d.la modules/mixer/libmixer.la modules/moire/libmoire.la modules/montage/libmontage.la modules/pixbounce/libpixbounce.la modules/plasma/libplasma.la modules/plato/libplato.la modules/ray/libray.la modules/rkt/librkt.la modules/roto/libroto.la modules/rtv/librtv.la modules/shapes/libshapes.la modules/snow/libsnow.la modules/sparkler/libsparkler.la modules/spiro/libspiro.la modules/stars/libstars.la modules/strobe/libstrobe.la modules/submit/libsubmit.la modules/swab/libswab.la modules/swarm/libswarm.la modules/voronoi/libvoronoi.la libs/grid/libgrid.la libs/puddle/libpuddle.la libs/ray/libray.la libs/rocket/librocket.la libs/sig/libsig.la libs/txt/libtxt.la libs/ascii/libascii.la libs/din/libdin.la diff --git a/src/til_str.c b/src/til_str.c new file mode 100644 index 0000000..442e923 --- /dev/null +++ b/src/til_str.c @@ -0,0 +1,180 @@ +#include <assert.h> +#include <stdarg.h> +#include <stdlib.h> +#include <string.h> + +#include "til_str.h" +#include "til_util.h" + +/* This implements very rudimentary growable strings, and hasn't been optimized at all. + * + * The impetus for adding this is to get rid of the open_memstream() usage which is mostly + * just being used as a convenient way to build up a buffer from format strings. So this + * basically implements just that ability using variadic functions... + * + * Why no open_memstream()? Because Windows, that's why. For some reason even mingw doesn't + * have open_memstream(). FILE* I keep forgetting that you're dead to me. + */ + +struct til_str_t { + struct { + size_t allocated, used; /* used is length, including '\0' terminator */ + } size; + char *buf; +}; + + +#define TIL_STR_MIN_SIZE 64 + + +/* alloc always returns a buf w/nul terminator present */ +static til_str_t * til_str_nulstr(size_t minsize) +{ + til_str_t *str; + + str = calloc(1, sizeof(*str)); + if (!str) + return NULL; + + str->size.used = 1; + str->size.allocated = MAX(minsize, TIL_STR_MIN_SIZE); + + str->buf = calloc(1, str->size.allocated); + if (!str->buf) { + free(str); + return NULL; + } + + return str; +} + + +/* allocate a new til_str, starting with a dup of seed, just use "" for an empty str, there is no NULL str */ +til_str_t * til_str_new(const char *seed) +{ + til_str_t *str; + size_t len; + + assert(seed); + + len = strlen(seed); + str = til_str_nulstr(len + 1); + if (!str) + return NULL; + + memcpy(str->buf, seed, len); + str->size.used += len; + + return str; +} + + +void * til_str_free(til_str_t *str) +{ + if (str) { + free(str->buf); + free(str); + } + + return NULL; +} + + +/* allocate a new til_str from a fmt string + args */ +til_str_t * til_str_newf(const char *format, ...) +{ + til_str_t *str; + va_list ap; + + assert(format); + + va_start(ap, format); + + str = til_str_nulstr(vsnprintf(NULL, 0, format, ap) + 1); + if (!str) + return NULL; + + str->size.used += vsnprintf(str->buf, str->size.allocated, format, ap); + + va_end(ap); + + assert(str->size.used <= str->size.allocated); + + return str; +} + + +/* append to an existing til_str_t from a fmt string + args */ +int til_str_appendf(til_str_t *str, const char *format, ...) +{ + size_t len; + va_list ap; + + assert(str); + assert(format); + + va_start(ap, format); + len = vsnprintf(NULL, 0, format, ap); + va_end(ap); + + if (str->size.used + len > str->size.allocated) { + char *new; + + new = realloc(str->buf, str->size.used + len); + if (!new) + return -ENOMEM; + + str->buf = new; + str->size.allocated = str->size.used + len; + } + + va_start(ap, format); + str->size.used += vsnprintf(&str->buf[str->size.used - 1], + str->size.allocated - (str->size.used - 1), + format, ap); + va_end(ap); + + assert(str->size.used <= str->size.allocated); + + return 0; +} + + +/* strdup() the /used/ contents of str */ +char * til_str_strdup(const til_str_t *str) +{ + assert(str); + + return strdup(str->buf); +} + + +/* a valid \0-terminated string is _always_ maintained @ str->buf so callers can just use it as a string.. + * but must not hang onto that pointer across more til_str() calls on the same str. + */ +const char * til_str_buf(const til_str_t *str, size_t *res_len) +{ + assert(str); + + if (res_len) + *res_len = str->size.used; + + return str->buf; +} + + +/* for when you just want the buf without the dup overhead, and don't need the str anymore */ +char * til_str_to_buf(til_str_t *str, size_t *res_len) +{ + char *buf; + + assert(str); + + if (res_len) + *res_len = str->size.used; + + buf = str->buf; + free(str); + + return buf; +} diff --git a/src/til_str.h b/src/til_str.h new file mode 100644 index 0000000..e653884 --- /dev/null +++ b/src/til_str.h @@ -0,0 +1,14 @@ +#ifndef _TIL_STR_H +#define _TIL_STR_H + +typedef struct til_str_t til_str_t; + +til_str_t * til_str_new(const char *seed); +void * til_str_free(til_str_t *str); +til_str_t * til_str_newf(const char *format, ...); +int til_str_appendf(til_str_t *str, const char *format, ...); +char * til_str_strdup(const til_str_t *str); +const char * til_str_buf(const til_str_t *str, size_t *res_len); +char * til_str_to_buf(til_str_t *str, size_t *res_len); + +#endif |