mirror of
https://github.com/yuzu-emu/unicorn.git
synced 2025-01-03 17:05:47 +00:00
qapi-visit: Expose visit_type_FOO_members()
Dan Berrange reported a case where he needs to work with a QCryptoBlockOptions union type using the OptsVisitor, but only visit one of the branches of that type (the discriminator is not visited directly, but learned externally). When things were boxed, it was easy: just visit the variant directly, which took care of both allocating the variant and visiting its members, then store that pointer in the union type. But now that things are unboxed, we need a way to visit the members without allocation, done by exposing visit_type_FOO_members() to the user. Before the patch, we had quite a bit of code associated with object_members_seen to make sure that a declaration of the helper was in scope before any use of the function. But now that the helper is public and declared in the header, the .c file no longer needs to worry about topological sorting (the helper is always in scope), which leads to some nice cleanups. Backports commit 4d91e9115cc6700113e772b19d1f39bbcf345977 from qemu
This commit is contained in:
parent
d28c6244c0
commit
8728fea067
|
@ -16,7 +16,7 @@
|
|||
#include "qemu-common.h"
|
||||
#include "qapi-visit.h"
|
||||
|
||||
static void visit_type_DummyForceArrays_members(Visitor *v, DummyForceArrays *obj, Error **errp)
|
||||
void visit_type_DummyForceArrays_members(Visitor *v, DummyForceArrays *obj, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
|
||||
|
@ -63,7 +63,7 @@ void visit_type_QapiErrorClass(Visitor *v, const char *name, QapiErrorClass *obj
|
|||
*obj = value;
|
||||
}
|
||||
|
||||
static void visit_type_X86CPUFeatureWordInfo_members(Visitor *v, X86CPUFeatureWordInfo *obj, Error **errp)
|
||||
void visit_type_X86CPUFeatureWordInfo_members(Visitor *v, X86CPUFeatureWordInfo *obj, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
|
||||
|
|
|
@ -42,8 +42,12 @@ void visit_type_uint8List(Visitor *v, const char *name, uint8List **obj, Error *
|
|||
|
||||
#endif /* QAPI_VISIT_BUILTIN */
|
||||
|
||||
|
||||
void visit_type_DummyForceArrays_members(Visitor *v, DummyForceArrays *obj, Error **errp);
|
||||
void visit_type_DummyForceArrays(Visitor *v, const char *name, DummyForceArrays **obj, Error **errp);
|
||||
void visit_type_QapiErrorClass(Visitor *v, const char *name, QapiErrorClass *obj, Error **errp);
|
||||
|
||||
void visit_type_X86CPUFeatureWordInfo_members(Visitor *v, X86CPUFeatureWordInfo *obj, Error **errp);
|
||||
void visit_type_X86CPUFeatureWordInfo(Visitor *v, const char *name, X86CPUFeatureWordInfo **obj, Error **errp);
|
||||
void visit_type_X86CPUFeatureWordInfoList(Visitor *v, const char *name, X86CPUFeatureWordInfoList **obj, Error **errp);
|
||||
void visit_type_X86CPURegister32(Visitor *v, const char *name, X86CPURegister32 *obj, Error **errp);
|
||||
|
|
|
@ -15,10 +15,6 @@
|
|||
from qapi import *
|
||||
import re
|
||||
|
||||
# visit_type_FOO_members() is always emitted; track if a forward declaration
|
||||
# or implementation has already been output.
|
||||
object_members_seen = set()
|
||||
|
||||
def gen_visit_decl(name, scalar=False):
|
||||
c_type = c_name(name) + ' *'
|
||||
if not scalar:
|
||||
|
@ -29,32 +25,18 @@ void visit_type_%(c_name)s(Visitor *v, const char *name, %(c_type)sobj, Error **
|
|||
c_name=c_name(name), c_type=c_type)
|
||||
|
||||
|
||||
def gen_visit_members_decl(typ):
|
||||
if typ.name in object_members_seen:
|
||||
return ''
|
||||
object_members_seen.add(typ.name)
|
||||
def gen_visit_members_decl(name):
|
||||
return mcgen('''
|
||||
|
||||
static void visit_type_%(c_type)s_members(Visitor *v, %(c_type)s *obj, Error **errp);
|
||||
void visit_type_%(c_name)s_members(Visitor *v, %(c_name)s *obj, Error **errp);
|
||||
''',
|
||||
c_type=typ.c_name())
|
||||
c_name=c_name(name))
|
||||
|
||||
|
||||
def gen_visit_object_members(name, base, members, variants):
|
||||
ret = ''
|
||||
ret = mcgen('''
|
||||
|
||||
if base:
|
||||
ret += gen_visit_members_decl(base)
|
||||
if variants:
|
||||
for var in variants.variants:
|
||||
# Ugly special case for simple union TODO get rid of it
|
||||
if not var.simple_union_type():
|
||||
ret += gen_visit_members_decl(var.type)
|
||||
|
||||
object_members_seen.add(name)
|
||||
ret += mcgen('''
|
||||
|
||||
static void visit_type_%(c_name)s_members(Visitor *v, %(c_name)s *obj, Error **errp)
|
||||
void visit_type_%(c_name)s_members(Visitor *v, %(c_name)s *obj, Error **errp)
|
||||
{
|
||||
Error *err = NULL;
|
||||
|
||||
|
@ -174,8 +156,6 @@ def gen_visit_alternate(name, variants):
|
|||
for var in variants.variants:
|
||||
if var.type.alternate_qtype() == 'QTYPE_QINT':
|
||||
promote_int = 'false'
|
||||
if isinstance(var.type, QAPISchemaObjectType):
|
||||
ret += gen_visit_members_decl(var.type)
|
||||
|
||||
ret += mcgen('''
|
||||
|
||||
|
@ -324,6 +304,7 @@ class QAPISchemaGenVisitVisitor(QAPISchemaVisitor):
|
|||
self.defn += defn
|
||||
|
||||
def visit_object_type(self, name, info, base, members, variants):
|
||||
self.decl += gen_visit_members_decl(name)
|
||||
self.decl += gen_visit_decl(name)
|
||||
self.defn += gen_visit_object(name, base, members, variants)
|
||||
|
||||
|
|
Loading…
Reference in a new issue