A future patch will add support for passing a qapi union type as the 'data' of a command. But to do that, the user function for implementing the command, as called by the generated marshal command, must take the corresponding C struct as a single boxed pointer, rather than a breakdown into one parameter per member. This patch adds the internal plubming of a 'box' flag associated with each command and event. For this patch, no behavior changes, other than the testsuite outputting the value of the new flag (always False for now).
Signed-off-by: Eric Blake <ebl...@redhat.com> --- scripts/qapi-commands.py | 93 ++++++++++++++++++--------------- scripts/qapi-event.py | 64 +++++++++++++---------- scripts/qapi-introspect.py | 4 +- scripts/qapi.py | 52 +++++++++++------- tests/qapi-schema/args-member-array.out | 2 +- tests/qapi-schema/event-case.out | 1 + tests/qapi-schema/ident-with-escape.out | 2 +- tests/qapi-schema/indented-expr.out | 4 +- tests/qapi-schema/qapi-schema-test.out | 17 +++--- tests/qapi-schema/returns-int.out | 2 +- tests/qapi-schema/test-qapi.py | 8 +-- 11 files changed, 142 insertions(+), 107 deletions(-) diff --git a/scripts/qapi-commands.py b/scripts/qapi-commands.py index 810a897..ed4b38f 100644 --- a/scripts/qapi-commands.py +++ b/scripts/qapi-commands.py @@ -16,13 +16,13 @@ from qapi import * import re -def gen_command_decl(name, arg_type, ret_type): +def gen_command_decl(name, arg_type, box, ret_type): return mcgen(''' %(c_type)s qmp_%(c_name)s(%(params)s); ''', c_type=(ret_type and ret_type.c_type()) or 'void', c_name=c_name(name), - params=gen_params(arg_type, 'Error **errp')) + params=gen_params(arg_type, box, 'Error **errp')) def gen_err_check(err): @@ -36,15 +36,18 @@ if (%(err)s) { err=err) -def gen_call(name, arg_type, ret_type): +def gen_call(name, arg_type, box, ret_type): ret = '' argstr = '' - if arg_type: - for memb in arg_type.members: - if memb.optional: - argstr += 'has_%s, ' % c_name(memb.name) - argstr += '%s, ' % c_name(memb.name) + if box: + assert False # not implemented + else: + if arg_type: + for memb in arg_type.members: + if memb.optional: + argstr += 'has_%s, ' % c_name(memb.name) + argstr += '%s, ' % c_name(memb.name) lhs = '' if ret_type: @@ -67,7 +70,7 @@ qmp_marshal_output_%(c_name)s(retval, ret, &local_err); return ret -def gen_marshal_vars(arg_type, ret_type): +def gen_marshal_vars(arg_type, box, ret_type): ret = mcgen(''' Error *local_err = NULL; ''') @@ -87,18 +90,21 @@ QapiDeallocVisitor *md; Visitor *v; ''') - for memb in arg_type.members: - if memb.optional: - ret += mcgen(''' + if box: + assert False # not implemented + else: + for memb in arg_type.members: + if memb.optional: + ret += mcgen(''' bool has_%(c_name)s = false; ''', - c_name=c_name(memb.name)) - ret += mcgen(''' + c_name=c_name(memb.name)) + ret += mcgen(''' %(c_type)s %(c_name)s = %(c_null)s; ''', - c_name=c_name(memb.name), - c_type=memb.type.c_type(), - c_null=memb.type.c_null()) + c_name=c_name(memb.name), + c_type=memb.type.c_type(), + c_null=memb.type.c_null()) ret += '\n' else: ret += mcgen(''' @@ -110,7 +116,7 @@ bool has_%(c_name)s = false; return ret -def gen_marshal_input_visit(arg_type, dealloc=False): +def gen_marshal_input_visit(arg_type, box, dealloc=False): ret = '' if not arg_type: @@ -133,28 +139,31 @@ v = qapi_dealloc_get_visitor(md); v = qmp_input_get_visitor(mi); ''') - for memb in arg_type.members: - if memb.optional: - ret += mcgen(''' + if box: + assert False # not implemented + else: + for memb in arg_type.members: + if memb.optional: + ret += mcgen(''' visit_optional(v, &has_%(c_name)s, "%(name)s", %(errp)s); ''', - c_name=c_name(memb.name), name=memb.name, - errp=errparg) - ret += gen_err_check(errarg) - ret += mcgen(''' + c_name=c_name(memb.name), name=memb.name, + errp=errparg) + ret += gen_err_check(errarg) + ret += mcgen(''' if (has_%(c_name)s) { ''', - c_name=c_name(memb.name)) - push_indent() - ret += mcgen(''' + c_name=c_name(memb.name)) + push_indent() + ret += mcgen(''' visit_type_%(c_type)s(v, &%(c_name)s, "%(name)s", %(errp)s); ''', - c_name=c_name(memb.name), name=memb.name, - c_type=memb.type.c_name(), errp=errparg) - ret += gen_err_check(errarg) - if memb.optional: - pop_indent() - ret += mcgen(''' + c_name=c_name(memb.name), name=memb.name, + c_type=memb.type.c_name(), errp=errparg) + ret += gen_err_check(errarg) + if memb.optional: + pop_indent() + ret += mcgen(''' } ''') @@ -209,7 +218,7 @@ def gen_marshal_decl(name): proto=gen_marshal_proto(name)) -def gen_marshal(name, arg_type, ret_type): +def gen_marshal(name, arg_type, box, ret_type): ret = mcgen(''' %(proto)s @@ -217,9 +226,9 @@ def gen_marshal(name, arg_type, ret_type): ''', proto=gen_marshal_proto(name)) - ret += gen_marshal_vars(arg_type, ret_type) - ret += gen_marshal_input_visit(arg_type) - ret += gen_call(name, arg_type, ret_type) + ret += gen_marshal_vars(arg_type, box, ret_type) + ret += gen_marshal_input_visit(arg_type, box) + ret += gen_call(name, arg_type, box, ret_type) if re.search('^ *goto out;', ret, re.MULTILINE): ret += mcgen(''' @@ -229,7 +238,7 @@ out: ret += mcgen(''' error_propagate(errp, local_err); ''') - ret += gen_marshal_input_visit(arg_type, dealloc=True) + ret += gen_marshal_input_visit(arg_type, box, dealloc=True) ret += mcgen(''' } ''') @@ -286,16 +295,16 @@ class QAPISchemaGenCommandVisitor(QAPISchemaVisitor): self._visited_ret_types = None def visit_command(self, name, info, arg_type, ret_type, - gen, success_response): + gen, success_response, box): if not gen: return - self.decl += gen_command_decl(name, arg_type, ret_type) + self.decl += gen_command_decl(name, arg_type, box, ret_type) if ret_type and ret_type not in self._visited_ret_types: self._visited_ret_types.add(ret_type) self.defn += gen_marshal_output(ret_type) if middle_mode: self.decl += gen_marshal_decl(name) - self.defn += gen_marshal(name, arg_type, ret_type) + self.defn += gen_marshal(name, arg_type, box, ret_type) if not middle_mode: self._regy += gen_register_command(name, success_response) diff --git a/scripts/qapi-event.py b/scripts/qapi-event.py index d15fad9..c333d05 100644 --- a/scripts/qapi-event.py +++ b/scripts/qapi-event.py @@ -14,21 +14,21 @@ from qapi import * -def gen_event_send_proto(name, arg_type): +def gen_event_send_proto(name, arg_type, box): return 'void qapi_event_send_%(c_name)s(%(param)s)' % { 'c_name': c_name(name.lower()), - 'param': gen_params(arg_type, 'Error **errp')} + 'param': gen_params(arg_type, box, 'Error **errp')} -def gen_event_send_decl(name, arg_type): +def gen_event_send_decl(name, arg_type, box): return mcgen(''' %(proto)s; ''', - proto=gen_event_send_proto(name, arg_type)) + proto=gen_event_send_proto(name, arg_type, box)) -def gen_event_send(name, arg_type): +def gen_event_send(name, arg_type, box): ret = mcgen(''' %(proto)s @@ -37,7 +37,7 @@ def gen_event_send(name, arg_type): Error *local_err = NULL; QMPEventFuncEmit emit; ''', - proto=gen_event_send_proto(name, arg_type)) + proto=gen_event_send_proto(name, arg_type, box)) if arg_type and arg_type.members: ret += mcgen(''' @@ -66,6 +66,12 @@ def gen_event_send(name, arg_type): v = qmp_output_get_visitor(qov); g_assert(v); +''') + + if box: + assert False # not implemented + else: + ret += mcgen(''' /* Fake visit, as if all members are under a structure */ visit_start_struct(v, NULL, "", "%(name)s", 0, &local_err); if (local_err) { @@ -73,40 +79,40 @@ def gen_event_send(name, arg_type): } ''', - name=name) + name=name) - for memb in arg_type.members: - if memb.optional: - ret += mcgen(''' + for memb in arg_type.members: + if memb.optional: + ret += mcgen(''' if (has_%(c_name)s) { ''', - c_name=c_name(memb.name)) - push_indent() + c_name=c_name(memb.name)) + push_indent() - # Ugly: need to cast away the const - if memb.type.name == "str": - cast = '(char **)' - else: - cast = '' + # Ugly: need to cast away the const + if memb.type.name == "str": + cast = '(char **)' + else: + cast = '' - ret += mcgen(''' + ret += mcgen(''' visit_type_%(c_type)s(v, %(cast)s&%(c_name)s, "%(name)s", &local_err); if (local_err) { goto clean; } ''', - cast=cast, - c_name=c_name(memb.name), - c_type=memb.type.c_name(), - name=memb.name) + cast=cast, + c_name=c_name(memb.name), + c_type=memb.type.c_name(), + name=memb.name) - if memb.optional: - pop_indent() - ret += mcgen(''' + if memb.optional: + pop_indent() + ret += mcgen(''' } ''') - ret += mcgen(''' + ret += mcgen(''' visit_end_struct(v, &local_err); if (local_err) { @@ -154,9 +160,9 @@ class QAPISchemaGenEventVisitor(QAPISchemaVisitor): self.defn += gen_enum_lookup(event_enum_name, self._event_names) self._event_names = None - def visit_event(self, name, info, arg_type): - self.decl += gen_event_send_decl(name, arg_type) - self.defn += gen_event_send(name, arg_type) + def visit_event(self, name, info, arg_type, box): + self.decl += gen_event_send_decl(name, arg_type, box) + self.defn += gen_event_send(name, arg_type, box) self._event_names.append(name) diff --git a/scripts/qapi-introspect.py b/scripts/qapi-introspect.py index eef5f1d..64069ad 100644 --- a/scripts/qapi-introspect.py +++ b/scripts/qapi-introspect.py @@ -151,14 +151,14 @@ const char %(c_name)s[] = %(c_string)s; for m in variants.variants]}) def visit_command(self, name, info, arg_type, ret_type, - gen, success_response): + gen, success_response, box): arg_type = arg_type or self._schema.the_empty_object_type ret_type = ret_type or self._schema.the_empty_object_type self._gen_json(name, 'command', {'arg-type': self._use_type(arg_type), 'ret-type': self._use_type(ret_type)}) - def visit_event(self, name, info, arg_type): + def visit_event(self, name, info, arg_type, box): arg_type = arg_type or self._schema.the_empty_object_type self._gen_json(name, 'event', {'arg-type': self._use_type(arg_type)}) diff --git a/scripts/qapi.py b/scripts/qapi.py index 475d2c2..b407779 100644 --- a/scripts/qapi.py +++ b/scripts/qapi.py @@ -798,10 +798,10 @@ class QAPISchemaVisitor(object): pass def visit_command(self, name, info, arg_type, ret_type, - gen, success_response): + gen, success_response, box): pass - def visit_event(self, name, info, arg_type): + def visit_event(self, name, info, arg_type, box): pass @@ -1036,7 +1036,8 @@ class QAPISchemaAlternateType(QAPISchemaType): class QAPISchemaCommand(QAPISchemaEntity): - def __init__(self, name, info, arg_type, ret_type, gen, success_response): + def __init__(self, name, info, arg_type, ret_type, gen, success_response, + box): QAPISchemaEntity.__init__(self, name, info) assert not arg_type or isinstance(arg_type, str) assert not ret_type or isinstance(ret_type, str) @@ -1046,12 +1047,14 @@ class QAPISchemaCommand(QAPISchemaEntity): self.ret_type = None self.gen = gen self.success_response = success_response + self.box = box def check(self, schema): if self._arg_type_name: self.arg_type = schema.lookup_type(self._arg_type_name) assert isinstance(self.arg_type, QAPISchemaObjectType) - assert not self.arg_type.variants # not implemented + if not self.box: + assert not self.arg_type.variants if self._ret_type_name: self.ret_type = schema.lookup_type(self._ret_type_name) assert isinstance(self.ret_type, QAPISchemaType) @@ -1059,24 +1062,26 @@ class QAPISchemaCommand(QAPISchemaEntity): def visit(self, visitor): visitor.visit_command(self.name, self.info, self.arg_type, self.ret_type, - self.gen, self.success_response) + self.gen, self.success_response, self.box) class QAPISchemaEvent(QAPISchemaEntity): - def __init__(self, name, info, arg_type): + def __init__(self, name, info, arg_type, box): QAPISchemaEntity.__init__(self, name, info) assert not arg_type or isinstance(arg_type, str) self._arg_type_name = arg_type self.arg_type = None + self.box = box def check(self, schema): if self._arg_type_name: self.arg_type = schema.lookup_type(self._arg_type_name) assert isinstance(self.arg_type, QAPISchemaObjectType) - assert not self.arg_type.variants # not implemented + if not self.box: + assert not self.arg_type.variants def visit(self, visitor): - visitor.visit_event(self.name, self.info, self.arg_type) + visitor.visit_event(self.name, self.info, self.arg_type, self.box) class QAPISchema(object): @@ -1232,22 +1237,24 @@ class QAPISchema(object): rets = expr.get('returns') gen = expr.get('gen', True) success_response = expr.get('success-response', True) - if isinstance(data, OrderedDict): + box = expr.get('box', False) + if isinstance(data, dict): data = self._make_implicit_object_type(name, 'arg', self._make_members(data)) if isinstance(rets, list): assert len(rets) == 1 rets = self._make_array_type(rets[0]) self._def_entity(QAPISchemaCommand(name, info, data, rets, gen, - success_response)) + success_response, box)) def _def_event(self, expr, info): name = expr['event'] data = expr.get('data') - if isinstance(data, OrderedDict): + box = expr.get('box', False) + if isinstance(data, dict): data = self._make_implicit_object_type(name, 'arg', self._make_members(data)) - self._def_entity(QAPISchemaEvent(name, info, data)) + self._def_entity(QAPISchemaEvent(name, info, data, box)) def _def_exprs(self): for expr_elem in self.exprs: @@ -1471,18 +1478,23 @@ extern const char *const %(c_name)s_lookup[]; c_name=c_name(name)) return ret -def gen_params(arg_type, extra): + +def gen_params(arg_type, box, extra): if not arg_type: return extra - assert not arg_type.variants ret = '' sep = '' - for memb in arg_type.members: - ret += sep - sep = ', ' - if memb.optional: - ret += 'bool has_%s, ' % c_name(memb.name) - ret += '%s %s' % (memb.type.c_type(is_param=True), c_name(memb.name)) + if box: + assert False # not implemented + else: + assert not arg_type.variants + for memb in arg_type.members: + ret += sep + sep = ', ' + if memb.optional: + ret += 'bool has_%s, ' % c_name(memb.name) + ret += '%s %s' % (memb.type.c_type(is_param=True), + c_name(memb.name)) if extra: ret += sep + extra return ret diff --git a/tests/qapi-schema/args-member-array.out b/tests/qapi-schema/args-member-array.out index b3b92df..f5dc409 100644 --- a/tests/qapi-schema/args-member-array.out +++ b/tests/qapi-schema/args-member-array.out @@ -6,4 +6,4 @@ enum abc ['a', 'b', 'c'] object def member array: abcList optional=False command okay :obj-okay-arg -> None - gen=True success_response=True + gen=True success_response=True box=False diff --git a/tests/qapi-schema/event-case.out b/tests/qapi-schema/event-case.out index cdfd264..97a682a 100644 --- a/tests/qapi-schema/event-case.out +++ b/tests/qapi-schema/event-case.out @@ -1,2 +1,3 @@ object :empty event oops None + box=False diff --git a/tests/qapi-schema/ident-with-escape.out b/tests/qapi-schema/ident-with-escape.out index f4542b1..2381c33 100644 --- a/tests/qapi-schema/ident-with-escape.out +++ b/tests/qapi-schema/ident-with-escape.out @@ -2,4 +2,4 @@ object :empty object :obj-fooA-arg member bar1: str optional=False command fooA :obj-fooA-arg -> None - gen=True success_response=True + gen=True success_response=True box=False diff --git a/tests/qapi-schema/indented-expr.out b/tests/qapi-schema/indented-expr.out index 226d300..f5d2695 100644 --- a/tests/qapi-schema/indented-expr.out +++ b/tests/qapi-schema/indented-expr.out @@ -1,5 +1,5 @@ object :empty command eins None -> None - gen=True success_response=True + gen=True success_response=True box=False command zwei None -> None - gen=True success_response=True + gen=True success_response=True box=False diff --git a/tests/qapi-schema/qapi-schema-test.out b/tests/qapi-schema/qapi-schema-test.out index b6e423e..6f775e5 100644 --- a/tests/qapi-schema/qapi-schema-test.out +++ b/tests/qapi-schema/qapi-schema-test.out @@ -72,9 +72,13 @@ alternate AltTwo case s: str case n: number event EVENT_A None + box=False event EVENT_B None + box=False event EVENT_C :obj-EVENT_C-arg + box=False event EVENT_D :obj-EVENT_D-arg + box=False enum EnumOne ['value1', 'value2', 'value3'] object EventStructOne member struct1: UserDefOne optional=False @@ -152,6 +156,7 @@ object UserDefUnionBase object UserDefZero member integer: int optional=False event __ORG.QEMU_X-EVENT __org.qemu_x-Struct + box=False alternate __org.qemu_x-Alt case __org.qemu_x-branch: str case b: __org.qemu_x-Base @@ -172,14 +177,14 @@ object __org.qemu_x-Union2 tag __org.qemu_x-member1 case __org.qemu_x-value: __org.qemu_x-Struct2 command __org.qemu_x-command :obj-__org.qemu_x-command-arg -> __org.qemu_x-Union1 - gen=True success_response=True + gen=True success_response=True box=False command guest-sync :obj-guest-sync-arg -> any - gen=True success_response=True + gen=True success_response=True box=False command user_def_cmd None -> None - gen=True success_response=True + gen=True success_response=True box=False command user_def_cmd1 :obj-user_def_cmd1-arg -> None - gen=True success_response=True + gen=True success_response=True box=False command user_def_cmd2 :obj-user_def_cmd2-arg -> UserDefTwo - gen=True success_response=True + gen=True success_response=True box=False command user_def_cmd3 :obj-user_def_cmd3-arg -> int - gen=True success_response=True + gen=True success_response=True box=False diff --git a/tests/qapi-schema/returns-int.out b/tests/qapi-schema/returns-int.out index a2da259..911212d 100644 --- a/tests/qapi-schema/returns-int.out +++ b/tests/qapi-schema/returns-int.out @@ -1,3 +1,3 @@ object :empty command guest-get-time None -> int - gen=True success_response=True + gen=True success_response=True box=False diff --git a/tests/qapi-schema/test-qapi.py b/tests/qapi-schema/test-qapi.py index 75f8da8..9811103 100644 --- a/tests/qapi-schema/test-qapi.py +++ b/tests/qapi-schema/test-qapi.py @@ -34,13 +34,15 @@ class QAPISchemaTestVisitor(QAPISchemaVisitor): self._print_variants(variants) def visit_command(self, name, info, arg_type, ret_type, - gen, success_response): + gen, success_response, box): print 'command %s %s -> %s' % \ (name, arg_type and arg_type.name, ret_type and ret_type.name) - print ' gen=%s success_response=%s' % (gen, success_response) + print ' gen=%s success_response=%s box=%s' % (gen, success_response, + box) - def visit_event(self, name, info, arg_type): + def visit_event(self, name, info, arg_type, box): print 'event %s %s' % (name, arg_type and arg_type.name) + print ' box=%s' % box @staticmethod def _print_variants(variants): -- 2.4.3