2015-08-21 07:04:50 +00:00
|
|
|
#
|
|
|
|
# QAPI helper library
|
|
|
|
#
|
|
|
|
# Copyright IBM, Corp. 2011
|
2018-02-19 19:18:46 +00:00
|
|
|
# Copyright (c) 2013-2015 Red Hat Inc.
|
2015-08-21 07:04:50 +00:00
|
|
|
#
|
|
|
|
# Authors:
|
|
|
|
# Anthony Liguori <aliguori@us.ibm.com>
|
|
|
|
# Markus Armbruster <armbru@redhat.com>
|
|
|
|
#
|
|
|
|
# This work is licensed under the terms of the GNU GPL, version 2.
|
|
|
|
# See the COPYING file in the top-level directory.
|
|
|
|
|
|
|
|
import re
|
|
|
|
from ordereddict import OrderedDict
|
2018-02-19 20:31:47 +00:00
|
|
|
import errno
|
2018-02-19 20:19:07 +00:00
|
|
|
import getopt
|
2015-08-21 07:04:50 +00:00
|
|
|
import os
|
|
|
|
import sys
|
2018-02-19 19:40:55 +00:00
|
|
|
import string
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 18:14:35 +00:00
|
|
|
builtin_types = {
|
2015-08-21 07:04:50 +00:00
|
|
|
'str': 'QTYPE_QSTRING',
|
|
|
|
'int': 'QTYPE_QINT',
|
|
|
|
'number': 'QTYPE_QFLOAT',
|
|
|
|
'bool': 'QTYPE_QBOOL',
|
|
|
|
'int8': 'QTYPE_QINT',
|
|
|
|
'int16': 'QTYPE_QINT',
|
|
|
|
'int32': 'QTYPE_QINT',
|
|
|
|
'int64': 'QTYPE_QINT',
|
|
|
|
'uint8': 'QTYPE_QINT',
|
|
|
|
'uint16': 'QTYPE_QINT',
|
|
|
|
'uint32': 'QTYPE_QINT',
|
|
|
|
'uint64': 'QTYPE_QINT',
|
2018-02-19 18:16:16 +00:00
|
|
|
'size': 'QTYPE_QINT',
|
2018-02-19 22:45:25 +00:00
|
|
|
'any': None, # any qtype_code possible, actually
|
2015-08-21 07:04:50 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 19:06:22 +00:00
|
|
|
# Whitelist of commands allowed to return a non-dictionary
|
|
|
|
returns_whitelist = [
|
|
|
|
# From QMP:
|
|
|
|
'human-monitor-command',
|
2018-02-19 22:47:51 +00:00
|
|
|
'qom-get',
|
2018-02-19 19:06:22 +00:00
|
|
|
'query-migrate-cache-size',
|
|
|
|
'query-tpm-models',
|
|
|
|
'query-tpm-types',
|
|
|
|
'ringbuf-read',
|
|
|
|
|
|
|
|
# From QGA:
|
|
|
|
'guest-file-open',
|
|
|
|
'guest-fsfreeze-freeze',
|
|
|
|
'guest-fsfreeze-freeze-list',
|
|
|
|
'guest-fsfreeze-status',
|
|
|
|
'guest-fsfreeze-thaw',
|
|
|
|
'guest-get-time',
|
|
|
|
'guest-set-vcpus',
|
|
|
|
'guest-sync',
|
|
|
|
'guest-sync-delimited',
|
|
|
|
|
|
|
|
# From qapi-schema-test:
|
|
|
|
'user_def_cmd3',
|
|
|
|
]
|
|
|
|
|
2018-02-19 18:56:02 +00:00
|
|
|
enum_types = []
|
|
|
|
struct_types = []
|
|
|
|
union_types = []
|
|
|
|
events = []
|
|
|
|
all_names = {}
|
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
# Parsing the schema into expressions
|
|
|
|
#
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def error_path(parent):
|
|
|
|
res = ""
|
|
|
|
while parent:
|
|
|
|
res = ("In file included from %s:%d:\n" % (parent['file'],
|
|
|
|
parent['line'])) + res
|
|
|
|
parent = parent['parent']
|
|
|
|
return res
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
class QAPISchemaError(Exception):
|
|
|
|
def __init__(self, schema, msg):
|
2018-02-19 22:57:18 +00:00
|
|
|
Exception.__init__(self)
|
2018-02-19 20:34:51 +00:00
|
|
|
self.fname = schema.fname
|
2015-08-21 07:04:50 +00:00
|
|
|
self.msg = msg
|
|
|
|
self.col = 1
|
|
|
|
self.line = schema.line
|
|
|
|
for ch in schema.src[schema.line_pos:schema.pos]:
|
|
|
|
if ch == '\t':
|
|
|
|
self.col = (self.col + 7) % 8 + 1
|
|
|
|
else:
|
|
|
|
self.col += 1
|
2018-02-19 20:34:51 +00:00
|
|
|
self.info = schema.incl_info
|
2015-08-21 07:04:50 +00:00
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return error_path(self.info) + \
|
2018-02-19 20:34:51 +00:00
|
|
|
"%s:%d:%d: %s" % (self.fname, self.line, self.col, self.msg)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
class QAPIExprError(Exception):
|
|
|
|
def __init__(self, expr_info, msg):
|
2018-02-19 22:57:18 +00:00
|
|
|
Exception.__init__(self)
|
2018-02-19 23:50:21 +00:00
|
|
|
assert expr_info
|
2015-08-21 07:04:50 +00:00
|
|
|
self.info = expr_info
|
|
|
|
self.msg = msg
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return error_path(self.info['parent']) + \
|
|
|
|
"%s:%d: %s" % (self.info['file'], self.info['line'], self.msg)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 21:40:55 +00:00
|
|
|
class QAPISchemaParser(object):
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
def __init__(self, fp, previously_included=[], incl_info=None):
|
2018-02-19 20:34:51 +00:00
|
|
|
abs_fname = os.path.abspath(fp.name)
|
2018-02-19 20:39:05 +00:00
|
|
|
fname = fp.name
|
2018-02-19 20:34:51 +00:00
|
|
|
self.fname = fname
|
|
|
|
previously_included.append(abs_fname)
|
|
|
|
self.incl_info = incl_info
|
2015-08-21 07:04:50 +00:00
|
|
|
self.src = fp.read()
|
|
|
|
if self.src == '' or self.src[-1] != '\n':
|
|
|
|
self.src += '\n'
|
|
|
|
self.cursor = 0
|
|
|
|
self.line = 1
|
|
|
|
self.line_pos = 0
|
|
|
|
self.exprs = []
|
|
|
|
self.accept()
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
while self.tok is not None:
|
2018-02-19 20:34:51 +00:00
|
|
|
expr_info = {'file': fname, 'line': self.line,
|
|
|
|
'parent': self.incl_info}
|
2015-08-21 07:04:50 +00:00
|
|
|
expr = self.get_expr(False)
|
|
|
|
if isinstance(expr, dict) and "include" in expr:
|
|
|
|
if len(expr) != 1:
|
2018-02-19 23:07:49 +00:00
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Invalid 'include' directive")
|
2015-08-21 07:04:50 +00:00
|
|
|
include = expr["include"]
|
|
|
|
if not isinstance(include, str):
|
|
|
|
raise QAPIExprError(expr_info,
|
2018-02-19 22:55:32 +00:00
|
|
|
"Value of 'include' must be a string")
|
2018-02-19 20:34:51 +00:00
|
|
|
incl_abs_fname = os.path.join(os.path.dirname(abs_fname),
|
|
|
|
include)
|
2018-02-19 20:40:42 +00:00
|
|
|
# catch inclusion cycle
|
|
|
|
inf = expr_info
|
|
|
|
while inf:
|
|
|
|
if incl_abs_fname == os.path.abspath(inf['file']):
|
2015-08-21 07:04:50 +00:00
|
|
|
raise QAPIExprError(expr_info, "Inclusion loop for %s"
|
|
|
|
% include)
|
2018-02-19 20:40:42 +00:00
|
|
|
inf = inf['parent']
|
2015-08-21 07:04:50 +00:00
|
|
|
# skip multiple include of the same file
|
2018-02-19 20:34:51 +00:00
|
|
|
if incl_abs_fname in previously_included:
|
2015-08-21 07:04:50 +00:00
|
|
|
continue
|
|
|
|
try:
|
2018-02-19 20:34:51 +00:00
|
|
|
fobj = open(incl_abs_fname, 'r')
|
2015-08-21 07:04:50 +00:00
|
|
|
except IOError, e:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
'%s: %s' % (e.strerror, include))
|
2018-02-19 21:40:55 +00:00
|
|
|
exprs_include = QAPISchemaParser(fobj, previously_included,
|
|
|
|
expr_info)
|
2015-08-21 07:04:50 +00:00
|
|
|
self.exprs.extend(exprs_include.exprs)
|
|
|
|
else:
|
|
|
|
expr_elem = {'expr': expr,
|
|
|
|
'info': expr_info}
|
|
|
|
self.exprs.append(expr_elem)
|
|
|
|
|
|
|
|
def accept(self):
|
|
|
|
while True:
|
|
|
|
self.tok = self.src[self.cursor]
|
|
|
|
self.pos = self.cursor
|
|
|
|
self.cursor += 1
|
|
|
|
self.val = None
|
|
|
|
|
|
|
|
if self.tok == '#':
|
|
|
|
self.cursor = self.src.find('\n', self.cursor)
|
2018-02-20 00:06:18 +00:00
|
|
|
elif self.tok in "{}:,[]":
|
2015-08-21 07:04:50 +00:00
|
|
|
return
|
|
|
|
elif self.tok == "'":
|
|
|
|
string = ''
|
|
|
|
esc = False
|
|
|
|
while True:
|
|
|
|
ch = self.src[self.cursor]
|
|
|
|
self.cursor += 1
|
|
|
|
if ch == '\n':
|
|
|
|
raise QAPISchemaError(self,
|
|
|
|
'Missing terminating "\'"')
|
|
|
|
if esc:
|
qapi: Support (subset of) \u escapes in strings
The handling of \ inside QAPI strings was less than ideal, and
really only worked JSON's \/, \\, \", and our extension of \'
(an obvious extension, when you realize we use '' instead of ""
for strings). For other things, like '\n', it resulted in a
literal 'n' instead of a newline.
Of course, at the moment, we really have no use for escaped
characters, as QAPI has to map to C identifiers, and we currently
support ASCII only for that. But down the road, we may add
support for default values for string parameters to a command
or struct; if that happens, it would be nice to correctly support
all JSON escape sequences, such as \n or \uXXXX. This gets us
closer, by supporting Unicode escapes in the ASCII range.
Since JSON does not require \OCTAL or \xXX escapes, and our QMP
implementation does not understand them either, I intentionally
reject it here, but it would be an easy addition if we desired it.
Likewise, intentionally refusing the NUL byte means we don't have
to worry about C strings being shorter than the qapi input.
Backports commit a7f5966b297330f6492020019544ae87c45d699b from qemu
2018-02-19 19:36:46 +00:00
|
|
|
if ch == 'b':
|
|
|
|
string += '\b'
|
|
|
|
elif ch == 'f':
|
|
|
|
string += '\f'
|
|
|
|
elif ch == 'n':
|
|
|
|
string += '\n'
|
|
|
|
elif ch == 'r':
|
|
|
|
string += '\r'
|
|
|
|
elif ch == 't':
|
|
|
|
string += '\t'
|
|
|
|
elif ch == 'u':
|
|
|
|
value = 0
|
2018-02-19 23:07:49 +00:00
|
|
|
for _ in range(0, 4):
|
qapi: Support (subset of) \u escapes in strings
The handling of \ inside QAPI strings was less than ideal, and
really only worked JSON's \/, \\, \", and our extension of \'
(an obvious extension, when you realize we use '' instead of ""
for strings). For other things, like '\n', it resulted in a
literal 'n' instead of a newline.
Of course, at the moment, we really have no use for escaped
characters, as QAPI has to map to C identifiers, and we currently
support ASCII only for that. But down the road, we may add
support for default values for string parameters to a command
or struct; if that happens, it would be nice to correctly support
all JSON escape sequences, such as \n or \uXXXX. This gets us
closer, by supporting Unicode escapes in the ASCII range.
Since JSON does not require \OCTAL or \xXX escapes, and our QMP
implementation does not understand them either, I intentionally
reject it here, but it would be an easy addition if we desired it.
Likewise, intentionally refusing the NUL byte means we don't have
to worry about C strings being shorter than the qapi input.
Backports commit a7f5966b297330f6492020019544ae87c45d699b from qemu
2018-02-19 19:36:46 +00:00
|
|
|
ch = self.src[self.cursor]
|
|
|
|
self.cursor += 1
|
|
|
|
if ch not in "0123456789abcdefABCDEF":
|
|
|
|
raise QAPISchemaError(self,
|
|
|
|
'\\u escape needs 4 '
|
|
|
|
'hex digits')
|
|
|
|
value = (value << 4) + int(ch, 16)
|
|
|
|
# If Python 2 and 3 didn't disagree so much on
|
|
|
|
# how to handle Unicode, then we could allow
|
|
|
|
# Unicode string defaults. But most of QAPI is
|
|
|
|
# ASCII-only, so we aren't losing much for now.
|
|
|
|
if not value or value > 0x7f:
|
|
|
|
raise QAPISchemaError(self,
|
|
|
|
'For now, \\u escape '
|
|
|
|
'only supports non-zero '
|
|
|
|
'values up to \\u007f')
|
|
|
|
string += chr(value)
|
|
|
|
elif ch in "\\/'\"":
|
|
|
|
string += ch
|
|
|
|
else:
|
|
|
|
raise QAPISchemaError(self,
|
2018-02-19 23:07:49 +00:00
|
|
|
"Unknown escape \\%s" % ch)
|
2015-08-21 07:04:50 +00:00
|
|
|
esc = False
|
|
|
|
elif ch == "\\":
|
|
|
|
esc = True
|
|
|
|
elif ch == "'":
|
|
|
|
self.val = string
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
string += ch
|
2018-02-19 20:41:36 +00:00
|
|
|
elif self.src.startswith("true", self.pos):
|
|
|
|
self.val = True
|
|
|
|
self.cursor += 3
|
|
|
|
return
|
|
|
|
elif self.src.startswith("false", self.pos):
|
|
|
|
self.val = False
|
|
|
|
self.cursor += 4
|
|
|
|
return
|
|
|
|
elif self.src.startswith("null", self.pos):
|
|
|
|
self.val = None
|
|
|
|
self.cursor += 3
|
|
|
|
return
|
2015-08-21 07:04:50 +00:00
|
|
|
elif self.tok == '\n':
|
|
|
|
if self.cursor == len(self.src):
|
|
|
|
self.tok = None
|
|
|
|
return
|
|
|
|
self.line += 1
|
|
|
|
self.line_pos = self.cursor
|
|
|
|
elif not self.tok.isspace():
|
|
|
|
raise QAPISchemaError(self, 'Stray "%s"' % self.tok)
|
|
|
|
|
|
|
|
def get_members(self):
|
|
|
|
expr = OrderedDict()
|
|
|
|
if self.tok == '}':
|
|
|
|
self.accept()
|
|
|
|
return expr
|
|
|
|
if self.tok != "'":
|
|
|
|
raise QAPISchemaError(self, 'Expected string or "}"')
|
|
|
|
while True:
|
|
|
|
key = self.val
|
|
|
|
self.accept()
|
|
|
|
if self.tok != ':':
|
|
|
|
raise QAPISchemaError(self, 'Expected ":"')
|
|
|
|
self.accept()
|
|
|
|
if key in expr:
|
|
|
|
raise QAPISchemaError(self, 'Duplicate key "%s"' % key)
|
|
|
|
expr[key] = self.get_expr(True)
|
|
|
|
if self.tok == '}':
|
|
|
|
self.accept()
|
|
|
|
return expr
|
|
|
|
if self.tok != ',':
|
|
|
|
raise QAPISchemaError(self, 'Expected "," or "}"')
|
|
|
|
self.accept()
|
|
|
|
if self.tok != "'":
|
|
|
|
raise QAPISchemaError(self, 'Expected string')
|
|
|
|
|
|
|
|
def get_values(self):
|
|
|
|
expr = []
|
|
|
|
if self.tok == ']':
|
|
|
|
self.accept()
|
|
|
|
return expr
|
2018-02-19 23:07:49 +00:00
|
|
|
if self.tok not in "{['tfn":
|
qapi: Allow true, false and null in schema json
In the near term, we will use it for a sensible-looking
'gen':false inside command declarations, instead of the
current ugly 'gen':'no'.
In the long term, it will allow conversion from shorthand
with defaults mentioned only in side-band documentation:
'data':{'*flag':'bool', '*string':'str'}
into an explicit default value documentation, as in:
'data':{'flag':{'type':'bool', 'optional':true, 'default':true},
'string':{'type':'str', 'optional':true, 'default':null}}
We still don't parse integer values (also necessary before
we can allow explicit defaults), but that can come in a later
series.
Backports commit e53188ada516c814a729551be2448684d6d8ce08 from qemu
2018-02-19 18:57:43 +00:00
|
|
|
raise QAPISchemaError(self, 'Expected "{", "[", "]", string, '
|
|
|
|
'boolean or "null"')
|
2015-08-21 07:04:50 +00:00
|
|
|
while True:
|
|
|
|
expr.append(self.get_expr(True))
|
|
|
|
if self.tok == ']':
|
|
|
|
self.accept()
|
|
|
|
return expr
|
|
|
|
if self.tok != ',':
|
|
|
|
raise QAPISchemaError(self, 'Expected "," or "]"')
|
|
|
|
self.accept()
|
|
|
|
|
|
|
|
def get_expr(self, nested):
|
|
|
|
if self.tok != '{' and not nested:
|
|
|
|
raise QAPISchemaError(self, 'Expected "{"')
|
|
|
|
if self.tok == '{':
|
|
|
|
self.accept()
|
|
|
|
expr = self.get_members()
|
|
|
|
elif self.tok == '[':
|
|
|
|
self.accept()
|
|
|
|
expr = self.get_values()
|
qapi: Allow true, false and null in schema json
In the near term, we will use it for a sensible-looking
'gen':false inside command declarations, instead of the
current ugly 'gen':'no'.
In the long term, it will allow conversion from shorthand
with defaults mentioned only in side-band documentation:
'data':{'*flag':'bool', '*string':'str'}
into an explicit default value documentation, as in:
'data':{'flag':{'type':'bool', 'optional':true, 'default':true},
'string':{'type':'str', 'optional':true, 'default':null}}
We still don't parse integer values (also necessary before
we can allow explicit defaults), but that can come in a later
series.
Backports commit e53188ada516c814a729551be2448684d6d8ce08 from qemu
2018-02-19 18:57:43 +00:00
|
|
|
elif self.tok in "'tfn":
|
2015-08-21 07:04:50 +00:00
|
|
|
expr = self.val
|
|
|
|
self.accept()
|
|
|
|
else:
|
|
|
|
raise QAPISchemaError(self, 'Expected "{", "[" or string')
|
|
|
|
return expr
|
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
# Semantic analysis of schema expressions
|
2018-02-19 21:43:28 +00:00
|
|
|
# TODO fold into QAPISchema
|
|
|
|
# TODO catching name collisions in generated code would be nice
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def find_base_fields(base):
|
|
|
|
base_struct_define = find_struct(base)
|
|
|
|
if not base_struct_define:
|
|
|
|
return None
|
|
|
|
return base_struct_define['data']
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 18:44:13 +00:00
|
|
|
# Return the qtype of an alternate branch, or None on error.
|
|
|
|
def find_alternate_member_qtype(qapi_type):
|
2018-02-19 23:07:49 +00:00
|
|
|
if qapi_type in builtin_types:
|
2018-02-19 18:34:18 +00:00
|
|
|
return builtin_types[qapi_type]
|
|
|
|
elif find_struct(qapi_type):
|
|
|
|
return "QTYPE_QDICT"
|
|
|
|
elif find_enum(qapi_type):
|
|
|
|
return "QTYPE_QSTRING"
|
2018-02-19 18:44:13 +00:00
|
|
|
elif find_union(qapi_type):
|
|
|
|
return "QTYPE_QDICT"
|
2018-02-19 18:34:18 +00:00
|
|
|
return None
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
# Return the discriminator enum define if discriminator is specified as an
|
|
|
|
# enum type, otherwise return None.
|
|
|
|
def discriminator_find_enum_define(expr):
|
|
|
|
base = expr.get('base')
|
|
|
|
discriminator = expr.get('discriminator')
|
|
|
|
|
|
|
|
if not (discriminator and base):
|
|
|
|
return None
|
|
|
|
|
|
|
|
base_fields = find_base_fields(base)
|
|
|
|
if not base_fields:
|
|
|
|
return None
|
|
|
|
|
|
|
|
discriminator_type = base_fields.get(discriminator)
|
|
|
|
if not discriminator_type:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return find_enum(discriminator_type)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 21:22:16 +00:00
|
|
|
# FIXME should enforce "other than downstream extensions [...], all
|
|
|
|
# names should begin with a letter".
|
2018-02-19 19:04:46 +00:00
|
|
|
valid_name = re.compile('^[a-zA-Z_][a-zA-Z0-9_.-]*$')
|
2018-02-19 23:07:49 +00:00
|
|
|
def check_name(expr_info, source, name, allow_optional=False,
|
|
|
|
enum_member=False):
|
2018-02-19 19:04:46 +00:00
|
|
|
global valid_name
|
|
|
|
membername = name
|
|
|
|
|
|
|
|
if not isinstance(name, str):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s requires a string name" % source)
|
|
|
|
if name.startswith('*'):
|
|
|
|
membername = name[1:]
|
|
|
|
if not allow_optional:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s does not allow optional name '%s'"
|
|
|
|
% (source, name))
|
|
|
|
# Enum members can start with a digit, because the generated C
|
|
|
|
# code always prefixes it with the enum name
|
|
|
|
if enum_member:
|
|
|
|
membername = '_' + membername
|
qapi: Reserve 'q_*' and 'has_*' member names
c_name() produces names starting with 'q_' when protecting a
dictionary member name that would fail to directly compile, but
in doing so can cause clashes with any member name already
beginning with 'q-' or 'q_'. Likewise, we create a C name 'has_'
for any optional member that can clash with any member name
beginning with 'has-' or 'has_'.
Technically, rather than blindly reserving the namespace,
we could try to complain about user names only when an actual
collision occurs, or even teach c_name() how to munge names
to avoid collisions. But it is not trivial, especially when
collisions can occur across multiple types (such as via
inheritance or flat unions). Besides, no existing .json
files are trying to use these names. So it's easier to just
outright forbid the potential for collision. We can always
relax things in the future if a real need arises for QMP to
express member names that have been forbidden here.
'has_' only has to be reserved for struct/union member names,
while 'q_' is reserved everywhere (matching the fact that
only members can be optional, while we use c_name() for munging
both members and entities). Note that we could relax 'q_'
restrictions on entities independently from member names; for
example, c_name('qmp_' + 'unix') would result in a different
function name than our current 'qmp_' + c_name('unix').
Update and add tests to cover the new error messages.
Backports commit 9fb081e0b98409556d023c7193eeb68947cd1211 from qemu
2018-02-20 00:09:15 +00:00
|
|
|
# Reserve the entire 'q_' namespace for c_name()
|
|
|
|
if not valid_name.match(membername) or \
|
|
|
|
c_name(membername, False).startswith('q_'):
|
2018-02-19 19:04:46 +00:00
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s uses invalid name '%s'" % (source, name))
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
|
|
|
def add_name(name, info, meta, implicit=False):
|
2018-02-19 20:47:16 +00:00
|
|
|
global all_names
|
|
|
|
check_name(info, "'%s'" % meta, name)
|
2018-02-19 21:22:16 +00:00
|
|
|
# FIXME should reject names that differ only in '_' vs. '.'
|
|
|
|
# vs. '-', because they're liable to clash in generated C.
|
2018-02-19 20:47:16 +00:00
|
|
|
if name in all_names:
|
|
|
|
raise QAPIExprError(info,
|
|
|
|
"%s '%s' is already defined"
|
|
|
|
% (all_names[name], name))
|
2018-02-20 00:08:11 +00:00
|
|
|
if not implicit and (name.endswith('Kind') or name.endswith('List')):
|
2018-02-19 20:47:16 +00:00
|
|
|
raise QAPIExprError(info,
|
2018-02-20 00:08:11 +00:00
|
|
|
"%s '%s' should not end in '%s'"
|
|
|
|
% (meta, name, name[-4:]))
|
2018-02-19 20:47:16 +00:00
|
|
|
all_names[name] = meta
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def add_struct(definition, info):
|
|
|
|
global struct_types
|
|
|
|
name = definition['struct']
|
|
|
|
add_name(name, info, 'struct')
|
|
|
|
struct_types.append(definition)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def find_struct(name):
|
|
|
|
global struct_types
|
|
|
|
for struct in struct_types:
|
|
|
|
if struct['struct'] == name:
|
|
|
|
return struct
|
|
|
|
return None
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def add_union(definition, info):
|
|
|
|
global union_types
|
|
|
|
name = definition['union']
|
|
|
|
add_name(name, info, 'union')
|
|
|
|
union_types.append(definition)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def find_union(name):
|
|
|
|
global union_types
|
|
|
|
for union in union_types:
|
|
|
|
if union['union'] == name:
|
|
|
|
return union
|
|
|
|
return None
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
|
|
|
def add_enum(name, info, enum_values=None, implicit=False):
|
2018-02-19 20:47:16 +00:00
|
|
|
global enum_types
|
|
|
|
add_name(name, info, 'enum', implicit)
|
|
|
|
enum_types.append({"enum_name": name, "enum_values": enum_values})
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def find_enum(name):
|
|
|
|
global enum_types
|
|
|
|
for enum in enum_types:
|
|
|
|
if enum['enum_name'] == name:
|
|
|
|
return enum
|
|
|
|
return None
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
def is_enum(name):
|
2018-02-19 23:07:49 +00:00
|
|
|
return find_enum(name) is not None
|
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
def check_type(expr_info, source, value, allow_array=False,
|
|
|
|
allow_dict=False, allow_optional=False,
|
|
|
|
allow_metas=[]):
|
2018-02-19 19:01:11 +00:00
|
|
|
global all_names
|
|
|
|
|
|
|
|
if value is None:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Check if array type for value is okay
|
|
|
|
if isinstance(value, list):
|
|
|
|
if not allow_array:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s cannot be an array" % source)
|
|
|
|
if len(value) != 1 or not isinstance(value[0], str):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s: array type must contain single type name"
|
|
|
|
% source)
|
|
|
|
value = value[0]
|
|
|
|
|
|
|
|
# Check if type name for value is okay
|
|
|
|
if isinstance(value, str):
|
2018-02-19 23:07:49 +00:00
|
|
|
if value not in all_names:
|
2018-02-19 19:01:11 +00:00
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s uses unknown type '%s'"
|
qapi: Simplify error reporting for array types
check_type() first checks and peels off the array type, then checks
the element type. For two out of four error messages, it takes pains
to report errors for "array of T" instead of just T. Odd. Let's
examine the errors.
* Unknown element type, e.g.
tests/qapi-schema/args-array-unknown.json:
Member 'array' of 'data' for command 'oops' uses unknown type
'array of NoSuchType'
To make sense of this, you need to know that 'array of NoSuchType'
refers to '[NoSuchType]'. Easy enough. However, simply reporting
Member 'array' of 'data' for command 'oops' uses unknown type
'NoSuchType'
is at least as easy to understand.
* Element type's meta-type is inadmissible, e.g.
tests/qapi-schema/returns-whitelist.json:
'returns' for command 'no-way-this-will-get-whitelisted' cannot
use built-in type 'array of int'
'array of int' is technically not a built-in type, but that's
pedantry. However, simply reporting
'returns' for command 'no-way-this-will-get-whitelisted' cannot
use built-in type 'int'
avoids the issue, and is at least as easy to understand.
* The remaining two errors are unreachable, because the array checking
ensures that value is a string.
Thus, reporting some errors for "array of T" instead of just T works,
but doesn't really improve things. Drop it.
Backports commit eddf817bd823a90df209dfbdc2a0b2ec33b7cb77 from qemu
2018-02-19 21:31:42 +00:00
|
|
|
% (source, value))
|
2018-02-19 19:01:11 +00:00
|
|
|
if not all_names[value] in allow_metas:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s cannot use %s type '%s'"
|
qapi: Simplify error reporting for array types
check_type() first checks and peels off the array type, then checks
the element type. For two out of four error messages, it takes pains
to report errors for "array of T" instead of just T. Odd. Let's
examine the errors.
* Unknown element type, e.g.
tests/qapi-schema/args-array-unknown.json:
Member 'array' of 'data' for command 'oops' uses unknown type
'array of NoSuchType'
To make sense of this, you need to know that 'array of NoSuchType'
refers to '[NoSuchType]'. Easy enough. However, simply reporting
Member 'array' of 'data' for command 'oops' uses unknown type
'NoSuchType'
is at least as easy to understand.
* Element type's meta-type is inadmissible, e.g.
tests/qapi-schema/returns-whitelist.json:
'returns' for command 'no-way-this-will-get-whitelisted' cannot
use built-in type 'array of int'
'array of int' is technically not a built-in type, but that's
pedantry. However, simply reporting
'returns' for command 'no-way-this-will-get-whitelisted' cannot
use built-in type 'int'
avoids the issue, and is at least as easy to understand.
* The remaining two errors are unreachable, because the array checking
ensures that value is a string.
Thus, reporting some errors for "array of T" instead of just T works,
but doesn't really improve things. Drop it.
Backports commit eddf817bd823a90df209dfbdc2a0b2ec33b7cb77 from qemu
2018-02-19 21:31:42 +00:00
|
|
|
% (source, all_names[value], value))
|
2018-02-19 19:01:11 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if not allow_dict:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s should be a type name" % source)
|
2018-02-19 21:30:47 +00:00
|
|
|
if not isinstance(value, OrderedDict):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"%s should be a dictionary or type name" % source)
|
|
|
|
|
|
|
|
# value is a dictionary, check that each member is okay
|
2018-02-19 19:01:11 +00:00
|
|
|
for (key, arg) in value.items():
|
2018-02-19 19:04:46 +00:00
|
|
|
check_name(expr_info, "Member of %s" % source, key,
|
|
|
|
allow_optional=allow_optional)
|
qapi: Reserve 'q_*' and 'has_*' member names
c_name() produces names starting with 'q_' when protecting a
dictionary member name that would fail to directly compile, but
in doing so can cause clashes with any member name already
beginning with 'q-' or 'q_'. Likewise, we create a C name 'has_'
for any optional member that can clash with any member name
beginning with 'has-' or 'has_'.
Technically, rather than blindly reserving the namespace,
we could try to complain about user names only when an actual
collision occurs, or even teach c_name() how to munge names
to avoid collisions. But it is not trivial, especially when
collisions can occur across multiple types (such as via
inheritance or flat unions). Besides, no existing .json
files are trying to use these names. So it's easier to just
outright forbid the potential for collision. We can always
relax things in the future if a real need arises for QMP to
express member names that have been forbidden here.
'has_' only has to be reserved for struct/union member names,
while 'q_' is reserved everywhere (matching the fact that
only members can be optional, while we use c_name() for munging
both members and entities). Note that we could relax 'q_'
restrictions on entities independently from member names; for
example, c_name('qmp_' + 'unix') would result in a different
function name than our current 'qmp_' + c_name('unix').
Update and add tests to cover the new error messages.
Backports commit 9fb081e0b98409556d023c7193eeb68947cd1211 from qemu
2018-02-20 00:09:15 +00:00
|
|
|
if c_name(key, False).startswith('has_'):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Member of %s uses reserved name '%s'"
|
|
|
|
% (source, key))
|
2018-02-19 19:28:44 +00:00
|
|
|
# Todo: allow dictionaries to represent default values of
|
|
|
|
# an optional argument.
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "Member '%s' of %s" % (key, source), arg,
|
2018-02-19 22:49:43 +00:00
|
|
|
allow_array=True,
|
2018-02-19 19:01:11 +00:00
|
|
|
allow_metas=['built-in', 'union', 'alternate', 'struct',
|
2018-02-19 19:28:44 +00:00
|
|
|
'enum'])
|
2018-02-19 19:01:11 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
|
|
|
def check_member_clash(expr_info, base_name, data, source=""):
|
2018-02-19 19:38:42 +00:00
|
|
|
base = find_struct(base_name)
|
|
|
|
assert base
|
|
|
|
base_members = base['data']
|
|
|
|
for key in data.keys():
|
|
|
|
if key.startswith('*'):
|
|
|
|
key = key[1:]
|
|
|
|
if key in base_members or "*" + key in base_members:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Member name '%s'%s clashes with base '%s'"
|
|
|
|
% (key, source, base_name))
|
|
|
|
if base.get('base'):
|
|
|
|
check_member_clash(expr_info, base['base'], data, source)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 19:01:11 +00:00
|
|
|
def check_command(expr, expr_info):
|
|
|
|
name = expr['command']
|
2018-02-19 19:09:01 +00:00
|
|
|
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "'data' for command '%s'" % name,
|
2018-02-19 19:04:46 +00:00
|
|
|
expr.get('data'), allow_dict=True, allow_optional=True,
|
2018-02-19 22:49:43 +00:00
|
|
|
allow_metas=['struct'])
|
2018-02-19 19:06:22 +00:00
|
|
|
returns_meta = ['union', 'struct']
|
|
|
|
if name in returns_whitelist:
|
|
|
|
returns_meta += ['built-in', 'alternate', 'enum']
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "'returns' for command '%s'" % name,
|
qapi: Command returning anonymous type doesn't work, outlaw
Reproducer: with
{ 'command': 'user_def_cmd4', 'returns': { 'a': 'int' } }
added to qapi-schema-test.json, qapi-commands.py dies when it tries to
generate the command handler function
Traceback (most recent call last):
File "/work/armbru/qemu/scripts/qapi-commands.py", line 359, in <module>
ret = generate_command_decl(cmd['command'], arglist, ret_type) + "\n"
File "/work/armbru/qemu/scripts/qapi-commands.py", line 29, in generate_command_decl
ret_type=c_type(ret_type), name=c_name(name),
File "/work/armbru/qemu/scripts/qapi.py", line 927, in c_type
assert isinstance(value, str) and value != ""
AssertionError
because the return type doesn't exist.
Simply outlaw this usage, and drop or dumb down test cases accordingly.
Backports commit 9b090d42aea9a0abbf39a1d75561a186057b5fe6 from qemu
2018-02-19 21:24:11 +00:00
|
|
|
expr.get('returns'), allow_array=True,
|
2018-02-19 22:49:43 +00:00
|
|
|
allow_optional=True, allow_metas=returns_meta)
|
2018-02-19 19:01:11 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def check_event(expr, expr_info):
|
2018-02-19 18:56:02 +00:00
|
|
|
global events
|
|
|
|
name = expr['event']
|
|
|
|
|
|
|
|
if name.upper() == 'MAX':
|
|
|
|
raise QAPIExprError(expr_info, "Event name 'MAX' cannot be created")
|
|
|
|
events.append(name)
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "'data' for event '%s'" % name,
|
2018-02-19 19:04:46 +00:00
|
|
|
expr.get('data'), allow_dict=True, allow_optional=True,
|
2018-02-19 21:23:21 +00:00
|
|
|
allow_metas=['struct'])
|
2018-02-19 18:56:02 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def check_union(expr, expr_info):
|
|
|
|
name = expr['union']
|
|
|
|
base = expr.get('base')
|
|
|
|
discriminator = expr.get('discriminator')
|
|
|
|
members = expr['data']
|
2018-02-19 23:09:08 +00:00
|
|
|
values = {'MAX': '(automatic)', 'KIND': '(automatic)'}
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 18:44:13 +00:00
|
|
|
# Two types of unions, determined by discriminator.
|
|
|
|
|
|
|
|
# With no discriminator it is a simple union.
|
|
|
|
if discriminator is None:
|
2015-08-21 07:04:50 +00:00
|
|
|
enum_define = None
|
2018-02-19 23:07:49 +00:00
|
|
|
allow_metas = ['built-in', 'union', 'alternate', 'struct', 'enum']
|
2018-02-19 18:34:18 +00:00
|
|
|
if base is not None:
|
|
|
|
raise QAPIExprError(expr_info,
|
2018-02-19 18:44:13 +00:00
|
|
|
"Simple union '%s' must not have a base"
|
2018-02-19 18:34:18 +00:00
|
|
|
% name)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
|
|
|
# Else, it's a flat union.
|
|
|
|
else:
|
2018-02-19 18:34:18 +00:00
|
|
|
# The object must have a string member 'base'.
|
2018-02-19 23:10:23 +00:00
|
|
|
check_type(expr_info, "'base' for union '%s'" % name,
|
|
|
|
base, allow_metas=['struct'])
|
|
|
|
if not base:
|
2015-08-21 07:04:50 +00:00
|
|
|
raise QAPIExprError(expr_info,
|
2018-02-19 23:10:23 +00:00
|
|
|
"Flat union '%s' must have a base"
|
2015-08-21 07:04:50 +00:00
|
|
|
% name)
|
2018-02-19 18:34:18 +00:00
|
|
|
base_fields = find_base_fields(base)
|
2018-02-19 23:10:23 +00:00
|
|
|
assert base_fields
|
|
|
|
|
2018-02-19 19:04:46 +00:00
|
|
|
# The value of member 'discriminator' must name a non-optional
|
2018-02-19 19:13:27 +00:00
|
|
|
# member of the base struct.
|
2018-02-19 19:04:46 +00:00
|
|
|
check_name(expr_info, "Discriminator of flat union '%s'" % name,
|
|
|
|
discriminator)
|
2015-08-21 07:04:50 +00:00
|
|
|
discriminator_type = base_fields.get(discriminator)
|
|
|
|
if not discriminator_type:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Discriminator '%s' is not a member of base "
|
2018-02-19 19:13:27 +00:00
|
|
|
"struct '%s'"
|
2015-08-21 07:04:50 +00:00
|
|
|
% (discriminator, base))
|
|
|
|
enum_define = find_enum(discriminator_type)
|
2018-02-19 23:07:49 +00:00
|
|
|
allow_metas = ['struct']
|
2015-08-21 07:04:50 +00:00
|
|
|
# Do not allow string discriminator
|
|
|
|
if not enum_define:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Discriminator '%s' must be of enumeration "
|
|
|
|
"type" % discriminator)
|
|
|
|
|
|
|
|
# Check every branch
|
|
|
|
for (key, value) in members.items():
|
2018-02-19 19:04:46 +00:00
|
|
|
check_name(expr_info, "Member of union '%s'" % name, key)
|
|
|
|
|
2018-02-19 19:01:11 +00:00
|
|
|
# Each value must name a known type; furthermore, in flat unions,
|
2018-02-19 19:38:42 +00:00
|
|
|
# branches must be a struct with no overlapping member names
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "Member '%s' of union '%s'" % (key, name),
|
2018-02-19 20:48:10 +00:00
|
|
|
value, allow_array=not base, allow_metas=allow_metas)
|
2018-02-19 19:38:42 +00:00
|
|
|
if base:
|
|
|
|
branch_struct = find_struct(value)
|
|
|
|
assert branch_struct
|
|
|
|
check_member_clash(expr_info, base, branch_struct['data'],
|
|
|
|
" of branch '%s'" % key)
|
2018-02-19 19:01:11 +00:00
|
|
|
|
2018-02-19 18:34:18 +00:00
|
|
|
# If the discriminator names an enum type, then all members
|
2018-02-19 23:09:08 +00:00
|
|
|
# of 'data' must also be members of the enum type, which in turn
|
|
|
|
# must not collide with the discriminator name.
|
2018-02-19 18:34:18 +00:00
|
|
|
if enum_define:
|
2018-02-19 23:07:49 +00:00
|
|
|
if key not in enum_define['enum_values']:
|
2018-02-19 18:34:18 +00:00
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Discriminator value '%s' is not found in "
|
|
|
|
"enum '%s'" %
|
|
|
|
(key, enum_define["enum_name"]))
|
2018-02-19 23:09:08 +00:00
|
|
|
if discriminator in enum_define['enum_values']:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Discriminator name '%s' collides with "
|
|
|
|
"enum value in '%s'" %
|
|
|
|
(discriminator, enum_define["enum_name"]))
|
2018-02-19 18:34:18 +00:00
|
|
|
|
|
|
|
# Otherwise, check for conflicts in the generated enum
|
|
|
|
else:
|
2018-02-19 19:46:43 +00:00
|
|
|
c_key = camel_to_upper(key)
|
2018-02-19 18:34:18 +00:00
|
|
|
if c_key in values:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Union '%s' member '%s' clashes with '%s'"
|
|
|
|
% (name, key, values[c_key]))
|
|
|
|
values[c_key] = key
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 18:44:13 +00:00
|
|
|
def check_alternate(expr, expr_info):
|
2018-02-19 18:49:54 +00:00
|
|
|
name = expr['alternate']
|
2018-02-19 18:44:13 +00:00
|
|
|
members = expr['data']
|
2018-02-19 23:07:49 +00:00
|
|
|
values = {'MAX': '(automatic)'}
|
2018-02-19 18:44:13 +00:00
|
|
|
types_seen = {}
|
|
|
|
|
|
|
|
# Check every branch
|
|
|
|
for (key, value) in members.items():
|
2018-02-19 19:04:46 +00:00
|
|
|
check_name(expr_info, "Member of alternate '%s'" % name, key)
|
|
|
|
|
2018-02-19 18:44:13 +00:00
|
|
|
# Check for conflicts in the generated enum
|
2018-02-19 19:46:43 +00:00
|
|
|
c_key = camel_to_upper(key)
|
2018-02-19 18:44:13 +00:00
|
|
|
if c_key in values:
|
|
|
|
raise QAPIExprError(expr_info,
|
2018-02-19 18:49:54 +00:00
|
|
|
"Alternate '%s' member '%s' clashes with '%s'"
|
|
|
|
% (name, key, values[c_key]))
|
2018-02-19 18:44:13 +00:00
|
|
|
values[c_key] = key
|
2018-02-19 18:34:18 +00:00
|
|
|
|
2018-02-19 18:44:13 +00:00
|
|
|
# Ensure alternates have no type conflicts.
|
2018-02-19 19:01:11 +00:00
|
|
|
check_type(expr_info, "Member '%s' of alternate '%s'" % (key, name),
|
|
|
|
value,
|
|
|
|
allow_metas=['built-in', 'union', 'struct', 'enum'])
|
2018-02-19 18:44:13 +00:00
|
|
|
qtype = find_alternate_member_qtype(value)
|
2018-02-19 19:01:11 +00:00
|
|
|
assert qtype
|
2018-02-19 18:44:13 +00:00
|
|
|
if qtype in types_seen:
|
|
|
|
raise QAPIExprError(expr_info,
|
2018-02-19 18:49:54 +00:00
|
|
|
"Alternate '%s' member '%s' can't "
|
2018-02-19 18:44:13 +00:00
|
|
|
"be distinguished from member '%s'"
|
|
|
|
% (name, key, types_seen[qtype]))
|
|
|
|
types_seen[qtype] = key
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 18:23:52 +00:00
|
|
|
def check_enum(expr, expr_info):
|
|
|
|
name = expr['enum']
|
|
|
|
members = expr.get('data')
|
2018-02-19 21:38:38 +00:00
|
|
|
prefix = expr.get('prefix')
|
2018-02-19 23:07:49 +00:00
|
|
|
values = {'MAX': '(automatic)'}
|
2018-02-19 18:23:52 +00:00
|
|
|
|
|
|
|
if not isinstance(members, list):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Enum '%s' requires an array for 'data'" % name)
|
2018-02-19 21:38:38 +00:00
|
|
|
if prefix is not None and not isinstance(prefix, str):
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Enum '%s' requires a string for 'prefix'" % name)
|
2018-02-19 18:23:52 +00:00
|
|
|
for member in members:
|
2018-02-19 23:07:49 +00:00
|
|
|
check_name(expr_info, "Member of enum '%s'" % name, member,
|
2018-02-19 19:04:46 +00:00
|
|
|
enum_member=True)
|
2018-02-19 19:46:43 +00:00
|
|
|
key = camel_to_upper(member)
|
2018-02-19 18:23:52 +00:00
|
|
|
if key in values:
|
|
|
|
raise QAPIExprError(expr_info,
|
|
|
|
"Enum '%s' member '%s' clashes with '%s'"
|
|
|
|
% (name, member, values[key]))
|
|
|
|
values[key] = member
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 19:01:11 +00:00
|
|
|
def check_struct(expr, expr_info):
|
2018-02-19 19:13:27 +00:00
|
|
|
name = expr['struct']
|
2018-02-19 19:01:11 +00:00
|
|
|
members = expr['data']
|
|
|
|
|
2018-02-19 19:13:27 +00:00
|
|
|
check_type(expr_info, "'data' for struct '%s'" % name, members,
|
2018-02-19 19:04:46 +00:00
|
|
|
allow_dict=True, allow_optional=True)
|
2018-02-19 19:13:27 +00:00
|
|
|
check_type(expr_info, "'base' for struct '%s'" % name, expr.get('base'),
|
2018-02-19 19:01:11 +00:00
|
|
|
allow_metas=['struct'])
|
2018-02-19 19:38:42 +00:00
|
|
|
if expr.get('base'):
|
|
|
|
check_member_clash(expr_info, expr['base'], expr['data'])
|
2018-02-19 19:01:11 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 18:52:10 +00:00
|
|
|
def check_keys(expr_elem, meta, required, optional=[]):
|
|
|
|
expr = expr_elem['expr']
|
|
|
|
info = expr_elem['info']
|
|
|
|
name = expr[meta]
|
|
|
|
if not isinstance(name, str):
|
|
|
|
raise QAPIExprError(info,
|
|
|
|
"'%s' key must have a string value" % meta)
|
2018-02-19 23:07:49 +00:00
|
|
|
required = required + [meta]
|
2018-02-19 18:52:10 +00:00
|
|
|
for (key, value) in expr.items():
|
2018-02-19 23:07:49 +00:00
|
|
|
if key not in required and key not in optional:
|
2018-02-19 18:52:10 +00:00
|
|
|
raise QAPIExprError(info,
|
|
|
|
"Unknown key '%s' in %s '%s'"
|
|
|
|
% (key, meta, name))
|
2018-02-19 23:07:49 +00:00
|
|
|
if (key == 'gen' or key == 'success-response') and value is not False:
|
2018-02-19 19:09:01 +00:00
|
|
|
raise QAPIExprError(info,
|
|
|
|
"'%s' of %s '%s' should only use false value"
|
|
|
|
% (key, meta, name))
|
2018-02-19 18:52:10 +00:00
|
|
|
for key in required:
|
2018-02-19 23:07:49 +00:00
|
|
|
if key not in expr:
|
2018-02-19 18:52:10 +00:00
|
|
|
raise QAPIExprError(info,
|
|
|
|
"Key '%s' is missing from %s '%s'"
|
|
|
|
% (key, meta, name))
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:44:18 +00:00
|
|
|
def check_exprs(exprs):
|
2018-02-19 18:56:02 +00:00
|
|
|
global all_names
|
|
|
|
|
2018-02-19 20:44:18 +00:00
|
|
|
# Learn the types and check for valid expression keys
|
|
|
|
for builtin in builtin_types.keys():
|
|
|
|
all_names[builtin] = 'built-in'
|
|
|
|
for expr_elem in exprs:
|
|
|
|
expr = expr_elem['expr']
|
|
|
|
info = expr_elem['info']
|
2018-02-19 23:07:49 +00:00
|
|
|
if 'enum' in expr:
|
2018-02-19 21:38:38 +00:00
|
|
|
check_keys(expr_elem, 'enum', ['data'], ['prefix'])
|
2018-02-19 20:44:18 +00:00
|
|
|
add_enum(expr['enum'], info, expr['data'])
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'union' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_keys(expr_elem, 'union', ['data'],
|
|
|
|
['base', 'discriminator'])
|
|
|
|
add_union(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'alternate' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_keys(expr_elem, 'alternate', ['data'])
|
|
|
|
add_name(expr['alternate'], info, 'alternate')
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'struct' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_keys(expr_elem, 'struct', ['data'], ['base'])
|
|
|
|
add_struct(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'command' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_keys(expr_elem, 'command', [],
|
|
|
|
['data', 'returns', 'gen', 'success-response'])
|
|
|
|
add_name(expr['command'], info, 'command')
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'event' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_keys(expr_elem, 'event', [], ['data'])
|
|
|
|
add_name(expr['event'], info, 'event')
|
|
|
|
else:
|
|
|
|
raise QAPIExprError(expr_elem['info'],
|
|
|
|
"Expression is missing metatype")
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 20:44:18 +00:00
|
|
|
# Try again for hidden UnionKind enum
|
|
|
|
for expr_elem in exprs:
|
|
|
|
expr = expr_elem['expr']
|
2018-02-19 23:07:49 +00:00
|
|
|
if 'union' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
if not discriminator_find_enum_define(expr):
|
|
|
|
add_enum('%sKind' % expr['union'], expr_elem['info'],
|
2018-02-19 18:56:02 +00:00
|
|
|
implicit=True)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'alternate' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
add_enum('%sKind' % expr['alternate'], expr_elem['info'],
|
|
|
|
implicit=True)
|
|
|
|
|
|
|
|
# Validate that exprs make sense
|
|
|
|
for expr_elem in exprs:
|
|
|
|
expr = expr_elem['expr']
|
|
|
|
info = expr_elem['info']
|
2018-02-19 18:37:31 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
if 'enum' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_enum(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'union' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_union(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'alternate' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_alternate(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'struct' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_struct(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'command' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_command(expr, info)
|
2018-02-19 23:07:49 +00:00
|
|
|
elif 'event' in expr:
|
2018-02-19 20:44:18 +00:00
|
|
|
check_event(expr, info)
|
|
|
|
else:
|
|
|
|
assert False, 'unexpected meta type'
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
return exprs
|
2018-02-19 20:44:18 +00:00
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
#
|
|
|
|
# Schema compiler frontend
|
|
|
|
#
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
class QAPISchemaEntity(object):
|
|
|
|
def __init__(self, name, info):
|
|
|
|
assert isinstance(name, str)
|
|
|
|
self.name = name
|
2018-02-20 00:04:36 +00:00
|
|
|
# For explicitly defined entities, info points to the (explicit)
|
|
|
|
# definition. For builtins (and their arrays), info is None.
|
|
|
|
# For implicitly defined entities, info points to a place that
|
|
|
|
# triggered the implicit definition (there may be more than one
|
|
|
|
# such place).
|
2018-02-19 21:43:28 +00:00
|
|
|
self.info = info
|
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def c_name(self):
|
|
|
|
return c_name(self.name)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
def check(self, schema):
|
|
|
|
pass
|
|
|
|
|
2018-02-19 23:53:03 +00:00
|
|
|
def is_implicit(self):
|
|
|
|
return not self.info
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class QAPISchemaVisitor(object):
|
|
|
|
def visit_begin(self, schema):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_end(self):
|
|
|
|
pass
|
|
|
|
|
2018-02-19 23:49:02 +00:00
|
|
|
def visit_needed(self, entity):
|
|
|
|
# Default to visiting everything
|
|
|
|
return True
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit_builtin_type(self, name, info, json_type):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_enum_type(self, name, info, values, prefix):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_array_type(self, name, info, element_type):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_object_type(self, name, info, base, members, variants):
|
|
|
|
pass
|
|
|
|
|
qapi: New QMP command query-qmp-schema for QMP introspection
qapi/introspect.json defines the introspection schema. It's designed
for QMP introspection, but should do for similar uses, such as QGA.
The introspection schema does not reflect all the rules and
restrictions that apply to QAPI schemata. A valid QAPI schema has an
introspection value conforming to the introspection schema, but the
converse is not true.
Introspection lowers away a number of schema details, and makes
implicit things explicit:
* The built-in types are declared with their JSON type.
All integer types are mapped to 'int', because how many bits we use
internally is an implementation detail. It could be pressed into
external interface service as very approximate range information,
but that's a bad idea. If we need range information, we better do
it properly.
* Implicit type definitions are made explicit, and given
auto-generated names:
- Array types, named by appending "List" to the name of their
element type, like in generated C.
- The enumeration types implicitly defined by simple union types,
named by appending "Kind" to the name of their simple union type,
like in generated C.
- Types that don't occur in generated C. Their names start with ':'
so they don't clash with the user's names.
* All type references are by name.
* The struct and union types are generalized into an object type.
* Base types are flattened.
* Commands take a single argument and return a single result.
Dictionary argument or list result is an implicit type definition.
The empty object type is used when a command takes no arguments or
produces no results.
The argument is always of object type, but the introspection schema
doesn't reflect that.
The 'gen': false directive is omitted as implementation detail.
The 'success-response' directive is omitted as well for now, even
though it's not an implementation detail, because it's not used by
QMP.
* Events carry a single data value.
Implicit type definition and empty object type use, just like for
commands.
The value is of object type, but the introspection schema doesn't
reflect that.
* Types not used by commands or events are omitted.
Indirect use counts as use.
* Optional members have a default, which can only be null right now
Instead of a mandatory "optional" flag, we have an optional default.
No default means mandatory, default null means optional without
default value. Non-null is available for optional with default
(possible future extension).
* Clients should *not* look up types by name, because type names are
not ABI. Look up the command or event you're interested in, then
follow the references.
TODO Should we hide the type names to eliminate the temptation?
New generator scripts/qapi-introspect.py computes an introspection
value for its input, and generates a C variable holding it.
It can generate awfully long lines. Marked TODO.
A new test-qmp-input-visitor test case feeds its result for both
tests/qapi-schema/qapi-schema-test.json and qapi-schema.json to a
QmpInputVisitor to verify it actually conforms to the schema.
New QMP command query-qmp-schema takes its return value from that
variable. Its reply is some 85KiBytes for me right now.
If this turns out to be too much, we have a couple of options:
* We can use shorter names in the JSON. Not the QMP style.
* Optionally return the sub-schema for commands and events given as
arguments.
Right now qmp_query_schema() sends the string literal computed by
qmp-introspect.py. To compute sub-schema at run time, we'd have to
duplicate parts of qapi-introspect.py in C. Unattractive.
* Let clients cache the output of query-qmp-schema.
It changes only on QEMU upgrades, i.e. rarely. Provide a command
query-qmp-schema-hash. Clients can have a cache indexed by hash,
and re-query the schema only when they don't have it cached. Even
simpler: put the hash in the QMP greeting.
Backports commit 39a181581650f4d50f4445bc6276d9716cece050 from qemu
2018-02-19 22:54:00 +00:00
|
|
|
def visit_object_type_flat(self, name, info, members, variants):
|
|
|
|
pass
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit_alternate_type(self, name, info, variants):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_command(self, name, info, arg_type, ret_type,
|
|
|
|
gen, success_response):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def visit_event(self, name, info, arg_type):
|
|
|
|
pass
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaType(QAPISchemaEntity):
|
2018-02-19 21:47:32 +00:00
|
|
|
def c_type(self, is_param=False):
|
|
|
|
return c_name(self.name) + pointer_suffix
|
|
|
|
|
|
|
|
def c_null(self):
|
|
|
|
return 'NULL'
|
|
|
|
|
|
|
|
def json_type(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def alternate_qtype(self):
|
|
|
|
json2qtype = {
|
|
|
|
'string': 'QTYPE_QSTRING',
|
|
|
|
'number': 'QTYPE_QFLOAT',
|
|
|
|
'int': 'QTYPE_QINT',
|
|
|
|
'boolean': 'QTYPE_QBOOL',
|
|
|
|
'object': 'QTYPE_QDICT'
|
|
|
|
}
|
|
|
|
return json2qtype.get(self.json_type())
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class QAPISchemaBuiltinType(QAPISchemaType):
|
2018-02-19 21:47:32 +00:00
|
|
|
def __init__(self, name, json_type, c_type, c_null):
|
2018-02-19 21:43:28 +00:00
|
|
|
QAPISchemaType.__init__(self, name, None)
|
2018-02-19 21:47:32 +00:00
|
|
|
assert not c_type or isinstance(c_type, str)
|
|
|
|
assert json_type in ('string', 'number', 'int', 'boolean', 'null',
|
|
|
|
'value')
|
|
|
|
self._json_type_name = json_type
|
|
|
|
self._c_type_name = c_type
|
|
|
|
self._c_null_val = c_null
|
|
|
|
|
|
|
|
def c_name(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
def c_type(self, is_param=False):
|
|
|
|
if is_param and self.name == 'str':
|
|
|
|
return 'const ' + self._c_type_name
|
|
|
|
return self._c_type_name
|
|
|
|
|
|
|
|
def c_null(self):
|
|
|
|
return self._c_null_val
|
|
|
|
|
|
|
|
def json_type(self):
|
|
|
|
return self._json_type_name
|
2018-02-19 21:43:28 +00:00
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_builtin_type(self.name, self.info, self.json_type())
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaEnumType(QAPISchemaType):
|
|
|
|
def __init__(self, name, info, values, prefix):
|
|
|
|
QAPISchemaType.__init__(self, name, info)
|
|
|
|
for v in values:
|
|
|
|
assert isinstance(v, str)
|
|
|
|
assert prefix is None or isinstance(prefix, str)
|
|
|
|
self.values = values
|
|
|
|
self.prefix = prefix
|
|
|
|
|
|
|
|
def check(self, schema):
|
|
|
|
assert len(set(self.values)) == len(self.values)
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def is_implicit(self):
|
|
|
|
# See QAPISchema._make_implicit_enum_type()
|
2018-02-20 00:06:18 +00:00
|
|
|
return self.name.endswith('Kind')
|
2018-02-20 00:04:36 +00:00
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def c_type(self, is_param=False):
|
|
|
|
return c_name(self.name)
|
|
|
|
|
|
|
|
def c_null(self):
|
|
|
|
return c_enum_const(self.name, (self.values + ['MAX'])[0],
|
|
|
|
self.prefix)
|
|
|
|
|
|
|
|
def json_type(self):
|
|
|
|
return 'string'
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_enum_type(self.name, self.info,
|
|
|
|
self.values, self.prefix)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaArrayType(QAPISchemaType):
|
|
|
|
def __init__(self, name, info, element_type):
|
|
|
|
QAPISchemaType.__init__(self, name, info)
|
|
|
|
assert isinstance(element_type, str)
|
|
|
|
self._element_type_name = element_type
|
|
|
|
self.element_type = None
|
|
|
|
|
|
|
|
def check(self, schema):
|
|
|
|
self.element_type = schema.lookup_type(self._element_type_name)
|
|
|
|
assert self.element_type
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def is_implicit(self):
|
|
|
|
return True
|
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def json_type(self):
|
|
|
|
return 'array'
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_array_type(self.name, self.info, self.element_type)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaObjectType(QAPISchemaType):
|
|
|
|
def __init__(self, name, info, base, local_members, variants):
|
|
|
|
QAPISchemaType.__init__(self, name, info)
|
|
|
|
assert base is None or isinstance(base, str)
|
|
|
|
for m in local_members:
|
|
|
|
assert isinstance(m, QAPISchemaObjectTypeMember)
|
|
|
|
assert (variants is None or
|
|
|
|
isinstance(variants, QAPISchemaObjectTypeVariants))
|
|
|
|
self._base_name = base
|
|
|
|
self.base = None
|
|
|
|
self.local_members = local_members
|
|
|
|
self.variants = variants
|
|
|
|
self.members = None
|
|
|
|
|
|
|
|
def check(self, schema):
|
|
|
|
assert self.members is not False # not running in cycles
|
|
|
|
if self.members:
|
|
|
|
return
|
|
|
|
self.members = False # mark as being checked
|
|
|
|
if self._base_name:
|
|
|
|
self.base = schema.lookup_type(self._base_name)
|
|
|
|
assert isinstance(self.base, QAPISchemaObjectType)
|
|
|
|
assert not self.base.variants # not implemented
|
|
|
|
self.base.check(schema)
|
|
|
|
members = list(self.base.members)
|
|
|
|
else:
|
|
|
|
members = []
|
|
|
|
seen = {}
|
|
|
|
for m in members:
|
2018-02-19 23:50:21 +00:00
|
|
|
assert c_name(m.name) not in seen
|
2018-02-19 21:43:28 +00:00
|
|
|
seen[m.name] = m
|
|
|
|
for m in self.local_members:
|
|
|
|
m.check(schema, members, seen)
|
|
|
|
if self.variants:
|
|
|
|
self.variants.check(schema, members, seen)
|
|
|
|
self.members = members
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def is_implicit(self):
|
|
|
|
# See QAPISchema._make_implicit_object_type()
|
|
|
|
return self.name[0] == ':'
|
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def c_name(self):
|
2018-02-19 23:53:03 +00:00
|
|
|
assert not self.is_implicit()
|
2018-02-19 21:47:32 +00:00
|
|
|
return QAPISchemaType.c_name(self)
|
|
|
|
|
|
|
|
def c_type(self, is_param=False):
|
2018-02-19 23:53:03 +00:00
|
|
|
assert not self.is_implicit()
|
2018-02-19 21:47:32 +00:00
|
|
|
return QAPISchemaType.c_type(self)
|
|
|
|
|
|
|
|
def json_type(self):
|
|
|
|
return 'object'
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_object_type(self.name, self.info,
|
|
|
|
self.base, self.local_members, self.variants)
|
qapi: New QMP command query-qmp-schema for QMP introspection
qapi/introspect.json defines the introspection schema. It's designed
for QMP introspection, but should do for similar uses, such as QGA.
The introspection schema does not reflect all the rules and
restrictions that apply to QAPI schemata. A valid QAPI schema has an
introspection value conforming to the introspection schema, but the
converse is not true.
Introspection lowers away a number of schema details, and makes
implicit things explicit:
* The built-in types are declared with their JSON type.
All integer types are mapped to 'int', because how many bits we use
internally is an implementation detail. It could be pressed into
external interface service as very approximate range information,
but that's a bad idea. If we need range information, we better do
it properly.
* Implicit type definitions are made explicit, and given
auto-generated names:
- Array types, named by appending "List" to the name of their
element type, like in generated C.
- The enumeration types implicitly defined by simple union types,
named by appending "Kind" to the name of their simple union type,
like in generated C.
- Types that don't occur in generated C. Their names start with ':'
so they don't clash with the user's names.
* All type references are by name.
* The struct and union types are generalized into an object type.
* Base types are flattened.
* Commands take a single argument and return a single result.
Dictionary argument or list result is an implicit type definition.
The empty object type is used when a command takes no arguments or
produces no results.
The argument is always of object type, but the introspection schema
doesn't reflect that.
The 'gen': false directive is omitted as implementation detail.
The 'success-response' directive is omitted as well for now, even
though it's not an implementation detail, because it's not used by
QMP.
* Events carry a single data value.
Implicit type definition and empty object type use, just like for
commands.
The value is of object type, but the introspection schema doesn't
reflect that.
* Types not used by commands or events are omitted.
Indirect use counts as use.
* Optional members have a default, which can only be null right now
Instead of a mandatory "optional" flag, we have an optional default.
No default means mandatory, default null means optional without
default value. Non-null is available for optional with default
(possible future extension).
* Clients should *not* look up types by name, because type names are
not ABI. Look up the command or event you're interested in, then
follow the references.
TODO Should we hide the type names to eliminate the temptation?
New generator scripts/qapi-introspect.py computes an introspection
value for its input, and generates a C variable holding it.
It can generate awfully long lines. Marked TODO.
A new test-qmp-input-visitor test case feeds its result for both
tests/qapi-schema/qapi-schema-test.json and qapi-schema.json to a
QmpInputVisitor to verify it actually conforms to the schema.
New QMP command query-qmp-schema takes its return value from that
variable. Its reply is some 85KiBytes for me right now.
If this turns out to be too much, we have a couple of options:
* We can use shorter names in the JSON. Not the QMP style.
* Optionally return the sub-schema for commands and events given as
arguments.
Right now qmp_query_schema() sends the string literal computed by
qmp-introspect.py. To compute sub-schema at run time, we'd have to
duplicate parts of qapi-introspect.py in C. Unattractive.
* Let clients cache the output of query-qmp-schema.
It changes only on QEMU upgrades, i.e. rarely. Provide a command
query-qmp-schema-hash. Clients can have a cache indexed by hash,
and re-query the schema only when they don't have it cached. Even
simpler: put the hash in the QMP greeting.
Backports commit 39a181581650f4d50f4445bc6276d9716cece050 from qemu
2018-02-19 22:54:00 +00:00
|
|
|
visitor.visit_object_type_flat(self.name, self.info,
|
|
|
|
self.members, self.variants)
|
2018-02-19 21:49:57 +00:00
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaObjectTypeMember(object):
|
|
|
|
def __init__(self, name, typ, optional):
|
|
|
|
assert isinstance(name, str)
|
|
|
|
assert isinstance(typ, str)
|
|
|
|
assert isinstance(optional, bool)
|
|
|
|
self.name = name
|
|
|
|
self._type_name = typ
|
|
|
|
self.type = None
|
|
|
|
self.optional = optional
|
|
|
|
|
|
|
|
def check(self, schema, all_members, seen):
|
|
|
|
assert self.name not in seen
|
|
|
|
self.type = schema.lookup_type(self._type_name)
|
|
|
|
assert self.type
|
|
|
|
all_members.append(self)
|
|
|
|
seen[self.name] = self
|
|
|
|
|
|
|
|
|
|
|
|
class QAPISchemaObjectTypeVariants(object):
|
2018-02-19 23:57:18 +00:00
|
|
|
def __init__(self, tag_name, tag_member, variants):
|
|
|
|
# Flat unions pass tag_name but not tag_member.
|
|
|
|
# Simple unions and alternates pass tag_member but not tag_name.
|
|
|
|
# After check(), tag_member is always set, and tag_name remains
|
|
|
|
# a reliable witness of being used by a flat union.
|
|
|
|
assert bool(tag_member) != bool(tag_name)
|
|
|
|
assert (isinstance(tag_name, str) or
|
|
|
|
isinstance(tag_member, QAPISchemaObjectTypeMember))
|
2018-02-19 21:43:28 +00:00
|
|
|
for v in variants:
|
|
|
|
assert isinstance(v, QAPISchemaObjectTypeVariant)
|
|
|
|
self.tag_name = tag_name
|
2018-02-19 23:57:18 +00:00
|
|
|
self.tag_member = tag_member
|
2018-02-19 21:43:28 +00:00
|
|
|
self.variants = variants
|
|
|
|
|
|
|
|
def check(self, schema, members, seen):
|
|
|
|
if self.tag_name:
|
|
|
|
self.tag_member = seen[self.tag_name]
|
|
|
|
else:
|
|
|
|
self.tag_member.check(schema, members, seen)
|
|
|
|
assert isinstance(self.tag_member.type, QAPISchemaEnumType)
|
|
|
|
for v in self.variants:
|
|
|
|
vseen = dict(seen)
|
|
|
|
v.check(schema, self.tag_member.type, vseen)
|
|
|
|
|
|
|
|
class QAPISchemaObjectTypeVariant(QAPISchemaObjectTypeMember):
|
|
|
|
def __init__(self, name, typ):
|
|
|
|
QAPISchemaObjectTypeMember.__init__(self, name, typ, False)
|
|
|
|
|
|
|
|
def check(self, schema, tag_type, seen):
|
|
|
|
QAPISchemaObjectTypeMember.check(self, schema, [], seen)
|
|
|
|
assert self.name in tag_type.values
|
|
|
|
|
qapi-types: Convert to QAPISchemaVisitor, fixing flat unions
Fixes flat unions to get the base's base members. Test case is from
commit 2fc0043, in qapi-schema-test.json:
{ 'union': 'UserDefFlatUnion',
'base': 'UserDefUnionBase',
'discriminator': 'enum1',
'data': { 'value1' : 'UserDefA',
'value2' : 'UserDefB',
'value3' : 'UserDefB' } }
{ 'struct': 'UserDefUnionBase',
'base': 'UserDefZero',
'data': { 'string': 'str', 'enum1': 'EnumOne' } }
{ 'struct': 'UserDefZero',
'data': { 'integer': 'int' } }
Patch's effect on UserDefFlatUnion:
struct UserDefFlatUnion {
/* Members inherited from UserDefUnionBase: */
+ int64_t integer;
char *string;
EnumOne enum1;
/* Own members: */
union { /* union tag is @enum1 */
void *data;
UserDefA *value1;
UserDefB *value2;
UserDefB *value3;
};
};
Flat union visitors remain broken. They'll be fixed next.
Code is generated in a different order now, but that doesn't matter.
The two guards QAPI_TYPES_BUILTIN_STRUCT_DECL and
QAPI_TYPES_BUILTIN_CLEANUP_DECL are replaced by just
QAPI_TYPES_BUILTIN.
Two ugly special cases for simple unions now stand out like sore
thumbs:
1. The type tag is named 'type' everywhere, except in generated C,
where it's 'kind'.
2. QAPISchema lowers simple unions to semantically equivalent flat
unions. However, the C generated for a simple unions differs from
the C generated for its equivalent flat union, and we therefore
need special code to preserve that pointless difference for now.
Mark both TODO.
Backports commit 2b162ccbe875e5323fc04c1009addbdea4d35220 from qemu
2018-02-19 21:57:19 +00:00
|
|
|
# This function exists to support ugly simple union special cases
|
|
|
|
# TODO get rid of them, and drop the function
|
|
|
|
def simple_union_type(self):
|
2018-02-19 23:53:03 +00:00
|
|
|
if (self.type.is_implicit() and
|
|
|
|
isinstance(self.type, QAPISchemaObjectType)):
|
qapi-types: Convert to QAPISchemaVisitor, fixing flat unions
Fixes flat unions to get the base's base members. Test case is from
commit 2fc0043, in qapi-schema-test.json:
{ 'union': 'UserDefFlatUnion',
'base': 'UserDefUnionBase',
'discriminator': 'enum1',
'data': { 'value1' : 'UserDefA',
'value2' : 'UserDefB',
'value3' : 'UserDefB' } }
{ 'struct': 'UserDefUnionBase',
'base': 'UserDefZero',
'data': { 'string': 'str', 'enum1': 'EnumOne' } }
{ 'struct': 'UserDefZero',
'data': { 'integer': 'int' } }
Patch's effect on UserDefFlatUnion:
struct UserDefFlatUnion {
/* Members inherited from UserDefUnionBase: */
+ int64_t integer;
char *string;
EnumOne enum1;
/* Own members: */
union { /* union tag is @enum1 */
void *data;
UserDefA *value1;
UserDefB *value2;
UserDefB *value3;
};
};
Flat union visitors remain broken. They'll be fixed next.
Code is generated in a different order now, but that doesn't matter.
The two guards QAPI_TYPES_BUILTIN_STRUCT_DECL and
QAPI_TYPES_BUILTIN_CLEANUP_DECL are replaced by just
QAPI_TYPES_BUILTIN.
Two ugly special cases for simple unions now stand out like sore
thumbs:
1. The type tag is named 'type' everywhere, except in generated C,
where it's 'kind'.
2. QAPISchema lowers simple unions to semantically equivalent flat
unions. However, the C generated for a simple unions differs from
the C generated for its equivalent flat union, and we therefore
need special code to preserve that pointless difference for now.
Mark both TODO.
Backports commit 2b162ccbe875e5323fc04c1009addbdea4d35220 from qemu
2018-02-19 21:57:19 +00:00
|
|
|
assert len(self.type.members) == 1
|
|
|
|
assert not self.type.variants
|
|
|
|
return self.type.members[0].type
|
|
|
|
return None
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaAlternateType(QAPISchemaType):
|
|
|
|
def __init__(self, name, info, variants):
|
|
|
|
QAPISchemaType.__init__(self, name, info)
|
|
|
|
assert isinstance(variants, QAPISchemaObjectTypeVariants)
|
|
|
|
assert not variants.tag_name
|
|
|
|
self.variants = variants
|
|
|
|
|
|
|
|
def check(self, schema):
|
|
|
|
self.variants.check(schema, [], {})
|
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def json_type(self):
|
|
|
|
return 'value'
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_alternate_type(self.name, self.info, self.variants)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaCommand(QAPISchemaEntity):
|
|
|
|
def __init__(self, name, info, arg_type, ret_type, gen, success_response):
|
|
|
|
QAPISchemaEntity.__init__(self, name, info)
|
|
|
|
assert not arg_type or isinstance(arg_type, str)
|
|
|
|
assert not ret_type or isinstance(ret_type, str)
|
|
|
|
self._arg_type_name = arg_type
|
|
|
|
self.arg_type = None
|
|
|
|
self._ret_type_name = ret_type
|
|
|
|
self.ret_type = None
|
|
|
|
self.gen = gen
|
|
|
|
self.success_response = success_response
|
|
|
|
|
|
|
|
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 self._ret_type_name:
|
|
|
|
self.ret_type = schema.lookup_type(self._ret_type_name)
|
|
|
|
assert isinstance(self.ret_type, QAPISchemaType)
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_command(self.name, self.info,
|
|
|
|
self.arg_type, self.ret_type,
|
|
|
|
self.gen, self.success_response)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchemaEvent(QAPISchemaEntity):
|
|
|
|
def __init__(self, name, info, arg_type):
|
|
|
|
QAPISchemaEntity.__init__(self, name, info)
|
|
|
|
assert not arg_type or isinstance(arg_type, str)
|
|
|
|
self._arg_type_name = arg_type
|
|
|
|
self.arg_type = None
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
|
|
|
visitor.visit_event(self.name, self.info, self.arg_type)
|
|
|
|
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
class QAPISchema(object):
|
|
|
|
def __init__(self, fname):
|
|
|
|
try:
|
|
|
|
self.exprs = check_exprs(QAPISchemaParser(open(fname, "r")).exprs)
|
2018-02-19 23:50:21 +00:00
|
|
|
self._entity_dict = {}
|
2018-02-20 00:04:36 +00:00
|
|
|
self._predefining = True
|
2018-02-19 23:50:21 +00:00
|
|
|
self._def_predefineds()
|
2018-02-20 00:04:36 +00:00
|
|
|
self._predefining = False
|
2018-02-19 23:50:21 +00:00
|
|
|
self._def_exprs()
|
|
|
|
self.check()
|
2018-02-19 21:43:28 +00:00
|
|
|
except (QAPISchemaError, QAPIExprError), err:
|
|
|
|
print >>sys.stderr, err
|
|
|
|
exit(1)
|
|
|
|
|
|
|
|
def _def_entity(self, ent):
|
2018-02-20 00:04:36 +00:00
|
|
|
# Only the predefined types are allowed to not have info
|
|
|
|
assert ent.info or self._predefining
|
2018-02-19 21:43:28 +00:00
|
|
|
assert ent.name not in self._entity_dict
|
|
|
|
self._entity_dict[ent.name] = ent
|
|
|
|
|
|
|
|
def lookup_entity(self, name, typ=None):
|
|
|
|
ent = self._entity_dict.get(name)
|
|
|
|
if typ and not isinstance(ent, typ):
|
|
|
|
return None
|
|
|
|
return ent
|
|
|
|
|
|
|
|
def lookup_type(self, name):
|
|
|
|
return self.lookup_entity(name, QAPISchemaType)
|
|
|
|
|
2018-02-19 21:47:32 +00:00
|
|
|
def _def_builtin_type(self, name, json_type, c_type, c_null):
|
|
|
|
self._def_entity(QAPISchemaBuiltinType(name, json_type,
|
|
|
|
c_type, c_null))
|
qapi: Lazy creation of array types
Commit ac88219a had several TODO markers about whether we needed
to automatically create the corresponding array type alongside
any other type. It turns out that most of the time, we don't!
There are a few exceptions: 1) We have a few situations where we
use an array type in internal code but do not expose that type
through QMP; fix it by declaring a dummy type that forces the
generator to see that we want to use the array type.
2) The builtin arrays (such as intList for QAPI ['int']) must
always be generated, because of the way our QAPI_TYPES_BUILTIN
compile guard works: we have situations (at the very least
tests/test-qmp-output-visitor.c) that include both top-level
"qapi-types.h" (via "error.h") and a secondary
"test-qapi-types.h". If we were to only emit the builtin types
when used locally, then the first .h file would not include all
types, but the second .h does not declare anything at all because
the first .h set QAPI_TYPES_BUILTIN, and we would end up with
compilation error due to things like unknown type 'int8List'.
Actually, we may need to revisit how we do type guards, and
change from a single QAPI_TYPES_BUILTIN over to a different
usage pattern that does one #ifdef per qapi type - right now,
the only types that are declared multiple times between two qapi
.json files for inclusion by a single .c file happen to be the
builtin arrays. But now that we have QAPI 'include' statements,
it is logical to assume that we will soon reach a point where
we want to reuse non-builtin types (yes, I'm thinking about what
it will take to add introspection to QGA, where we will want to
reuse the SchemaInfo type and friends). One #ifdef per type
will help ensure that generating the same qapi type into more
than one qapi-types.h won't cause collisions when both are
included in the same .c file; but we also have to solve how to
avoid creating duplicate qapi-types.c entry points. So that
is a problem left for another day.
Generated code for qapi-types and qapi-visit is drastically
reduced; less than a third of the arrays that were blindly
created were actually needed (a quick grep shows we dropped
from 219 to 69 *List types), and the .o files lost more than
30% of their bulk. [For best results, diff the generated
files with 'git diff --patience --no-index pre post'.]
Interestingly, the introspection output is unchanged - this is
because we already cull all types that are not indirectly
reachable from a command or event, so introspection was already
using only a subset of array types. The subset of types
introspected is now a much larger percentage of the overall set
of array types emitted in qapi-types.h (since the larger set
shrunk), but still not 100% (evidence that the array types
emitted for our new Dummy structs, and the new struct itself,
don't affect QMP).
Backports commit 9f08c8ec73878122ad4b061ed334f0437afaaa32 from qemu
2018-02-19 23:55:33 +00:00
|
|
|
# TODO As long as we have QAPI_TYPES_BUILTIN to share multiple
|
|
|
|
# qapi-types.h from a single .c, all arrays of builtins must be
|
|
|
|
# declared in the first file whether or not they are used. Nicer
|
|
|
|
# would be to use lazy instantiation, while figuring out how to
|
|
|
|
# avoid compilation issues with multiple qapi-types.h.
|
2018-02-20 00:04:36 +00:00
|
|
|
self._make_array_type(name, None)
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
def _def_predefineds(self):
|
2018-02-19 21:47:32 +00:00
|
|
|
for t in [('str', 'string', 'char' + pointer_suffix, 'NULL'),
|
|
|
|
('number', 'number', 'double', '0'),
|
|
|
|
('int', 'int', 'int64_t', '0'),
|
|
|
|
('int8', 'int', 'int8_t', '0'),
|
|
|
|
('int16', 'int', 'int16_t', '0'),
|
|
|
|
('int32', 'int', 'int32_t', '0'),
|
|
|
|
('int64', 'int', 'int64_t', '0'),
|
|
|
|
('uint8', 'int', 'uint8_t', '0'),
|
|
|
|
('uint16', 'int', 'uint16_t', '0'),
|
|
|
|
('uint32', 'int', 'uint32_t', '0'),
|
|
|
|
('uint64', 'int', 'uint64_t', '0'),
|
|
|
|
('size', 'int', 'uint64_t', '0'),
|
|
|
|
('bool', 'boolean', 'bool', 'false'),
|
2018-02-19 22:45:25 +00:00
|
|
|
('any', 'value', 'QObject' + pointer_suffix, 'NULL')]:
|
2018-02-19 21:47:32 +00:00
|
|
|
self._def_builtin_type(*t)
|
qapi: New QMP command query-qmp-schema for QMP introspection
qapi/introspect.json defines the introspection schema. It's designed
for QMP introspection, but should do for similar uses, such as QGA.
The introspection schema does not reflect all the rules and
restrictions that apply to QAPI schemata. A valid QAPI schema has an
introspection value conforming to the introspection schema, but the
converse is not true.
Introspection lowers away a number of schema details, and makes
implicit things explicit:
* The built-in types are declared with their JSON type.
All integer types are mapped to 'int', because how many bits we use
internally is an implementation detail. It could be pressed into
external interface service as very approximate range information,
but that's a bad idea. If we need range information, we better do
it properly.
* Implicit type definitions are made explicit, and given
auto-generated names:
- Array types, named by appending "List" to the name of their
element type, like in generated C.
- The enumeration types implicitly defined by simple union types,
named by appending "Kind" to the name of their simple union type,
like in generated C.
- Types that don't occur in generated C. Their names start with ':'
so they don't clash with the user's names.
* All type references are by name.
* The struct and union types are generalized into an object type.
* Base types are flattened.
* Commands take a single argument and return a single result.
Dictionary argument or list result is an implicit type definition.
The empty object type is used when a command takes no arguments or
produces no results.
The argument is always of object type, but the introspection schema
doesn't reflect that.
The 'gen': false directive is omitted as implementation detail.
The 'success-response' directive is omitted as well for now, even
though it's not an implementation detail, because it's not used by
QMP.
* Events carry a single data value.
Implicit type definition and empty object type use, just like for
commands.
The value is of object type, but the introspection schema doesn't
reflect that.
* Types not used by commands or events are omitted.
Indirect use counts as use.
* Optional members have a default, which can only be null right now
Instead of a mandatory "optional" flag, we have an optional default.
No default means mandatory, default null means optional without
default value. Non-null is available for optional with default
(possible future extension).
* Clients should *not* look up types by name, because type names are
not ABI. Look up the command or event you're interested in, then
follow the references.
TODO Should we hide the type names to eliminate the temptation?
New generator scripts/qapi-introspect.py computes an introspection
value for its input, and generates a C variable holding it.
It can generate awfully long lines. Marked TODO.
A new test-qmp-input-visitor test case feeds its result for both
tests/qapi-schema/qapi-schema-test.json and qapi-schema.json to a
QmpInputVisitor to verify it actually conforms to the schema.
New QMP command query-qmp-schema takes its return value from that
variable. Its reply is some 85KiBytes for me right now.
If this turns out to be too much, we have a couple of options:
* We can use shorter names in the JSON. Not the QMP style.
* Optionally return the sub-schema for commands and events given as
arguments.
Right now qmp_query_schema() sends the string literal computed by
qmp-introspect.py. To compute sub-schema at run time, we'd have to
duplicate parts of qapi-introspect.py in C. Unattractive.
* Let clients cache the output of query-qmp-schema.
It changes only on QEMU upgrades, i.e. rarely. Provide a command
query-qmp-schema-hash. Clients can have a cache indexed by hash,
and re-query the schema only when they don't have it cached. Even
simpler: put the hash in the QMP greeting.
Backports commit 39a181581650f4d50f4445bc6276d9716cece050 from qemu
2018-02-19 22:54:00 +00:00
|
|
|
self.the_empty_object_type = QAPISchemaObjectType(':empty', None, None,
|
|
|
|
[], None)
|
|
|
|
self._def_entity(self.the_empty_object_type)
|
2018-02-19 21:43:28 +00:00
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_implicit_enum_type(self, name, info, values):
|
2018-02-19 23:53:03 +00:00
|
|
|
name = name + 'Kind' # Use namespace reserved by add_name()
|
2018-02-20 00:04:36 +00:00
|
|
|
self._def_entity(QAPISchemaEnumType(name, info, values, None))
|
2018-02-19 21:43:28 +00:00
|
|
|
return name
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_array_type(self, element_type, info):
|
2018-02-20 00:08:11 +00:00
|
|
|
name = element_type + 'List' # Use namespace reserved by add_name()
|
2018-02-19 21:43:28 +00:00
|
|
|
if not self.lookup_type(name):
|
2018-02-20 00:04:36 +00:00
|
|
|
self._def_entity(QAPISchemaArrayType(name, info, element_type))
|
2018-02-19 21:43:28 +00:00
|
|
|
return name
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_implicit_object_type(self, name, info, role, members):
|
2018-02-19 21:43:28 +00:00
|
|
|
if not members:
|
|
|
|
return None
|
|
|
|
name = ':obj-%s-%s' % (name, role)
|
|
|
|
if not self.lookup_entity(name, QAPISchemaObjectType):
|
2018-02-20 00:04:36 +00:00
|
|
|
self._def_entity(QAPISchemaObjectType(name, info, None,
|
2018-02-19 21:43:28 +00:00
|
|
|
members, None))
|
|
|
|
return name
|
|
|
|
|
|
|
|
def _def_enum_type(self, expr, info):
|
|
|
|
name = expr['enum']
|
|
|
|
data = expr['data']
|
|
|
|
prefix = expr.get('prefix')
|
|
|
|
self._def_entity(QAPISchemaEnumType(name, info, data, prefix))
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_member(self, name, typ, info):
|
2018-02-19 21:43:28 +00:00
|
|
|
optional = False
|
|
|
|
if name.startswith('*'):
|
|
|
|
name = name[1:]
|
|
|
|
optional = True
|
|
|
|
if isinstance(typ, list):
|
|
|
|
assert len(typ) == 1
|
2018-02-20 00:04:36 +00:00
|
|
|
typ = self._make_array_type(typ[0], info)
|
2018-02-19 21:43:28 +00:00
|
|
|
return QAPISchemaObjectTypeMember(name, typ, optional)
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_members(self, data, info):
|
|
|
|
return [self._make_member(key, value, info)
|
2018-02-19 21:43:28 +00:00
|
|
|
for (key, value) in data.iteritems()]
|
|
|
|
|
|
|
|
def _def_struct_type(self, expr, info):
|
|
|
|
name = expr['struct']
|
|
|
|
base = expr.get('base')
|
|
|
|
data = expr['data']
|
|
|
|
self._def_entity(QAPISchemaObjectType(name, info, base,
|
2018-02-20 00:04:36 +00:00
|
|
|
self._make_members(data, info),
|
2018-02-19 21:43:28 +00:00
|
|
|
None))
|
|
|
|
|
|
|
|
def _make_variant(self, case, typ):
|
|
|
|
return QAPISchemaObjectTypeVariant(case, typ)
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_simple_variant(self, case, typ, info):
|
2018-02-19 21:43:28 +00:00
|
|
|
if isinstance(typ, list):
|
|
|
|
assert len(typ) == 1
|
2018-02-20 00:04:36 +00:00
|
|
|
typ = self._make_array_type(typ[0], info)
|
|
|
|
typ = self._make_implicit_object_type(
|
|
|
|
typ, info, 'wrapper', [self._make_member('data', typ, info)])
|
2018-02-19 21:43:28 +00:00
|
|
|
return QAPISchemaObjectTypeVariant(case, typ)
|
|
|
|
|
2018-02-20 00:04:36 +00:00
|
|
|
def _make_implicit_tag(self, type_name, info, variants):
|
|
|
|
typ = self._make_implicit_enum_type(type_name, info,
|
2018-02-19 23:57:18 +00:00
|
|
|
[v.name for v in variants])
|
|
|
|
return QAPISchemaObjectTypeMember('type', typ, False)
|
2018-02-19 21:43:28 +00:00
|
|
|
|
|
|
|
def _def_union_type(self, expr, info):
|
|
|
|
name = expr['union']
|
|
|
|
data = expr['data']
|
|
|
|
base = expr.get('base')
|
|
|
|
tag_name = expr.get('discriminator')
|
2018-02-19 23:57:18 +00:00
|
|
|
tag_member = None
|
2018-02-19 21:43:28 +00:00
|
|
|
if tag_name:
|
|
|
|
variants = [self._make_variant(key, value)
|
|
|
|
for (key, value) in data.iteritems()]
|
|
|
|
else:
|
2018-02-20 00:04:36 +00:00
|
|
|
variants = [self._make_simple_variant(key, value, info)
|
2018-02-19 21:43:28 +00:00
|
|
|
for (key, value) in data.iteritems()]
|
2018-02-20 00:04:36 +00:00
|
|
|
tag_member = self._make_implicit_tag(name, info, variants)
|
2018-02-19 21:43:28 +00:00
|
|
|
self._def_entity(
|
|
|
|
QAPISchemaObjectType(name, info, base,
|
2018-02-20 00:04:36 +00:00
|
|
|
self._make_members(OrderedDict(), info),
|
2018-02-19 21:43:28 +00:00
|
|
|
QAPISchemaObjectTypeVariants(tag_name,
|
2018-02-19 23:57:18 +00:00
|
|
|
tag_member,
|
2018-02-19 21:43:28 +00:00
|
|
|
variants)))
|
|
|
|
|
|
|
|
def _def_alternate_type(self, expr, info):
|
|
|
|
name = expr['alternate']
|
|
|
|
data = expr['data']
|
|
|
|
variants = [self._make_variant(key, value)
|
|
|
|
for (key, value) in data.iteritems()]
|
2018-02-20 00:04:36 +00:00
|
|
|
tag_member = self._make_implicit_tag(name, info, variants)
|
2018-02-19 21:43:28 +00:00
|
|
|
self._def_entity(
|
|
|
|
QAPISchemaAlternateType(name, info,
|
|
|
|
QAPISchemaObjectTypeVariants(None,
|
2018-02-19 23:57:18 +00:00
|
|
|
tag_member,
|
2018-02-19 21:43:28 +00:00
|
|
|
variants)))
|
|
|
|
|
|
|
|
def _def_command(self, expr, info):
|
|
|
|
name = expr['command']
|
|
|
|
data = expr.get('data')
|
|
|
|
rets = expr.get('returns')
|
|
|
|
gen = expr.get('gen', True)
|
|
|
|
success_response = expr.get('success-response', True)
|
|
|
|
if isinstance(data, OrderedDict):
|
2018-02-20 00:04:36 +00:00
|
|
|
data = self._make_implicit_object_type(
|
|
|
|
name, info, 'arg', self._make_members(data, info))
|
2018-02-19 21:43:28 +00:00
|
|
|
if isinstance(rets, list):
|
|
|
|
assert len(rets) == 1
|
2018-02-20 00:04:36 +00:00
|
|
|
rets = self._make_array_type(rets[0], info)
|
2018-02-19 21:43:28 +00:00
|
|
|
self._def_entity(QAPISchemaCommand(name, info, data, rets, gen,
|
|
|
|
success_response))
|
|
|
|
|
|
|
|
def _def_event(self, expr, info):
|
|
|
|
name = expr['event']
|
|
|
|
data = expr.get('data')
|
|
|
|
if isinstance(data, OrderedDict):
|
2018-02-20 00:04:36 +00:00
|
|
|
data = self._make_implicit_object_type(
|
|
|
|
name, info, 'arg', self._make_members(data, info))
|
2018-02-19 21:43:28 +00:00
|
|
|
self._def_entity(QAPISchemaEvent(name, info, data))
|
|
|
|
|
|
|
|
def _def_exprs(self):
|
|
|
|
for expr_elem in self.exprs:
|
|
|
|
expr = expr_elem['expr']
|
|
|
|
info = expr_elem['info']
|
|
|
|
if 'enum' in expr:
|
|
|
|
self._def_enum_type(expr, info)
|
|
|
|
elif 'struct' in expr:
|
|
|
|
self._def_struct_type(expr, info)
|
|
|
|
elif 'union' in expr:
|
|
|
|
self._def_union_type(expr, info)
|
|
|
|
elif 'alternate' in expr:
|
|
|
|
self._def_alternate_type(expr, info)
|
|
|
|
elif 'command' in expr:
|
|
|
|
self._def_command(expr, info)
|
|
|
|
elif 'event' in expr:
|
|
|
|
self._def_event(expr, info)
|
|
|
|
else:
|
|
|
|
assert False
|
|
|
|
|
|
|
|
def check(self):
|
|
|
|
for ent in self._entity_dict.values():
|
|
|
|
ent.check(self)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 21:49:57 +00:00
|
|
|
def visit(self, visitor):
|
2018-02-19 23:49:02 +00:00
|
|
|
visitor.visit_begin(self)
|
|
|
|
for (name, entity) in sorted(self._entity_dict.items()):
|
|
|
|
if visitor.visit_needed(entity):
|
|
|
|
entity.visit(visitor)
|
2018-02-19 21:49:57 +00:00
|
|
|
visitor.visit_end()
|
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
# Code generation helpers
|
|
|
|
#
|
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def camel_case(name):
|
|
|
|
new_name = ''
|
|
|
|
first = True
|
|
|
|
for ch in name:
|
|
|
|
if ch in ['_', '-']:
|
|
|
|
first = True
|
|
|
|
elif first:
|
|
|
|
new_name += ch.upper()
|
|
|
|
first = False
|
|
|
|
else:
|
|
|
|
new_name += ch.lower()
|
|
|
|
return new_name
|
|
|
|
|
2018-02-19 19:51:01 +00:00
|
|
|
# ENUMName -> ENUM_NAME, EnumName1 -> ENUM_NAME1
|
|
|
|
# ENUM_NAME -> ENUM_NAME, ENUM_NAME1 -> ENUM_NAME1, ENUM_Name2 -> ENUM_NAME2
|
|
|
|
# ENUM24_Name -> ENUM24_NAME
|
|
|
|
def camel_to_upper(value):
|
|
|
|
c_fun_str = c_name(value, False)
|
|
|
|
if value.isupper():
|
|
|
|
return c_fun_str
|
|
|
|
|
|
|
|
new_name = ''
|
|
|
|
l = len(c_fun_str)
|
|
|
|
for i in range(l):
|
|
|
|
c = c_fun_str[i]
|
|
|
|
# When c is upper and no "_" appears before, do more checks
|
|
|
|
if c.isupper() and (i > 0) and c_fun_str[i - 1] != "_":
|
2018-02-19 23:07:49 +00:00
|
|
|
if i < l - 1 and c_fun_str[i + 1].islower():
|
|
|
|
new_name += '_'
|
|
|
|
elif c_fun_str[i - 1].isdigit():
|
2018-02-19 19:51:01 +00:00
|
|
|
new_name += '_'
|
|
|
|
new_name += c
|
|
|
|
return new_name.lstrip('_').upper()
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 21:38:38 +00:00
|
|
|
def c_enum_const(type_name, const_name, prefix=None):
|
|
|
|
if prefix is not None:
|
|
|
|
type_name = prefix
|
2018-02-19 19:51:01 +00:00
|
|
|
return camel_to_upper(type_name + '_' + const_name)
|
|
|
|
|
2018-02-19 19:44:59 +00:00
|
|
|
c_name_trans = string.maketrans('.-', '__')
|
2018-02-19 19:40:55 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
qapi: Make c_type() consistently convert qapi names
Continuing the string of cleanups for supporting downstream names
containing '.', this patch focuses on ensuring c_type() can
handle a downstream name. This patch alone does not fix the
places where generator output should be calling this function
but was open-coding things instead, but it gets us a step closer.
In particular, the changes to c_list_type() and type_name() mean
that type_name(FOO) now handles the case when FOO contains '.',
'-', or is a ticklish identifier other than a builtin (builtins
are exempted because ['int'] must remain mapped to 'intList' and
not 'q_intList'). Meanwhile, ['unix'] now maps to 'q_unixList'
rather than 'unixList', to match the fact that 'unix' is ticklish;
however, our naming conventions state that complex types should
start with a capital, so no type name following conventions will
ever have the 'q_' prepended.
Likewise, changes to c_type() mean that c_type(FOO) properly
handles an enum or complex type FOO with '.' or '-' in the
name, or is a ticklish identifier (again, a ticklish identifier
as a type name violates conventions).
Backports commit c6405b54b7b09a876f2f2fba2aa6f8ac87189cb9 from qemu
2018-02-19 19:56:46 +00:00
|
|
|
# Map @name to a valid C identifier.
|
|
|
|
# If @protect, avoid returning certain ticklish identifiers (like
|
|
|
|
# C keywords) by prepending "q_".
|
|
|
|
#
|
|
|
|
# Used for converting 'name' from a 'name':'type' qapi definition
|
|
|
|
# into a generated struct member, as well as converting type names
|
|
|
|
# into substrings of a generated C function name.
|
|
|
|
# '__a.b_c' -> '__a_b_c', 'x-foo' -> 'x_foo'
|
|
|
|
# protect=True: 'int' -> 'q_int'; protect=False: 'int' -> 'int'
|
2018-02-19 19:44:59 +00:00
|
|
|
def c_name(name, protect=True):
|
2015-08-21 07:04:50 +00:00
|
|
|
# ANSI X3J11/88-090, 3.1.1
|
|
|
|
c89_words = set(['auto', 'break', 'case', 'char', 'const', 'continue',
|
2018-02-19 23:07:49 +00:00
|
|
|
'default', 'do', 'double', 'else', 'enum', 'extern',
|
|
|
|
'float', 'for', 'goto', 'if', 'int', 'long', 'register',
|
|
|
|
'return', 'short', 'signed', 'sizeof', 'static',
|
|
|
|
'struct', 'switch', 'typedef', 'union', 'unsigned',
|
|
|
|
'void', 'volatile', 'while'])
|
2015-08-21 07:04:50 +00:00
|
|
|
# ISO/IEC 9899:1999, 6.4.1
|
|
|
|
c99_words = set(['inline', 'restrict', '_Bool', '_Complex', '_Imaginary'])
|
|
|
|
# ISO/IEC 9899:2011, 6.4.1
|
2018-02-19 23:07:49 +00:00
|
|
|
c11_words = set(['_Alignas', '_Alignof', '_Atomic', '_Generic',
|
|
|
|
'_Noreturn', '_Static_assert', '_Thread_local'])
|
2015-08-21 07:04:50 +00:00
|
|
|
# GCC http://gcc.gnu.org/onlinedocs/gcc-4.7.1/gcc/C-Extensions.html
|
|
|
|
# excluding _.*
|
|
|
|
gcc_words = set(['asm', 'typeof'])
|
|
|
|
# C++ ISO/IEC 14882:2003 2.11
|
|
|
|
cpp_words = set(['bool', 'catch', 'class', 'const_cast', 'delete',
|
|
|
|
'dynamic_cast', 'explicit', 'false', 'friend', 'mutable',
|
|
|
|
'namespace', 'new', 'operator', 'private', 'protected',
|
|
|
|
'public', 'reinterpret_cast', 'static_cast', 'template',
|
|
|
|
'this', 'throw', 'true', 'try', 'typeid', 'typename',
|
|
|
|
'using', 'virtual', 'wchar_t',
|
|
|
|
# alternative representations
|
|
|
|
'and', 'and_eq', 'bitand', 'bitor', 'compl', 'not',
|
|
|
|
'not_eq', 'or', 'or_eq', 'xor', 'xor_eq'])
|
|
|
|
# namespace pollution:
|
|
|
|
polluted_words = set(['unix', 'errno'])
|
2018-02-19 23:07:49 +00:00
|
|
|
if protect and (name in c89_words | c99_words | c11_words | gcc_words
|
|
|
|
| cpp_words | polluted_words):
|
2015-08-21 07:04:50 +00:00
|
|
|
return "q_" + name
|
2018-02-19 19:44:59 +00:00
|
|
|
return name.translate(c_name_trans)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
|
|
|
eatspace = '\033EATSPACE.'
|
2018-02-19 19:53:53 +00:00
|
|
|
pointer_suffix = ' *' + eatspace
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def genindent(count):
|
|
|
|
ret = ""
|
2018-02-19 23:07:49 +00:00
|
|
|
for _ in range(count):
|
2015-08-21 07:04:50 +00:00
|
|
|
ret += " "
|
|
|
|
return ret
|
|
|
|
|
|
|
|
indent_level = 0
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def push_indent(indent_amount=4):
|
|
|
|
global indent_level
|
|
|
|
indent_level += indent_amount
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def pop_indent(indent_amount=4):
|
|
|
|
global indent_level
|
|
|
|
indent_level -= indent_amount
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 21:04:34 +00:00
|
|
|
# Generate @code with @kwds interpolated.
|
|
|
|
# Obey indent_level, and strip eatspace.
|
2015-08-21 07:04:50 +00:00
|
|
|
def cgen(code, **kwds):
|
2018-02-19 21:04:34 +00:00
|
|
|
raw = code % kwds
|
|
|
|
if indent_level:
|
|
|
|
indent = genindent(indent_level)
|
2018-02-19 21:34:14 +00:00
|
|
|
# re.subn() lacks flags support before Python 2.7, use re.compile()
|
|
|
|
raw = re.subn(re.compile("^.", re.MULTILINE),
|
|
|
|
indent + r'\g<0>', raw)
|
2018-02-19 21:04:34 +00:00
|
|
|
raw = raw[0]
|
|
|
|
return re.sub(re.escape(eatspace) + ' *', '', raw)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def mcgen(code, **kwds):
|
2018-02-19 21:04:34 +00:00
|
|
|
if code[0] == '\n':
|
|
|
|
code = code[1:]
|
|
|
|
return cgen(code, **kwds)
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def guardname(filename):
|
2018-02-19 21:05:50 +00:00
|
|
|
return c_name(filename, protect=False).upper()
|
2015-08-21 07:04:50 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def guardstart(name):
|
|
|
|
return mcgen('''
|
|
|
|
|
|
|
|
#ifndef %(name)s
|
|
|
|
#define %(name)s
|
|
|
|
|
|
|
|
''',
|
|
|
|
name=guardname(name))
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2015-08-21 07:04:50 +00:00
|
|
|
def guardend(name):
|
|
|
|
return mcgen('''
|
|
|
|
|
|
|
|
#endif /* %(name)s */
|
|
|
|
|
|
|
|
''',
|
|
|
|
name=guardname(name))
|
2018-02-19 20:19:07 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
def gen_enum_lookup(name, values, prefix=None):
|
2018-02-19 22:08:05 +00:00
|
|
|
ret = mcgen('''
|
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
const char *const %(c_name)s_lookup[] = {
|
2018-02-19 22:08:05 +00:00
|
|
|
''',
|
2018-02-19 22:33:32 +00:00
|
|
|
c_name=c_name(name))
|
2018-02-19 22:08:05 +00:00
|
|
|
for value in values:
|
|
|
|
index = c_enum_const(name, value, prefix)
|
|
|
|
ret += mcgen('''
|
|
|
|
"%(value)s",
|
|
|
|
''',
|
2018-02-19 22:33:32 +00:00
|
|
|
value=value)
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 22:11:58 +00:00
|
|
|
# Unicorn: We don't use C99 [ARRAY_INDICING] = Thing because
|
|
|
|
# MSVC is still in the stone-age with this part of C compiler
|
|
|
|
# support.
|
2018-02-19 22:08:05 +00:00
|
|
|
max_index = c_enum_const(name, 'MAX', prefix)
|
|
|
|
ret += mcgen('''
|
|
|
|
NULL,
|
|
|
|
};
|
2018-02-19 22:33:32 +00:00
|
|
|
''')
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
return ret
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
def gen_enum(name, values, prefix=None):
|
|
|
|
# append automatically generated _MAX value
|
|
|
|
enum_values = values + ['MAX']
|
|
|
|
ret = mcgen('''
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
typedef enum %(c_name)s {
|
2018-02-19 22:08:05 +00:00
|
|
|
''',
|
2018-02-19 22:33:32 +00:00
|
|
|
c_name=c_name(name))
|
2018-02-19 22:08:05 +00:00
|
|
|
|
|
|
|
i = 0
|
|
|
|
for value in enum_values:
|
2018-02-19 22:33:32 +00:00
|
|
|
ret += mcgen('''
|
|
|
|
%(c_enum)s = %(i)d,
|
2018-02-19 22:08:05 +00:00
|
|
|
''',
|
2018-02-19 22:33:32 +00:00
|
|
|
c_enum=c_enum_const(name, value, prefix),
|
2018-02-19 22:08:05 +00:00
|
|
|
i=i)
|
|
|
|
i += 1
|
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
ret += mcgen('''
|
|
|
|
} %(c_name)s;
|
2018-02-19 22:08:05 +00:00
|
|
|
''',
|
2018-02-19 22:33:32 +00:00
|
|
|
c_name=c_name(name))
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 22:33:32 +00:00
|
|
|
ret += mcgen('''
|
|
|
|
extern const char *const %(c_name)s_lookup[];
|
|
|
|
''',
|
|
|
|
c_name=c_name(name))
|
|
|
|
return ret
|
2018-02-19 22:08:05 +00:00
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 22:38:50 +00:00
|
|
|
def gen_params(arg_type, 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 extra:
|
|
|
|
ret += sep + extra
|
|
|
|
return ret
|
|
|
|
|
2018-02-19 23:27:47 +00:00
|
|
|
|
2018-02-19 23:43:38 +00:00
|
|
|
def gen_err_check(label='out', skiperr=False):
|
|
|
|
if skiperr:
|
2018-02-19 23:27:47 +00:00
|
|
|
return ''
|
|
|
|
return mcgen('''
|
2018-02-19 23:43:38 +00:00
|
|
|
if (err) {
|
2018-02-19 23:27:47 +00:00
|
|
|
goto %(label)s;
|
|
|
|
}
|
|
|
|
''',
|
2018-02-19 23:43:38 +00:00
|
|
|
label=label)
|
2018-02-19 23:27:47 +00:00
|
|
|
|
|
|
|
|
2018-02-19 23:43:38 +00:00
|
|
|
def gen_visit_fields(members, prefix='', need_cast=False, skiperr=False):
|
qapi: Share gen_visit_fields()
Consolidate the code between visit, command marshalling, and
event generation that iterates over the members of a struct.
It reduces code duplication in the generator, so that a future
patch can reduce the size of generated code while touching only
one instead of three locations.
There are no changes to the generated marshal code.
The visitor code becomes slightly more verbose, but remains
semantically equivalent, and is actually easier to read as
it follows a more common idiom:
| visit_optional(v, &(*obj)->has_device, "device", &err);
|- if (!err && (*obj)->has_device) {
|- visit_type_str(v, &(*obj)->device, "device", &err);
|- }
| if (err) {
| goto out;
| }
|+ if ((*obj)->has_device) {
|+ visit_type_str(v, &(*obj)->device, "device", &err);
|+ if (err) {
|+ goto out;
|+ }
|+ }
The event code becomes slightly more verbose, but this is
arguably a bug fix: although the visitors are not well
documented, use of an optional member should not be attempted
unless guarded by a prior call to visit_optional(). Works only
because the output qmp visitor has a no-op visit_optional():
|+ visit_optional(v, &has_offset, "offset", &err);
|+ if (err) {
|+ goto out;
|+ }
| if (has_offset) {
| visit_type_int(v, &offset, "offset", &err);
Backports commit 82ca8e469666b169ccf818a0e36136aee97d7db0 from qemu
2018-02-19 23:41:37 +00:00
|
|
|
ret = ''
|
2018-02-19 23:43:38 +00:00
|
|
|
if skiperr:
|
qapi: Share gen_visit_fields()
Consolidate the code between visit, command marshalling, and
event generation that iterates over the members of a struct.
It reduces code duplication in the generator, so that a future
patch can reduce the size of generated code while touching only
one instead of three locations.
There are no changes to the generated marshal code.
The visitor code becomes slightly more verbose, but remains
semantically equivalent, and is actually easier to read as
it follows a more common idiom:
| visit_optional(v, &(*obj)->has_device, "device", &err);
|- if (!err && (*obj)->has_device) {
|- visit_type_str(v, &(*obj)->device, "device", &err);
|- }
| if (err) {
| goto out;
| }
|+ if ((*obj)->has_device) {
|+ visit_type_str(v, &(*obj)->device, "device", &err);
|+ if (err) {
|+ goto out;
|+ }
|+ }
The event code becomes slightly more verbose, but this is
arguably a bug fix: although the visitors are not well
documented, use of an optional member should not be attempted
unless guarded by a prior call to visit_optional(). Works only
because the output qmp visitor has a no-op visit_optional():
|+ visit_optional(v, &has_offset, "offset", &err);
|+ if (err) {
|+ goto out;
|+ }
| if (has_offset) {
| visit_type_int(v, &offset, "offset", &err);
Backports commit 82ca8e469666b169ccf818a0e36136aee97d7db0 from qemu
2018-02-19 23:41:37 +00:00
|
|
|
errparg = 'NULL'
|
2018-02-19 23:43:38 +00:00
|
|
|
else:
|
|
|
|
errparg = '&err'
|
qapi: Share gen_visit_fields()
Consolidate the code between visit, command marshalling, and
event generation that iterates over the members of a struct.
It reduces code duplication in the generator, so that a future
patch can reduce the size of generated code while touching only
one instead of three locations.
There are no changes to the generated marshal code.
The visitor code becomes slightly more verbose, but remains
semantically equivalent, and is actually easier to read as
it follows a more common idiom:
| visit_optional(v, &(*obj)->has_device, "device", &err);
|- if (!err && (*obj)->has_device) {
|- visit_type_str(v, &(*obj)->device, "device", &err);
|- }
| if (err) {
| goto out;
| }
|+ if ((*obj)->has_device) {
|+ visit_type_str(v, &(*obj)->device, "device", &err);
|+ if (err) {
|+ goto out;
|+ }
|+ }
The event code becomes slightly more verbose, but this is
arguably a bug fix: although the visitors are not well
documented, use of an optional member should not be attempted
unless guarded by a prior call to visit_optional(). Works only
because the output qmp visitor has a no-op visit_optional():
|+ visit_optional(v, &has_offset, "offset", &err);
|+ if (err) {
|+ goto out;
|+ }
| if (has_offset) {
| visit_type_int(v, &offset, "offset", &err);
Backports commit 82ca8e469666b169ccf818a0e36136aee97d7db0 from qemu
2018-02-19 23:41:37 +00:00
|
|
|
|
|
|
|
for memb in members:
|
|
|
|
if memb.optional:
|
|
|
|
ret += mcgen('''
|
|
|
|
visit_optional(v, &%(prefix)shas_%(c_name)s, "%(name)s", %(errp)s);
|
|
|
|
''',
|
|
|
|
prefix=prefix, c_name=c_name(memb.name),
|
|
|
|
name=memb.name, errp=errparg)
|
2018-02-19 23:43:38 +00:00
|
|
|
ret += gen_err_check(skiperr=skiperr)
|
qapi: Share gen_visit_fields()
Consolidate the code between visit, command marshalling, and
event generation that iterates over the members of a struct.
It reduces code duplication in the generator, so that a future
patch can reduce the size of generated code while touching only
one instead of three locations.
There are no changes to the generated marshal code.
The visitor code becomes slightly more verbose, but remains
semantically equivalent, and is actually easier to read as
it follows a more common idiom:
| visit_optional(v, &(*obj)->has_device, "device", &err);
|- if (!err && (*obj)->has_device) {
|- visit_type_str(v, &(*obj)->device, "device", &err);
|- }
| if (err) {
| goto out;
| }
|+ if ((*obj)->has_device) {
|+ visit_type_str(v, &(*obj)->device, "device", &err);
|+ if (err) {
|+ goto out;
|+ }
|+ }
The event code becomes slightly more verbose, but this is
arguably a bug fix: although the visitors are not well
documented, use of an optional member should not be attempted
unless guarded by a prior call to visit_optional(). Works only
because the output qmp visitor has a no-op visit_optional():
|+ visit_optional(v, &has_offset, "offset", &err);
|+ if (err) {
|+ goto out;
|+ }
| if (has_offset) {
| visit_type_int(v, &offset, "offset", &err);
Backports commit 82ca8e469666b169ccf818a0e36136aee97d7db0 from qemu
2018-02-19 23:41:37 +00:00
|
|
|
ret += mcgen('''
|
|
|
|
if (%(prefix)shas_%(c_name)s) {
|
|
|
|
''',
|
|
|
|
prefix=prefix, c_name=c_name(memb.name))
|
|
|
|
push_indent()
|
|
|
|
|
|
|
|
# Ugly: sometimes we need to cast away const
|
|
|
|
if need_cast and memb.type.name == 'str':
|
|
|
|
cast = '(char **)'
|
|
|
|
else:
|
|
|
|
cast = ''
|
|
|
|
|
|
|
|
ret += mcgen('''
|
|
|
|
visit_type_%(c_type)s(v, %(cast)s&%(prefix)s%(c_name)s, "%(name)s", %(errp)s);
|
|
|
|
''',
|
|
|
|
c_type=memb.type.c_name(), prefix=prefix, cast=cast,
|
|
|
|
c_name=c_name(memb.name), name=memb.name,
|
|
|
|
errp=errparg)
|
2018-02-19 23:43:38 +00:00
|
|
|
ret += gen_err_check(skiperr=skiperr)
|
qapi: Share gen_visit_fields()
Consolidate the code between visit, command marshalling, and
event generation that iterates over the members of a struct.
It reduces code duplication in the generator, so that a future
patch can reduce the size of generated code while touching only
one instead of three locations.
There are no changes to the generated marshal code.
The visitor code becomes slightly more verbose, but remains
semantically equivalent, and is actually easier to read as
it follows a more common idiom:
| visit_optional(v, &(*obj)->has_device, "device", &err);
|- if (!err && (*obj)->has_device) {
|- visit_type_str(v, &(*obj)->device, "device", &err);
|- }
| if (err) {
| goto out;
| }
|+ if ((*obj)->has_device) {
|+ visit_type_str(v, &(*obj)->device, "device", &err);
|+ if (err) {
|+ goto out;
|+ }
|+ }
The event code becomes slightly more verbose, but this is
arguably a bug fix: although the visitors are not well
documented, use of an optional member should not be attempted
unless guarded by a prior call to visit_optional(). Works only
because the output qmp visitor has a no-op visit_optional():
|+ visit_optional(v, &has_offset, "offset", &err);
|+ if (err) {
|+ goto out;
|+ }
| if (has_offset) {
| visit_type_int(v, &offset, "offset", &err);
Backports commit 82ca8e469666b169ccf818a0e36136aee97d7db0 from qemu
2018-02-19 23:41:37 +00:00
|
|
|
|
|
|
|
if memb.optional:
|
|
|
|
pop_indent()
|
|
|
|
ret += mcgen('''
|
|
|
|
}
|
|
|
|
''')
|
|
|
|
return ret
|
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
# Common command line parsing
|
|
|
|
#
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
|
|
|
def parse_command_line(extra_options="", extra_long_options=[]):
|
2018-02-19 20:19:07 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
opts, args = getopt.gnu_getopt(sys.argv[1:],
|
2018-02-19 20:22:24 +00:00
|
|
|
"chp:o:" + extra_options,
|
2018-02-19 20:19:07 +00:00
|
|
|
["source", "header", "prefix=",
|
2018-02-19 20:22:24 +00:00
|
|
|
"output-dir="] + extra_long_options)
|
2018-02-19 20:19:07 +00:00
|
|
|
except getopt.GetoptError, err:
|
2018-02-19 20:20:05 +00:00
|
|
|
print >>sys.stderr, "%s: %s" % (sys.argv[0], str(err))
|
2018-02-19 20:19:07 +00:00
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
output_dir = ""
|
|
|
|
prefix = ""
|
|
|
|
do_c = False
|
|
|
|
do_h = False
|
|
|
|
extra_opts = []
|
|
|
|
|
|
|
|
for oa in opts:
|
|
|
|
o, a = oa
|
|
|
|
if o in ("-p", "--prefix"):
|
2018-02-19 21:07:38 +00:00
|
|
|
match = re.match('([A-Za-z_.-][A-Za-z0-9_.-]*)?', a)
|
|
|
|
if match.end() != len(a):
|
|
|
|
print >>sys.stderr, \
|
|
|
|
"%s: 'funny character '%s' in argument of --prefix" \
|
|
|
|
% (sys.argv[0], a[match.end()])
|
|
|
|
sys.exit(1)
|
2018-02-19 20:19:07 +00:00
|
|
|
prefix = a
|
|
|
|
elif o in ("-o", "--output-dir"):
|
|
|
|
output_dir = a + "/"
|
|
|
|
elif o in ("-c", "--source"):
|
|
|
|
do_c = True
|
|
|
|
elif o in ("-h", "--header"):
|
|
|
|
do_h = True
|
|
|
|
else:
|
|
|
|
extra_opts.append(oa)
|
|
|
|
|
|
|
|
if not do_c and not do_h:
|
|
|
|
do_c = True
|
|
|
|
do_h = True
|
|
|
|
|
2018-02-19 20:22:24 +00:00
|
|
|
if len(args) != 1:
|
|
|
|
print >>sys.stderr, "%s: need exactly one argument" % sys.argv[0]
|
2018-02-19 20:20:05 +00:00
|
|
|
sys.exit(1)
|
2018-02-19 20:34:51 +00:00
|
|
|
fname = args[0]
|
2018-02-19 20:20:05 +00:00
|
|
|
|
2018-02-19 20:34:51 +00:00
|
|
|
return (fname, output_dir, do_c, do_h, prefix, extra_opts)
|
2018-02-19 20:31:47 +00:00
|
|
|
|
2018-02-19 20:47:16 +00:00
|
|
|
#
|
|
|
|
# Generate output files with boilerplate
|
|
|
|
#
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:31:47 +00:00
|
|
|
def open_output(output_dir, do_c, do_h, prefix, c_file, h_file,
|
|
|
|
c_comment, h_comment):
|
2018-02-19 21:05:50 +00:00
|
|
|
guard = guardname(prefix + h_file)
|
2018-02-19 20:31:47 +00:00
|
|
|
c_file = output_dir + prefix + c_file
|
|
|
|
h_file = output_dir + prefix + h_file
|
|
|
|
|
2018-02-19 21:32:48 +00:00
|
|
|
if output_dir:
|
|
|
|
try:
|
|
|
|
os.makedirs(output_dir)
|
|
|
|
except os.error, e:
|
|
|
|
if e.errno != errno.EEXIST:
|
|
|
|
raise
|
2018-02-19 20:31:47 +00:00
|
|
|
|
|
|
|
def maybe_open(really, name, opt):
|
|
|
|
if really:
|
|
|
|
return open(name, opt)
|
|
|
|
else:
|
|
|
|
import StringIO
|
|
|
|
return StringIO.StringIO()
|
|
|
|
|
|
|
|
fdef = maybe_open(do_c, c_file, 'w')
|
|
|
|
fdecl = maybe_open(do_h, h_file, 'w')
|
|
|
|
|
|
|
|
fdef.write(mcgen('''
|
|
|
|
/* AUTOMATICALLY GENERATED, DO NOT MODIFY */
|
|
|
|
%(comment)s
|
|
|
|
''',
|
2018-02-19 23:07:49 +00:00
|
|
|
comment=c_comment))
|
2018-02-19 20:31:47 +00:00
|
|
|
|
|
|
|
fdecl.write(mcgen('''
|
|
|
|
/* AUTOMATICALLY GENERATED, DO NOT MODIFY */
|
|
|
|
%(comment)s
|
|
|
|
#ifndef %(guard)s
|
|
|
|
#define %(guard)s
|
|
|
|
|
|
|
|
''',
|
2018-02-19 23:07:49 +00:00
|
|
|
comment=h_comment, guard=guard))
|
2018-02-19 20:31:47 +00:00
|
|
|
|
|
|
|
return (fdef, fdecl)
|
|
|
|
|
2018-02-19 23:07:49 +00:00
|
|
|
|
2018-02-19 20:31:47 +00:00
|
|
|
def close_output(fdef, fdecl):
|
|
|
|
fdecl.write('''
|
|
|
|
#endif
|
|
|
|
''')
|
|
|
|
fdecl.close()
|
|
|
|
fdef.close()
|
|
|
|
|