keyval: New keyval_parse()
keyval_parse() parses KEY=VALUE,... into a QDict. Works like qemu_opts_parse(), except: * Returns a QDict instead of a QemuOpts (d'oh). * Supports nesting, unlike QemuOpts: a KEY is split into key fragments at '.' (dotted key convention; the block layer does something similar on top of QemuOpts). The key fragments are QDict keys, and the last one's value is updated to VALUE. * Each key fragment may be up to 127 bytes long. qemu_opts_parse() limits the entire key to 127 bytes. * Overlong key fragments are rejected. qemu_opts_parse() silently truncates them. * Empty key fragments are rejected. qemu_opts_parse() happily accepts empty keys. * It does not store the returned value. qemu_opts_parse() stores it in the QemuOptsList. * It does not treat parameter "id" specially. qemu_opts_parse() ignores all but the first "id", and fails when its value isn't id_wellformed(), or duplicate (a QemuOpts with the same ID is already stored). It also screws up when a value contains ",id=". * Implied value is not supported. qemu_opts_parse() desugars "foo" to "foo=on", and "nofoo" to "foo=off". * An implied key's value can't be empty, and can't contain ','. I intend to grow this into a saner replacement for QemuOpts. It'll take time, though. Note: keyval_parse() provides no way to do lists, and its key syntax is incompatible with the __RFQDN_ prefix convention for downstream extensions, because it blindly splits at '.', even in __RFQDN_. Both issues will be addressed later in the series. Signed-off-by: Markus Armbruster <armbru@redhat.com> Message-Id: <1488317230-26248-4-git-send-email-armbru@redhat.com>
This commit is contained in:
		| @@ -141,4 +141,7 @@ void qemu_opts_print_help(QemuOptsList *list); | |||||||
| void qemu_opts_free(QemuOptsList *list); | void qemu_opts_free(QemuOptsList *list); | ||||||
| QemuOptsList *qemu_opts_append(QemuOptsList *dst, QemuOptsList *list); | QemuOptsList *qemu_opts_append(QemuOptsList *dst, QemuOptsList *list); | ||||||
|  |  | ||||||
|  | QDict *keyval_parse(const char *params, const char *implied_key, | ||||||
|  |                     Error **errp); | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								tests/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								tests/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -47,6 +47,7 @@ test-io-channel-file.txt | |||||||
| test-io-channel-socket | test-io-channel-socket | ||||||
| test-io-channel-tls | test-io-channel-tls | ||||||
| test-io-task | test-io-task | ||||||
|  | test-keyval | ||||||
| test-logging | test-logging | ||||||
| test-mul64 | test-mul64 | ||||||
| test-opts-visitor | test-opts-visitor | ||||||
|   | |||||||
| @@ -94,6 +94,8 @@ check-unit-y += tests/check-qom-proplist$(EXESUF) | |||||||
| gcov-files-check-qom-proplist-y = qom/object.c | gcov-files-check-qom-proplist-y = qom/object.c | ||||||
| check-unit-y += tests/test-qemu-opts$(EXESUF) | check-unit-y += tests/test-qemu-opts$(EXESUF) | ||||||
| gcov-files-test-qemu-opts-y = util/qemu-option.c | gcov-files-test-qemu-opts-y = util/qemu-option.c | ||||||
|  | check-unit-y += tests/test-keyval$(EXESUF) | ||||||
|  | gcov-files-test-keyval-y = util/keyval.c | ||||||
| check-unit-y += tests/test-write-threshold$(EXESUF) | check-unit-y += tests/test-write-threshold$(EXESUF) | ||||||
| gcov-files-test-write-threshold-y = block/write-threshold.c | gcov-files-test-write-threshold-y = block/write-threshold.c | ||||||
| check-unit-y += tests/test-crypto-hash$(EXESUF) | check-unit-y += tests/test-crypto-hash$(EXESUF) | ||||||
| @@ -720,6 +722,7 @@ tests/vhost-user-test$(EXESUF): tests/vhost-user-test.o $(test-util-obj-y) \ | |||||||
| 	$(chardev-obj-y) | 	$(chardev-obj-y) | ||||||
| tests/qemu-iotests/socket_scm_helper$(EXESUF): tests/qemu-iotests/socket_scm_helper.o | tests/qemu-iotests/socket_scm_helper$(EXESUF): tests/qemu-iotests/socket_scm_helper.o | ||||||
| tests/test-qemu-opts$(EXESUF): tests/test-qemu-opts.o $(test-util-obj-y) | tests/test-qemu-opts$(EXESUF): tests/test-qemu-opts.o $(test-util-obj-y) | ||||||
|  | tests/test-keyval$(EXESUF): tests/test-keyval.o $(test-util-obj-y) | ||||||
| tests/test-write-threshold$(EXESUF): tests/test-write-threshold.o $(test-block-obj-y) | tests/test-write-threshold$(EXESUF): tests/test-write-threshold.o $(test-block-obj-y) | ||||||
| tests/test-netfilter$(EXESUF): tests/test-netfilter.o $(qtest-obj-y) | tests/test-netfilter$(EXESUF): tests/test-netfilter.o $(qtest-obj-y) | ||||||
| tests/test-filter-mirror$(EXESUF): tests/test-filter-mirror.o $(qtest-obj-y) | tests/test-filter-mirror$(EXESUF): tests/test-filter-mirror.o $(qtest-obj-y) | ||||||
|   | |||||||
							
								
								
									
										180
									
								
								tests/test-keyval.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										180
									
								
								tests/test-keyval.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,180 @@ | |||||||
|  | /* | ||||||
|  |  * Unit tests for parsing of KEY=VALUE,... strings | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2017 Red Hat Inc. | ||||||
|  |  * | ||||||
|  |  * Authors: | ||||||
|  |  *  Markus Armbruster <armbru@redhat.com>, | ||||||
|  |  * | ||||||
|  |  * This work is licensed under the terms of the GNU GPL, version 2 or later. | ||||||
|  |  * See the COPYING file in the top-level directory. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "qemu/osdep.h" | ||||||
|  | #include "qapi/error.h" | ||||||
|  | #include "qemu/option.h" | ||||||
|  |  | ||||||
|  | static void test_keyval_parse(void) | ||||||
|  | { | ||||||
|  |     Error *err = NULL; | ||||||
|  |     QDict *qdict, *sub_qdict; | ||||||
|  |     char long_key[129]; | ||||||
|  |     char *params; | ||||||
|  |  | ||||||
|  |     /* Nothing */ | ||||||
|  |     qdict = keyval_parse("", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 0); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Empty key (qemu_opts_parse() accepts this) */ | ||||||
|  |     qdict = keyval_parse("=val", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Empty key fragment */ | ||||||
|  |     qdict = keyval_parse(".", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |     qdict = keyval_parse("key.", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Overlong key */ | ||||||
|  |     memset(long_key, 'a', 127); | ||||||
|  |     long_key[127] = 'z'; | ||||||
|  |     long_key[128] = 0; | ||||||
|  |     params = g_strdup_printf("k.%s=v", long_key); | ||||||
|  |     qdict = keyval_parse(params + 2, NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Overlong key fragment */ | ||||||
|  |     qdict = keyval_parse(params, NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |     g_free(params); | ||||||
|  |  | ||||||
|  |     /* Long key (qemu_opts_parse() accepts and truncates silently) */ | ||||||
|  |     params = g_strdup_printf("k.%s=v", long_key + 1); | ||||||
|  |     qdict = keyval_parse(params + 2, NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, long_key + 1), ==, "v"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Long key fragment */ | ||||||
|  |     qdict = keyval_parse(params, NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     sub_qdict = qdict_get_qdict(qdict, "k"); | ||||||
|  |     g_assert(sub_qdict); | ||||||
|  |     g_assert_cmpuint(qdict_size(sub_qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(sub_qdict, long_key + 1), ==, "v"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |     g_free(params); | ||||||
|  |  | ||||||
|  |     /* Multiple keys, last one wins */ | ||||||
|  |     qdict = keyval_parse("a=1,b=2,,x,a=3", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 2); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "a"), ==, "3"); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "b"), ==, "2,x"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Even when it doesn't in qemu_opts_parse() */ | ||||||
|  |     qdict = keyval_parse("id=foo,id=bar", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "id"), ==, "bar"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Dotted keys */ | ||||||
|  |     qdict = keyval_parse("a.b.c=1,a.b.c=2,d=3", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 2); | ||||||
|  |     sub_qdict = qdict_get_qdict(qdict, "a"); | ||||||
|  |     g_assert(sub_qdict); | ||||||
|  |     g_assert_cmpuint(qdict_size(sub_qdict), ==, 1); | ||||||
|  |     sub_qdict = qdict_get_qdict(sub_qdict, "b"); | ||||||
|  |     g_assert(sub_qdict); | ||||||
|  |     g_assert_cmpuint(qdict_size(sub_qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(sub_qdict, "c"), ==, "2"); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "d"), ==, "3"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Inconsistent dotted keys */ | ||||||
|  |     qdict = keyval_parse("a.b=1,a=2", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |     qdict = keyval_parse("a.b=1,a.b.c=2", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Trailing comma is ignored */ | ||||||
|  |     qdict = keyval_parse("x=y,", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "x"), ==, "y"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Except when it isn't */ | ||||||
|  |     qdict = keyval_parse(",", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Value containing ,id= not misinterpreted as qemu_opts_parse() does */ | ||||||
|  |     qdict = keyval_parse("x=,,id=bar", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "x"), ==, ",id=bar"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Anti-social ID is left to caller (qemu_opts_parse() rejects it) */ | ||||||
|  |     qdict = keyval_parse("id=666", NULL, &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "id"), ==, "666"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Implied value not supported (unlike qemu_opts_parse()) */ | ||||||
|  |     qdict = keyval_parse("an,noaus,noaus=", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Implied value, key "no" (qemu_opts_parse(): negated empty key) */ | ||||||
|  |     qdict = keyval_parse("no", NULL, &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Implied key */ | ||||||
|  |     qdict = keyval_parse("an,aus=off,noaus=", "implied", &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 3); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "implied"), ==, "an"); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "aus"), ==, "off"); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(qdict, "noaus"), ==, ""); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Implied dotted key */ | ||||||
|  |     qdict = keyval_parse("val", "eins.zwei", &error_abort); | ||||||
|  |     g_assert_cmpuint(qdict_size(qdict), ==, 1); | ||||||
|  |     sub_qdict = qdict_get_qdict(qdict, "eins"); | ||||||
|  |     g_assert(sub_qdict); | ||||||
|  |     g_assert_cmpuint(qdict_size(sub_qdict), ==, 1); | ||||||
|  |     g_assert_cmpstr(qdict_get_try_str(sub_qdict, "zwei"), ==, "val"); | ||||||
|  |     QDECREF(qdict); | ||||||
|  |  | ||||||
|  |     /* Implied key with empty value (qemu_opts_parse() accepts this) */ | ||||||
|  |     qdict = keyval_parse(",", "implied", &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Likewise (qemu_opts_parse(): implied key with comma value) */ | ||||||
|  |     qdict = keyval_parse(",,,a=1", "implied", &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  |  | ||||||
|  |     /* Empty key is not an implied key */ | ||||||
|  |     qdict = keyval_parse("=val", "implied", &err); | ||||||
|  |     error_free_or_abort(&err); | ||||||
|  |     g_assert(!qdict); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int main(int argc, char *argv[]) | ||||||
|  | { | ||||||
|  |     g_test_init(&argc, &argv, NULL); | ||||||
|  |     g_test_add_func("/keyval/keyval_parse", test_keyval_parse); | ||||||
|  |     g_test_run(); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
| @@ -24,6 +24,7 @@ util-obj-y += error.o qemu-error.o | |||||||
| util-obj-y += id.o | util-obj-y += id.o | ||||||
| util-obj-y += iov.o qemu-config.o qemu-sockets.o uri.o notify.o | util-obj-y += iov.o qemu-config.o qemu-sockets.o uri.o notify.o | ||||||
| util-obj-y += qemu-option.o qemu-progress.o | util-obj-y += qemu-option.o qemu-progress.o | ||||||
|  | util-obj-y += keyval.o | ||||||
| util-obj-y += hexdump.o | util-obj-y += hexdump.o | ||||||
| util-obj-y += crc32c.o | util-obj-y += crc32c.o | ||||||
| util-obj-y += uuid.o | util-obj-y += uuid.o | ||||||
|   | |||||||
							
								
								
									
										231
									
								
								util/keyval.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										231
									
								
								util/keyval.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,231 @@ | |||||||
|  | /* | ||||||
|  |  * Parsing KEY=VALUE,... strings | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2017 Red Hat Inc. | ||||||
|  |  * | ||||||
|  |  * Authors: | ||||||
|  |  *  Markus Armbruster <armbru@redhat.com>, | ||||||
|  |  * | ||||||
|  |  * This work is licensed under the terms of the GNU GPL, version 2 or later. | ||||||
|  |  * See the COPYING file in the top-level directory. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * KEY=VALUE,... syntax: | ||||||
|  |  * | ||||||
|  |  *   key-vals     = [ key-val { ',' key-val } [ ',' ] ] | ||||||
|  |  *   key-val      = key '=' val | ||||||
|  |  *   key          = key-fragment { '.' key-fragment } | ||||||
|  |  *   key-fragment = / [^=,.]* / | ||||||
|  |  *   val          = { / [^,]* / | ',,' } | ||||||
|  |  * | ||||||
|  |  * Semantics defined by reduction to JSON: | ||||||
|  |  * | ||||||
|  |  *   key-vals defines a tree of objects rooted at R | ||||||
|  |  *   where for each key-val = key-fragment . ... = val in key-vals | ||||||
|  |  *       R op key-fragment op ... = val' | ||||||
|  |  *       where (left-associative) op is member reference L.key-fragment | ||||||
|  |  *             val' is val with ',,' replaced by ',' | ||||||
|  |  *   and only R may be empty. | ||||||
|  |  * | ||||||
|  |  *   Duplicate keys are permitted; all but the last one are ignored. | ||||||
|  |  * | ||||||
|  |  *   The equations must have a solution.  Counter-example: a.b=1,a=2 | ||||||
|  |  *   doesn't have one, because R.a must be an object to satisfy a.b=1 | ||||||
|  |  *   and a string to satisfy a=2. | ||||||
|  |  * | ||||||
|  |  * The length of any key-fragment must be between 1 and 127. | ||||||
|  |  * | ||||||
|  |  * Design flaw: there is no way to denote an empty non-root object. | ||||||
|  |  * While interpreting "key absent" as empty object seems natural | ||||||
|  |  * (removing a key-val from the input string removes the member when | ||||||
|  |  * there are more, so why not when it's the last), it doesn't work: | ||||||
|  |  * "key absent" already means "optional object absent", which isn't | ||||||
|  |  * the same as "empty object present". | ||||||
|  |  * | ||||||
|  |  * Additional syntax for use with an implied key: | ||||||
|  |  * | ||||||
|  |  *   key-vals-ik  = val-no-key [ ',' key-vals ] | ||||||
|  |  *   val-no-key   = / [^=,]* / | ||||||
|  |  * | ||||||
|  |  * where no-key is syntactic sugar for implied-key=val-no-key. | ||||||
|  |  * | ||||||
|  |  * TODO support lists | ||||||
|  |  * TODO support key-fragment with __RFQDN_ prefix (downstream extensions) | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include "qemu/osdep.h" | ||||||
|  | #include "qapi/error.h" | ||||||
|  | #include "qapi/qmp/qstring.h" | ||||||
|  | #include "qemu/option.h" | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Ensure @cur maps @key_in_cur the right way. | ||||||
|  |  * If @value is null, it needs to map to a QDict, else to this | ||||||
|  |  * QString. | ||||||
|  |  * If @cur doesn't have @key_in_cur, put an empty QDict or @value, | ||||||
|  |  * respectively. | ||||||
|  |  * Else, if it needs to map to a QDict, and already does, do nothing. | ||||||
|  |  * Else, if it needs to map to this QString, and already maps to a | ||||||
|  |  * QString, replace it by @value. | ||||||
|  |  * Else, fail because we have conflicting needs on how to map | ||||||
|  |  * @key_in_cur. | ||||||
|  |  * In any case, take over the reference to @value, i.e. if the caller | ||||||
|  |  * wants to hold on to a reference, it needs to QINCREF(). | ||||||
|  |  * Use @key up to @key_cursor to identify the key in error messages. | ||||||
|  |  * On success, return the mapped value. | ||||||
|  |  * On failure, store an error through @errp and return NULL. | ||||||
|  |  */ | ||||||
|  | static QObject *keyval_parse_put(QDict *cur, | ||||||
|  |                                  const char *key_in_cur, QString *value, | ||||||
|  |                                  const char *key, const char *key_cursor, | ||||||
|  |                                  Error **errp) | ||||||
|  | { | ||||||
|  |     QObject *old, *new; | ||||||
|  |  | ||||||
|  |     old = qdict_get(cur, key_in_cur); | ||||||
|  |     if (old) { | ||||||
|  |         if (qobject_type(old) != (value ? QTYPE_QSTRING : QTYPE_QDICT)) { | ||||||
|  |             error_setg(errp, "Parameters '%.*s.*' used inconsistently", | ||||||
|  |                        (int)(key_cursor - key), key); | ||||||
|  |             QDECREF(value); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |         if (!value) { | ||||||
|  |             return old;         /* already QDict, do nothing */ | ||||||
|  |         } | ||||||
|  |         new = QOBJECT(value);   /* replacement */ | ||||||
|  |     } else { | ||||||
|  |         new = value ? QOBJECT(value) : QOBJECT(qdict_new()); | ||||||
|  |     } | ||||||
|  |     qdict_put_obj(cur, key_in_cur, new); | ||||||
|  |     return new; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Parse one KEY=VALUE from @params, store result in @qdict. | ||||||
|  |  * The first fragment of KEY applies to @qdict.  Subsequent fragments | ||||||
|  |  * apply to nested QDicts, which are created on demand.  @implied_key | ||||||
|  |  * is as in keyval_parse(). | ||||||
|  |  * On success, return a pointer to the next KEY=VALUE, or else to '\0'. | ||||||
|  |  * On failure, return NULL. | ||||||
|  |  */ | ||||||
|  | static const char *keyval_parse_one(QDict *qdict, const char *params, | ||||||
|  |                                     const char *implied_key, | ||||||
|  |                                     Error **errp) | ||||||
|  | { | ||||||
|  |     const char *key, *key_end, *s; | ||||||
|  |     size_t len; | ||||||
|  |     char key_in_cur[128]; | ||||||
|  |     QDict *cur; | ||||||
|  |     QObject *next; | ||||||
|  |     QString *val; | ||||||
|  |  | ||||||
|  |     key = params; | ||||||
|  |     len = strcspn(params, "=,"); | ||||||
|  |     if (implied_key && len && key[len] != '=') { | ||||||
|  |         /* Desugar implied key */ | ||||||
|  |         key = implied_key; | ||||||
|  |         len = strlen(implied_key); | ||||||
|  |     } | ||||||
|  |     key_end = key + len; | ||||||
|  |  | ||||||
|  |     /* | ||||||
|  |      * Loop over key fragments: @s points to current fragment, it | ||||||
|  |      * applies to @cur.  @key_in_cur[] holds the previous fragment. | ||||||
|  |      */ | ||||||
|  |     cur = qdict; | ||||||
|  |     s = key; | ||||||
|  |     for (;;) { | ||||||
|  |         for (len = 0; s + len < key_end && s[len] != '.'; len++) { | ||||||
|  |         } | ||||||
|  |         if (!len) { | ||||||
|  |             assert(key != implied_key); | ||||||
|  |             error_setg(errp, "Invalid parameter '%.*s'", | ||||||
|  |                        (int)(key_end - key), key); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |         if (len >= sizeof(key_in_cur)) { | ||||||
|  |             assert(key != implied_key); | ||||||
|  |             error_setg(errp, "Parameter%s '%.*s' is too long", | ||||||
|  |                        s != key || s + len != key_end ? " fragment" : "", | ||||||
|  |                        (int)len, s); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (s != key) { | ||||||
|  |             next = keyval_parse_put(cur, key_in_cur, NULL, | ||||||
|  |                                     key, s - 1, errp); | ||||||
|  |             if (!next) { | ||||||
|  |                 return NULL; | ||||||
|  |             } | ||||||
|  |             cur = qobject_to_qdict(next); | ||||||
|  |             assert(cur); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         memcpy(key_in_cur, s, len); | ||||||
|  |         key_in_cur[len] = 0; | ||||||
|  |         s += len; | ||||||
|  |  | ||||||
|  |         if (*s != '.') { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         s++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (key == implied_key) { | ||||||
|  |         assert(!*s); | ||||||
|  |         s = params; | ||||||
|  |     } else { | ||||||
|  |         if (*s != '=') { | ||||||
|  |             error_setg(errp, "Expected '=' after parameter '%.*s'", | ||||||
|  |                        (int)(s - key), key); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |         s++; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     val = qstring_new(); | ||||||
|  |     for (;;) { | ||||||
|  |         if (!*s) { | ||||||
|  |             break; | ||||||
|  |         } else if (*s == ',') { | ||||||
|  |             s++; | ||||||
|  |             if (*s != ',') { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         qstring_append_chr(val, *s++); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (!keyval_parse_put(cur, key_in_cur, val, key, key_end, errp)) { | ||||||
|  |         return NULL; | ||||||
|  |     } | ||||||
|  |     return s; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Parse @params in QEMU's traditional KEY=VALUE,... syntax. | ||||||
|  |  * If @implied_key, the first KEY= can be omitted.  @implied_key is | ||||||
|  |  * implied then, and VALUE can't be empty or contain ',' or '='. | ||||||
|  |  * On success, return a dictionary of the parsed keys and values. | ||||||
|  |  * On failure, store an error through @errp and return NULL. | ||||||
|  |  */ | ||||||
|  | QDict *keyval_parse(const char *params, const char *implied_key, | ||||||
|  |                     Error **errp) | ||||||
|  | { | ||||||
|  |     QDict *qdict = qdict_new(); | ||||||
|  |     const char *s; | ||||||
|  |  | ||||||
|  |     s = params; | ||||||
|  |     while (*s) { | ||||||
|  |         s = keyval_parse_one(qdict, s, implied_key, errp); | ||||||
|  |         if (!s) { | ||||||
|  |             QDECREF(qdict); | ||||||
|  |             return NULL; | ||||||
|  |         } | ||||||
|  |         implied_key = NULL; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return qdict; | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user