|  | #include "git-compat-util.h" | 
|  | #include "strvec.h" | 
|  | #include "strbuf.h" | 
|  |  | 
|  | const char *empty_strvec[] = { NULL }; | 
|  |  | 
|  | void strvec_init(struct strvec *array) | 
|  | { | 
|  | struct strvec blank = STRVEC_INIT; | 
|  | memcpy(array, &blank, sizeof(*array)); | 
|  | } | 
|  |  | 
|  | void strvec_push_nodup(struct strvec *array, char *value) | 
|  | { | 
|  | if (array->v == empty_strvec) | 
|  | array->v = NULL; | 
|  |  | 
|  | ALLOC_GROW(array->v, array->nr + 2, array->alloc); | 
|  | array->v[array->nr++] = value; | 
|  | array->v[array->nr] = NULL; | 
|  | } | 
|  |  | 
|  | const char *strvec_push(struct strvec *array, const char *value) | 
|  | { | 
|  | strvec_push_nodup(array, xstrdup(value)); | 
|  | return array->v[array->nr - 1]; | 
|  | } | 
|  |  | 
|  | const char *strvec_pushf(struct strvec *array, const char *fmt, ...) | 
|  | { | 
|  | va_list ap; | 
|  | struct strbuf v = STRBUF_INIT; | 
|  |  | 
|  | va_start(ap, fmt); | 
|  | strbuf_vaddf(&v, fmt, ap); | 
|  | va_end(ap); | 
|  |  | 
|  | strvec_push_nodup(array, strbuf_detach(&v, NULL)); | 
|  | return array->v[array->nr - 1]; | 
|  | } | 
|  |  | 
|  | void strvec_pushl(struct strvec *array, ...) | 
|  | { | 
|  | va_list ap; | 
|  | const char *arg; | 
|  |  | 
|  | va_start(ap, array); | 
|  | while ((arg = va_arg(ap, const char *))) | 
|  | strvec_push(array, arg); | 
|  | va_end(ap); | 
|  | } | 
|  |  | 
|  | void strvec_pushv(struct strvec *array, const char **items) | 
|  | { | 
|  | for (; *items; items++) | 
|  | strvec_push(array, *items); | 
|  | } | 
|  |  | 
|  | void strvec_splice(struct strvec *array, size_t idx, size_t len, | 
|  | const char **replacement, size_t replacement_len) | 
|  | { | 
|  | if (idx + len > array->nr) | 
|  | BUG("range outside of array boundary"); | 
|  | if (replacement_len > len) { | 
|  | if (array->v == empty_strvec) | 
|  | array->v = NULL; | 
|  | ALLOC_GROW(array->v, array->nr + (replacement_len - len) + 1, | 
|  | array->alloc); | 
|  | array->v[array->nr + (replacement_len - len)] = NULL; | 
|  | } | 
|  | for (size_t i = 0; i < len; i++) | 
|  | free((char *)array->v[idx + i]); | 
|  | if ((replacement_len != len) && array->nr) | 
|  | memmove(array->v + idx + replacement_len, array->v + idx + len, | 
|  | (array->nr - idx - len + 1) * sizeof(char *)); | 
|  | array->nr += replacement_len - len; | 
|  | for (size_t i = 0; i < replacement_len; i++) | 
|  | array->v[idx + i] = xstrdup(replacement[i]); | 
|  | } | 
|  |  | 
|  | const char *strvec_replace(struct strvec *array, size_t idx, const char *replacement) | 
|  | { | 
|  | char *to_free; | 
|  | if (idx >= array->nr) | 
|  | BUG("index outside of array boundary"); | 
|  | to_free = (char *) array->v[idx]; | 
|  | array->v[idx] = xstrdup(replacement); | 
|  | free(to_free); | 
|  | return array->v[idx]; | 
|  | } | 
|  |  | 
|  | void strvec_remove(struct strvec *array, size_t idx) | 
|  | { | 
|  | if (idx >= array->nr) | 
|  | BUG("index outside of array boundary"); | 
|  | free((char *)array->v[idx]); | 
|  | memmove(array->v + idx, array->v + idx + 1, (array->nr - idx) * sizeof(char *)); | 
|  | array->nr--; | 
|  | } | 
|  |  | 
|  | void strvec_pop(struct strvec *array) | 
|  | { | 
|  | if (!array->nr) | 
|  | return; | 
|  | free((char *)array->v[array->nr - 1]); | 
|  | array->v[array->nr - 1] = NULL; | 
|  | array->nr--; | 
|  | } | 
|  |  | 
|  | void strvec_split(struct strvec *array, const char *to_split) | 
|  | { | 
|  | while (isspace(*to_split)) | 
|  | to_split++; | 
|  | for (;;) { | 
|  | const char *p = to_split; | 
|  |  | 
|  | if (!*p) | 
|  | break; | 
|  |  | 
|  | while (*p && !isspace(*p)) | 
|  | p++; | 
|  | strvec_push_nodup(array, xstrndup(to_split, p - to_split)); | 
|  |  | 
|  | while (isspace(*p)) | 
|  | p++; | 
|  | to_split = p; | 
|  | } | 
|  | } | 
|  |  | 
|  | void strvec_clear(struct strvec *array) | 
|  | { | 
|  | if (array->v != empty_strvec) { | 
|  | for (size_t i = 0; i < array->nr; i++) | 
|  | free((char *)array->v[i]); | 
|  | free(array->v); | 
|  | } | 
|  | strvec_init(array); | 
|  | } | 
|  |  | 
|  | const char **strvec_detach(struct strvec *array) | 
|  | { | 
|  | if (array->v == empty_strvec) | 
|  | return xcalloc(1, sizeof(const char *)); | 
|  | else { | 
|  | const char **ret = array->v; | 
|  | strvec_init(array); | 
|  | return ret; | 
|  | } | 
|  | } |