schema.py 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236
  1. # -*- coding: utf-8 -*-
  2. #
  3. # QAPI schema internal representation
  4. #
  5. # Copyright (c) 2015-2019 Red Hat Inc.
  6. #
  7. # Authors:
  8. # Markus Armbruster <armbru@redhat.com>
  9. # Eric Blake <eblake@redhat.com>
  10. # Marc-André Lureau <marcandre.lureau@redhat.com>
  11. #
  12. # This work is licensed under the terms of the GNU GPL, version 2.
  13. # See the COPYING file in the top-level directory.
  14. # pylint: disable=too-many-lines
  15. # TODO catching name collisions in generated code would be nice
  16. from collections import OrderedDict
  17. import os
  18. import re
  19. from typing import List, Optional
  20. from .common import (
  21. POINTER_SUFFIX,
  22. c_name,
  23. cgen_ifcond,
  24. docgen_ifcond,
  25. gen_endif,
  26. gen_if,
  27. )
  28. from .error import QAPIError, QAPISemError, QAPISourceError
  29. from .expr import check_exprs
  30. from .parser import QAPIExpression, QAPISchemaParser
  31. class QAPISchemaIfCond:
  32. def __init__(self, ifcond=None):
  33. self.ifcond = ifcond
  34. def _cgen(self):
  35. return cgen_ifcond(self.ifcond)
  36. def gen_if(self):
  37. return gen_if(self._cgen())
  38. def gen_endif(self):
  39. return gen_endif(self._cgen())
  40. def docgen(self):
  41. return docgen_ifcond(self.ifcond)
  42. def is_present(self):
  43. return bool(self.ifcond)
  44. class QAPISchemaEntity:
  45. meta: Optional[str] = None
  46. def __init__(self, name: str, info, doc, ifcond=None, features=None):
  47. assert name is None or isinstance(name, str)
  48. for f in features or []:
  49. assert isinstance(f, QAPISchemaFeature)
  50. f.set_defined_in(name)
  51. self.name = name
  52. self._module = None
  53. # For explicitly defined entities, info points to the (explicit)
  54. # definition. For builtins (and their arrays), info is None.
  55. # For implicitly defined entities, info points to a place that
  56. # triggered the implicit definition (there may be more than one
  57. # such place).
  58. self.info = info
  59. self.doc = doc
  60. self._ifcond = ifcond or QAPISchemaIfCond()
  61. self.features = features or []
  62. self._checked = False
  63. def __repr__(self):
  64. if self.name is None:
  65. return "<%s at 0x%x>" % (type(self).__name__, id(self))
  66. return "<%s:%s at 0x%x>" % (type(self).__name__, self.name,
  67. id(self))
  68. def c_name(self):
  69. return c_name(self.name)
  70. def check(self, schema):
  71. # pylint: disable=unused-argument
  72. assert not self._checked
  73. seen = {}
  74. for f in self.features:
  75. f.check_clash(self.info, seen)
  76. self._checked = True
  77. def connect_doc(self, doc=None):
  78. doc = doc or self.doc
  79. if doc:
  80. for f in self.features:
  81. doc.connect_feature(f)
  82. def _set_module(self, schema, info):
  83. assert self._checked
  84. fname = info.fname if info else QAPISchemaModule.BUILTIN_MODULE_NAME
  85. self._module = schema.module_by_fname(fname)
  86. self._module.add_entity(self)
  87. def set_module(self, schema):
  88. self._set_module(schema, self.info)
  89. @property
  90. def ifcond(self):
  91. assert self._checked
  92. return self._ifcond
  93. def is_implicit(self):
  94. return not self.info
  95. def visit(self, visitor):
  96. # pylint: disable=unused-argument
  97. assert self._checked
  98. def describe(self):
  99. assert self.meta
  100. return "%s '%s'" % (self.meta, self.name)
  101. class QAPISchemaVisitor:
  102. def visit_begin(self, schema):
  103. pass
  104. def visit_end(self):
  105. pass
  106. def visit_module(self, name):
  107. pass
  108. def visit_needed(self, entity):
  109. # pylint: disable=unused-argument
  110. # Default to visiting everything
  111. return True
  112. def visit_include(self, name, info):
  113. pass
  114. def visit_builtin_type(self, name, info, json_type):
  115. pass
  116. def visit_enum_type(self, name, info, ifcond, features, members, prefix):
  117. pass
  118. def visit_array_type(self, name, info, ifcond, element_type):
  119. pass
  120. def visit_object_type(self, name, info, ifcond, features,
  121. base, members, variants):
  122. pass
  123. def visit_object_type_flat(self, name, info, ifcond, features,
  124. members, variants):
  125. pass
  126. def visit_alternate_type(self, name, info, ifcond, features, variants):
  127. pass
  128. def visit_command(self, name, info, ifcond, features,
  129. arg_type, ret_type, gen, success_response, boxed,
  130. allow_oob, allow_preconfig, coroutine):
  131. pass
  132. def visit_event(self, name, info, ifcond, features, arg_type, boxed):
  133. pass
  134. class QAPISchemaModule:
  135. BUILTIN_MODULE_NAME = './builtin'
  136. def __init__(self, name):
  137. self.name = name
  138. self._entity_list = []
  139. @staticmethod
  140. def is_system_module(name: str) -> bool:
  141. """
  142. System modules are internally defined modules.
  143. Their names start with the "./" prefix.
  144. """
  145. return name.startswith('./')
  146. @classmethod
  147. def is_user_module(cls, name: str) -> bool:
  148. """
  149. User modules are those defined by the user in qapi JSON files.
  150. They do not start with the "./" prefix.
  151. """
  152. return not cls.is_system_module(name)
  153. @classmethod
  154. def is_builtin_module(cls, name: str) -> bool:
  155. """
  156. The built-in module is a single System module for the built-in types.
  157. It is always "./builtin".
  158. """
  159. return name == cls.BUILTIN_MODULE_NAME
  160. def add_entity(self, ent):
  161. self._entity_list.append(ent)
  162. def visit(self, visitor):
  163. visitor.visit_module(self.name)
  164. for entity in self._entity_list:
  165. if visitor.visit_needed(entity):
  166. entity.visit(visitor)
  167. class QAPISchemaInclude(QAPISchemaEntity):
  168. def __init__(self, sub_module, info):
  169. super().__init__(None, info, None)
  170. self._sub_module = sub_module
  171. def visit(self, visitor):
  172. super().visit(visitor)
  173. visitor.visit_include(self._sub_module.name, self.info)
  174. class QAPISchemaType(QAPISchemaEntity):
  175. # Return the C type for common use.
  176. # For the types we commonly box, this is a pointer type.
  177. def c_type(self):
  178. pass
  179. # Return the C type to be used in a parameter list.
  180. def c_param_type(self):
  181. return self.c_type()
  182. # Return the C type to be used where we suppress boxing.
  183. def c_unboxed_type(self):
  184. return self.c_type()
  185. def json_type(self):
  186. pass
  187. def alternate_qtype(self):
  188. json2qtype = {
  189. 'null': 'QTYPE_QNULL',
  190. 'string': 'QTYPE_QSTRING',
  191. 'number': 'QTYPE_QNUM',
  192. 'int': 'QTYPE_QNUM',
  193. 'boolean': 'QTYPE_QBOOL',
  194. 'array': 'QTYPE_QLIST',
  195. 'object': 'QTYPE_QDICT'
  196. }
  197. return json2qtype.get(self.json_type())
  198. def doc_type(self):
  199. if self.is_implicit():
  200. return None
  201. return self.name
  202. def need_has_if_optional(self):
  203. # When FOO is a pointer, has_FOO == !!FOO, i.e. has_FOO is redundant.
  204. # Except for arrays; see QAPISchemaArrayType.need_has_if_optional().
  205. return not self.c_type().endswith(POINTER_SUFFIX)
  206. def check(self, schema):
  207. super().check(schema)
  208. for feat in self.features:
  209. if feat.is_special():
  210. raise QAPISemError(
  211. self.info,
  212. f"feature '{feat.name}' is not supported for types")
  213. def describe(self):
  214. assert self.meta
  215. return "%s type '%s'" % (self.meta, self.name)
  216. class QAPISchemaBuiltinType(QAPISchemaType):
  217. meta = 'built-in'
  218. def __init__(self, name, json_type, c_type):
  219. super().__init__(name, None, None)
  220. assert not c_type or isinstance(c_type, str)
  221. assert json_type in ('string', 'number', 'int', 'boolean', 'null',
  222. 'value')
  223. self._json_type_name = json_type
  224. self._c_type_name = c_type
  225. def c_name(self):
  226. return self.name
  227. def c_type(self):
  228. return self._c_type_name
  229. def c_param_type(self):
  230. if self.name == 'str':
  231. return 'const ' + self._c_type_name
  232. return self._c_type_name
  233. def json_type(self):
  234. return self._json_type_name
  235. def doc_type(self):
  236. return self.json_type()
  237. def visit(self, visitor):
  238. super().visit(visitor)
  239. visitor.visit_builtin_type(self.name, self.info, self.json_type())
  240. class QAPISchemaEnumType(QAPISchemaType):
  241. meta = 'enum'
  242. def __init__(self, name, info, doc, ifcond, features, members, prefix):
  243. super().__init__(name, info, doc, ifcond, features)
  244. for m in members:
  245. assert isinstance(m, QAPISchemaEnumMember)
  246. m.set_defined_in(name)
  247. assert prefix is None or isinstance(prefix, str)
  248. self.members = members
  249. self.prefix = prefix
  250. def check(self, schema):
  251. super().check(schema)
  252. seen = {}
  253. for m in self.members:
  254. m.check_clash(self.info, seen)
  255. def connect_doc(self, doc=None):
  256. super().connect_doc(doc)
  257. doc = doc or self.doc
  258. for m in self.members:
  259. m.connect_doc(doc)
  260. def is_implicit(self):
  261. # See QAPISchema._def_predefineds()
  262. return self.name == 'QType'
  263. def c_type(self):
  264. return c_name(self.name)
  265. def member_names(self):
  266. return [m.name for m in self.members]
  267. def json_type(self):
  268. return 'string'
  269. def visit(self, visitor):
  270. super().visit(visitor)
  271. visitor.visit_enum_type(
  272. self.name, self.info, self.ifcond, self.features,
  273. self.members, self.prefix)
  274. class QAPISchemaArrayType(QAPISchemaType):
  275. meta = 'array'
  276. def __init__(self, name, info, element_type):
  277. super().__init__(name, info, None)
  278. assert isinstance(element_type, str)
  279. self._element_type_name = element_type
  280. self.element_type = None
  281. def need_has_if_optional(self):
  282. # When FOO is an array, we still need has_FOO to distinguish
  283. # absent (!has_FOO) from present and empty (has_FOO && !FOO).
  284. return True
  285. def check(self, schema):
  286. super().check(schema)
  287. self.element_type = schema.resolve_type(
  288. self._element_type_name, self.info,
  289. self.info and self.info.defn_meta)
  290. assert not isinstance(self.element_type, QAPISchemaArrayType)
  291. def set_module(self, schema):
  292. self._set_module(schema, self.element_type.info)
  293. @property
  294. def ifcond(self):
  295. assert self._checked
  296. return self.element_type.ifcond
  297. def is_implicit(self):
  298. return True
  299. def c_type(self):
  300. return c_name(self.name) + POINTER_SUFFIX
  301. def json_type(self):
  302. return 'array'
  303. def doc_type(self):
  304. elt_doc_type = self.element_type.doc_type()
  305. if not elt_doc_type:
  306. return None
  307. return 'array of ' + elt_doc_type
  308. def visit(self, visitor):
  309. super().visit(visitor)
  310. visitor.visit_array_type(self.name, self.info, self.ifcond,
  311. self.element_type)
  312. def describe(self):
  313. assert self.meta
  314. return "%s type ['%s']" % (self.meta, self._element_type_name)
  315. class QAPISchemaObjectType(QAPISchemaType):
  316. def __init__(self, name, info, doc, ifcond, features,
  317. base, local_members, variants):
  318. # struct has local_members, optional base, and no variants
  319. # union has base, variants, and no local_members
  320. super().__init__(name, info, doc, ifcond, features)
  321. self.meta = 'union' if variants else 'struct'
  322. assert base is None or isinstance(base, str)
  323. for m in local_members:
  324. assert isinstance(m, QAPISchemaObjectTypeMember)
  325. m.set_defined_in(name)
  326. if variants is not None:
  327. assert isinstance(variants, QAPISchemaVariants)
  328. variants.set_defined_in(name)
  329. self._base_name = base
  330. self.base = None
  331. self.local_members = local_members
  332. self.variants = variants
  333. self.members = None
  334. def check(self, schema):
  335. # This calls another type T's .check() exactly when the C
  336. # struct emitted by gen_object() contains that T's C struct
  337. # (pointers don't count).
  338. if self.members is not None:
  339. # A previous .check() completed: nothing to do
  340. return
  341. if self._checked:
  342. # Recursed: C struct contains itself
  343. raise QAPISemError(self.info,
  344. "object %s contains itself" % self.name)
  345. super().check(schema)
  346. assert self._checked and self.members is None
  347. seen = OrderedDict()
  348. if self._base_name:
  349. self.base = schema.resolve_type(self._base_name, self.info,
  350. "'base'")
  351. if (not isinstance(self.base, QAPISchemaObjectType)
  352. or self.base.variants):
  353. raise QAPISemError(
  354. self.info,
  355. "'base' requires a struct type, %s isn't"
  356. % self.base.describe())
  357. self.base.check(schema)
  358. self.base.check_clash(self.info, seen)
  359. for m in self.local_members:
  360. m.check(schema)
  361. m.check_clash(self.info, seen)
  362. members = seen.values()
  363. if self.variants:
  364. self.variants.check(schema, seen)
  365. self.variants.check_clash(self.info, seen)
  366. self.members = members # mark completed
  367. # Check that the members of this type do not cause duplicate JSON members,
  368. # and update seen to track the members seen so far. Report any errors
  369. # on behalf of info, which is not necessarily self.info
  370. def check_clash(self, info, seen):
  371. assert self._checked
  372. for m in self.members:
  373. m.check_clash(info, seen)
  374. if self.variants:
  375. self.variants.check_clash(info, seen)
  376. def connect_doc(self, doc=None):
  377. super().connect_doc(doc)
  378. doc = doc or self.doc
  379. if self.base and self.base.is_implicit():
  380. self.base.connect_doc(doc)
  381. for m in self.local_members:
  382. m.connect_doc(doc)
  383. def is_implicit(self):
  384. # See QAPISchema._make_implicit_object_type(), as well as
  385. # _def_predefineds()
  386. return self.name.startswith('q_')
  387. def is_empty(self):
  388. assert self.members is not None
  389. return not self.members and not self.variants
  390. def has_conditional_members(self):
  391. assert self.members is not None
  392. return any(m.ifcond.is_present() for m in self.members)
  393. def c_name(self):
  394. assert self.name != 'q_empty'
  395. return super().c_name()
  396. def c_type(self):
  397. assert not self.is_implicit()
  398. return c_name(self.name) + POINTER_SUFFIX
  399. def c_unboxed_type(self):
  400. return c_name(self.name)
  401. def json_type(self):
  402. return 'object'
  403. def visit(self, visitor):
  404. super().visit(visitor)
  405. visitor.visit_object_type(
  406. self.name, self.info, self.ifcond, self.features,
  407. self.base, self.local_members, self.variants)
  408. visitor.visit_object_type_flat(
  409. self.name, self.info, self.ifcond, self.features,
  410. self.members, self.variants)
  411. class QAPISchemaAlternateType(QAPISchemaType):
  412. meta = 'alternate'
  413. def __init__(self, name, info, doc, ifcond, features, variants):
  414. super().__init__(name, info, doc, ifcond, features)
  415. assert isinstance(variants, QAPISchemaVariants)
  416. assert variants.tag_member
  417. variants.set_defined_in(name)
  418. variants.tag_member.set_defined_in(self.name)
  419. self.variants = variants
  420. def check(self, schema):
  421. super().check(schema)
  422. self.variants.tag_member.check(schema)
  423. # Not calling self.variants.check_clash(), because there's nothing
  424. # to clash with
  425. self.variants.check(schema, {})
  426. # Alternate branch names have no relation to the tag enum values;
  427. # so we have to check for potential name collisions ourselves.
  428. seen = {}
  429. types_seen = {}
  430. for v in self.variants.variants:
  431. v.check_clash(self.info, seen)
  432. qtype = v.type.alternate_qtype()
  433. if not qtype:
  434. raise QAPISemError(
  435. self.info,
  436. "%s cannot use %s"
  437. % (v.describe(self.info), v.type.describe()))
  438. conflicting = set([qtype])
  439. if qtype == 'QTYPE_QSTRING':
  440. if isinstance(v.type, QAPISchemaEnumType):
  441. for m in v.type.members:
  442. if m.name in ['on', 'off']:
  443. conflicting.add('QTYPE_QBOOL')
  444. if re.match(r'[-+0-9.]', m.name):
  445. # lazy, could be tightened
  446. conflicting.add('QTYPE_QNUM')
  447. else:
  448. conflicting.add('QTYPE_QNUM')
  449. conflicting.add('QTYPE_QBOOL')
  450. for qt in conflicting:
  451. if qt in types_seen:
  452. raise QAPISemError(
  453. self.info,
  454. "%s can't be distinguished from '%s'"
  455. % (v.describe(self.info), types_seen[qt]))
  456. types_seen[qt] = v.name
  457. def connect_doc(self, doc=None):
  458. super().connect_doc(doc)
  459. doc = doc or self.doc
  460. for v in self.variants.variants:
  461. v.connect_doc(doc)
  462. def c_type(self):
  463. return c_name(self.name) + POINTER_SUFFIX
  464. def json_type(self):
  465. return 'value'
  466. def visit(self, visitor):
  467. super().visit(visitor)
  468. visitor.visit_alternate_type(
  469. self.name, self.info, self.ifcond, self.features, self.variants)
  470. class QAPISchemaVariants:
  471. def __init__(self, tag_name, info, tag_member, variants):
  472. # Unions pass tag_name but not tag_member.
  473. # Alternates pass tag_member but not tag_name.
  474. # After check(), tag_member is always set.
  475. assert bool(tag_member) != bool(tag_name)
  476. assert (isinstance(tag_name, str) or
  477. isinstance(tag_member, QAPISchemaObjectTypeMember))
  478. for v in variants:
  479. assert isinstance(v, QAPISchemaVariant)
  480. self._tag_name = tag_name
  481. self.info = info
  482. self.tag_member = tag_member
  483. self.variants = variants
  484. def set_defined_in(self, name):
  485. for v in self.variants:
  486. v.set_defined_in(name)
  487. def check(self, schema, seen):
  488. if self._tag_name: # union
  489. self.tag_member = seen.get(c_name(self._tag_name))
  490. base = "'base'"
  491. # Pointing to the base type when not implicit would be
  492. # nice, but we don't know it here
  493. if not self.tag_member or self._tag_name != self.tag_member.name:
  494. raise QAPISemError(
  495. self.info,
  496. "discriminator '%s' is not a member of %s"
  497. % (self._tag_name, base))
  498. # Here we do:
  499. base_type = schema.lookup_type(self.tag_member.defined_in)
  500. assert base_type
  501. if not base_type.is_implicit():
  502. base = "base type '%s'" % self.tag_member.defined_in
  503. if not isinstance(self.tag_member.type, QAPISchemaEnumType):
  504. raise QAPISemError(
  505. self.info,
  506. "discriminator member '%s' of %s must be of enum type"
  507. % (self._tag_name, base))
  508. if self.tag_member.optional:
  509. raise QAPISemError(
  510. self.info,
  511. "discriminator member '%s' of %s must not be optional"
  512. % (self._tag_name, base))
  513. if self.tag_member.ifcond.is_present():
  514. raise QAPISemError(
  515. self.info,
  516. "discriminator member '%s' of %s must not be conditional"
  517. % (self._tag_name, base))
  518. else: # alternate
  519. assert isinstance(self.tag_member.type, QAPISchemaEnumType)
  520. assert not self.tag_member.optional
  521. assert not self.tag_member.ifcond.is_present()
  522. if self._tag_name: # union
  523. # branches that are not explicitly covered get an empty type
  524. cases = {v.name for v in self.variants}
  525. for m in self.tag_member.type.members:
  526. if m.name not in cases:
  527. v = QAPISchemaVariant(m.name, self.info,
  528. 'q_empty', m.ifcond)
  529. v.set_defined_in(self.tag_member.defined_in)
  530. self.variants.append(v)
  531. if not self.variants:
  532. raise QAPISemError(self.info, "union has no branches")
  533. for v in self.variants:
  534. v.check(schema)
  535. # Union names must match enum values; alternate names are
  536. # checked separately. Use 'seen' to tell the two apart.
  537. if seen:
  538. if v.name not in self.tag_member.type.member_names():
  539. raise QAPISemError(
  540. self.info,
  541. "branch '%s' is not a value of %s"
  542. % (v.name, self.tag_member.type.describe()))
  543. if not isinstance(v.type, QAPISchemaObjectType):
  544. raise QAPISemError(
  545. self.info,
  546. "%s cannot use %s"
  547. % (v.describe(self.info), v.type.describe()))
  548. v.type.check(schema)
  549. def check_clash(self, info, seen):
  550. for v in self.variants:
  551. # Reset seen map for each variant, since qapi names from one
  552. # branch do not affect another branch
  553. v.type.check_clash(info, dict(seen))
  554. class QAPISchemaMember:
  555. """ Represents object members, enum members and features """
  556. role = 'member'
  557. def __init__(self, name, info, ifcond=None):
  558. assert isinstance(name, str)
  559. self.name = name
  560. self.info = info
  561. self.ifcond = ifcond or QAPISchemaIfCond()
  562. self.defined_in = None
  563. def set_defined_in(self, name):
  564. assert not self.defined_in
  565. self.defined_in = name
  566. def check_clash(self, info, seen):
  567. cname = c_name(self.name)
  568. if cname in seen:
  569. raise QAPISemError(
  570. info,
  571. "%s collides with %s"
  572. % (self.describe(info), seen[cname].describe(info)))
  573. seen[cname] = self
  574. def connect_doc(self, doc):
  575. if doc:
  576. doc.connect_member(self)
  577. def describe(self, info):
  578. role = self.role
  579. meta = 'type'
  580. defined_in = self.defined_in
  581. assert defined_in
  582. if defined_in.startswith('q_obj_'):
  583. # See QAPISchema._make_implicit_object_type() - reverse the
  584. # mapping there to create a nice human-readable description
  585. defined_in = defined_in[6:]
  586. if defined_in.endswith('-arg'):
  587. # Implicit type created for a command's dict 'data'
  588. assert role == 'member'
  589. role = 'parameter'
  590. meta = 'command'
  591. defined_in = defined_in[:-4]
  592. elif defined_in.endswith('-base'):
  593. # Implicit type created for a union's dict 'base'
  594. role = 'base ' + role
  595. defined_in = defined_in[:-5]
  596. else:
  597. assert False
  598. if defined_in != info.defn_name:
  599. return "%s '%s' of %s '%s'" % (role, self.name, meta, defined_in)
  600. return "%s '%s'" % (role, self.name)
  601. class QAPISchemaEnumMember(QAPISchemaMember):
  602. role = 'value'
  603. def __init__(self, name, info, ifcond=None, features=None):
  604. super().__init__(name, info, ifcond)
  605. for f in features or []:
  606. assert isinstance(f, QAPISchemaFeature)
  607. f.set_defined_in(name)
  608. self.features = features or []
  609. def connect_doc(self, doc):
  610. super().connect_doc(doc)
  611. if doc:
  612. for f in self.features:
  613. doc.connect_feature(f)
  614. class QAPISchemaFeature(QAPISchemaMember):
  615. role = 'feature'
  616. def is_special(self):
  617. return self.name in ('deprecated', 'unstable')
  618. class QAPISchemaObjectTypeMember(QAPISchemaMember):
  619. def __init__(self, name, info, typ, optional, ifcond=None, features=None):
  620. super().__init__(name, info, ifcond)
  621. assert isinstance(typ, str)
  622. assert isinstance(optional, bool)
  623. for f in features or []:
  624. assert isinstance(f, QAPISchemaFeature)
  625. f.set_defined_in(name)
  626. self._type_name = typ
  627. self.type = None
  628. self.optional = optional
  629. self.features = features or []
  630. def need_has(self):
  631. assert self.type
  632. return self.optional and self.type.need_has_if_optional()
  633. def check(self, schema):
  634. assert self.defined_in
  635. self.type = schema.resolve_type(self._type_name, self.info,
  636. self.describe)
  637. seen = {}
  638. for f in self.features:
  639. f.check_clash(self.info, seen)
  640. def connect_doc(self, doc):
  641. super().connect_doc(doc)
  642. if doc:
  643. for f in self.features:
  644. doc.connect_feature(f)
  645. class QAPISchemaVariant(QAPISchemaObjectTypeMember):
  646. role = 'branch'
  647. def __init__(self, name, info, typ, ifcond=None):
  648. super().__init__(name, info, typ, False, ifcond)
  649. class QAPISchemaCommand(QAPISchemaEntity):
  650. meta = 'command'
  651. def __init__(self, name, info, doc, ifcond, features,
  652. arg_type, ret_type,
  653. gen, success_response, boxed, allow_oob, allow_preconfig,
  654. coroutine):
  655. super().__init__(name, info, doc, ifcond, features)
  656. assert not arg_type or isinstance(arg_type, str)
  657. assert not ret_type or isinstance(ret_type, str)
  658. self._arg_type_name = arg_type
  659. self.arg_type = None
  660. self._ret_type_name = ret_type
  661. self.ret_type = None
  662. self.gen = gen
  663. self.success_response = success_response
  664. self.boxed = boxed
  665. self.allow_oob = allow_oob
  666. self.allow_preconfig = allow_preconfig
  667. self.coroutine = coroutine
  668. def check(self, schema):
  669. super().check(schema)
  670. if self._arg_type_name:
  671. self.arg_type = schema.resolve_type(
  672. self._arg_type_name, self.info, "command's 'data'")
  673. if not isinstance(self.arg_type, QAPISchemaObjectType):
  674. raise QAPISemError(
  675. self.info,
  676. "command's 'data' cannot take %s"
  677. % self.arg_type.describe())
  678. if self.arg_type.variants and not self.boxed:
  679. raise QAPISemError(
  680. self.info,
  681. "command's 'data' can take %s only with 'boxed': true"
  682. % self.arg_type.describe())
  683. self.arg_type.check(schema)
  684. if self.arg_type.has_conditional_members() and not self.boxed:
  685. raise QAPISemError(
  686. self.info,
  687. "conditional command arguments require 'boxed': true")
  688. if self._ret_type_name:
  689. self.ret_type = schema.resolve_type(
  690. self._ret_type_name, self.info, "command's 'returns'")
  691. if self.name not in self.info.pragma.command_returns_exceptions:
  692. typ = self.ret_type
  693. if isinstance(typ, QAPISchemaArrayType):
  694. typ = self.ret_type.element_type
  695. assert typ
  696. if not isinstance(typ, QAPISchemaObjectType):
  697. raise QAPISemError(
  698. self.info,
  699. "command's 'returns' cannot take %s"
  700. % self.ret_type.describe())
  701. def connect_doc(self, doc=None):
  702. super().connect_doc(doc)
  703. doc = doc or self.doc
  704. if doc:
  705. if self.arg_type and self.arg_type.is_implicit():
  706. self.arg_type.connect_doc(doc)
  707. def visit(self, visitor):
  708. super().visit(visitor)
  709. visitor.visit_command(
  710. self.name, self.info, self.ifcond, self.features,
  711. self.arg_type, self.ret_type, self.gen, self.success_response,
  712. self.boxed, self.allow_oob, self.allow_preconfig,
  713. self.coroutine)
  714. class QAPISchemaEvent(QAPISchemaEntity):
  715. meta = 'event'
  716. def __init__(self, name, info, doc, ifcond, features, arg_type, boxed):
  717. super().__init__(name, info, doc, ifcond, features)
  718. assert not arg_type or isinstance(arg_type, str)
  719. self._arg_type_name = arg_type
  720. self.arg_type = None
  721. self.boxed = boxed
  722. def check(self, schema):
  723. super().check(schema)
  724. if self._arg_type_name:
  725. self.arg_type = schema.resolve_type(
  726. self._arg_type_name, self.info, "event's 'data'")
  727. if not isinstance(self.arg_type, QAPISchemaObjectType):
  728. raise QAPISemError(
  729. self.info,
  730. "event's 'data' cannot take %s"
  731. % self.arg_type.describe())
  732. if self.arg_type.variants and not self.boxed:
  733. raise QAPISemError(
  734. self.info,
  735. "event's 'data' can take %s only with 'boxed': true"
  736. % self.arg_type.describe())
  737. self.arg_type.check(schema)
  738. if self.arg_type.has_conditional_members() and not self.boxed:
  739. raise QAPISemError(
  740. self.info,
  741. "conditional event arguments require 'boxed': true")
  742. def connect_doc(self, doc=None):
  743. super().connect_doc(doc)
  744. doc = doc or self.doc
  745. if doc:
  746. if self.arg_type and self.arg_type.is_implicit():
  747. self.arg_type.connect_doc(doc)
  748. def visit(self, visitor):
  749. super().visit(visitor)
  750. visitor.visit_event(
  751. self.name, self.info, self.ifcond, self.features,
  752. self.arg_type, self.boxed)
  753. class QAPISchema:
  754. def __init__(self, fname):
  755. self.fname = fname
  756. try:
  757. parser = QAPISchemaParser(fname)
  758. except OSError as err:
  759. raise QAPIError(
  760. f"can't read schema file '{fname}': {err.strerror}"
  761. ) from err
  762. exprs = check_exprs(parser.exprs)
  763. self.docs = parser.docs
  764. self._entity_list = []
  765. self._entity_dict = {}
  766. self._module_dict = OrderedDict()
  767. self._schema_dir = os.path.dirname(fname)
  768. self._make_module(QAPISchemaModule.BUILTIN_MODULE_NAME)
  769. self._make_module(fname)
  770. self._predefining = True
  771. self._def_predefineds()
  772. self._predefining = False
  773. self._def_exprs(exprs)
  774. self.check()
  775. def _def_entity(self, ent):
  776. # Only the predefined types are allowed to not have info
  777. assert ent.info or self._predefining
  778. self._entity_list.append(ent)
  779. if ent.name is None:
  780. return
  781. # TODO reject names that differ only in '_' vs. '.' vs. '-',
  782. # because they're liable to clash in generated C.
  783. other_ent = self._entity_dict.get(ent.name)
  784. if other_ent:
  785. if other_ent.info:
  786. where = QAPISourceError(other_ent.info, "previous definition")
  787. raise QAPISemError(
  788. ent.info,
  789. "'%s' is already defined\n%s" % (ent.name, where))
  790. raise QAPISemError(
  791. ent.info, "%s is already defined" % other_ent.describe())
  792. self._entity_dict[ent.name] = ent
  793. def lookup_entity(self, name, typ=None):
  794. ent = self._entity_dict.get(name)
  795. if typ and not isinstance(ent, typ):
  796. return None
  797. return ent
  798. def lookup_type(self, name):
  799. return self.lookup_entity(name, QAPISchemaType)
  800. def resolve_type(self, name, info, what):
  801. typ = self.lookup_type(name)
  802. if not typ:
  803. if callable(what):
  804. what = what(info)
  805. raise QAPISemError(
  806. info, "%s uses unknown type '%s'" % (what, name))
  807. return typ
  808. def _module_name(self, fname: str) -> str:
  809. if QAPISchemaModule.is_system_module(fname):
  810. return fname
  811. return os.path.relpath(fname, self._schema_dir)
  812. def _make_module(self, fname):
  813. name = self._module_name(fname)
  814. if name not in self._module_dict:
  815. self._module_dict[name] = QAPISchemaModule(name)
  816. return self._module_dict[name]
  817. def module_by_fname(self, fname):
  818. name = self._module_name(fname)
  819. return self._module_dict[name]
  820. def _def_include(self, expr: QAPIExpression):
  821. include = expr['include']
  822. assert expr.doc is None
  823. self._def_entity(
  824. QAPISchemaInclude(self._make_module(include), expr.info))
  825. def _def_builtin_type(self, name, json_type, c_type):
  826. self._def_entity(QAPISchemaBuiltinType(name, json_type, c_type))
  827. # Instantiating only the arrays that are actually used would
  828. # be nice, but we can't as long as their generated code
  829. # (qapi-builtin-types.[ch]) may be shared by some other
  830. # schema.
  831. self._make_array_type(name, None)
  832. def _def_predefineds(self):
  833. for t in [('str', 'string', 'char' + POINTER_SUFFIX),
  834. ('number', 'number', 'double'),
  835. ('int', 'int', 'int64_t'),
  836. ('int8', 'int', 'int8_t'),
  837. ('int16', 'int', 'int16_t'),
  838. ('int32', 'int', 'int32_t'),
  839. ('int64', 'int', 'int64_t'),
  840. ('uint8', 'int', 'uint8_t'),
  841. ('uint16', 'int', 'uint16_t'),
  842. ('uint32', 'int', 'uint32_t'),
  843. ('uint64', 'int', 'uint64_t'),
  844. ('size', 'int', 'uint64_t'),
  845. ('bool', 'boolean', 'bool'),
  846. ('any', 'value', 'QObject' + POINTER_SUFFIX),
  847. ('null', 'null', 'QNull' + POINTER_SUFFIX)]:
  848. self._def_builtin_type(*t)
  849. self.the_empty_object_type = QAPISchemaObjectType(
  850. 'q_empty', None, None, None, None, None, [], None)
  851. self._def_entity(self.the_empty_object_type)
  852. qtypes = ['none', 'qnull', 'qnum', 'qstring', 'qdict', 'qlist',
  853. 'qbool']
  854. qtype_values = self._make_enum_members(
  855. [{'name': n} for n in qtypes], None)
  856. self._def_entity(QAPISchemaEnumType('QType', None, None, None, None,
  857. qtype_values, 'QTYPE'))
  858. def _make_features(self, features, info):
  859. if features is None:
  860. return []
  861. return [QAPISchemaFeature(f['name'], info,
  862. QAPISchemaIfCond(f.get('if')))
  863. for f in features]
  864. def _make_enum_member(self, name, ifcond, features, info):
  865. return QAPISchemaEnumMember(name, info,
  866. QAPISchemaIfCond(ifcond),
  867. self._make_features(features, info))
  868. def _make_enum_members(self, values, info):
  869. return [self._make_enum_member(v['name'], v.get('if'),
  870. v.get('features'), info)
  871. for v in values]
  872. def _make_array_type(self, element_type, info):
  873. name = element_type + 'List' # reserved by check_defn_name_str()
  874. if not self.lookup_type(name):
  875. self._def_entity(QAPISchemaArrayType(name, info, element_type))
  876. return name
  877. def _make_implicit_object_type(self, name, info, ifcond, role, members):
  878. if not members:
  879. return None
  880. # See also QAPISchemaObjectTypeMember.describe()
  881. name = 'q_obj_%s-%s' % (name, role)
  882. typ = self.lookup_entity(name, QAPISchemaObjectType)
  883. if typ:
  884. # The implicit object type has multiple users. This can
  885. # only be a duplicate definition, which will be flagged
  886. # later.
  887. pass
  888. else:
  889. self._def_entity(QAPISchemaObjectType(
  890. name, info, None, ifcond, None, None, members, None))
  891. return name
  892. def _def_enum_type(self, expr: QAPIExpression):
  893. name = expr['enum']
  894. data = expr['data']
  895. prefix = expr.get('prefix')
  896. ifcond = QAPISchemaIfCond(expr.get('if'))
  897. info = expr.info
  898. features = self._make_features(expr.get('features'), info)
  899. self._def_entity(QAPISchemaEnumType(
  900. name, info, expr.doc, ifcond, features,
  901. self._make_enum_members(data, info), prefix))
  902. def _make_member(self, name, typ, ifcond, features, info):
  903. optional = False
  904. if name.startswith('*'):
  905. name = name[1:]
  906. optional = True
  907. if isinstance(typ, list):
  908. assert len(typ) == 1
  909. typ = self._make_array_type(typ[0], info)
  910. return QAPISchemaObjectTypeMember(name, info, typ, optional, ifcond,
  911. self._make_features(features, info))
  912. def _make_members(self, data, info):
  913. return [self._make_member(key, value['type'],
  914. QAPISchemaIfCond(value.get('if')),
  915. value.get('features'), info)
  916. for (key, value) in data.items()]
  917. def _def_struct_type(self, expr: QAPIExpression):
  918. name = expr['struct']
  919. base = expr.get('base')
  920. data = expr['data']
  921. info = expr.info
  922. ifcond = QAPISchemaIfCond(expr.get('if'))
  923. features = self._make_features(expr.get('features'), info)
  924. self._def_entity(QAPISchemaObjectType(
  925. name, info, expr.doc, ifcond, features, base,
  926. self._make_members(data, info),
  927. None))
  928. def _make_variant(self, case, typ, ifcond, info):
  929. if isinstance(typ, list):
  930. assert len(typ) == 1
  931. typ = self._make_array_type(typ[0], info)
  932. return QAPISchemaVariant(case, info, typ, ifcond)
  933. def _def_union_type(self, expr: QAPIExpression):
  934. name = expr['union']
  935. base = expr['base']
  936. tag_name = expr['discriminator']
  937. data = expr['data']
  938. assert isinstance(data, dict)
  939. info = expr.info
  940. ifcond = QAPISchemaIfCond(expr.get('if'))
  941. features = self._make_features(expr.get('features'), info)
  942. if isinstance(base, dict):
  943. base = self._make_implicit_object_type(
  944. name, info, ifcond,
  945. 'base', self._make_members(base, info))
  946. variants = [
  947. self._make_variant(key, value['type'],
  948. QAPISchemaIfCond(value.get('if')),
  949. info)
  950. for (key, value) in data.items()]
  951. members: List[QAPISchemaObjectTypeMember] = []
  952. self._def_entity(
  953. QAPISchemaObjectType(name, info, expr.doc, ifcond, features,
  954. base, members,
  955. QAPISchemaVariants(
  956. tag_name, info, None, variants)))
  957. def _def_alternate_type(self, expr: QAPIExpression):
  958. name = expr['alternate']
  959. data = expr['data']
  960. assert isinstance(data, dict)
  961. ifcond = QAPISchemaIfCond(expr.get('if'))
  962. info = expr.info
  963. features = self._make_features(expr.get('features'), info)
  964. variants = [
  965. self._make_variant(key, value['type'],
  966. QAPISchemaIfCond(value.get('if')),
  967. info)
  968. for (key, value) in data.items()]
  969. tag_member = QAPISchemaObjectTypeMember('type', info, 'QType', False)
  970. self._def_entity(
  971. QAPISchemaAlternateType(
  972. name, info, expr.doc, ifcond, features,
  973. QAPISchemaVariants(None, info, tag_member, variants)))
  974. def _def_command(self, expr: QAPIExpression):
  975. name = expr['command']
  976. data = expr.get('data')
  977. rets = expr.get('returns')
  978. gen = expr.get('gen', True)
  979. success_response = expr.get('success-response', True)
  980. boxed = expr.get('boxed', False)
  981. allow_oob = expr.get('allow-oob', False)
  982. allow_preconfig = expr.get('allow-preconfig', False)
  983. coroutine = expr.get('coroutine', False)
  984. ifcond = QAPISchemaIfCond(expr.get('if'))
  985. info = expr.info
  986. features = self._make_features(expr.get('features'), info)
  987. if isinstance(data, OrderedDict):
  988. data = self._make_implicit_object_type(
  989. name, info, ifcond,
  990. 'arg', self._make_members(data, info))
  991. if isinstance(rets, list):
  992. assert len(rets) == 1
  993. rets = self._make_array_type(rets[0], info)
  994. self._def_entity(QAPISchemaCommand(name, info, expr.doc, ifcond,
  995. features, data, rets,
  996. gen, success_response,
  997. boxed, allow_oob, allow_preconfig,
  998. coroutine))
  999. def _def_event(self, expr: QAPIExpression):
  1000. name = expr['event']
  1001. data = expr.get('data')
  1002. boxed = expr.get('boxed', False)
  1003. ifcond = QAPISchemaIfCond(expr.get('if'))
  1004. info = expr.info
  1005. features = self._make_features(expr.get('features'), info)
  1006. if isinstance(data, OrderedDict):
  1007. data = self._make_implicit_object_type(
  1008. name, info, ifcond,
  1009. 'arg', self._make_members(data, info))
  1010. self._def_entity(QAPISchemaEvent(name, info, expr.doc, ifcond,
  1011. features, data, boxed))
  1012. def _def_exprs(self, exprs):
  1013. for expr in exprs:
  1014. if 'enum' in expr:
  1015. self._def_enum_type(expr)
  1016. elif 'struct' in expr:
  1017. self._def_struct_type(expr)
  1018. elif 'union' in expr:
  1019. self._def_union_type(expr)
  1020. elif 'alternate' in expr:
  1021. self._def_alternate_type(expr)
  1022. elif 'command' in expr:
  1023. self._def_command(expr)
  1024. elif 'event' in expr:
  1025. self._def_event(expr)
  1026. elif 'include' in expr:
  1027. self._def_include(expr)
  1028. else:
  1029. assert False
  1030. def check(self):
  1031. for ent in self._entity_list:
  1032. ent.check(self)
  1033. ent.connect_doc()
  1034. for ent in self._entity_list:
  1035. ent.set_module(self)
  1036. for doc in self.docs:
  1037. doc.check()
  1038. def visit(self, visitor):
  1039. visitor.visit_begin(self)
  1040. for mod in self._module_dict.values():
  1041. mod.visit(visitor)
  1042. visitor.visit_end()