qobject-input-visitor.c 17.6 KB
Newer Older
M
Michael Roth 已提交
1 2 3
/*
 * Input Visitor
 *
4
 * Copyright (C) 2012-2017 Red Hat, Inc.
M
Michael Roth 已提交
5 6 7 8 9 10 11 12 13 14
 * Copyright IBM, Corp. 2011
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *
 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
 * See the COPYING.LIB file in the top-level directory.
 *
 */

P
Peter Maydell 已提交
15
#include "qemu/osdep.h"
16
#include "qapi/error.h"
17
#include "qapi/qobject-input-visitor.h"
18
#include "qapi/visitor-impl.h"
19
#include "qemu/queue.h"
M
Michael Roth 已提交
20
#include "qemu-common.h"
21 22
#include "qapi/qmp/types.h"
#include "qapi/qmp/qerror.h"
23
#include "qemu/cutils.h"
M
Michael Roth 已提交
24

25 26 27
typedef struct StackObject {
    const char *name;            /* Name of @obj in its parent, if any */
    QObject *obj;                /* QDict or QList being visited */
E
Eric Blake 已提交
28
    void *qapi; /* sanity check that caller uses same pointer */
E
Eric Blake 已提交
29

30 31 32
    GHashTable *h;              /* If @obj is QDict: unvisited keys */
    const QListEntry *entry;    /* If @obj is QList: unvisited tail */
    unsigned index;             /* If @obj is QList: list index of @entry */
33

34
    QSLIST_ENTRY(StackObject) node; /* parent */
M
Michael Roth 已提交
35 36
} StackObject;

37
struct QObjectInputVisitor {
M
Michael Roth 已提交
38
    Visitor visitor;
E
Eric Blake 已提交
39

40 41 42 43 44
    /* Root of visit at visitor creation. */
    QObject *root;

    /* Stack of objects being visited (all entries will be either
     * QDict or QList). */
45
    QSLIST_HEAD(, StackObject) stack;
E
Eric Blake 已提交
46

47
    GString *errname;           /* Accumulator for full_name() */
M
Michael Roth 已提交
48 49
};

50
static QObjectInputVisitor *to_qiv(Visitor *v)
M
Michael Roth 已提交
51
{
52
    return container_of(v, QObjectInputVisitor, visitor);
M
Michael Roth 已提交
53 54
}

55 56
static const char *full_name_nth(QObjectInputVisitor *qiv, const char *name,
                                 int n)
57 58 59 60 61 62 63 64 65 66 67
{
    StackObject *so;
    char buf[32];

    if (qiv->errname) {
        g_string_truncate(qiv->errname, 0);
    } else {
        qiv->errname = g_string_new("");
    }

    QSLIST_FOREACH(so , &qiv->stack, node) {
68 69 70 71
        if (n) {
            n--;
        } else if (qobject_type(so->obj) == QTYPE_QDICT) {
            g_string_prepend(qiv->errname, name ?: "<anonymous>");
72 73 74 75 76 77 78
            g_string_prepend_c(qiv->errname, '.');
        } else {
            snprintf(buf, sizeof(buf), "[%u]", so->index);
            g_string_prepend(qiv->errname, buf);
        }
        name = so->name;
    }
79
    assert(!n);
80 81 82 83 84

    if (name) {
        g_string_prepend(qiv->errname, name);
    } else if (qiv->errname->str[0] == '.') {
        g_string_erase(qiv->errname, 0, 1);
85
    } else if (!qiv->errname->str[0]) {
86 87 88 89 90 91
        return "<anonymous>";
    }

    return qiv->errname->str;
}

92 93 94 95 96
static const char *full_name(QObjectInputVisitor *qiv, const char *name)
{
    return full_name_nth(qiv, name, 0);
}

97 98 99
static QObject *qobject_input_try_get_object(QObjectInputVisitor *qiv,
                                             const char *name,
                                             bool consume)
M
Michael Roth 已提交
100
{
101 102
    StackObject *tos;
    QObject *qobj;
103
    QObject *ret;
E
Eric Blake 已提交
104

105
    if (QSLIST_EMPTY(&qiv->stack)) {
106
        /* Starting at root, name is ignored. */
107
        assert(qiv->root);
108 109 110 111
        return qiv->root;
    }

    /* We are in a container; find the next element. */
112
    tos = QSLIST_FIRST(&qiv->stack);
113
    qobj = tos->obj;
E
Eric Blake 已提交
114 115
    assert(qobj);

116 117
    if (qobject_type(qobj) == QTYPE_QDICT) {
        assert(name);
118 119 120 121
        ret = qdict_get(qobject_to_qdict(qobj), name);
        if (tos->h && consume && ret) {
            bool removed = g_hash_table_remove(tos->h, name);
            assert(removed);
122
        }
123
    } else {
E
Eric Blake 已提交
124
        assert(qobject_type(qobj) == QTYPE_QLIST);
125
        assert(!name);
126 127 128 129 130 131 132 133
        if (tos->entry) {
            ret = qlist_entry_obj(tos->entry);
            if (consume) {
                tos->entry = qlist_next(tos->entry);
            }
        } else {
            ret = NULL;
        }
134
        if (consume) {
135
            tos->index++;
136
        }
M
Michael Roth 已提交
137 138
    }

139
    return ret;
M
Michael Roth 已提交
140 141
}

142 143 144 145 146 147 148 149 150 151 152 153
static QObject *qobject_input_get_object(QObjectInputVisitor *qiv,
                                         const char *name,
                                         bool consume, Error **errp)
{
    QObject *obj = qobject_input_try_get_object(qiv, name, consume);

    if (!obj) {
        error_setg(errp, QERR_MISSING_PARAMETER, full_name(qiv, name));
    }
    return obj;
}

154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
static const char *qobject_input_get_keyval(QObjectInputVisitor *qiv,
                                            const char *name,
                                            Error **errp)
{
    QObject *qobj;
    QString *qstr;

    qobj = qobject_input_get_object(qiv, name, true, errp);
    if (!qobj) {
        return NULL;
    }

    qstr = qobject_to_qstring(qobj);
    if (!qstr) {
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "string");
        return NULL;
    }

    return qstring_get_str(qstr);
}

176 177 178 179 180 181
static void qdict_add_key(const char *key, QObject *obj, void *opaque)
{
    GHashTable *h = opaque;
    g_hash_table_insert(h, (gpointer) key, NULL);
}

182
static const QListEntry *qobject_input_push(QObjectInputVisitor *qiv,
183
                                            const char *name,
184
                                            QObject *obj, void *qapi)
M
Michael Roth 已提交
185
{
186
    GHashTable *h;
187
    StackObject *tos = g_new0(StackObject, 1);
M
Michael Roth 已提交
188

E
Eric Blake 已提交
189
    assert(obj);
190
    tos->name = name;
E
Eric Blake 已提交
191
    tos->obj = obj;
E
Eric Blake 已提交
192
    tos->qapi = qapi;
193

194
    if (qobject_type(obj) == QTYPE_QDICT) {
195 196
        h = g_hash_table_new(g_str_hash, g_str_equal);
        qdict_iter(qobject_to_qdict(obj), qdict_add_key, h);
E
Eric Blake 已提交
197
        tos->h = h;
198 199
    } else {
        assert(qobject_type(obj) == QTYPE_QLIST);
200
        tos->entry = qlist_first(qobject_to_qlist(obj));
201
        tos->index = -1;
202 203
    }

204
    QSLIST_INSERT_HEAD(&qiv->stack, tos, node);
205
    return tos->entry;
M
Michael Roth 已提交
206 207
}

208

209
static void qobject_input_check_struct(Visitor *v, Error **errp)
M
Michael Roth 已提交
210
{
211
    QObjectInputVisitor *qiv = to_qiv(v);
212
    StackObject *tos = QSLIST_FIRST(&qiv->stack);
213 214
    GHashTableIter iter;
    const char *key;
215

216
    assert(tos && !tos->entry);
217 218 219 220 221

    g_hash_table_iter_init(&iter, tos->h);
    if (g_hash_table_iter_next(&iter, (void **)&key, NULL)) {
        error_setg(errp, "Parameter '%s' is unexpected",
                   full_name(qiv, key));
222 223 224
    }
}

225
static void qobject_input_stack_object_free(StackObject *tos)
226
{
227 228 229
    if (tos->h) {
        g_hash_table_unref(tos->h);
    }
230

231 232
    g_free(tos);
}
233

234
static void qobject_input_pop(Visitor *v, void **obj)
235
{
236
    QObjectInputVisitor *qiv = to_qiv(v);
237
    StackObject *tos = QSLIST_FIRST(&qiv->stack);
238

239 240
    assert(tos && tos->qapi == obj);
    QSLIST_REMOVE_HEAD(&qiv->stack, node);
241
    qobject_input_stack_object_free(tos);
M
Michael Roth 已提交
242 243
}

244 245
static void qobject_input_start_struct(Visitor *v, const char *name, void **obj,
                                       size_t size, Error **errp)
M
Michael Roth 已提交
246
{
247 248
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
M
Michael Roth 已提交
249

250 251 252
    if (obj) {
        *obj = NULL;
    }
253 254 255 256
    if (!qobj) {
        return;
    }
    if (qobject_type(qobj) != QTYPE_QDICT) {
257 258
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "object");
M
Michael Roth 已提交
259 260 261
        return;
    }

262
    qobject_input_push(qiv, name, qobj, obj);
M
Michael Roth 已提交
263 264

    if (obj) {
265
        *obj = g_malloc0(size);
M
Michael Roth 已提交
266 267 268 269
    }
}


270 271 272
static void qobject_input_start_list(Visitor *v, const char *name,
                                     GenericList **list, size_t size,
                                     Error **errp)
M
Michael Roth 已提交
273
{
274 275
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
276
    const QListEntry *entry;
M
Michael Roth 已提交
277

278 279 280
    if (list) {
        *list = NULL;
    }
281 282 283 284
    if (!qobj) {
        return;
    }
    if (qobject_type(qobj) != QTYPE_QLIST) {
285 286
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "array");
M
Michael Roth 已提交
287 288 289
        return;
    }

290
    entry = qobject_input_push(qiv, name, qobj, list);
291 292
    if (entry && list) {
        *list = g_malloc0(size);
293
    }
M
Michael Roth 已提交
294 295
}

296 297
static GenericList *qobject_input_next_list(Visitor *v, GenericList *tail,
                                            size_t size)
M
Michael Roth 已提交
298
{
299
    QObjectInputVisitor *qiv = to_qiv(v);
300 301 302
    StackObject *tos = QSLIST_FIRST(&qiv->stack);

    assert(tos && tos->obj && qobject_type(tos->obj) == QTYPE_QLIST);
M
Michael Roth 已提交
303

304
    if (!tos->entry) {
M
Michael Roth 已提交
305 306
        return NULL;
    }
307 308
    tail->next = g_malloc0(size);
    return tail->next;
M
Michael Roth 已提交
309 310
}

311 312 313 314 315 316 317 318 319 320 321 322 323
static void qobject_input_check_list(Visitor *v, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
    StackObject *tos = QSLIST_FIRST(&qiv->stack);

    assert(tos && tos->obj && qobject_type(tos->obj) == QTYPE_QLIST);

    if (tos->entry) {
        error_setg(errp, "Only %u list elements expected in %s",
                   tos->index + 1, full_name_nth(qiv, NULL, 1));
    }
}

M
Michael Roth 已提交
324

325 326 327
static void qobject_input_start_alternate(Visitor *v, const char *name,
                                          GenericAlternate **obj, size_t size,
                                          bool promote_int, Error **errp)
K
Kevin Wolf 已提交
328
{
329 330
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, false, errp);
K
Kevin Wolf 已提交
331 332

    if (!qobj) {
333
        *obj = NULL;
K
Kevin Wolf 已提交
334 335
        return;
    }
336 337 338 339
    *obj = g_malloc0(size);
    (*obj)->type = qobject_type(qobj);
    if (promote_int && (*obj)->type == QTYPE_QINT) {
        (*obj)->type = QTYPE_QFLOAT;
340
    }
K
Kevin Wolf 已提交
341 342
}

343 344
static void qobject_input_type_int64(Visitor *v, const char *name, int64_t *obj,
                                     Error **errp)
M
Michael Roth 已提交
345
{
346 347
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
348
    QInt *qint;
M
Michael Roth 已提交
349

350 351 352 353
    if (!qobj) {
        return;
    }
    qint = qobject_to_qint(qobj);
354
    if (!qint) {
355 356
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "integer");
M
Michael Roth 已提交
357 358 359
        return;
    }

360
    *obj = qint_get_int(qint);
M
Michael Roth 已提交
361 362
}

363 364 365 366 367

static void qobject_input_type_int64_keyval(Visitor *v, const char *name,
                                            int64_t *obj, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
368
    const char *str = qobject_input_get_keyval(qiv, name, errp);
369

370
    if (!str) {
371 372 373
        return;
    }

374
    if (qemu_strtoi64(str, NULL, 0, obj) < 0) {
375 376 377 378 379 380
        /* TODO report -ERANGE more nicely */
        error_setg(errp, QERR_INVALID_PARAMETER_VALUE,
                   full_name(qiv, name), "integer");
    }
}

381 382
static void qobject_input_type_uint64(Visitor *v, const char *name,
                                      uint64_t *obj, Error **errp)
383 384
{
    /* FIXME: qobject_to_qint mishandles values over INT64_MAX */
385 386
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
387
    QInt *qint;
388

389 390 391 392
    if (!qobj) {
        return;
    }
    qint = qobject_to_qint(qobj);
393
    if (!qint) {
394 395
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "integer");
396 397 398 399 400 401
        return;
    }

    *obj = qint_get_int(qint);
}

402 403 404 405
static void qobject_input_type_uint64_keyval(Visitor *v, const char *name,
                                             uint64_t *obj, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
406
    const char *str = qobject_input_get_keyval(qiv, name, errp);
407

408
    if (!str) {
409 410 411
        return;
    }

412
    if (qemu_strtou64(str, NULL, 0, obj) < 0) {
413 414 415 416 417 418
        /* TODO report -ERANGE more nicely */
        error_setg(errp, QERR_INVALID_PARAMETER_VALUE,
                   full_name(qiv, name), "integer");
    }
}

419 420
static void qobject_input_type_bool(Visitor *v, const char *name, bool *obj,
                                    Error **errp)
M
Michael Roth 已提交
421
{
422 423
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
424
    QBool *qbool;
M
Michael Roth 已提交
425

426 427 428 429
    if (!qobj) {
        return;
    }
    qbool = qobject_to_qbool(qobj);
430
    if (!qbool) {
431 432
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "boolean");
M
Michael Roth 已提交
433 434 435
        return;
    }

436
    *obj = qbool_get_bool(qbool);
M
Michael Roth 已提交
437 438
}

439 440 441 442
static void qobject_input_type_bool_keyval(Visitor *v, const char *name,
                                           bool *obj, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
443
    const char *str = qobject_input_get_keyval(qiv, name, errp);
444

445
    if (!str) {
446 447 448 449 450 451 452 453 454 455 456 457 458
        return;
    }

    if (!strcmp(str, "on")) {
        *obj = true;
    } else if (!strcmp(str, "off")) {
        *obj = false;
    } else {
        error_setg(errp, QERR_INVALID_PARAMETER_VALUE,
                   full_name(qiv, name), "'on' or 'off'");
    }
}

459 460
static void qobject_input_type_str(Visitor *v, const char *name, char **obj,
                                   Error **errp)
M
Michael Roth 已提交
461
{
462 463
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
464
    QString *qstr;
M
Michael Roth 已提交
465

466 467 468 469 470
    *obj = NULL;
    if (!qobj) {
        return;
    }
    qstr = qobject_to_qstring(qobj);
471
    if (!qstr) {
472 473
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "string");
M
Michael Roth 已提交
474 475 476
        return;
    }

477
    *obj = g_strdup(qstring_get_str(qstr));
M
Michael Roth 已提交
478 479
}

480 481
static void qobject_input_type_number(Visitor *v, const char *name, double *obj,
                                      Error **errp)
M
Michael Roth 已提交
482
{
483 484
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
485 486
    QInt *qint;
    QFloat *qfloat;
M
Michael Roth 已提交
487

488 489 490
    if (!qobj) {
        return;
    }
491 492 493
    qint = qobject_to_qint(qobj);
    if (qint) {
        *obj = qint_get_int(qobject_to_qint(qobj));
M
Michael Roth 已提交
494 495 496
        return;
    }

497 498
    qfloat = qobject_to_qfloat(qobj);
    if (qfloat) {
499
        *obj = qfloat_get_double(qobject_to_qfloat(qobj));
500
        return;
501
    }
502

503 504
    error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
               full_name(qiv, name), "number");
M
Michael Roth 已提交
505 506
}

507 508 509 510
static void qobject_input_type_number_keyval(Visitor *v, const char *name,
                                             double *obj, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
511
    const char *str = qobject_input_get_keyval(qiv, name, errp);
512 513
    char *endp;

514
    if (!str) {
515 516 517 518 519 520 521 522 523 524 525 526
        return;
    }

    errno = 0;
    *obj = strtod(str, &endp);
    if (errno || endp == str || *endp) {
        /* TODO report -ERANGE more nicely */
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "number");
    }
}

527 528
static void qobject_input_type_any(Visitor *v, const char *name, QObject **obj,
                                   Error **errp)
529
{
530 531
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
532

533
    *obj = NULL;
534 535 536 537
    if (!qobj) {
        return;
    }

538 539 540 541
    qobject_incref(qobj);
    *obj = qobj;
}

542
static void qobject_input_type_null(Visitor *v, const char *name, Error **errp)
E
Eric Blake 已提交
543
{
544 545
    QObjectInputVisitor *qiv = to_qiv(v);
    QObject *qobj = qobject_input_get_object(qiv, name, true, errp);
546

547 548 549 550
    if (!qobj) {
        return;
    }

551
    if (qobject_type(qobj) != QTYPE_QNULL) {
552 553
        error_setg(errp, QERR_INVALID_PARAMETER_TYPE,
                   full_name(qiv, name), "null");
554
    }
E
Eric Blake 已提交
555 556
}

557 558 559 560
static void qobject_input_type_size_keyval(Visitor *v, const char *name,
                                           uint64_t *obj, Error **errp)
{
    QObjectInputVisitor *qiv = to_qiv(v);
561
    const char *str = qobject_input_get_keyval(qiv, name, errp);
562

563
    if (!str) {
564 565 566
        return;
    }

567
    if (qemu_strtosz(str, NULL, obj) < 0) {
568 569 570 571 572 573
        /* TODO report -ERANGE more nicely */
        error_setg(errp, QERR_INVALID_PARAMETER_VALUE,
                   full_name(qiv, name), "size");
    }
}

574
static void qobject_input_optional(Visitor *v, const char *name, bool *present)
M
Michael Roth 已提交
575
{
576
    QObjectInputVisitor *qiv = to_qiv(v);
577
    QObject *qobj = qobject_input_try_get_object(qiv, name, false);
M
Michael Roth 已提交
578 579 580 581 582 583 584 585 586

    if (!qobj) {
        *present = false;
        return;
    }

    *present = true;
}

587
static void qobject_input_free(Visitor *v)
E
Eric Blake 已提交
588
{
589
    QObjectInputVisitor *qiv = to_qiv(v);
590

591 592 593 594
    while (!QSLIST_EMPTY(&qiv->stack)) {
        StackObject *tos = QSLIST_FIRST(&qiv->stack);

        QSLIST_REMOVE_HEAD(&qiv->stack, node);
595
        qobject_input_stack_object_free(tos);
596
    }
E
Eric Blake 已提交
597

598
    qobject_decref(qiv->root);
599 600 601
    if (qiv->errname) {
        g_string_free(qiv->errname, TRUE);
    }
602
    g_free(qiv);
E
Eric Blake 已提交
603 604
}

605
static QObjectInputVisitor *qobject_input_visitor_base_new(QObject *obj)
M
Michael Roth 已提交
606
{
607
    QObjectInputVisitor *v = g_malloc0(sizeof(*v));
M
Michael Roth 已提交
608

609
    assert(obj);
M
Michael Roth 已提交
610

611
    v->visitor.type = VISITOR_INPUT;
612 613 614 615 616
    v->visitor.start_struct = qobject_input_start_struct;
    v->visitor.check_struct = qobject_input_check_struct;
    v->visitor.end_struct = qobject_input_pop;
    v->visitor.start_list = qobject_input_start_list;
    v->visitor.next_list = qobject_input_next_list;
617
    v->visitor.check_list = qobject_input_check_list;
618 619
    v->visitor.end_list = qobject_input_pop;
    v->visitor.start_alternate = qobject_input_start_alternate;
620 621 622 623 624 625 626 627 628 629 630 631 632
    v->visitor.optional = qobject_input_optional;
    v->visitor.free = qobject_input_free;

    v->root = obj;
    qobject_incref(obj);

    return v;
}

Visitor *qobject_input_visitor_new(QObject *obj)
{
    QObjectInputVisitor *v = qobject_input_visitor_base_new(obj);

633 634 635 636 637 638 639
    v->visitor.type_int64 = qobject_input_type_int64;
    v->visitor.type_uint64 = qobject_input_type_uint64;
    v->visitor.type_bool = qobject_input_type_bool;
    v->visitor.type_str = qobject_input_type_str;
    v->visitor.type_number = qobject_input_type_number;
    v->visitor.type_any = qobject_input_type_any;
    v->visitor.type_null = qobject_input_type_null;
M
Michael Roth 已提交
640

641
    return &v->visitor;
M
Michael Roth 已提交
642
}
643 644 645

Visitor *qobject_input_visitor_new_keyval(QObject *obj)
{
646
    QObjectInputVisitor *v = qobject_input_visitor_base_new(obj);
647 648 649 650 651 652 653 654 655 656 657 658

    v->visitor.type_int64 = qobject_input_type_int64_keyval;
    v->visitor.type_uint64 = qobject_input_type_uint64_keyval;
    v->visitor.type_bool = qobject_input_type_bool_keyval;
    v->visitor.type_str = qobject_input_type_str;
    v->visitor.type_number = qobject_input_type_number_keyval;
    v->visitor.type_any = qobject_input_type_any;
    v->visitor.type_null = qobject_input_type_null;
    v->visitor.type_size = qobject_input_type_size_keyval;

    return &v->visitor;
}