From: Ben Pfaff Date: Fri, 23 Mar 2018 22:46:58 +0000 (-0700) Subject: json: Avoid extra memory allocation and string copy parsing object members. X-Git-Tag: v2.12.3~1969 X-Git-Url: https://git.proxmox.com/?a=commitdiff_plain;h=828129d9271da18c2742648aa910e1fe399bb8db;p=mirror_ovs.git json: Avoid extra memory allocation and string copy parsing object members. Until now, every time the JSON parser added an object member, it made an extra copy of the member name and then freed the original copy. This is wasteful, so this commit eliminates the extra copy. Signed-off-by: Ben Pfaff Reviewed-by: Yifeng Sun --- diff --git a/include/openvswitch/json.h b/include/openvswitch/json.h index 61b9a02cf..bcf6a2782 100644 --- a/include/openvswitch/json.h +++ b/include/openvswitch/json.h @@ -91,6 +91,7 @@ struct json *json_array_create_3(struct json *, struct json *, struct json *); struct json *json_object_create(void); void json_object_put(struct json *, const char *name, struct json *value); +void json_object_put_nocopy(struct json *, char *name, struct json *value); void json_object_put_string(struct json *, const char *name, const char *value); void json_object_put_format(struct json *, diff --git a/include/openvswitch/shash.h b/include/openvswitch/shash.h index afc482364..c249e13e1 100644 --- a/include/openvswitch/shash.h +++ b/include/openvswitch/shash.h @@ -62,6 +62,7 @@ struct shash_node *shash_add_nocopy(struct shash *, char *, const void *); bool shash_add_once(struct shash *, const char *, const void *); void shash_add_assert(struct shash *, const char *, const void *); void *shash_replace(struct shash *, const char *, const void *data); +void *shash_replace_nocopy(struct shash *, char *name, const void *data); void shash_delete(struct shash *, struct shash_node *); char *shash_steal(struct shash *, struct shash_node *); struct shash_node *shash_find(const struct shash *, const char *); diff --git a/lib/json.c b/lib/json.c index 07ca87b21..603fd1df8 100644 --- a/lib/json.c +++ b/lib/json.c @@ -279,6 +279,12 @@ json_object_put(struct json *json, const char *name, struct json *value) json_destroy(shash_replace(json->u.object, name, value)); } +void +json_object_put_nocopy(struct json *json, char *name, struct json *value) +{ + json_destroy(shash_replace_nocopy(json->u.object, name, value)); +} + void json_object_put_string(struct json *json, const char *name, const char *value) { @@ -1217,8 +1223,7 @@ json_parser_put_value(struct json_parser *p, struct json *value) { struct json_parser_node *node = json_parser_top(p); if (node->json->type == JSON_OBJECT) { - json_object_put(node->json, p->member_name, value); - free(p->member_name); + json_object_put_nocopy(node->json, p->member_name, value); p->member_name = NULL; } else if (node->json->type == JSON_ARRAY) { json_array_add(node->json, value); diff --git a/lib/shash.c b/lib/shash.c index 318a30ffc..a8433629a 100644 --- a/lib/shash.c +++ b/lib/shash.c @@ -166,6 +166,29 @@ shash_replace(struct shash *sh, const char *name, const void *data) } } +/* Searches for 'name' in 'sh'. If it does not already exist, adds it along + * with 'data' and returns NULL. If it does already exist, replaces its data + * by 'data' and returns the data that it formerly contained. + * + * Takes ownership of 'name'. */ +void * +shash_replace_nocopy(struct shash *sh, char *name, const void *data) +{ + size_t hash = hash_name(name); + struct shash_node *node; + + node = shash_find__(sh, name, strlen(name), hash); + if (!node) { + shash_add_nocopy__(sh, name, data, hash); + return NULL; + } else { + free(name); + void *old_data = node->data; + node->data = CONST_CAST(void *, data); + return old_data; + } +} + /* Deletes 'node' from 'sh' and frees the node's name. The caller is still * responsible for freeing the node's data, if necessary. */ void