Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 1 | /* |
| 2 | * String parsing visitor |
| 3 | * |
Eric Blake | 08f9541 | 2016-01-29 06:48:59 -0700 | [diff] [blame] | 4 | * Copyright Red Hat, Inc. 2012-2016 |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 5 | * |
| 6 | * Author: Paolo Bonzini <pbonzini@redhat.com> |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 7 | * David Hildenbrand <david@redhat.com> |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 8 | * |
| 9 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| 10 | * See the COPYING.LIB file in the top-level directory. |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 11 | */ |
| 12 | |
Peter Maydell | cbf2115 | 2016-01-29 17:49:57 +0000 | [diff] [blame] | 13 | #include "qemu/osdep.h" |
Markus Armbruster | da34e65 | 2016-03-14 09:01:28 +0100 | [diff] [blame] | 14 | #include "qapi/error.h" |
Paolo Bonzini | 7b1b5d1 | 2012-12-17 18:19:43 +0100 | [diff] [blame] | 15 | #include "qapi/string-input-visitor.h" |
| 16 | #include "qapi/visitor-impl.h" |
| 17 | #include "qapi/qmp/qerror.h" |
Max Reitz | 84be629 | 2017-11-14 19:01:23 +0100 | [diff] [blame] | 18 | #include "qapi/qmp/qnull.h" |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 19 | #include "qemu/option.h" |
David Hildenbrand | 4b69d4c | 2018-11-21 17:44:15 +0100 | [diff] [blame] | 20 | #include "qemu/cutils.h" |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 21 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 22 | typedef enum ListMode { |
| 23 | /* no list parsing active / no list expected */ |
| 24 | LM_NONE, |
| 25 | /* we have an unparsed string remaining */ |
| 26 | LM_UNPARSED, |
| 27 | /* we have an unfinished int64 range */ |
| 28 | LM_INT64_RANGE, |
| 29 | /* we have an unfinished uint64 range */ |
| 30 | LM_UINT64_RANGE, |
| 31 | /* we have parsed the string completely and no range is remaining */ |
| 32 | LM_END, |
| 33 | } ListMode; |
| 34 | |
| 35 | /* protect against DOS attacks, limit the amount of elements per range */ |
| 36 | #define RANGE_MAX_ELEMENTS 65536 |
| 37 | |
| 38 | typedef union RangeElement { |
| 39 | int64_t i64; |
| 40 | uint64_t u64; |
| 41 | } RangeElement; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 42 | |
| 43 | struct StringInputVisitor |
| 44 | { |
| 45 | Visitor visitor; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 46 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 47 | /* List parsing state */ |
| 48 | ListMode lm; |
| 49 | RangeElement rangeNext; |
| 50 | RangeElement rangeEnd; |
| 51 | const char *unparsed_string; |
| 52 | void *list; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 53 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 54 | /* The original string to parse */ |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 55 | const char *string; |
| 56 | }; |
| 57 | |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 58 | static StringInputVisitor *to_siv(Visitor *v) |
| 59 | { |
| 60 | return container_of(v, StringInputVisitor, visitor); |
| 61 | } |
| 62 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 63 | static bool start_list(Visitor *v, const char *name, GenericList **list, |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 64 | size_t size, Error **errp) |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 65 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 66 | StringInputVisitor *siv = to_siv(v); |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 67 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 68 | assert(siv->lm == LM_NONE); |
Eric Blake | 1158bb2 | 2016-06-09 10:48:34 -0600 | [diff] [blame] | 69 | siv->list = list; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 70 | siv->unparsed_string = siv->string; |
Eric Blake | d9f62dd | 2016-04-28 15:45:31 -0600 | [diff] [blame] | 71 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 72 | if (!siv->string[0]) { |
| 73 | if (list) { |
| 74 | *list = NULL; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 75 | } |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 76 | siv->lm = LM_END; |
Eric Blake | d9f62dd | 2016-04-28 15:45:31 -0600 | [diff] [blame] | 77 | } else { |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 78 | if (list) { |
| 79 | *list = g_malloc0(size); |
| 80 | } |
| 81 | siv->lm = LM_UNPARSED; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 82 | } |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 83 | return true; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 84 | } |
| 85 | |
Eric Blake | d9f62dd | 2016-04-28 15:45:31 -0600 | [diff] [blame] | 86 | static GenericList *next_list(Visitor *v, GenericList *tail, size_t size) |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 87 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 88 | StringInputVisitor *siv = to_siv(v); |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 89 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 90 | switch (siv->lm) { |
| 91 | case LM_END: |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 92 | return NULL; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 93 | case LM_INT64_RANGE: |
| 94 | case LM_UINT64_RANGE: |
| 95 | case LM_UNPARSED: |
| 96 | /* we have an unparsed string or something left in a range */ |
| 97 | break; |
| 98 | default: |
| 99 | abort(); |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 100 | } |
| 101 | |
Eric Blake | d9f62dd | 2016-04-28 15:45:31 -0600 | [diff] [blame] | 102 | tail->next = g_malloc0(size); |
| 103 | return tail->next; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 104 | } |
| 105 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 106 | static bool check_list(Visitor *v, Error **errp) |
Markus Armbruster | a4a1c70 | 2017-03-03 13:32:45 +0100 | [diff] [blame] | 107 | { |
| 108 | const StringInputVisitor *siv = to_siv(v); |
Markus Armbruster | a4a1c70 | 2017-03-03 13:32:45 +0100 | [diff] [blame] | 109 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 110 | switch (siv->lm) { |
| 111 | case LM_INT64_RANGE: |
| 112 | case LM_UINT64_RANGE: |
| 113 | case LM_UNPARSED: |
| 114 | error_setg(errp, "Fewer list elements expected"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 115 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 116 | case LM_END: |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 117 | return true; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 118 | default: |
| 119 | abort(); |
Markus Armbruster | a4a1c70 | 2017-03-03 13:32:45 +0100 | [diff] [blame] | 120 | } |
Markus Armbruster | a4a1c70 | 2017-03-03 13:32:45 +0100 | [diff] [blame] | 121 | } |
| 122 | |
Eric Blake | 1158bb2 | 2016-06-09 10:48:34 -0600 | [diff] [blame] | 123 | static void end_list(Visitor *v, void **obj) |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 124 | { |
Eric Blake | 1158bb2 | 2016-06-09 10:48:34 -0600 | [diff] [blame] | 125 | StringInputVisitor *siv = to_siv(v); |
| 126 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 127 | assert(siv->lm != LM_NONE); |
Eric Blake | 1158bb2 | 2016-06-09 10:48:34 -0600 | [diff] [blame] | 128 | assert(siv->list == obj); |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 129 | siv->list = NULL; |
| 130 | siv->unparsed_string = NULL; |
| 131 | siv->lm = LM_NONE; |
| 132 | } |
| 133 | |
| 134 | static int try_parse_int64_list_entry(StringInputVisitor *siv, int64_t *obj) |
| 135 | { |
| 136 | const char *endptr; |
| 137 | int64_t start, end; |
| 138 | |
| 139 | /* parse a simple int64 or range */ |
| 140 | if (qemu_strtoi64(siv->unparsed_string, &endptr, 0, &start)) { |
| 141 | return -EINVAL; |
| 142 | } |
| 143 | end = start; |
| 144 | |
| 145 | switch (endptr[0]) { |
| 146 | case '\0': |
| 147 | siv->unparsed_string = endptr; |
| 148 | break; |
| 149 | case ',': |
| 150 | siv->unparsed_string = endptr + 1; |
| 151 | break; |
| 152 | case '-': |
| 153 | /* parse the end of the range */ |
| 154 | if (qemu_strtoi64(endptr + 1, &endptr, 0, &end)) { |
| 155 | return -EINVAL; |
| 156 | } |
| 157 | if (start > end || end - start >= RANGE_MAX_ELEMENTS) { |
| 158 | return -EINVAL; |
| 159 | } |
| 160 | switch (endptr[0]) { |
| 161 | case '\0': |
| 162 | siv->unparsed_string = endptr; |
| 163 | break; |
| 164 | case ',': |
| 165 | siv->unparsed_string = endptr + 1; |
| 166 | break; |
| 167 | default: |
| 168 | return -EINVAL; |
| 169 | } |
| 170 | break; |
| 171 | default: |
| 172 | return -EINVAL; |
| 173 | } |
| 174 | |
| 175 | /* we have a proper range (with maybe only one element) */ |
| 176 | siv->lm = LM_INT64_RANGE; |
| 177 | siv->rangeNext.i64 = start; |
| 178 | siv->rangeEnd.i64 = end; |
| 179 | return 0; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 180 | } |
| 181 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 182 | static bool parse_type_int64(Visitor *v, const char *name, int64_t *obj, |
Eric Blake | 4c40314 | 2016-01-29 06:48:49 -0700 | [diff] [blame] | 183 | Error **errp) |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 184 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 185 | StringInputVisitor *siv = to_siv(v); |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 186 | int64_t val; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 187 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 188 | switch (siv->lm) { |
| 189 | case LM_NONE: |
| 190 | /* just parse a simple int64, bail out if not completely consumed */ |
| 191 | if (qemu_strtoi64(siv->string, NULL, 0, &val)) { |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 192 | error_setg(errp, QERR_INVALID_PARAMETER_VALUE, |
| 193 | name ? name : "null", "int64"); |
| 194 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 195 | } |
| 196 | *obj = val; |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 197 | return true; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 198 | case LM_UNPARSED: |
| 199 | if (try_parse_int64_list_entry(siv, obj)) { |
| 200 | error_setg(errp, QERR_INVALID_PARAMETER_VALUE, name ? name : "null", |
| 201 | "list of int64 values or ranges"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 202 | return false; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 203 | } |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 204 | assert(siv->lm == LM_INT64_RANGE); |
| 205 | /* fall through */ |
| 206 | case LM_INT64_RANGE: |
| 207 | /* return the next element in the range */ |
| 208 | assert(siv->rangeNext.i64 <= siv->rangeEnd.i64); |
| 209 | *obj = siv->rangeNext.i64++; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 210 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 211 | if (siv->rangeNext.i64 > siv->rangeEnd.i64 || *obj == INT64_MAX) { |
| 212 | /* end of range, check if there is more to parse */ |
| 213 | siv->lm = siv->unparsed_string[0] ? LM_UNPARSED : LM_END; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 214 | } |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 215 | return true; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 216 | case LM_END: |
| 217 | error_setg(errp, "Fewer list elements expected"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 218 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 219 | default: |
| 220 | abort(); |
| 221 | } |
| 222 | } |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 223 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 224 | static int try_parse_uint64_list_entry(StringInputVisitor *siv, uint64_t *obj) |
| 225 | { |
| 226 | const char *endptr; |
| 227 | uint64_t start, end; |
| 228 | |
| 229 | /* parse a simple uint64 or range */ |
| 230 | if (qemu_strtou64(siv->unparsed_string, &endptr, 0, &start)) { |
| 231 | return -EINVAL; |
| 232 | } |
| 233 | end = start; |
| 234 | |
| 235 | switch (endptr[0]) { |
| 236 | case '\0': |
| 237 | siv->unparsed_string = endptr; |
| 238 | break; |
| 239 | case ',': |
| 240 | siv->unparsed_string = endptr + 1; |
| 241 | break; |
| 242 | case '-': |
| 243 | /* parse the end of the range */ |
| 244 | if (qemu_strtou64(endptr + 1, &endptr, 0, &end)) { |
| 245 | return -EINVAL; |
| 246 | } |
| 247 | if (start > end || end - start >= RANGE_MAX_ELEMENTS) { |
| 248 | return -EINVAL; |
| 249 | } |
| 250 | switch (endptr[0]) { |
| 251 | case '\0': |
| 252 | siv->unparsed_string = endptr; |
| 253 | break; |
| 254 | case ',': |
| 255 | siv->unparsed_string = endptr + 1; |
| 256 | break; |
| 257 | default: |
| 258 | return -EINVAL; |
| 259 | } |
| 260 | break; |
| 261 | default: |
| 262 | return -EINVAL; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 263 | } |
| 264 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 265 | /* we have a proper range (with maybe only one element) */ |
| 266 | siv->lm = LM_UINT64_RANGE; |
| 267 | siv->rangeNext.u64 = start; |
| 268 | siv->rangeEnd.u64 = end; |
| 269 | return 0; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 270 | } |
| 271 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 272 | static bool parse_type_uint64(Visitor *v, const char *name, uint64_t *obj, |
Eric Blake | f755dea | 2016-01-29 06:48:50 -0700 | [diff] [blame] | 273 | Error **errp) |
| 274 | { |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 275 | StringInputVisitor *siv = to_siv(v); |
| 276 | uint64_t val; |
| 277 | |
| 278 | switch (siv->lm) { |
| 279 | case LM_NONE: |
| 280 | /* just parse a simple uint64, bail out if not completely consumed */ |
| 281 | if (qemu_strtou64(siv->string, NULL, 0, &val)) { |
| 282 | error_setg(errp, QERR_INVALID_PARAMETER_VALUE, name ? name : "null", |
| 283 | "uint64"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 284 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 285 | } |
| 286 | *obj = val; |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 287 | return true; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 288 | case LM_UNPARSED: |
| 289 | if (try_parse_uint64_list_entry(siv, obj)) { |
| 290 | error_setg(errp, QERR_INVALID_PARAMETER_VALUE, name ? name : "null", |
| 291 | "list of uint64 values or ranges"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 292 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 293 | } |
| 294 | assert(siv->lm == LM_UINT64_RANGE); |
| 295 | /* fall through */ |
| 296 | case LM_UINT64_RANGE: |
| 297 | /* return the next element in the range */ |
| 298 | assert(siv->rangeNext.u64 <= siv->rangeEnd.u64); |
| 299 | *obj = siv->rangeNext.u64++; |
| 300 | |
| 301 | if (siv->rangeNext.u64 > siv->rangeEnd.u64 || *obj == UINT64_MAX) { |
| 302 | /* end of range, check if there is more to parse */ |
| 303 | siv->lm = siv->unparsed_string[0] ? LM_UNPARSED : LM_END; |
| 304 | } |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 305 | return true; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 306 | case LM_END: |
| 307 | error_setg(errp, "Fewer list elements expected"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 308 | return false; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 309 | default: |
| 310 | abort(); |
Eric Blake | f755dea | 2016-01-29 06:48:50 -0700 | [diff] [blame] | 311 | } |
| 312 | } |
| 313 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 314 | static bool parse_type_size(Visitor *v, const char *name, uint64_t *obj, |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 315 | Error **errp) |
| 316 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 317 | StringInputVisitor *siv = to_siv(v); |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 318 | uint64_t val; |
| 319 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 320 | assert(siv->lm == LM_NONE); |
Markus Armbruster | 668f62e | 2020-07-07 18:06:02 +0200 | [diff] [blame] | 321 | if (!parse_option_size(name, siv->string, &val, errp)) { |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 322 | return false; |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 323 | } |
| 324 | |
| 325 | *obj = val; |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 326 | return true; |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 327 | } |
| 328 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 329 | static bool parse_type_bool(Visitor *v, const char *name, bool *obj, |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 330 | Error **errp) |
| 331 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 332 | StringInputVisitor *siv = to_siv(v); |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 333 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 334 | assert(siv->lm == LM_NONE); |
Paolo Bonzini | 372bcb2 | 2020-11-03 11:13:39 -0500 | [diff] [blame] | 335 | return qapi_bool_parse(name ? name : "null", siv->string, obj, errp); |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 336 | } |
| 337 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 338 | static bool parse_type_str(Visitor *v, const char *name, char **obj, |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 339 | Error **errp) |
| 340 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 341 | StringInputVisitor *siv = to_siv(v); |
Markus Armbruster | f332e83 | 2017-03-03 13:32:36 +0100 | [diff] [blame] | 342 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 343 | assert(siv->lm == LM_NONE); |
Markus Armbruster | f332e83 | 2017-03-03 13:32:36 +0100 | [diff] [blame] | 344 | *obj = g_strdup(siv->string); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 345 | return true; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 346 | } |
| 347 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 348 | static bool parse_type_number(Visitor *v, const char *name, double *obj, |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 349 | Error **errp) |
| 350 | { |
Eric Blake | d7bea75 | 2016-01-29 06:48:38 -0700 | [diff] [blame] | 351 | StringInputVisitor *siv = to_siv(v); |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 352 | double val; |
| 353 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 354 | assert(siv->lm == LM_NONE); |
David Hildenbrand | 4b69d4c | 2018-11-21 17:44:15 +0100 | [diff] [blame] | 355 | if (qemu_strtod_finite(siv->string, NULL, &val)) { |
Markus Armbruster | c6bd8c7 | 2015-03-17 11:54:50 +0100 | [diff] [blame] | 356 | error_setg(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null", |
| 357 | "number"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 358 | return false; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 359 | } |
| 360 | |
| 361 | *obj = val; |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 362 | return true; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 363 | } |
| 364 | |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 365 | static bool parse_type_null(Visitor *v, const char *name, QNull **obj, |
Markus Armbruster | d2f95f4 | 2017-06-26 18:22:59 +0200 | [diff] [blame] | 366 | Error **errp) |
Greg Kurz | a733371 | 2016-12-16 16:26:09 +0100 | [diff] [blame] | 367 | { |
| 368 | StringInputVisitor *siv = to_siv(v); |
| 369 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 370 | assert(siv->lm == LM_NONE); |
Markus Armbruster | d2f95f4 | 2017-06-26 18:22:59 +0200 | [diff] [blame] | 371 | *obj = NULL; |
| 372 | |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 373 | if (siv->string[0]) { |
Greg Kurz | a733371 | 2016-12-16 16:26:09 +0100 | [diff] [blame] | 374 | error_setg(errp, QERR_INVALID_PARAMETER_TYPE, name ? name : "null", |
| 375 | "null"); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 376 | return false; |
Greg Kurz | a733371 | 2016-12-16 16:26:09 +0100 | [diff] [blame] | 377 | } |
Markus Armbruster | d2f95f4 | 2017-06-26 18:22:59 +0200 | [diff] [blame] | 378 | |
| 379 | *obj = qnull(); |
Markus Armbruster | 012d4c9 | 2020-07-07 18:05:45 +0200 | [diff] [blame] | 380 | return true; |
Greg Kurz | a733371 | 2016-12-16 16:26:09 +0100 | [diff] [blame] | 381 | } |
| 382 | |
Eric Blake | 2c0ef9f | 2016-06-09 10:48:35 -0600 | [diff] [blame] | 383 | static void string_input_free(Visitor *v) |
| 384 | { |
| 385 | StringInputVisitor *siv = to_siv(v); |
| 386 | |
Eric Blake | 7a0525c | 2016-06-09 10:48:37 -0600 | [diff] [blame] | 387 | g_free(siv); |
Eric Blake | 2c0ef9f | 2016-06-09 10:48:35 -0600 | [diff] [blame] | 388 | } |
| 389 | |
Eric Blake | 7a0525c | 2016-06-09 10:48:37 -0600 | [diff] [blame] | 390 | Visitor *string_input_visitor_new(const char *str) |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 391 | { |
| 392 | StringInputVisitor *v; |
| 393 | |
Markus Armbruster | f332e83 | 2017-03-03 13:32:36 +0100 | [diff] [blame] | 394 | assert(str); |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 395 | v = g_malloc0(sizeof(*v)); |
| 396 | |
Eric Blake | 983f52d | 2016-04-28 15:45:09 -0600 | [diff] [blame] | 397 | v->visitor.type = VISITOR_INPUT; |
Eric Blake | 4c40314 | 2016-01-29 06:48:49 -0700 | [diff] [blame] | 398 | v->visitor.type_int64 = parse_type_int64; |
Eric Blake | f755dea | 2016-01-29 06:48:50 -0700 | [diff] [blame] | 399 | v->visitor.type_uint64 = parse_type_uint64; |
Paolo Bonzini | a5829cc | 2014-02-08 11:01:44 +0100 | [diff] [blame] | 400 | v->visitor.type_size = parse_type_size; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 401 | v->visitor.type_bool = parse_type_bool; |
| 402 | v->visitor.type_str = parse_type_str; |
| 403 | v->visitor.type_number = parse_type_number; |
Greg Kurz | a733371 | 2016-12-16 16:26:09 +0100 | [diff] [blame] | 404 | v->visitor.type_null = parse_type_null; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 405 | v->visitor.start_list = start_list; |
| 406 | v->visitor.next_list = next_list; |
Markus Armbruster | a4a1c70 | 2017-03-03 13:32:45 +0100 | [diff] [blame] | 407 | v->visitor.check_list = check_list; |
Hu Tao | 659268f | 2014-06-10 19:15:27 +0800 | [diff] [blame] | 408 | v->visitor.end_list = end_list; |
Eric Blake | 2c0ef9f | 2016-06-09 10:48:35 -0600 | [diff] [blame] | 409 | v->visitor.free = string_input_free; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 410 | |
| 411 | v->string = str; |
David Hildenbrand | c9fba9d | 2018-11-21 17:44:18 +0100 | [diff] [blame] | 412 | v->lm = LM_NONE; |
Eric Blake | 7a0525c | 2016-06-09 10:48:37 -0600 | [diff] [blame] | 413 | return &v->visitor; |
Paolo Bonzini | a020f98 | 2012-02-09 09:36:37 +0100 | [diff] [blame] | 414 | } |