From 6ff901d7b016c3ad49c7dedcebbbe6420d1e2d93 Mon Sep 17 00:00:00 2001 From: Rusty Russell Date: Wed, 4 Jul 2018 11:16:07 +0930 Subject: [PATCH] params: simplify lifetimes of params. @wythe points out we don't need to keep the around now param_is_set() is removed. We can in fact go further and avoid marshalling them into temporary objects at the caller altogether. This means internally we have an array of struct param, rather than an array of 'struct param *', which causes most of the noise in this patch. Signed-off-by: Rusty Russell --- lightningd/params.c | 147 ++++++++++++++++------------------- lightningd/params.h | 49 ++++++------ lightningd/test/run-params.c | 20 +++-- 3 files changed, 105 insertions(+), 111 deletions(-) diff --git a/lightningd/params.c b/lightningd/params.c index c999b06d05f5..483fdbfa3539 100644 --- a/lightningd/params.c +++ b/lightningd/params.c @@ -16,16 +16,20 @@ struct param { size_t argsize; }; -struct param *param_add_(const tal_t *ctx, - const char *name, param_cb cb, void *arg, - size_t argsize) +static void param_add(struct param **params, + const char *name, param_cb cb, void *arg, + const tal_t *ctx, size_t argsize) { #if DEVELOPER assert(name); assert(cb); assert(arg); #endif - struct param *last = tal(tmpctx, struct param); + struct param *last; + + tal_resize(params, tal_count(*params) + 1); + last = &(*params)[tal_count(*params) - 1]; + last->ctx = ctx; last->is_set = false; last->name = name; @@ -35,22 +39,6 @@ struct param *param_add_(const tal_t *ctx, /* Non-NULL means we are supposed to allocate iff found */ if (last->ctx) *(void **)last->arg = NULL; - return last; -} - -struct param *param_opt_add_(const tal_t *ctx, const char *name, - const jsmntok_t **tok) -{ - struct param *last = tal(tmpctx, struct param); - assert(ctx); - last->ctx = ctx; - last->is_set = false; - last->name = name; - last->cb = (param_cb)json_tok_tok; - last->arg = tok; - last->argsize = sizeof(*tok); - *tok = NULL; - return last; } struct fail_format { @@ -109,17 +97,17 @@ static bool make_callback(struct command *cmd, return true; } -static struct param **post_check(struct command *cmd, struct param **params) +static struct param *post_check(struct command *cmd, struct param *params) { - struct param **first = params; - struct param **last = first + tal_count(params); + struct param *first = params; + struct param *last = first + tal_count(params); /* Make sure required params were provided. */ - while (first != last && (*first)->argsize == 0) { - if (!(*first)->is_set) { + while (first != last && first->argsize == 0) { + if (!first->is_set) { command_fail(cmd, JSONRPC2_INVALID_PARAMS, "missing required parameter: '%s'", - (*first)->name); + first->name); return NULL; } first++; @@ -127,19 +115,19 @@ static struct param **post_check(struct command *cmd, struct param **params) return params; } -static struct param **parse_by_position(struct command *cmd, - struct param **params, - const char *buffer, - const jsmntok_t tokens[]) +static bool parse_by_position(struct command *cmd, + struct param *params, + const char *buffer, + const jsmntok_t tokens[]) { const jsmntok_t *tok = tokens + 1; const jsmntok_t *end = json_next(tokens); - struct param **first = params; - struct param **last = first + tal_count(params); + struct param *first = params; + struct param *last = first + tal_count(params); while (first != last && tok != end) { if (!json_tok_is_null(buffer, tok)) - if (!make_callback(cmd, *first, buffer, tok)) + if (!make_callback(cmd, first, buffer, tok)) return NULL; tok = json_next(tok); first++; @@ -151,31 +139,31 @@ static struct param **parse_by_position(struct command *cmd, "too many parameters:" " got %u, expected %zu", tokens->size, tal_count(params)); - return NULL; + return false; } return post_check(cmd, params); } -static struct param *find_param(struct param **params, const char *start, +static struct param *find_param(struct param *params, const char *start, size_t n) { - struct param **first = params; - struct param **last = first + tal_count(params); + struct param *first = params; + struct param *last = first + tal_count(params); while (first != last) { - if (strncmp((*first)->name, start, n) == 0) - if (strlen((*first)->name) == n) - return *first; + if (strncmp(first->name, start, n) == 0) + if (strlen(first->name) == n) + return first; first++; } return NULL; } -static struct param **parse_by_name(struct command *cmd, - struct param **params, - const char *buffer, - const jsmntok_t tokens[]) +static bool parse_by_name(struct command *cmd, + struct param *params, + const char *buffer, + const jsmntok_t tokens[]) { const jsmntok_t *first = tokens + 1; const jsmntok_t *last = json_next(tokens); @@ -188,36 +176,36 @@ static struct param **parse_by_name(struct command *cmd, "unknown parameter: '%.*s'", first->end - first->start, buffer + first->start); - return NULL; + return false; } if (p->is_set) { command_fail(cmd, JSONRPC2_INVALID_PARAMS, "duplicate json names: '%s'", p->name); - return NULL; + return false; } if (!make_callback(cmd, p, buffer, first + 1)) - return NULL; + return false; first = json_next(first + 1); } return post_check(cmd, params); } #if DEVELOPER -static int comp_by_name(struct param *const *a, struct param *const *b, +static int comp_by_name(const struct param *a, const struct param *b, void *unused) { - return strcmp((*a)->name, (*b)->name); + return strcmp(a->name, b->name); } -static int comp_by_arg(struct param *const *a, struct param *const *b, +static int comp_by_arg(const struct param *a, const struct param *b, void *unused) { /* size_t could be larger than int: don't turn a 4bn difference into 0 */ - if ((*a)->arg > (*b)->arg) + if (a->arg > b->arg) return 1; - else if ((*a)->arg < (*b)->arg) + else if (a->arg < b->arg) return -1; return 0; } @@ -225,10 +213,10 @@ static int comp_by_arg(struct param *const *a, struct param *const *b, /* This comparator is a bit different, but works well. * Return 0 if @a is optional and @b is required. Otherwise return 1. */ -static int comp_req_order(struct param *const *a, struct param *const *b, +static int comp_req_order(const struct param *a, const struct param *b, void *unused) { - if ((*a)->argsize != 0 && (*b)->argsize == 0) + if (a->argsize != 0 && b->argsize == 0) return 0; return 1; } @@ -237,12 +225,12 @@ static int comp_req_order(struct param *const *a, struct param *const *b, * Make sure 2 sequential items in @params are not equal (based on * provided comparator). */ -static void check_distinct(struct param **params, - int (*compar) (struct param *const *a, - struct param *const *b, void *unused)) +static void check_distinct(struct param *params, + int (*compar) (const struct param *a, + const struct param *b, void *unused)) { - struct param **first = params; - struct param **last = first + tal_count(params); + struct param *first = params; + struct param *last = first + tal_count(params); first++; while (first != last) { assert(compar(first - 1, first, NULL) != 0); @@ -250,9 +238,9 @@ static void check_distinct(struct param **params, } } -static void check_unique(struct param **copy, - int (*compar) (struct param *const *a, - struct param *const *b, void *unused)) +static void check_unique(struct param *copy, + int (*compar) (const struct param *a, + const struct param *b, void *unused)) { asort(copy, tal_count(copy), compar, NULL); check_distinct(copy, compar); @@ -261,7 +249,7 @@ static void check_unique(struct param **copy, /* * Verify consistent internal state. */ -static void check_params(struct param **params) +static void check_params(struct param *params) { if (tal_count(params) < 2) return; @@ -270,8 +258,8 @@ static void check_params(struct param **params) check_distinct(params, comp_req_order); /* duplicate so we can sort */ - struct param **copy = tal_dup_arr(params, struct param *, - params, tal_count(params), 0); + struct param *copy = tal_dup_arr(params, struct param, + params, tal_count(params), 0); /* check for repeated names and args */ check_unique(copy, comp_by_name); @@ -281,10 +269,10 @@ static void check_params(struct param **params) } #endif -static struct param **param_parse_arr(struct command *cmd, - const char *buffer, - const jsmntok_t tokens[], - struct param **params) +static bool param_parse_arr(struct command *cmd, + const char *buffer, + const jsmntok_t tokens[], + struct param *params) { #if DEVELOPER check_params(params); @@ -296,20 +284,23 @@ static struct param **param_parse_arr(struct command *cmd, command_fail(cmd, JSONRPC2_INVALID_PARAMS, "Expected array or object for params"); - return NULL; + return false; } -struct param **param_parse(struct command *cmd, const char *buffer, - const jsmntok_t tokens[], ...) +bool param_parse(struct command *cmd, const char *buffer, + const jsmntok_t tokens[], ...) { - struct param *def; - struct param **params = tal_arr(cmd, struct param *, 0); + struct param *params = tal_arr(tmpctx, struct param, 0); + const char *name; va_list ap; + va_start(ap, tokens); - while ((def = va_arg(ap, struct param *)) != NULL) { - tal_steal(params, def); - tal_resize(¶ms, tal_count(params) + 1); - params[tal_count(params) - 1] = def; + while ((name = va_arg(ap, const char *)) != NULL) { + param_cb cb = va_arg(ap, param_cb); + void *arg = va_arg(ap, void *); + const tal_t *ctx = va_arg(ap, const tal_t *); + size_t argsize = va_arg(ap, size_t); + param_add(¶ms, name, cb, arg, ctx, argsize); } va_end(ap); diff --git a/lightningd/params.h b/lightningd/params.h index b8f26f9d826c..9312018f5823 100644 --- a/lightningd/params.h +++ b/lightningd/params.h @@ -21,7 +21,7 @@ struct param; return; At this point in the code you can be assured the json tokens were successfully - parsed. If not, param_parse() returned NULL, having already called + parsed. If not, param_parse() returns false, having already called command_fail() with a descriptive error message. The data section of the json result contains the offending parameter and its value. @@ -40,8 +40,8 @@ struct param; Otherwise a generic message is provided. */ -struct param **param_parse(struct command *cmd, const char *buffer, - const jsmntok_t params[], ...); +bool param_parse(struct command *cmd, const char *buffer, + const jsmntok_t params[], ...); /* * This callback provided must follow this signature; e.g., @@ -59,28 +59,33 @@ typedef bool(*param_cb)(const char *buffer, const jsmntok_t *tok, void *arg); * * Returns an opaque pointer that can be later used in param_is_set(). */ -#define param_req(name, cb, arg) \ - param_add_(NULL, name"", \ - typesafe_cb_preargs(bool, void *, \ - (cb), (arg), \ - const char *, \ - const jsmntok_t *), \ - (arg), 0) +#define param_req(name, cb, arg) \ + name"", \ + typesafe_cb_preargs(bool, void *, \ + (cb), (arg), \ + const char *, \ + const jsmntok_t *), \ + (arg), NULL, 0 /* * Same as above but for optional parameters. */ -#define param_opt(ctx, name, cb, arg) \ - param_add_(ctx, name"", \ - typesafe_cb_preargs(bool, void *, \ - (cb), *(arg), \ - const char *, \ - const jsmntok_t *), \ - (arg), sizeof(**arg)) -struct param * param_add_(const tal_t *ctx, const char *name, param_cb cb, void *arg, size_t argsize); +#define param_opt(ctx, name, cb, arg) \ + name"", \ + typesafe_cb_preargs(bool, void *, \ + (cb), *(arg), \ + const char *, \ + const jsmntok_t *), \ + (arg), (ctx), sizeof(**arg) -#define param_opt_tok(ctx, name, arg) \ - param_opt_add_(ctx, name"", arg) - -struct param *param_opt_add_(const tal_t *ctx, const char *name, const jsmntok_t **tok); +/* + * For when you want an optional raw token. + * + * Note: weird sizeof() does type check that arg really is a (const) jsmntok_t **. + */ +#define param_opt_tok(ctx, name, arg) \ + name"", \ + json_tok_tok, \ + (arg) + 0*sizeof(*(arg) == (jsmntok_t *)NULL), \ + (ctx), sizeof(const jsmntok_t *) #endif /* LIGHTNING_LIGHTNINGD_PARAMS_H */ diff --git a/lightningd/test/run-params.c b/lightningd/test/run-params.c index ec1dd215e6c6..54ca06fec00a 100644 --- a/lightningd/test/run-params.c +++ b/lightningd/test/run-params.c @@ -353,15 +353,13 @@ static void add_members(struct param **params, char *name = tal_fmt(tmpctx, "%i", i); json_add_num(obj, name, i); json_add_num(arr, NULL, i); - /* Since name is not a literal, we do this manually. */ - params[i] = param_add_(NULL, name, - typesafe_cb_preargs(bool, void *, - json_tok_number, - &ints[i], - const char *, - const jsmntok_t *), - &ints[i], 0); - tal_steal(params, params[i]); + param_add(params, name, + typesafe_cb_preargs(bool, void *, + json_tok_number, + &ints[i], + const char *, + const jsmntok_t *), + &ints[i], NULL, 0); } } @@ -371,14 +369,14 @@ static void add_members(struct param **params, */ static void five_hundred_params(void) { - struct param **params = tal_arr(NULL, struct param *, 500); + struct param *params = tal_arr(NULL, struct param, 0); unsigned int *ints = tal_arr(params, unsigned int, 500); struct json_result *obj = new_json_result(params); struct json_result *arr = new_json_result(params); json_object_start(obj, NULL); json_array_start(arr, NULL); - add_members(params, obj, arr, ints); + add_members(¶ms, obj, arr, ints); json_object_end(obj); json_array_end(arr);