Markus Armbruster | 5ddeec8 | 2018-02-11 10:35:41 +0100 | [diff] [blame] | 1 | """ |
| 2 | QAPI introspection generator |
| 3 | |
John Snow | cf26906 | 2021-02-15 21:18:06 -0500 | [diff] [blame] | 4 | Copyright (C) 2015-2021 Red Hat, Inc. |
Markus Armbruster | 5ddeec8 | 2018-02-11 10:35:41 +0100 | [diff] [blame] | 5 | |
| 6 | Authors: |
| 7 | Markus Armbruster <armbru@redhat.com> |
John Snow | cf26906 | 2021-02-15 21:18:06 -0500 | [diff] [blame] | 8 | John Snow <jsnow@redhat.com> |
Markus Armbruster | 5ddeec8 | 2018-02-11 10:35:41 +0100 | [diff] [blame] | 9 | |
| 10 | This work is licensed under the terms of the GNU GPL, version 2. |
| 11 | See the COPYING file in the top-level directory. |
| 12 | """ |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 13 | |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 14 | from typing import ( |
| 15 | Any, |
| 16 | Dict, |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 17 | Generic, |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 18 | List, |
| 19 | Optional, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 20 | Sequence, |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 21 | TypeVar, |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 22 | Union, |
| 23 | ) |
John Snow | 5f50ced | 2021-02-15 21:17:57 -0500 | [diff] [blame] | 24 | |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 25 | from .common import c_name, mcgen |
John Snow | 7137a96 | 2020-10-09 12:15:27 -0400 | [diff] [blame] | 26 | from .gen import QAPISchemaMonolithicCVisitor |
John Snow | 67fea57 | 2020-10-09 12:15:29 -0400 | [diff] [blame] | 27 | from .schema import ( |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 28 | QAPISchema, |
John Snow | 67fea57 | 2020-10-09 12:15:29 -0400 | [diff] [blame] | 29 | QAPISchemaArrayType, |
| 30 | QAPISchemaBuiltinType, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 31 | QAPISchemaEntity, |
| 32 | QAPISchemaEnumMember, |
| 33 | QAPISchemaFeature, |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 34 | QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 35 | QAPISchemaObjectType, |
| 36 | QAPISchemaObjectTypeMember, |
John Snow | 67fea57 | 2020-10-09 12:15:29 -0400 | [diff] [blame] | 37 | QAPISchemaType, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 38 | QAPISchemaVariant, |
| 39 | QAPISchemaVariants, |
John Snow | 67fea57 | 2020-10-09 12:15:29 -0400 | [diff] [blame] | 40 | ) |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 41 | from .source import QAPISourceInfo |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 42 | |
| 43 | |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 44 | # This module constructs a tree data structure that is used to |
| 45 | # generate the introspection information for QEMU. It is shaped |
| 46 | # like a JSON value. |
| 47 | # |
| 48 | # A complexity over JSON is that our values may or may not be annotated. |
| 49 | # |
| 50 | # Un-annotated values may be: |
| 51 | # Scalar: str, bool, None. |
| 52 | # Non-scalar: List, Dict |
| 53 | # _value = Union[str, bool, None, Dict[str, JSONValue], List[JSONValue]] |
| 54 | # |
| 55 | # With optional annotations, the type of all values is: |
| 56 | # JSONValue = Union[_Value, Annotated[_Value]] |
| 57 | # |
| 58 | # Sadly, mypy does not support recursive types; so the _Stub alias is used to |
| 59 | # mark the imprecision in the type model where we'd otherwise use JSONValue. |
| 60 | _Stub = Any |
| 61 | _Scalar = Union[str, bool, None] |
| 62 | _NonScalar = Union[Dict[str, _Stub], List[_Stub]] |
| 63 | _Value = Union[_Scalar, _NonScalar] |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 64 | JSONValue = Union[_Value, 'Annotated[_Value]'] |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 65 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 66 | # These types are based on structures defined in QEMU's schema, so we |
| 67 | # lack precise types for them here. Python 3.6 does not offer |
| 68 | # TypedDict constructs, so they are broadly typed here as simple |
| 69 | # Python Dicts. |
| 70 | SchemaInfo = Dict[str, object] |
Markus Armbruster | 75ecee7 | 2021-10-25 06:24:01 +0200 | [diff] [blame] | 71 | SchemaInfoEnumMember = Dict[str, object] |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 72 | SchemaInfoObject = Dict[str, object] |
| 73 | SchemaInfoObjectVariant = Dict[str, object] |
| 74 | SchemaInfoObjectMember = Dict[str, object] |
| 75 | SchemaInfoCommand = Dict[str, object] |
| 76 | |
John Snow | 9db2734 | 2021-02-15 21:17:59 -0500 | [diff] [blame] | 77 | |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 78 | _ValueT = TypeVar('_ValueT', bound=_Value) |
| 79 | |
| 80 | |
| 81 | class Annotated(Generic[_ValueT]): |
| 82 | """ |
| 83 | Annotated generally contains a SchemaInfo-like type (as a dict), |
| 84 | But it also used to wrap comments/ifconds around scalar leaf values, |
| 85 | for the benefit of features and enums. |
| 86 | """ |
| 87 | # TODO: Remove after Python 3.7 adds @dataclass: |
| 88 | # pylint: disable=too-few-public-methods |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 89 | def __init__(self, value: _ValueT, ifcond: QAPISchemaIfCond, |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 90 | comment: Optional[str] = None): |
| 91 | self.value = value |
| 92 | self.comment: Optional[str] = comment |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 93 | self.ifcond = ifcond |
Markus Armbruster | 24cfd6a | 2020-03-17 12:54:40 +0100 | [diff] [blame] | 94 | |
| 95 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 96 | def _tree_to_qlit(obj: JSONValue, |
| 97 | level: int = 0, |
| 98 | dict_value: bool = False) -> str: |
John Snow | 5444ded | 2021-02-15 21:18:05 -0500 | [diff] [blame] | 99 | """ |
| 100 | Convert the type tree into a QLIT C string, recursively. |
| 101 | |
| 102 | :param obj: The value to convert. |
| 103 | This value may not be Annotated when dict_value is True. |
| 104 | :param level: The indentation level for this particular value. |
| 105 | :param dict_value: True when the value being processed belongs to a |
| 106 | dict key; which suppresses the output indent. |
| 107 | """ |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 108 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 109 | def indent(level: int) -> str: |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 110 | return level * 4 * ' ' |
| 111 | |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 112 | if isinstance(obj, Annotated): |
John Snow | 0555696 | 2021-02-15 21:17:55 -0500 | [diff] [blame] | 113 | # NB: _tree_to_qlit is called recursively on the values of a |
| 114 | # key:value pair; those values can't be decorated with |
| 115 | # comments or conditionals. |
| 116 | msg = "dict values cannot have attached comments or if-conditionals." |
| 117 | assert not dict_value, msg |
| 118 | |
Eric Blake | 8c64336 | 2018-08-27 16:39:43 -0500 | [diff] [blame] | 119 | ret = '' |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 120 | if obj.comment: |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 121 | ret += indent(level) + f"/* {obj.comment} */\n" |
Marc-André Lureau | 33aa326 | 2021-08-04 12:30:58 +0400 | [diff] [blame] | 122 | if obj.ifcond.is_present(): |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 123 | ret += obj.ifcond.gen_if() |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 124 | ret += _tree_to_qlit(obj.value, level) |
Marc-André Lureau | 33aa326 | 2021-08-04 12:30:58 +0400 | [diff] [blame] | 125 | if obj.ifcond.is_present(): |
Markus Armbruster | 1889e57 | 2021-08-31 14:37:58 +0200 | [diff] [blame] | 126 | ret += '\n' + obj.ifcond.gen_endif() |
Marc-André Lureau | d626b6c | 2018-07-03 17:56:42 +0200 | [diff] [blame] | 127 | return ret |
| 128 | |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 129 | ret = '' |
John Snow | 0555696 | 2021-02-15 21:17:55 -0500 | [diff] [blame] | 130 | if not dict_value: |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 131 | ret += indent(level) |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 132 | |
| 133 | # Scalars: |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 134 | if obj is None: |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 135 | ret += 'QLIT_QNULL' |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 136 | elif isinstance(obj, str): |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 137 | ret += f"QLIT_QSTR({to_c_string(obj)})" |
| 138 | elif isinstance(obj, bool): |
| 139 | ret += f"QLIT_QBOOL({str(obj).lower()})" |
| 140 | |
| 141 | # Non-scalars: |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 142 | elif isinstance(obj, list): |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 143 | ret += 'QLIT_QLIST(((QLitObject[]) {\n' |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 144 | for value in obj: |
| 145 | ret += _tree_to_qlit(value, level + 1).strip('\n') + '\n' |
| 146 | ret += indent(level + 1) + '{}\n' |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 147 | ret += indent(level) + '}))' |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 148 | elif isinstance(obj, dict): |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 149 | ret += 'QLIT_QDICT(((QLitDictEntry[]) {\n' |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 150 | for key, value in sorted(obj.items()): |
| 151 | ret += indent(level + 1) + "{{ {:s}, {:s} }},\n".format( |
| 152 | to_c_string(key), |
| 153 | _tree_to_qlit(value, level + 1, dict_value=True) |
| 154 | ) |
| 155 | ret += indent(level + 1) + '{}\n' |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 156 | ret += indent(level) + '}))' |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 157 | else: |
John Snow | 2a6c161 | 2021-02-15 21:18:01 -0500 | [diff] [blame] | 158 | raise NotImplementedError( |
| 159 | f"type '{type(obj).__name__}' not implemented" |
| 160 | ) |
John Snow | c0e8d9f | 2021-02-15 21:18:02 -0500 | [diff] [blame] | 161 | |
Marc-André Lureau | 40bb137 | 2018-07-03 17:56:41 +0200 | [diff] [blame] | 162 | if level > 0: |
| 163 | ret += ',' |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 164 | return ret |
| 165 | |
| 166 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 167 | def to_c_string(string: str) -> str: |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 168 | return '"' + string.replace('\\', r'\\').replace('"', r'\"') + '"' |
| 169 | |
| 170 | |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 171 | class QAPISchemaGenIntrospectVisitor(QAPISchemaMonolithicCVisitor): |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 172 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 173 | def __init__(self, prefix: str, unmask: bool): |
Markus Armbruster | 2cae67b | 2020-03-04 16:59:31 +0100 | [diff] [blame] | 174 | super().__init__( |
| 175 | prefix, 'qapi-introspect', |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 176 | ' * QAPI/QMP schema introspection', __doc__) |
| 177 | self._unmask = unmask |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 178 | self._schema: Optional[QAPISchema] = None |
| 179 | self._trees: List[Annotated[SchemaInfo]] = [] |
| 180 | self._used_types: List[QAPISchemaType] = [] |
| 181 | self._name_map: Dict[str, str] = {} |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 182 | self._genc.add(mcgen(''' |
| 183 | #include "qemu/osdep.h" |
Markus Armbruster | eb815e2 | 2018-02-11 10:36:05 +0100 | [diff] [blame] | 184 | #include "%(prefix)sqapi-introspect.h" |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 185 | |
| 186 | ''', |
| 187 | prefix=prefix)) |
| 188 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 189 | def visit_begin(self, schema: QAPISchema) -> None: |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 190 | self._schema = schema |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 191 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 192 | def visit_end(self) -> None: |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 193 | # visit the types that are actually used |
| 194 | for typ in self._used_types: |
| 195 | typ.visit(self) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 196 | # generate C |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 197 | name = c_name(self._prefix, protect=False) + 'qmp_schema_qlit' |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 198 | self._genh.add(mcgen(''' |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 199 | #include "qapi/qmp/qlit.h" |
| 200 | |
| 201 | extern const QLitObject %(c_name)s; |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 202 | ''', |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 203 | c_name=c_name(name))) |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 204 | self._genc.add(mcgen(''' |
Marc-André Lureau | 7d0f982 | 2018-03-05 18:29:51 +0100 | [diff] [blame] | 205 | const QLitObject %(c_name)s = %(c_string)s; |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 206 | ''', |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 207 | c_name=c_name(name), |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 208 | c_string=_tree_to_qlit(self._trees))) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 209 | self._schema = None |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 210 | self._trees = [] |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 211 | self._used_types = [] |
| 212 | self._name_map = {} |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 213 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 214 | def visit_needed(self, entity: QAPISchemaEntity) -> bool: |
Eric Blake | 25a0d9c | 2015-10-12 22:22:21 -0600 | [diff] [blame] | 215 | # Ignore types on first pass; visit_end() will pick up used types |
| 216 | return not isinstance(entity, QAPISchemaType) |
| 217 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 218 | def _name(self, name: str) -> str: |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 219 | if self._unmask: |
| 220 | return name |
| 221 | if name not in self._name_map: |
| 222 | self._name_map[name] = '%d' % len(self._name_map) |
| 223 | return self._name_map[name] |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 224 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 225 | def _use_type(self, typ: QAPISchemaType) -> str: |
John Snow | 6b67bca | 2021-02-15 21:17:52 -0500 | [diff] [blame] | 226 | assert self._schema is not None |
| 227 | |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 228 | # Map the various integer types to plain int |
| 229 | if typ.json_type() == 'int': |
| 230 | typ = self._schema.lookup_type('int') |
| 231 | elif (isinstance(typ, QAPISchemaArrayType) and |
| 232 | typ.element_type.json_type() == 'int'): |
| 233 | typ = self._schema.lookup_type('intList') |
| 234 | # Add type to work queue if new |
| 235 | if typ not in self._used_types: |
| 236 | self._used_types.append(typ) |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 237 | # Clients should examine commands and events, not types. Hide |
Eric Blake | 1aa806c | 2018-08-27 16:39:42 -0500 | [diff] [blame] | 238 | # type names as integers to reduce the temptation. Also, it |
| 239 | # saves a few characters on the wire. |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 240 | if isinstance(typ, QAPISchemaBuiltinType): |
| 241 | return typ.name |
Eric Blake | ce5fcb4 | 2015-11-05 23:35:35 -0700 | [diff] [blame] | 242 | if isinstance(typ, QAPISchemaArrayType): |
| 243 | return '[' + self._use_type(typ.element_type) + ']' |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 244 | return self._name(typ.name) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 245 | |
John Snow | 84bece7 | 2021-02-15 21:17:54 -0500 | [diff] [blame] | 246 | @staticmethod |
John Snow | cea53c3 | 2021-02-15 21:18:07 -0500 | [diff] [blame] | 247 | def _gen_features(features: Sequence[QAPISchemaFeature] |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 248 | ) -> List[Annotated[str]]: |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 249 | return [Annotated(f.name, f.ifcond) for f in features] |
John Snow | 84bece7 | 2021-02-15 21:17:54 -0500 | [diff] [blame] | 250 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 251 | def _gen_tree(self, name: str, mtype: str, obj: Dict[str, object], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 252 | ifcond: QAPISchemaIfCond = QAPISchemaIfCond(), |
John Snow | cea53c3 | 2021-02-15 21:18:07 -0500 | [diff] [blame] | 253 | features: Sequence[QAPISchemaFeature] = ()) -> None: |
John Snow | 5444ded | 2021-02-15 21:18:05 -0500 | [diff] [blame] | 254 | """ |
| 255 | Build and append a SchemaInfo object to self._trees. |
| 256 | |
| 257 | :param name: The SchemaInfo's name. |
| 258 | :param mtype: The SchemaInfo's meta-type. |
| 259 | :param obj: Additional SchemaInfo members, as appropriate for |
| 260 | the meta-type. |
| 261 | :param ifcond: Conditionals to apply to the SchemaInfo. |
| 262 | :param features: The SchemaInfo's features. |
| 263 | Will be omitted from the output if empty. |
| 264 | """ |
John Snow | 5f50ced | 2021-02-15 21:17:57 -0500 | [diff] [blame] | 265 | comment: Optional[str] = None |
Eric Blake | ce5fcb4 | 2015-11-05 23:35:35 -0700 | [diff] [blame] | 266 | if mtype not in ('command', 'event', 'builtin', 'array'): |
Eric Blake | 8c64336 | 2018-08-27 16:39:43 -0500 | [diff] [blame] | 267 | if not self._unmask: |
| 268 | # Output a comment to make it easy to map masked names |
| 269 | # back to the source when reading the generated output. |
John Snow | 5f50ced | 2021-02-15 21:17:57 -0500 | [diff] [blame] | 270 | comment = f'"{self._name(name)}" = {name}' |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 271 | name = self._name(name) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 272 | obj['name'] = name |
| 273 | obj['meta-type'] = mtype |
John Snow | 84bece7 | 2021-02-15 21:17:54 -0500 | [diff] [blame] | 274 | if features: |
| 275 | obj['features'] = self._gen_features(features) |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 276 | self._trees.append(Annotated(obj, ifcond, comment)) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 277 | |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 278 | def _gen_enum_member(self, member: QAPISchemaEnumMember |
Markus Armbruster | 75ecee7 | 2021-10-25 06:24:01 +0200 | [diff] [blame] | 279 | ) -> Annotated[SchemaInfoEnumMember]: |
| 280 | obj: SchemaInfoEnumMember = { |
| 281 | 'name': member.name, |
| 282 | } |
Markus Armbruster | b6c1875 | 2021-10-25 06:24:02 +0200 | [diff] [blame] | 283 | if member.features: |
| 284 | obj['features'] = self._gen_features(member.features) |
Markus Armbruster | 75ecee7 | 2021-10-25 06:24:01 +0200 | [diff] [blame] | 285 | return Annotated(obj, member.ifcond) |
| 286 | |
| 287 | def _gen_object_member(self, member: QAPISchemaObjectTypeMember |
| 288 | ) -> Annotated[SchemaInfoObjectMember]: |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 289 | obj: SchemaInfoObjectMember = { |
| 290 | 'name': member.name, |
| 291 | 'type': self._use_type(member.type) |
| 292 | } |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 293 | if member.optional: |
Markus Armbruster | 24cfd6a | 2020-03-17 12:54:40 +0100 | [diff] [blame] | 294 | obj['default'] = None |
John Snow | 84bece7 | 2021-02-15 21:17:54 -0500 | [diff] [blame] | 295 | if member.features: |
| 296 | obj['features'] = self._gen_features(member.features) |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 297 | return Annotated(obj, member.ifcond) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 298 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 299 | def _gen_variant(self, variant: QAPISchemaVariant |
| 300 | ) -> Annotated[SchemaInfoObjectVariant]: |
| 301 | obj: SchemaInfoObjectVariant = { |
| 302 | 'case': variant.name, |
| 303 | 'type': self._use_type(variant.type) |
| 304 | } |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 305 | return Annotated(obj, variant.ifcond) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 306 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 307 | def visit_builtin_type(self, name: str, info: Optional[QAPISourceInfo], |
| 308 | json_type: str) -> None: |
John Snow | 9b77d94 | 2021-02-15 21:18:08 -0500 | [diff] [blame] | 309 | self._gen_tree(name, 'builtin', {'json-type': json_type}) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 310 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 311 | def visit_enum_type(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 312 | ifcond: QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 313 | features: List[QAPISchemaFeature], |
| 314 | members: List[QAPISchemaEnumMember], |
| 315 | prefix: Optional[str]) -> None: |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 316 | self._gen_tree( |
| 317 | name, 'enum', |
Markus Armbruster | 75ecee7 | 2021-10-25 06:24:01 +0200 | [diff] [blame] | 318 | {'members': [self._gen_enum_member(m) for m in members], |
| 319 | 'values': [Annotated(m.name, m.ifcond) for m in members]}, |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 320 | ifcond, features |
| 321 | ) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 322 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 323 | def visit_array_type(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 324 | ifcond: QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 325 | element_type: QAPISchemaType) -> None: |
Eric Blake | ce5fcb4 | 2015-11-05 23:35:35 -0700 | [diff] [blame] | 326 | element = self._use_type(element_type) |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 327 | self._gen_tree('[' + element + ']', 'array', {'element-type': element}, |
John Snow | cea53c3 | 2021-02-15 21:18:07 -0500 | [diff] [blame] | 328 | ifcond) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 329 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 330 | def visit_object_type_flat(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 331 | ifcond: QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 332 | features: List[QAPISchemaFeature], |
| 333 | members: List[QAPISchemaObjectTypeMember], |
| 334 | variants: Optional[QAPISchemaVariants]) -> None: |
| 335 | obj: SchemaInfoObject = { |
Markus Armbruster | 75ecee7 | 2021-10-25 06:24:01 +0200 | [diff] [blame] | 336 | 'members': [self._gen_object_member(m) for m in members] |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 337 | } |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 338 | if variants: |
John Snow | cf5db21 | 2021-02-15 21:18:03 -0500 | [diff] [blame] | 339 | obj['tag'] = variants.tag_member.name |
| 340 | obj['variants'] = [self._gen_variant(v) for v in variants.variants] |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 341 | self._gen_tree(name, 'object', obj, ifcond, features) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 342 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 343 | def visit_alternate_type(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 344 | ifcond: QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 345 | features: List[QAPISchemaFeature], |
| 346 | variants: QAPISchemaVariants) -> None: |
John Snow | 4f7f97a | 2021-02-15 21:18:00 -0500 | [diff] [blame] | 347 | self._gen_tree( |
| 348 | name, 'alternate', |
| 349 | {'members': [Annotated({'type': self._use_type(m.type)}, |
| 350 | m.ifcond) |
| 351 | for m in variants.variants]}, |
| 352 | ifcond, features |
| 353 | ) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 354 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 355 | def visit_command(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 356 | ifcond: QAPISchemaIfCond, |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 357 | features: List[QAPISchemaFeature], |
| 358 | arg_type: Optional[QAPISchemaObjectType], |
| 359 | ret_type: Optional[QAPISchemaType], gen: bool, |
| 360 | success_response: bool, boxed: bool, allow_oob: bool, |
| 361 | allow_preconfig: bool, coroutine: bool) -> None: |
John Snow | 6b67bca | 2021-02-15 21:17:52 -0500 | [diff] [blame] | 362 | assert self._schema is not None |
| 363 | |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 364 | arg_type = arg_type or self._schema.the_empty_object_type |
| 365 | ret_type = ret_type or self._schema.the_empty_object_type |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 366 | obj: SchemaInfoCommand = { |
| 367 | 'arg-type': self._use_type(arg_type), |
| 368 | 'ret-type': self._use_type(ret_type) |
| 369 | } |
Markus Armbruster | 25b1ef3 | 2018-07-18 11:05:57 +0200 | [diff] [blame] | 370 | if allow_oob: |
| 371 | obj['allow-oob'] = allow_oob |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 372 | self._gen_tree(name, 'command', obj, ifcond, features) |
Peter Krempa | 23394b4 | 2019-10-18 10:14:51 +0200 | [diff] [blame] | 373 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 374 | def visit_event(self, name: str, info: Optional[QAPISourceInfo], |
Marc-André Lureau | f17539c | 2021-08-04 12:30:57 +0400 | [diff] [blame] | 375 | ifcond: QAPISchemaIfCond, |
| 376 | features: List[QAPISchemaFeature], |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 377 | arg_type: Optional[QAPISchemaObjectType], |
| 378 | boxed: bool) -> None: |
John Snow | 6b67bca | 2021-02-15 21:17:52 -0500 | [diff] [blame] | 379 | assert self._schema is not None |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 380 | |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 381 | arg_type = arg_type or self._schema.the_empty_object_type |
Markus Armbruster | 2e8a843 | 2020-03-17 12:54:39 +0100 | [diff] [blame] | 382 | self._gen_tree(name, 'event', {'arg-type': self._use_type(arg_type)}, |
Markus Armbruster | 013b4ef | 2020-03-17 12:54:37 +0100 | [diff] [blame] | 383 | ifcond, features) |
Markus Armbruster | 39a1815 | 2015-09-16 13:06:28 +0200 | [diff] [blame] | 384 | |
Markus Armbruster | 1a9a507 | 2015-09-16 13:06:29 +0200 | [diff] [blame] | 385 | |
John Snow | 82b52f6 | 2021-02-15 21:18:04 -0500 | [diff] [blame] | 386 | def gen_introspect(schema: QAPISchema, output_dir: str, prefix: str, |
| 387 | opt_unmask: bool) -> None: |
Markus Armbruster | 26df4e7 | 2018-02-26 13:39:37 -0600 | [diff] [blame] | 388 | vis = QAPISchemaGenIntrospectVisitor(prefix, opt_unmask) |
| 389 | schema.visit(vis) |
Markus Armbruster | 71b3f04 | 2018-02-26 13:50:08 -0600 | [diff] [blame] | 390 | vis.write(output_dir) |