girwriter.py 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611
  1. # -*- Mode: Python -*-
  2. # GObject-Introspection - a framework for introspecting GObject libraries
  3. # Copyright (C) 2008 Johan Dahlin
  4. # Copyright (C) 2008, 2009 Red Hat, Inc.
  5. #
  6. # This library is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU Lesser General Public
  8. # License as published by the Free Software Foundation; either
  9. # version 2 of the License, or (at your option) any later version.
  10. #
  11. # This library is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. # Lesser General Public License for more details.
  15. #
  16. # You should have received a copy of the GNU Lesser General Public
  17. # License along with this library; if not, write to the
  18. # Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  19. # Boston, MA 02111-1307, USA.
  20. #
  21. from __future__ import with_statement
  22. from __future__ import absolute_import
  23. from __future__ import division
  24. from __future__ import print_function
  25. from __future__ import unicode_literals
  26. from . import ast
  27. from .xmlwriter import XMLWriter
  28. # Bump this for *incompatible* changes to the .gir.
  29. # Compatible changes we just make inline
  30. COMPATIBLE_GIR_VERSION = '1.2'
  31. class GIRWriter(XMLWriter):
  32. def __init__(self, namespace):
  33. super(GIRWriter, self).__init__()
  34. self.write_comment(
  35. 'This file was automatically generated from C sources - DO NOT EDIT!\n'
  36. 'To affect the contents of this file, edit the original C definitions,\n'
  37. 'and/or use gtk-doc annotations. ')
  38. self._write_repository(namespace)
  39. def _write_repository(self, namespace):
  40. attrs = [
  41. ('version', COMPATIBLE_GIR_VERSION),
  42. ('xmlns', 'http://www.gtk.org/introspection/core/1.0'),
  43. ('xmlns:c', 'http://www.gtk.org/introspection/c/1.0'),
  44. ('xmlns:glib', 'http://www.gtk.org/introspection/glib/1.0')]
  45. with self.tagcontext('repository', attrs):
  46. for include in sorted(namespace.includes):
  47. self._write_include(include)
  48. for pkg in sorted(set(namespace.exported_packages)):
  49. self._write_pkgconfig_pkg(pkg)
  50. for c_include in sorted(set(namespace.c_includes)):
  51. self._write_c_include(c_include)
  52. self._namespace = namespace
  53. self._write_namespace(namespace)
  54. self._namespace = None
  55. def _write_include(self, include):
  56. attrs = [('name', include.name), ('version', include.version)]
  57. self.write_tag('include', attrs)
  58. def _write_pkgconfig_pkg(self, package):
  59. attrs = [('name', package)]
  60. self.write_tag('package', attrs)
  61. def _write_c_include(self, c_include):
  62. attrs = [('name', c_include)]
  63. self.write_tag('c:include', attrs)
  64. def _write_namespace(self, namespace):
  65. attrs = [('name', namespace.name),
  66. ('version', namespace.version),
  67. ('shared-library', ','.join(namespace.shared_libraries)),
  68. ('c:identifier-prefixes', ','.join(namespace.identifier_prefixes)),
  69. ('c:symbol-prefixes', ','.join(namespace.symbol_prefixes))]
  70. with self.tagcontext('namespace', attrs):
  71. # We define a custom sorting function here because
  72. # we want aliases to be first. They're a bit
  73. # special because the typelib compiler expands them.
  74. def nscmp(val):
  75. if isinstance(val, ast.Alias):
  76. return 0, val
  77. else:
  78. return 1, val
  79. for node in sorted(namespace.values(), key=nscmp):
  80. self._write_node(node)
  81. def _write_node(self, node):
  82. if isinstance(node, ast.Function):
  83. self._write_function(node)
  84. elif isinstance(node, ast.Enum):
  85. self._write_enum(node)
  86. elif isinstance(node, ast.Bitfield):
  87. self._write_bitfield(node)
  88. elif isinstance(node, (ast.Class, ast.Interface)):
  89. self._write_class(node)
  90. elif isinstance(node, ast.Callback):
  91. self._write_callback(node)
  92. elif isinstance(node, ast.Record):
  93. self._write_record(node)
  94. elif isinstance(node, ast.Union):
  95. self._write_union(node)
  96. elif isinstance(node, ast.Boxed):
  97. self._write_boxed(node)
  98. elif isinstance(node, ast.Member):
  99. # FIXME: atk_misc_instance singleton
  100. pass
  101. elif isinstance(node, ast.Alias):
  102. self._write_alias(node)
  103. elif isinstance(node, ast.Constant):
  104. self._write_constant(node)
  105. else:
  106. print('WRITER: Unhandled node', node)
  107. def _append_version(self, node, attrs):
  108. if node.version:
  109. attrs.append(('version', node.version))
  110. def _write_generic(self, node):
  111. for key, value in node.attributes.items():
  112. self.write_tag('attribute', [('name', key), ('value', value)])
  113. if hasattr(node, 'doc') and node.doc:
  114. self.write_tag('doc', [('xml:space', 'preserve')],
  115. node.doc)
  116. if hasattr(node, 'version_doc') and node.version_doc:
  117. self.write_tag('doc-version', [('xml:space', 'preserve')],
  118. node.version_doc)
  119. if hasattr(node, 'deprecated_doc') and node.deprecated_doc:
  120. self.write_tag('doc-deprecated', [('xml:space', 'preserve')],
  121. node.deprecated_doc)
  122. if hasattr(node, 'stability_doc') and node.stability_doc:
  123. self.write_tag('doc-stability', [('xml:space', 'preserve')],
  124. node.stability_doc)
  125. def _append_node_generic(self, node, attrs):
  126. if node.skip or not node.introspectable:
  127. attrs.append(('introspectable', '0'))
  128. if node.deprecated or node.deprecated_doc:
  129. # The deprecated attribute used to contain node.deprecated_doc as an attribute. As
  130. # an xml attribute cannot preserve whitespace, deprecated_doc has been moved into
  131. # it's own tag, written in _write_generic() above. We continue to write the deprecated
  132. # attribute for backwards compatibility
  133. attrs.append(('deprecated', '1'))
  134. if node.deprecated:
  135. attrs.append(('deprecated-version', node.deprecated))
  136. if node.stability:
  137. attrs.append(('stability', node.stability))
  138. def _append_throws(self, func, attrs):
  139. if func.throws:
  140. attrs.append(('throws', '1'))
  141. def _write_alias(self, alias):
  142. attrs = [('name', alias.name)]
  143. if alias.ctype is not None:
  144. attrs.append(('c:type', alias.ctype))
  145. self._append_node_generic(alias, attrs)
  146. with self.tagcontext('alias', attrs):
  147. self._write_generic(alias)
  148. self._write_type_ref(alias.target)
  149. def _write_callable(self, callable, tag_name, extra_attrs):
  150. attrs = [('name', callable.name)]
  151. attrs.extend(extra_attrs)
  152. self._append_version(callable, attrs)
  153. self._append_node_generic(callable, attrs)
  154. self._append_throws(callable, attrs)
  155. with self.tagcontext(tag_name, attrs):
  156. self._write_generic(callable)
  157. self._write_return_type(callable.retval, parent=callable)
  158. self._write_parameters(callable)
  159. def _write_function(self, func, tag_name='function'):
  160. if func.internal_skipped:
  161. return
  162. attrs = []
  163. if hasattr(func, 'symbol'):
  164. attrs.append(('c:identifier', func.symbol))
  165. if func.shadowed_by:
  166. attrs.append(('shadowed-by', func.shadowed_by))
  167. elif func.shadows:
  168. attrs.append(('shadows', func.shadows))
  169. if func.moved_to is not None:
  170. attrs.append(('moved-to', func.moved_to))
  171. self._write_callable(func, tag_name, attrs)
  172. def _write_method(self, method):
  173. self._write_function(method, tag_name='method')
  174. def _write_static_method(self, method):
  175. self._write_function(method, tag_name='function')
  176. def _write_constructor(self, method):
  177. self._write_function(method, tag_name='constructor')
  178. def _write_return_type(self, return_, parent=None):
  179. if not return_:
  180. return
  181. attrs = []
  182. if return_.transfer:
  183. attrs.append(('transfer-ownership', return_.transfer))
  184. if return_.skip:
  185. attrs.append(('skip', '1'))
  186. if return_.nullable and not return_.not_nullable:
  187. attrs.append(('nullable', '1'))
  188. with self.tagcontext('return-value', attrs):
  189. self._write_generic(return_)
  190. self._write_type(return_.type, parent=parent)
  191. def _write_parameters(self, callable):
  192. if not callable.parameters and callable.instance_parameter is None:
  193. return
  194. with self.tagcontext('parameters'):
  195. if callable.instance_parameter:
  196. self._write_parameter(callable, callable.instance_parameter, 'instance-parameter')
  197. for parameter in callable.parameters:
  198. self._write_parameter(callable, parameter)
  199. def _write_parameter(self, parent, parameter, nodename='parameter'):
  200. attrs = []
  201. if parameter.argname is not None:
  202. attrs.append(('name', parameter.argname))
  203. if (parameter.direction is not None) and (parameter.direction != 'in'):
  204. attrs.append(('direction', parameter.direction))
  205. attrs.append(('caller-allocates',
  206. '1' if parameter.caller_allocates else '0'))
  207. if parameter.transfer:
  208. attrs.append(('transfer-ownership',
  209. parameter.transfer))
  210. if parameter.nullable and not parameter.not_nullable:
  211. attrs.append(('nullable', '1'))
  212. if parameter.direction != ast.PARAM_DIRECTION_OUT:
  213. attrs.append(('allow-none', '1'))
  214. if parameter.optional:
  215. attrs.append(('optional', '1'))
  216. if parameter.direction == ast.PARAM_DIRECTION_OUT:
  217. attrs.append(('allow-none', '1'))
  218. if parameter.scope:
  219. attrs.append(('scope', parameter.scope))
  220. if parameter.closure_name is not None:
  221. idx = parent.get_parameter_index(parameter.closure_name)
  222. attrs.append(('closure', '%d' % (idx, )))
  223. if parameter.destroy_name is not None:
  224. idx = parent.get_parameter_index(parameter.destroy_name)
  225. attrs.append(('destroy', '%d' % (idx, )))
  226. if parameter.skip:
  227. attrs.append(('skip', '1'))
  228. with self.tagcontext(nodename, attrs):
  229. self._write_generic(parameter)
  230. self._write_type(parameter.type, parent=parent)
  231. def _type_to_name(self, typeval):
  232. if not typeval.resolved:
  233. raise AssertionError("Caught unresolved type %r (ctype=%r)" % (typeval, typeval.ctype))
  234. assert typeval.target_giname is not None
  235. prefix = self._namespace.name + '.'
  236. if typeval.target_giname.startswith(prefix):
  237. return typeval.target_giname[len(prefix):]
  238. return typeval.target_giname
  239. def _write_type_ref(self, ntype):
  240. """ Like _write_type, but only writes the type name rather than the full details """
  241. assert isinstance(ntype, ast.Type), ntype
  242. attrs = []
  243. if ntype.ctype:
  244. attrs.append(('c:type', ntype.ctype))
  245. if isinstance(ntype, ast.Array):
  246. if ntype.array_type != ast.Array.C:
  247. attrs.insert(0, ('name', ntype.array_type))
  248. elif isinstance(ntype, ast.List):
  249. if ntype.name:
  250. attrs.insert(0, ('name', ntype.name))
  251. elif isinstance(ntype, ast.Map):
  252. attrs.insert(0, ('name', 'GLib.HashTable'))
  253. else:
  254. if ntype.target_giname:
  255. attrs.insert(0, ('name', self._type_to_name(ntype)))
  256. elif ntype.target_fundamental:
  257. attrs.insert(0, ('name', ntype.target_fundamental))
  258. self.write_tag('type', attrs)
  259. def _write_type(self, ntype, relation=None, parent=None):
  260. assert isinstance(ntype, ast.Type), ntype
  261. attrs = []
  262. if ntype.complete_ctype:
  263. attrs.append(('c:type', ntype.complete_ctype))
  264. elif ntype.ctype:
  265. attrs.append(('c:type', ntype.ctype))
  266. if isinstance(ntype, ast.Varargs):
  267. self.write_tag('varargs', [])
  268. elif isinstance(ntype, ast.Array):
  269. if ntype.array_type != ast.Array.C:
  270. attrs.insert(0, ('name', ntype.array_type))
  271. # we insert an explicit 'zero-terminated' attribute
  272. # when it is false, or when it would not be implied
  273. # by the absence of length and fixed-size
  274. if not ntype.zeroterminated:
  275. attrs.insert(0, ('zero-terminated', '0'))
  276. elif (ntype.zeroterminated
  277. and (ntype.size is not None or ntype.length_param_name is not None)):
  278. attrs.insert(0, ('zero-terminated', '1'))
  279. if ntype.size is not None:
  280. attrs.append(('fixed-size', '%d' % (ntype.size, )))
  281. if ntype.length_param_name is not None:
  282. if isinstance(parent, ast.Callable):
  283. length = parent.get_parameter_index(ntype.length_param_name)
  284. elif isinstance(parent, ast.Compound):
  285. length = parent.get_field_index(ntype.length_param_name)
  286. else:
  287. assert False, "parent not a callable or compound: %r" % parent
  288. attrs.insert(0, ('length', '%d' % (length, )))
  289. with self.tagcontext('array', attrs):
  290. self._write_type(ntype.element_type)
  291. elif isinstance(ntype, ast.List):
  292. if ntype.name:
  293. attrs.insert(0, ('name', ntype.name))
  294. with self.tagcontext('type', attrs):
  295. self._write_type(ntype.element_type)
  296. elif isinstance(ntype, ast.Map):
  297. attrs.insert(0, ('name', 'GLib.HashTable'))
  298. with self.tagcontext('type', attrs):
  299. self._write_type(ntype.key_type)
  300. self._write_type(ntype.value_type)
  301. else:
  302. # REWRITEFIXME - enable this for 1.2
  303. if ntype.target_giname:
  304. attrs.insert(0, ('name', self._type_to_name(ntype)))
  305. elif ntype.target_fundamental:
  306. # attrs = [('fundamental', ntype.target_fundamental)]
  307. attrs.insert(0, ('name', ntype.target_fundamental))
  308. elif ntype.target_foreign:
  309. attrs.insert(0, ('foreign', '1'))
  310. self.write_tag('type', attrs)
  311. def _append_registered(self, node, attrs):
  312. assert isinstance(node, ast.Registered)
  313. if node.get_type:
  314. attrs.extend([('glib:type-name', node.gtype_name),
  315. ('glib:get-type', node.get_type)])
  316. def _write_enum(self, enum):
  317. attrs = [('name', enum.name)]
  318. self._append_version(enum, attrs)
  319. self._append_node_generic(enum, attrs)
  320. self._append_registered(enum, attrs)
  321. attrs.append(('c:type', enum.ctype))
  322. if enum.error_domain:
  323. attrs.append(('glib:error-domain', enum.error_domain))
  324. with self.tagcontext('enumeration', attrs):
  325. self._write_generic(enum)
  326. for member in enum.members:
  327. self._write_member(member)
  328. for method in sorted(enum.static_methods):
  329. self._write_static_method(method)
  330. def _write_bitfield(self, bitfield):
  331. attrs = [('name', bitfield.name)]
  332. self._append_version(bitfield, attrs)
  333. self._append_node_generic(bitfield, attrs)
  334. self._append_registered(bitfield, attrs)
  335. attrs.append(('c:type', bitfield.ctype))
  336. with self.tagcontext('bitfield', attrs):
  337. self._write_generic(bitfield)
  338. for member in bitfield.members:
  339. self._write_member(member)
  340. for method in sorted(bitfield.static_methods):
  341. self._write_static_method(method)
  342. def _write_member(self, member):
  343. attrs = [('name', member.name),
  344. ('value', str(member.value)),
  345. ('c:identifier', member.symbol)]
  346. if member.nick is not None:
  347. attrs.append(('glib:nick', member.nick))
  348. with self.tagcontext('member', attrs):
  349. self._write_generic(member)
  350. def _write_constant(self, constant):
  351. attrs = [('name', constant.name),
  352. ('value', constant.value),
  353. ('c:type', constant.ctype)]
  354. self._append_version(constant, attrs)
  355. self._append_node_generic(constant, attrs)
  356. with self.tagcontext('constant', attrs):
  357. self._write_generic(constant)
  358. self._write_type(constant.value_type)
  359. def _write_class(self, node):
  360. attrs = [('name', node.name),
  361. ('c:symbol-prefix', node.c_symbol_prefix),
  362. ('c:type', node.ctype)]
  363. self._append_version(node, attrs)
  364. self._append_node_generic(node, attrs)
  365. if isinstance(node, ast.Class):
  366. tag_name = 'class'
  367. if node.parent_type is not None:
  368. attrs.append(('parent',
  369. self._type_to_name(node.parent_type)))
  370. if node.is_abstract:
  371. attrs.append(('abstract', '1'))
  372. else:
  373. assert isinstance(node, ast.Interface)
  374. tag_name = 'interface'
  375. attrs.append(('glib:type-name', node.gtype_name))
  376. if node.get_type is not None:
  377. attrs.append(('glib:get-type', node.get_type))
  378. if node.glib_type_struct is not None:
  379. attrs.append(('glib:type-struct',
  380. self._type_to_name(node.glib_type_struct)))
  381. if isinstance(node, ast.Class):
  382. if node.fundamental:
  383. attrs.append(('glib:fundamental', '1'))
  384. if node.ref_func:
  385. attrs.append(('glib:ref-func', node.ref_func))
  386. if node.unref_func:
  387. attrs.append(('glib:unref-func', node.unref_func))
  388. if node.set_value_func:
  389. attrs.append(('glib:set-value-func', node.set_value_func))
  390. if node.get_value_func:
  391. attrs.append(('glib:get-value-func', node.get_value_func))
  392. with self.tagcontext(tag_name, attrs):
  393. self._write_generic(node)
  394. if isinstance(node, ast.Class):
  395. for iface in sorted(node.interfaces):
  396. self.write_tag('implements',
  397. [('name', self._type_to_name(iface))])
  398. if isinstance(node, ast.Interface):
  399. for iface in sorted(node.prerequisites):
  400. self.write_tag('prerequisite',
  401. [('name', self._type_to_name(iface))])
  402. if isinstance(node, ast.Class):
  403. for method in sorted(node.constructors):
  404. self._write_constructor(method)
  405. for method in sorted(node.static_methods):
  406. self._write_static_method(method)
  407. for vfunc in sorted(node.virtual_methods):
  408. self._write_vfunc(vfunc)
  409. for method in sorted(node.methods):
  410. self._write_method(method)
  411. for prop in sorted(node.properties):
  412. self._write_property(prop)
  413. for field in node.fields:
  414. self._write_field(field, node)
  415. for signal in sorted(node.signals):
  416. self._write_signal(signal)
  417. def _write_boxed(self, boxed):
  418. attrs = [('glib:name', boxed.name)]
  419. if boxed.c_symbol_prefix is not None:
  420. attrs.append(('c:symbol-prefix', boxed.c_symbol_prefix))
  421. self._append_registered(boxed, attrs)
  422. with self.tagcontext('glib:boxed', attrs):
  423. self._write_generic(boxed)
  424. for method in sorted(boxed.constructors):
  425. self._write_constructor(method)
  426. for method in sorted(boxed.methods):
  427. self._write_method(method)
  428. for method in sorted(boxed.static_methods):
  429. self._write_static_method(method)
  430. def _write_property(self, prop):
  431. attrs = [('name', prop.name)]
  432. self._append_version(prop, attrs)
  433. self._append_node_generic(prop, attrs)
  434. # Properties are assumed to be readable (see also generate.c)
  435. if not prop.readable:
  436. attrs.append(('readable', '0'))
  437. if prop.writable:
  438. attrs.append(('writable', '1'))
  439. if prop.construct:
  440. attrs.append(('construct', '1'))
  441. if prop.construct_only:
  442. attrs.append(('construct-only', '1'))
  443. if prop.transfer:
  444. attrs.append(('transfer-ownership', prop.transfer))
  445. with self.tagcontext('property', attrs):
  446. self._write_generic(prop)
  447. self._write_type(prop.type)
  448. def _write_vfunc(self, vf):
  449. attrs = []
  450. if vf.invoker:
  451. attrs.append(('invoker', vf.invoker))
  452. self._write_callable(vf, 'virtual-method', attrs)
  453. def _write_callback(self, callback):
  454. attrs = []
  455. if callback.ctype != callback.name:
  456. attrs.append(('c:type', callback.ctype))
  457. self._write_callable(callback, 'callback', attrs)
  458. def _write_record(self, record, extra_attrs=[]):
  459. is_gtype_struct = False
  460. attrs = list(extra_attrs)
  461. if record.name is not None:
  462. attrs.append(('name', record.name))
  463. if record.ctype is not None: # the record might be anonymous
  464. attrs.append(('c:type', record.ctype))
  465. if record.disguised:
  466. attrs.append(('disguised', '1'))
  467. if record.foreign:
  468. attrs.append(('foreign', '1'))
  469. if record.is_gtype_struct_for is not None:
  470. is_gtype_struct = True
  471. attrs.append(('glib:is-gtype-struct-for',
  472. self._type_to_name(record.is_gtype_struct_for)))
  473. self._append_version(record, attrs)
  474. self._append_node_generic(record, attrs)
  475. self._append_registered(record, attrs)
  476. if record.c_symbol_prefix:
  477. attrs.append(('c:symbol-prefix', record.c_symbol_prefix))
  478. with self.tagcontext('record', attrs):
  479. self._write_generic(record)
  480. if record.fields:
  481. for field in record.fields:
  482. self._write_field(field, record, is_gtype_struct)
  483. for method in sorted(record.constructors):
  484. self._write_constructor(method)
  485. for method in sorted(record.methods):
  486. self._write_method(method)
  487. for method in sorted(record.static_methods):
  488. self._write_static_method(method)
  489. def _write_union(self, union):
  490. attrs = []
  491. if union.name is not None:
  492. attrs.append(('name', union.name))
  493. if union.ctype is not None: # the union might be anonymous
  494. attrs.append(('c:type', union.ctype))
  495. self._append_version(union, attrs)
  496. self._append_node_generic(union, attrs)
  497. self._append_registered(union, attrs)
  498. if union.c_symbol_prefix:
  499. attrs.append(('c:symbol-prefix', union.c_symbol_prefix))
  500. with self.tagcontext('union', attrs):
  501. self._write_generic(union)
  502. if union.fields:
  503. for field in union.fields:
  504. self._write_field(field, union)
  505. for method in sorted(union.constructors):
  506. self._write_constructor(method)
  507. for method in sorted(union.methods):
  508. self._write_method(method)
  509. for method in sorted(union.static_methods):
  510. self._write_static_method(method)
  511. def _write_field(self, field, parent, is_gtype_struct=False):
  512. if field.anonymous_node:
  513. if isinstance(field.anonymous_node, ast.Callback):
  514. attrs = [('name', field.name)]
  515. self._append_node_generic(field, attrs)
  516. with self.tagcontext('field', attrs):
  517. self._write_callback(field.anonymous_node)
  518. elif isinstance(field.anonymous_node, ast.Record):
  519. self._write_record(field.anonymous_node)
  520. elif isinstance(field.anonymous_node, ast.Union):
  521. self._write_union(field.anonymous_node)
  522. else:
  523. raise AssertionError("Unknown field anonymous: %r" % (field.anonymous_node, ))
  524. else:
  525. attrs = [('name', field.name)]
  526. self._append_node_generic(field, attrs)
  527. # Fields are assumed to be read-only
  528. # (see also girparser.c and generate.c)
  529. if not field.readable:
  530. attrs.append(('readable', '0'))
  531. if field.writable:
  532. attrs.append(('writable', '1'))
  533. if field.bits:
  534. attrs.append(('bits', str(field.bits)))
  535. if field.private:
  536. attrs.append(('private', '1'))
  537. with self.tagcontext('field', attrs):
  538. self._write_generic(field)
  539. self._write_type(field.type, parent=parent)
  540. def _write_signal(self, signal):
  541. attrs = [('name', signal.name)]
  542. if signal.when:
  543. attrs.append(('when', signal.when))
  544. if signal.no_recurse:
  545. attrs.append(('no-recurse', '1'))
  546. if signal.detailed:
  547. attrs.append(('detailed', '1'))
  548. if signal.action:
  549. attrs.append(('action', '1'))
  550. if signal.no_hooks:
  551. attrs.append(('no-hooks', '1'))
  552. self._append_version(signal, attrs)
  553. self._append_node_generic(signal, attrs)
  554. with self.tagcontext('glib:signal', attrs):
  555. self._write_generic(signal)
  556. self._write_return_type(signal.retval)
  557. self._write_parameters(signal)