diff options
| author | Jeff King <peff@peff.net> | 2011-09-13 17:57:57 -0400 | 
|---|---|---|
| committer | Junio C Hamano <gitster@pobox.com> | 2011-09-14 11:56:36 -0700 | 
| commit | c1189caeaf726e6c16c8bca7da8bf0b243da478d (patch) | |
| tree | 8d8feeeef2e8584c11875da8dde2d04f334cb56b /argv-array.c | |
| parent | 7878b07c0d86d05fa505f2464557c69addcc2c05 (diff) | |
| download | git-c1189caeaf726e6c16c8bca7da8bf0b243da478d.tar.gz | |
refactor argv_array into generic code
The submodule code recently grew generic code to build a
dynamic argv array. Many other parts of the code can reuse
this, too, so let's make it generically available.
There are two enhancements not found in the original code:
  1. We now handle the NULL-termination invariant properly,
     even when no strings have been pushed (before, you
     could have an empty, NULL argv). This was not a problem
     for the submodule code, which always pushed at least
     one argument, but was not sufficiently safe for
     generic code.
  2. There is a formatted variant of the "push" function.
     This is a convenience function which was not needed by
     the submodule code, but will make it easier to port
     other users to the new code.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
Diffstat (limited to 'argv-array.c')
| -rw-r--r-- | argv-array.c | 51 | 
1 files changed, 51 insertions, 0 deletions
| diff --git a/argv-array.c b/argv-array.c new file mode 100644 index 0000000000..a4e04201e6 --- /dev/null +++ b/argv-array.c @@ -0,0 +1,51 @@ +#include "cache.h" +#include "argv-array.h" +#include "strbuf.h" + +static const char *empty_argv_storage = NULL; +const char **empty_argv = &empty_argv_storage; + +void argv_array_init(struct argv_array *array) +{ +	array->argv = empty_argv; +	array->argc = 0; +	array->alloc = 0; +} + +static void argv_array_push_nodup(struct argv_array *array, const char *value) +{ +	if (array->argv == empty_argv) +		array->argv = NULL; + +	ALLOC_GROW(array->argv, array->argc + 2, array->alloc); +	array->argv[array->argc++] = value; +	array->argv[array->argc] = NULL; +} + +void argv_array_push(struct argv_array *array, const char *value) +{ +	argv_array_push_nodup(array, xstrdup(value)); +} + +void argv_array_pushf(struct argv_array *array, const char *fmt, ...) +{ +	va_list ap; +	struct strbuf v = STRBUF_INIT; + +	va_start(ap, fmt); +	strbuf_vaddf(&v, fmt, ap); +	va_end(ap); + +	argv_array_push_nodup(array, strbuf_detach(&v, NULL)); +} + +void argv_array_clear(struct argv_array *array) +{ +	if (array->argv != empty_argv) { +		int i; +		for (i = 0; i < array->argc; i++) +			free((char **)array->argv[i]); +		free(array->argv); +	} +	argv_array_init(array); +} | 
