transformer.py 42 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007
  1. # -*- Mode: Python -*-
  2. # GObject-Introspection - a framework for introspecting GObject libraries
  3. # Copyright (C) 2008 Johan Dahlin
  4. #
  5. # This library is free software; you can redistribute it and/or
  6. # modify it under the terms of the GNU Lesser General Public
  7. # License as published by the Free Software Foundation; either
  8. # version 2 of the License, or (at your option) any later version.
  9. #
  10. # This library is distributed in the hope that it will be useful,
  11. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. # Lesser General Public License for more details.
  14. #
  15. # You should have received a copy of the GNU Lesser General Public
  16. # License along with this library; if not, write to the
  17. # Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  18. # Boston, MA 02111-1307, USA.
  19. #
  20. from __future__ import absolute_import
  21. from __future__ import division
  22. from __future__ import print_function
  23. from __future__ import unicode_literals
  24. import os
  25. import sys
  26. import subprocess
  27. from . import ast
  28. from . import message
  29. from . import utils
  30. from .cachestore import CacheStore
  31. from .girparser import GIRParser
  32. from .sourcescanner import (
  33. SourceSymbol, ctype_name, CTYPE_POINTER,
  34. CTYPE_BASIC_TYPE, CTYPE_UNION, CTYPE_ARRAY, CTYPE_TYPEDEF,
  35. CTYPE_VOID, CTYPE_ENUM, CTYPE_FUNCTION, CTYPE_STRUCT,
  36. CSYMBOL_TYPE_FUNCTION, CSYMBOL_TYPE_TYPEDEF, CSYMBOL_TYPE_STRUCT,
  37. CSYMBOL_TYPE_ENUM, CSYMBOL_TYPE_UNION, CSYMBOL_TYPE_OBJECT,
  38. CSYMBOL_TYPE_MEMBER, CSYMBOL_TYPE_ELLIPSIS, CSYMBOL_TYPE_CONST,
  39. TYPE_QUALIFIER_CONST, TYPE_QUALIFIER_VOLATILE)
  40. class TransformerException(Exception):
  41. pass
  42. class Transformer(object):
  43. namespace = property(lambda self: self._namespace)
  44. def __init__(self, namespace, accept_unprefixed=False,
  45. identifier_filter_cmd='', symbol_filter_cmd=''):
  46. self._cachestore = CacheStore()
  47. self._accept_unprefixed = accept_unprefixed
  48. self._namespace = namespace
  49. self._pkg_config_packages = set()
  50. self._typedefs_ns = {}
  51. self._parsed_includes = {} # <string namespace -> Namespace>
  52. self._includepaths = []
  53. self._passthrough_mode = False
  54. self._identifier_filter_cmd = identifier_filter_cmd
  55. self._symbol_filter_cmd = symbol_filter_cmd
  56. # Cache a list of struct/unions in C's "tag namespace". This helps
  57. # manage various orderings of typedefs and structs. See:
  58. # https://bugzilla.gnome.org/show_bug.cgi?id=581525
  59. self._tag_ns = {}
  60. def get_pkgconfig_packages(self):
  61. return self._pkg_config_packages
  62. def disable_cache(self):
  63. self._cachestore = None
  64. def set_passthrough_mode(self):
  65. self._passthrough_mode = True
  66. def _append_new_node(self, node):
  67. original = self._namespace.get(node.name)
  68. # Special case constants here; we allow duplication to sort-of
  69. # handle #ifdef. But this introduces an arch-dependency in the .gir
  70. # file. So far this has only come up scanning glib - in theory, other
  71. # modules will just depend on that.
  72. if isinstance(original, ast.Constant) and isinstance(node, ast.Constant):
  73. pass
  74. elif original is node:
  75. # Ignore attempts to add the same node to the namespace. This can
  76. # happen when parsing typedefs and structs in particular orderings:
  77. # typedef struct _Foo Foo;
  78. # struct _Foo {...};
  79. pass
  80. elif original:
  81. positions = set()
  82. positions.update(original.file_positions)
  83. positions.update(node.file_positions)
  84. message.fatal("Namespace conflict for '%s'" % (node.name, ),
  85. positions)
  86. else:
  87. self._namespace.append(node)
  88. def parse(self, symbols):
  89. for symbol in symbols:
  90. # WORKAROUND
  91. # https://bugzilla.gnome.org/show_bug.cgi?id=550616
  92. if symbol.ident in ['gst_g_error_get_type']:
  93. continue
  94. try:
  95. node = self._traverse_one(symbol)
  96. except TransformerException as e:
  97. message.warn_symbol(symbol, e)
  98. continue
  99. if node and node.name:
  100. self._append_new_node(node)
  101. if isinstance(node, ast.Compound) and node.tag_name and \
  102. node.tag_name not in self._tag_ns:
  103. self._tag_ns[node.tag_name] = node
  104. # Run through the tag namespace looking for structs that have not been
  105. # promoted into the main namespace. In this case we simply promote them
  106. # with their struct tag.
  107. for tag_name, struct in self._tag_ns.items():
  108. if not struct.name:
  109. try:
  110. name = self.strip_identifier(tag_name)
  111. struct.name = name
  112. self._append_new_node(struct)
  113. except TransformerException as e:
  114. message.warn_node(node, e)
  115. def set_include_paths(self, paths):
  116. self._includepaths = list(paths)
  117. def register_include(self, include):
  118. if include in self._namespace.includes:
  119. return
  120. self._namespace.includes.add(include)
  121. filename = self._find_include(include)
  122. self._parse_include(filename)
  123. def register_include_uninstalled(self, include_path):
  124. basename = os.path.basename(include_path)
  125. if not basename.endswith('.gir'):
  126. raise SystemExit("Include path '%s' must be a filename path "
  127. "ending in .gir" % (include_path, ))
  128. girname = basename[:-4]
  129. include = ast.Include.from_string(girname)
  130. if include in self._namespace.includes:
  131. return
  132. self._namespace.includes.add(include)
  133. self._parse_include(include_path, uninstalled=True)
  134. def lookup_giname(self, name):
  135. """Given a name of the form Foo or Bar.Foo,
  136. return the corresponding ast.Node, or None if none
  137. available. Will throw KeyError however for unknown
  138. namespaces."""
  139. if '.' not in name:
  140. return self._namespace.get(name)
  141. else:
  142. (ns, giname) = name.split('.', 1)
  143. if ns == self._namespace.name:
  144. return self._namespace.get(giname)
  145. # Fallback to the main namespace if not a dependency and matches a prefix
  146. if ns in self._namespace.identifier_prefixes and ns not in self._parsed_includes:
  147. message.warn(("Deprecated reference to identifier " +
  148. "prefix %s in GIName %s") % (ns, name))
  149. return self._namespace.get(giname)
  150. include = self._parsed_includes[ns]
  151. return include.get(giname)
  152. def lookup_typenode(self, typeobj):
  153. """Given a Type object, if it points to a giname,
  154. calls lookup_giname() on the name. Otherwise return
  155. None."""
  156. if typeobj.target_giname:
  157. return self.lookup_giname(typeobj.target_giname)
  158. return None
  159. # Private
  160. def _get_gi_data_dirs(self):
  161. data_dirs = utils.get_system_data_dirs()
  162. data_dirs.append(DATADIR)
  163. if os.name != 'nt':
  164. # For backwards compatibility, was always unconditionally added to the list.
  165. data_dirs.append('/usr/share')
  166. return data_dirs
  167. def _find_include(self, include):
  168. searchdirs = self._includepaths[:]
  169. for path in self._get_gi_data_dirs():
  170. searchdirs.append(os.path.join(path, 'gir-1.0'))
  171. searchdirs.append(os.path.join(DATADIR, 'gir-1.0'))
  172. girname = '%s-%s.gir' % (include.name, include.version)
  173. for d in searchdirs:
  174. path = os.path.join(d, girname)
  175. if os.path.exists(path):
  176. return path
  177. sys.stderr.write("Couldn't find include '%s' (search path: '%s')\n" %
  178. (girname, searchdirs))
  179. sys.exit(1)
  180. @classmethod
  181. def parse_from_gir(cls, filename, extra_include_dirs=None):
  182. self = cls(None)
  183. if extra_include_dirs is not None:
  184. self.set_include_paths(extra_include_dirs)
  185. self.set_passthrough_mode()
  186. parser = self._parse_include(filename)
  187. self._namespace = parser.get_namespace()
  188. del self._parsed_includes[self._namespace.name]
  189. return self
  190. def _parse_include(self, filename, uninstalled=False):
  191. parser = None
  192. if self._cachestore is not None:
  193. parser = self._cachestore.load(filename)
  194. if parser is None:
  195. parser = GIRParser(types_only=not self._passthrough_mode)
  196. parser.parse(filename)
  197. if self._cachestore is not None:
  198. self._cachestore.store(filename, parser)
  199. for include in parser.get_namespace().includes:
  200. if include.name not in self._parsed_includes:
  201. dep_filename = self._find_include(include)
  202. self._parse_include(dep_filename)
  203. if not uninstalled:
  204. for pkg in parser.get_namespace().exported_packages:
  205. self._pkg_config_packages.add(pkg)
  206. namespace = parser.get_namespace()
  207. self._parsed_includes[namespace.name] = namespace
  208. return parser
  209. def _iter_namespaces(self):
  210. """Return an iterator over all included namespaces; the
  211. currently-scanned namespace is first."""
  212. yield self._namespace
  213. for ns in self._parsed_includes.values():
  214. yield ns
  215. def _sort_matches(self, val):
  216. """Key sort which ensures items in self._namespace are last by returning
  217. a tuple key starting with 1 for self._namespace entries and 0 for
  218. everythin else.
  219. """
  220. if val[0] == self._namespace:
  221. return 1, val[2]
  222. else:
  223. return 0, val[2]
  224. def _split_c_string_for_namespace_matches(self, name, is_identifier=False):
  225. if not is_identifier and self._symbol_filter_cmd:
  226. proc = subprocess.Popen(self._symbol_filter_cmd,
  227. stdin=subprocess.PIPE,
  228. stdout=subprocess.PIPE,
  229. stderr=subprocess.PIPE,
  230. shell=True)
  231. _name = name
  232. proc_name, err = proc.communicate(name.encode())
  233. if proc.returncode:
  234. raise ValueError('filter: "%s" exited: %d with error: %s' %
  235. (self._symbol_filter_cmd, proc.returncode, err))
  236. name = proc_name.decode('ascii')
  237. matches = [] # Namespaces which might contain this name
  238. unprefixed_namespaces = [] # Namespaces with no prefix, last resort
  239. for ns in self._iter_namespaces():
  240. if is_identifier:
  241. prefixes = ns.identifier_prefixes
  242. elif name[0].isupper():
  243. prefixes = ns._ucase_symbol_prefixes
  244. else:
  245. prefixes = ns.symbol_prefixes
  246. if prefixes:
  247. for prefix in prefixes:
  248. if (not is_identifier) and (not prefix.endswith('_')):
  249. prefix = prefix + '_'
  250. if name.startswith(prefix):
  251. matches.append((ns, name[len(prefix):], len(prefix)))
  252. break
  253. else:
  254. unprefixed_namespaces.append(ns)
  255. if matches:
  256. matches.sort(key=self._sort_matches)
  257. return list(map(lambda x: (x[0], x[1]), matches))
  258. elif self._accept_unprefixed:
  259. return [(self._namespace, name)]
  260. elif unprefixed_namespaces:
  261. # A bit of a hack; this function ideally shouldn't look through the
  262. # contents of namespaces; but since we aren't scanning anything
  263. # without a prefix, it's not too bad.
  264. for ns in unprefixed_namespaces:
  265. if name in ns:
  266. return [(ns, name)]
  267. raise ValueError("Unknown namespace for %s '%s'"
  268. % ('identifier' if is_identifier else 'symbol', name, ))
  269. def split_ctype_namespaces(self, ident):
  270. """Given a StudlyCaps string identifier like FooBar, return a
  271. list of (namespace, stripped_identifier) sorted by namespace length,
  272. or raise ValueError. As a special case, if the current namespace matches,
  273. it is always biggest (i.e. last)."""
  274. return self._split_c_string_for_namespace_matches(ident, is_identifier=True)
  275. def split_csymbol_namespaces(self, symbol):
  276. """Given a C symbol like foo_bar_do_baz, return a list of
  277. (namespace, stripped_symbol) sorted by namespace match probablity, or
  278. raise ValueError."""
  279. return self._split_c_string_for_namespace_matches(symbol, is_identifier=False)
  280. def split_csymbol(self, symbol):
  281. """Given a C symbol like foo_bar_do_baz, return the most probable
  282. (namespace, stripped_symbol) match, or raise ValueError."""
  283. matches = self._split_c_string_for_namespace_matches(symbol, is_identifier=False)
  284. return matches[-1]
  285. def strip_identifier(self, ident):
  286. if self._identifier_filter_cmd:
  287. proc = subprocess.Popen(self._identifier_filter_cmd,
  288. stdin=subprocess.PIPE,
  289. stdout=subprocess.PIPE,
  290. stderr=subprocess.PIPE,
  291. shell=True)
  292. proc_ident, err = proc.communicate(ident.encode())
  293. if proc.returncode:
  294. raise ValueError('filter: "%s" exited: %d with error: %s' %
  295. (self._identifier_filter_cmd, proc.returncode, err))
  296. ident = proc_ident.decode('ascii')
  297. hidden = ident.startswith('_')
  298. if hidden:
  299. ident = ident[1:]
  300. try:
  301. matches = self.split_ctype_namespaces(ident)
  302. except ValueError as e:
  303. raise TransformerException(str(e))
  304. for ns, name in matches:
  305. if ns is self._namespace:
  306. if hidden:
  307. return '_' + name
  308. return name
  309. (ns, name) = matches[-1]
  310. raise TransformerException(
  311. "Skipping foreign identifier '%s' from namespace %s" % (ident, ns.name, ))
  312. return None
  313. def _strip_symbol(self, symbol):
  314. ident = symbol.ident
  315. hidden = ident.startswith('_')
  316. if hidden:
  317. ident = ident[1:]
  318. try:
  319. (ns, name) = self.split_csymbol(ident)
  320. except ValueError as e:
  321. raise TransformerException(str(e))
  322. if ns != self._namespace:
  323. raise TransformerException(
  324. "Skipping foreign symbol from namespace %s" % (ns.name, ))
  325. if hidden:
  326. return '_' + name
  327. return name
  328. def _traverse_one(self, symbol, stype=None, parent_symbol=None):
  329. assert isinstance(symbol, SourceSymbol), symbol
  330. if stype is None:
  331. stype = symbol.type
  332. if stype == CSYMBOL_TYPE_FUNCTION:
  333. return self._create_function(symbol)
  334. elif stype == CSYMBOL_TYPE_TYPEDEF:
  335. return self._create_typedef(symbol)
  336. elif stype == CSYMBOL_TYPE_STRUCT:
  337. return self._create_tag_ns_compound(ast.Record, symbol)
  338. elif stype == CSYMBOL_TYPE_ENUM:
  339. return self._create_enum(symbol)
  340. elif stype == CSYMBOL_TYPE_MEMBER:
  341. return self._create_member(symbol, parent_symbol)
  342. elif stype == CSYMBOL_TYPE_UNION:
  343. return self._create_tag_ns_compound(ast.Union, symbol)
  344. elif stype == CSYMBOL_TYPE_CONST:
  345. return self._create_const(symbol)
  346. # Ignore variable declarations in the header
  347. elif stype == CSYMBOL_TYPE_OBJECT:
  348. pass
  349. else:
  350. print("transformer: unhandled symbol: '%s'" % (symbol, ))
  351. def _enum_common_prefix(self, symbol):
  352. def common_prefix(a, b):
  353. commonparts = []
  354. for aword, bword in zip(a.split('_'), b.split('_')):
  355. if aword != bword:
  356. return '_'.join(commonparts) + '_'
  357. commonparts.append(aword)
  358. return min(a, b)
  359. # Nothing less than 2 has a common prefix
  360. if len(list(symbol.base_type.child_list)) < 2:
  361. return None
  362. prefix = None
  363. for child in symbol.base_type.child_list:
  364. if prefix is None:
  365. prefix = child.ident
  366. else:
  367. prefix = common_prefix(prefix, child.ident)
  368. if prefix == '':
  369. return None
  370. return prefix
  371. def _create_enum(self, symbol):
  372. prefix = self._enum_common_prefix(symbol)
  373. if prefix:
  374. prefixlen = len(prefix)
  375. else:
  376. prefixlen = 0
  377. members = []
  378. for child in symbol.base_type.child_list:
  379. if child.private:
  380. continue
  381. if prefixlen > 0:
  382. name = child.ident[prefixlen:]
  383. else:
  384. # Ok, the enum members don't have a consistent prefix
  385. # among them, so let's just remove the global namespace
  386. # prefix.
  387. name = self._strip_symbol(child)
  388. members.append(ast.Member(name.lower(),
  389. child.const_int,
  390. child.ident,
  391. None))
  392. enum_name = self.strip_identifier(symbol.ident)
  393. if symbol.base_type.is_bitfield:
  394. klass = ast.Bitfield
  395. else:
  396. klass = ast.Enum
  397. node = klass(enum_name, symbol.ident, members=members)
  398. node.add_symbol_reference(symbol)
  399. return node
  400. def _create_function(self, symbol):
  401. # Drop functions that start with _ very early on here
  402. if symbol.ident.startswith('_'):
  403. return None
  404. parameters = list(self._create_parameters(symbol, symbol.base_type))
  405. return_ = self._create_return(symbol.base_type.base_type)
  406. name = self._strip_symbol(symbol)
  407. func = ast.Function(name, return_, parameters, False, symbol.ident)
  408. func.add_symbol_reference(symbol)
  409. return func
  410. def _create_source_type(self, source_type):
  411. assert source_type is not None
  412. if source_type.type == CTYPE_VOID:
  413. value = 'void'
  414. elif source_type.type == CTYPE_BASIC_TYPE:
  415. value = source_type.name
  416. elif source_type.type == CTYPE_TYPEDEF:
  417. value = source_type.name
  418. elif source_type.type == CTYPE_ARRAY:
  419. return self._create_source_type(source_type.base_type)
  420. elif source_type.type == CTYPE_POINTER:
  421. value = self._create_source_type(source_type.base_type) + '*'
  422. else:
  423. value = 'gpointer'
  424. return value
  425. def _create_complete_source_type(self, source_type):
  426. assert source_type is not None
  427. const = (source_type.type_qualifier & TYPE_QUALIFIER_CONST)
  428. volatile = (source_type.type_qualifier & TYPE_QUALIFIER_VOLATILE)
  429. if source_type.type == CTYPE_VOID:
  430. return 'void'
  431. elif source_type.type in [CTYPE_BASIC_TYPE,
  432. CTYPE_TYPEDEF,
  433. CTYPE_STRUCT,
  434. CTYPE_UNION,
  435. CTYPE_ENUM]:
  436. value = source_type.name
  437. if not value:
  438. value = 'gpointer'
  439. if const:
  440. value = 'const ' + value
  441. if volatile:
  442. value = 'volatile ' + value
  443. elif source_type.type == CTYPE_ARRAY:
  444. return self._create_complete_source_type(source_type.base_type)
  445. elif source_type.type == CTYPE_POINTER:
  446. value = self._create_complete_source_type(source_type.base_type) + '*'
  447. # TODO: handle pointer to function as a special case?
  448. if const:
  449. value += ' const'
  450. if volatile:
  451. value += ' volatile'
  452. else:
  453. if const:
  454. value = 'gconstpointer'
  455. else:
  456. value = 'gpointer'
  457. if volatile:
  458. value = 'volatile ' + value
  459. return value
  460. return value
  461. def _create_parameters(self, symbol, base_type):
  462. for i, child in enumerate(base_type.child_list):
  463. yield self._create_parameter(symbol, i, child)
  464. def _synthesize_union_type(self, symbol, parent_symbol):
  465. # Synthesize a named union so that it can be referenced.
  466. parent_ident = parent_symbol.ident
  467. # FIXME: Should split_ctype_namespaces handle the hidden case?
  468. hidden = parent_ident.startswith('_')
  469. if hidden:
  470. parent_ident = parent_ident[1:]
  471. matches = self.split_ctype_namespaces(parent_ident)
  472. (namespace, parent_name) = matches[-1]
  473. assert namespace and parent_name
  474. if hidden:
  475. parent_name = '_' + parent_name
  476. fake_union = ast.Union("%s__%s__union" % (parent_name, symbol.ident))
  477. # _parse_fields accesses <type>.base_type.child_list, so we have to
  478. # pass symbol.base_type even though that refers to the array, not the
  479. # union.
  480. self._parse_fields(symbol.base_type, fake_union)
  481. self._append_new_node(fake_union)
  482. fake_type = ast.Type(
  483. target_giname="%s.%s" % (namespace.name, fake_union.name))
  484. return fake_type
  485. def _create_member(self, symbol, parent_symbol=None):
  486. source_type = symbol.base_type
  487. if (source_type.type == CTYPE_POINTER
  488. and symbol.base_type.base_type.type == CTYPE_FUNCTION):
  489. node = self._create_callback(symbol, member=True)
  490. elif source_type.type == CTYPE_STRUCT and source_type.name is None:
  491. node = self._create_member_compound(ast.Record, symbol)
  492. elif source_type.type == CTYPE_UNION and source_type.name is None:
  493. node = self._create_member_compound(ast.Union, symbol)
  494. else:
  495. # Special handling for fields; we don't have annotations on them
  496. # to apply later, yet.
  497. if source_type.type == CTYPE_ARRAY:
  498. complete_ctype = self._create_complete_source_type(source_type)
  499. # If the array contains anonymous unions, like in the GValue
  500. # struct, we need to handle this specially. This is necessary
  501. # to be able to properly calculate the size of the compound
  502. # type (e.g. GValue) that contains this array, see
  503. # <https://bugzilla.gnome.org/show_bug.cgi?id=657040>.
  504. if (source_type.base_type.type == CTYPE_UNION
  505. and source_type.base_type.name is None):
  506. synthesized_type = self._synthesize_union_type(symbol, parent_symbol)
  507. ftype = ast.Array(None, synthesized_type, complete_ctype=complete_ctype)
  508. else:
  509. ctype = self._create_source_type(source_type)
  510. canonical_ctype = self._canonicalize_ctype(ctype)
  511. if canonical_ctype[-1] == '*':
  512. derefed_name = canonical_ctype[:-1]
  513. else:
  514. derefed_name = canonical_ctype
  515. if complete_ctype[-1] == '*':
  516. derefed_complete_ctype = complete_ctype[:-1]
  517. else:
  518. derefed_complete_ctype = complete_ctype
  519. from_ctype = self.create_type_from_ctype_string(ctype,
  520. complete_ctype=complete_ctype)
  521. ftype = ast.Array(None, from_ctype,
  522. ctype=derefed_name,
  523. complete_ctype=derefed_complete_ctype)
  524. child_list = list(symbol.base_type.child_list)
  525. ftype.zeroterminated = False
  526. if child_list:
  527. ftype.size = child_list[0].const_int
  528. else:
  529. ftype = self._create_type_from_base(symbol.base_type)
  530. # ast.Fields are assumed to be read-write
  531. # (except for Objects, see also glibtransformer.py)
  532. node = ast.Field(symbol.ident, ftype,
  533. readable=True, writable=True,
  534. bits=symbol.const_int)
  535. if symbol.private:
  536. node.readable = False
  537. node.writable = False
  538. node.private = True
  539. return node
  540. def _create_typedef(self, symbol):
  541. ctype = symbol.base_type.type
  542. if (ctype == CTYPE_POINTER and symbol.base_type.base_type.type == CTYPE_FUNCTION):
  543. node = self._create_typedef_callback(symbol)
  544. elif (ctype == CTYPE_FUNCTION):
  545. node = self._create_typedef_callback(symbol)
  546. elif (ctype == CTYPE_POINTER and symbol.base_type.base_type.type == CTYPE_STRUCT):
  547. node = self._create_typedef_compound(ast.Record, symbol, disguised=True)
  548. elif ctype == CTYPE_STRUCT:
  549. node = self._create_typedef_compound(ast.Record, symbol)
  550. elif ctype == CTYPE_UNION:
  551. node = self._create_typedef_compound(ast.Union, symbol)
  552. elif ctype == CTYPE_ENUM:
  553. return self._create_enum(symbol)
  554. elif ctype in (CTYPE_TYPEDEF,
  555. CTYPE_POINTER,
  556. CTYPE_BASIC_TYPE,
  557. CTYPE_VOID):
  558. name = self.strip_identifier(symbol.ident)
  559. if symbol.base_type.name:
  560. complete_ctype = self._create_complete_source_type(symbol.base_type)
  561. target = self.create_type_from_ctype_string(symbol.base_type.name,
  562. complete_ctype=complete_ctype)
  563. else:
  564. target = ast.TYPE_ANY
  565. if name in ast.type_names:
  566. return None
  567. # https://bugzilla.gnome.org/show_bug.cgi?id=755882
  568. if name.endswith('_autoptr'):
  569. return None
  570. return ast.Alias(name, target, ctype=symbol.ident)
  571. else:
  572. raise NotImplementedError(
  573. "symbol '%s' of type %s" % (symbol.ident, ctype_name(ctype)))
  574. return node
  575. def _canonicalize_ctype(self, ctype):
  576. # First look up the ctype including any pointers;
  577. # a few type names like 'char*' have their own aliases
  578. # and we need pointer information for those.
  579. firstpass = ast.type_names.get(ctype)
  580. # If we have a particular alias for this, skip deep
  581. # canonicalization to prevent changing
  582. # e.g. char* -> int8*
  583. if firstpass:
  584. return firstpass.target_fundamental
  585. if not ctype.endswith('*'):
  586. return ctype
  587. # We have a pointer type.
  588. # Strip the end pointer, canonicalize our base type
  589. base = ctype[:-1]
  590. canonical_base = self._canonicalize_ctype(base)
  591. # Append the pointer again
  592. canonical = canonical_base + '*'
  593. return canonical
  594. def _create_type_from_base(self, source_type, is_parameter=False, is_return=False):
  595. ctype = self._create_source_type(source_type)
  596. complete_ctype = self._create_complete_source_type(source_type)
  597. const = ((source_type.type == CTYPE_POINTER) and
  598. (source_type.base_type.type_qualifier & TYPE_QUALIFIER_CONST))
  599. return self.create_type_from_ctype_string(ctype, is_const=const,
  600. is_parameter=is_parameter, is_return=is_return,
  601. complete_ctype=complete_ctype)
  602. def _create_bare_container_type(self, base, ctype=None,
  603. is_const=False, complete_ctype=None):
  604. if base in ('GList', 'GSList', 'GLib.List', 'GLib.SList'):
  605. if base in ('GList', 'GSList'):
  606. name = 'GLib.' + base[1:]
  607. else:
  608. name = base
  609. return ast.List(name, ast.TYPE_ANY, ctype=ctype,
  610. is_const=is_const, complete_ctype=complete_ctype)
  611. elif base in ('GArray', 'GPtrArray', 'GByteArray',
  612. 'GLib.Array', 'GLib.PtrArray', 'GLib.ByteArray',
  613. 'GObject.Array', 'GObject.PtrArray', 'GObject.ByteArray'):
  614. if '.' in base:
  615. name = 'GLib.' + base.split('.', 1)[1]
  616. else:
  617. name = 'GLib.' + base[1:]
  618. return ast.Array(name, ast.TYPE_ANY, ctype=ctype,
  619. is_const=is_const, complete_ctype=complete_ctype)
  620. elif base in ('GHashTable', 'GLib.HashTable', 'GObject.HashTable'):
  621. return ast.Map(ast.TYPE_ANY, ast.TYPE_ANY, ctype=ctype, is_const=is_const,
  622. complete_ctype=complete_ctype)
  623. return None
  624. def create_type_from_ctype_string(self, ctype, is_const=False,
  625. is_parameter=False, is_return=False,
  626. complete_ctype=None):
  627. canonical = self._canonicalize_ctype(ctype)
  628. base = canonical.replace('*', '')
  629. # Special default: char ** -> ast.Array, same for GStrv
  630. if (is_return and canonical == 'utf8*') or base == 'GStrv':
  631. bare_utf8 = ast.TYPE_STRING.clone()
  632. bare_utf8.ctype = None
  633. return ast.Array(None, bare_utf8, ctype=ctype,
  634. is_const=is_const, complete_ctype=complete_ctype)
  635. fundamental = ast.type_names.get(base)
  636. if fundamental is not None:
  637. return ast.Type(target_fundamental=fundamental.target_fundamental,
  638. ctype=ctype,
  639. is_const=is_const, complete_ctype=complete_ctype)
  640. container = self._create_bare_container_type(base, ctype=ctype, is_const=is_const,
  641. complete_ctype=complete_ctype)
  642. if container:
  643. return container
  644. return ast.Type(ctype=ctype, is_const=is_const, complete_ctype=complete_ctype)
  645. def _create_parameter(self, parent_symbol, index, symbol):
  646. if symbol.type == CSYMBOL_TYPE_ELLIPSIS:
  647. return ast.Parameter('...', ast.Varargs())
  648. else:
  649. ptype = self._create_type_from_base(symbol.base_type, is_parameter=True)
  650. if symbol.ident is None:
  651. if symbol.base_type and symbol.base_type.type != CTYPE_VOID:
  652. message.warn_symbol(parent_symbol, "missing parameter name; undocumentable")
  653. ident = 'arg%d' % (index, )
  654. else:
  655. ident = symbol.ident
  656. return ast.Parameter(ident, ptype)
  657. def _create_return(self, source_type):
  658. typeval = self._create_type_from_base(source_type, is_return=True)
  659. return ast.Return(typeval)
  660. def _create_const(self, symbol):
  661. if symbol.ident.startswith('_'):
  662. return None
  663. # Don't create constants for non-public things
  664. # http://bugzilla.gnome.org/show_bug.cgi?id=572790
  665. if (symbol.source_filename is None or not symbol.source_filename.endswith('.h')):
  666. return None
  667. name = self._strip_symbol(symbol)
  668. if symbol.const_string is not None:
  669. typeval = ast.TYPE_STRING
  670. value = symbol.const_string
  671. elif symbol.const_int is not None:
  672. if symbol.base_type is not None:
  673. typeval = self._create_type_from_base(symbol.base_type)
  674. else:
  675. typeval = ast.TYPE_INT
  676. unaliased = typeval
  677. self._resolve_type_from_ctype(unaliased)
  678. if typeval.target_giname and typeval.ctype:
  679. target = self.lookup_giname(typeval.target_giname)
  680. target = self.resolve_aliases(target)
  681. if isinstance(target, ast.Type):
  682. unaliased = target
  683. if unaliased == ast.TYPE_UINT64:
  684. value = str(symbol.const_int % 2 ** 64)
  685. elif unaliased == ast.TYPE_UINT32:
  686. value = str(symbol.const_int % 2 ** 32)
  687. elif unaliased == ast.TYPE_UINT16:
  688. value = str(symbol.const_int % 2 ** 16)
  689. elif unaliased == ast.TYPE_UINT8:
  690. value = str(symbol.const_int % 2 ** 16)
  691. else:
  692. value = str(symbol.const_int)
  693. elif symbol.const_boolean is not None:
  694. typeval = ast.TYPE_BOOLEAN
  695. value = "true" if symbol.const_boolean else "false"
  696. elif symbol.const_double is not None:
  697. typeval = ast.TYPE_DOUBLE
  698. value = '%f' % (symbol.const_double, )
  699. else:
  700. raise AssertionError()
  701. const = ast.Constant(name, typeval, value,
  702. symbol.ident)
  703. const.add_symbol_reference(symbol)
  704. return const
  705. def _create_typedef_compound(self, compound_class, symbol, disguised=False):
  706. name = self.strip_identifier(symbol.ident)
  707. assert symbol.base_type
  708. if symbol.base_type.name:
  709. tag_name = symbol.base_type.name
  710. else:
  711. tag_name = None
  712. # If the struct already exists in the tag namespace, use it.
  713. if tag_name in self._tag_ns:
  714. compound = self._tag_ns[tag_name]
  715. if compound.name:
  716. # If the struct name is set it means the struct has already been
  717. # promoted from the tag namespace to the main namespace by a
  718. # prior typedef struct. If we get here it means this is another
  719. # typedef of that struct. Instead of creating an alias to the
  720. # primary typedef that has been promoted, we create a new Record
  721. # with shared fields. This handles the case where we want to
  722. # give structs like GInitiallyUnowned its own Record:
  723. # typedef struct _GObject GObject;
  724. # typedef struct _GObject GInitiallyUnowned;
  725. # See: http://bugzilla.gnome.org/show_bug.cgi?id=569408
  726. new_compound = compound_class(name, symbol.ident, tag_name=tag_name)
  727. new_compound.fields = compound.fields
  728. new_compound.add_symbol_reference(symbol)
  729. return new_compound
  730. else:
  731. # If the struct does not have its name set, it exists only in
  732. # the tag namespace. Set it here and return it which will
  733. # promote it to the main namespace. Essentially the first
  734. # typedef for a struct clobbers its name and ctype which is what
  735. # will be visible to GI.
  736. compound.name = name
  737. compound.ctype = symbol.ident
  738. else:
  739. # Create a new struct with a typedef name and tag name when available.
  740. # Structs with a typedef name are promoted into the main namespace
  741. # by it being returned to the "parse" function and are also added to
  742. # the tag namespace if it has a tag_name set.
  743. compound = compound_class(name, symbol.ident, disguised=disguised, tag_name=tag_name)
  744. if tag_name:
  745. # Force the struct as disguised for now since we do not yet know
  746. # if it has fields that will be parsed. Note that this is using
  747. # an erroneous definition of disguised and we should eventually
  748. # only look at the field count when needed.
  749. compound.disguised = True
  750. else:
  751. # Case where we have an anonymous struct which is typedef'd:
  752. # typedef struct {...} Struct;
  753. # we need to parse the fields because we never get a struct
  754. # in the tag namespace which is normally where fields are parsed.
  755. self._parse_fields(symbol, compound)
  756. compound.add_symbol_reference(symbol)
  757. return compound
  758. def _create_tag_ns_compound(self, compound_class, symbol):
  759. # Get or create a struct from C's tag namespace
  760. if symbol.ident in self._tag_ns:
  761. compound = self._tag_ns[symbol.ident]
  762. else:
  763. compound = compound_class(None, symbol.ident, tag_name=symbol.ident)
  764. # Make sure disguised is False as we are now about to parse the
  765. # fields of the real struct.
  766. compound.disguised = False
  767. # Fields may need to be parsed in either of the above cases because the
  768. # Record can be created with a typedef prior to the struct definition.
  769. self._parse_fields(symbol, compound)
  770. compound.add_symbol_reference(symbol)
  771. return compound
  772. def _create_member_compound(self, compound_class, symbol):
  773. compound = compound_class(symbol.ident, symbol.ident)
  774. self._parse_fields(symbol, compound)
  775. compound.add_symbol_reference(symbol)
  776. return compound
  777. def _create_typedef_callback(self, symbol):
  778. callback = self._create_callback(symbol)
  779. if not callback:
  780. return None
  781. return callback
  782. def _parse_fields(self, symbol, compound):
  783. for child in symbol.base_type.child_list:
  784. child_node = self._traverse_one(child, parent_symbol=symbol)
  785. if not child_node:
  786. continue
  787. if isinstance(child_node, ast.Field):
  788. field = child_node
  789. else:
  790. field = ast.Field(child.ident, None, True, False,
  791. anonymous_node=child_node)
  792. compound.fields.append(field)
  793. def _create_callback(self, symbol, member=False):
  794. if (symbol.base_type.type == CTYPE_FUNCTION): # function
  795. paramtype = symbol.base_type
  796. retvaltype = symbol.base_type.base_type
  797. elif (symbol.base_type.type == CTYPE_POINTER): # function pointer
  798. paramtype = symbol.base_type.base_type
  799. retvaltype = symbol.base_type.base_type.base_type
  800. parameters = list(self._create_parameters(symbol, paramtype))
  801. retval = self._create_return(retvaltype)
  802. # Mark the 'user_data' arguments
  803. for i, param in enumerate(parameters):
  804. if (param.type.target_fundamental == 'gpointer' and param.argname == 'user_data'):
  805. param.closure_name = param.argname
  806. if member:
  807. name = symbol.ident
  808. elif symbol.ident.find('_') > 0:
  809. name = self._strip_symbol(symbol)
  810. else:
  811. name = self.strip_identifier(symbol.ident)
  812. callback = ast.Callback(name, retval, parameters, False,
  813. ctype=symbol.ident)
  814. callback.add_symbol_reference(symbol)
  815. return callback
  816. def create_type_from_user_string(self, typestr):
  817. """Parse a C type string (as might be given from an
  818. annotation) and resolve it. For compatibility, we can consume
  819. both GI type string (utf8, Foo.Bar) style, as well as C (char *, FooBar) style.
  820. Note that type resolution may not succeed."""
  821. if '.' in typestr:
  822. container = self._create_bare_container_type(typestr)
  823. if container:
  824. typeval = container
  825. else:
  826. typeval = self._namespace.type_from_name(typestr)
  827. else:
  828. typeval = self.create_type_from_ctype_string(typestr)
  829. self.resolve_type(typeval)
  830. if typeval.resolved:
  831. # Explicitly clear out the c_type; there isn't one in this case.
  832. typeval.ctype = None
  833. return typeval
  834. def _resolve_type_from_ctype_all_namespaces(self, typeval, pointer_stripped):
  835. # If we can't determine the namespace from the type name,
  836. # fall back to trying all of our includes. An example of this is mutter,
  837. # which has nominal namespace of "Meta", but a few classes are
  838. # "Mutter". We don't export that data in introspection currently.
  839. # Basically the library should be fixed, but we'll hack around it here.
  840. for namespace in self._parsed_includes.values():
  841. target = namespace.get_by_ctype(pointer_stripped)
  842. if target:
  843. typeval.target_giname = '%s.%s' % (namespace.name, target.name)
  844. return True
  845. return False
  846. def _resolve_type_from_ctype(self, typeval):
  847. assert typeval.ctype is not None
  848. pointer_stripped = typeval.ctype.replace('*', '')
  849. try:
  850. matches = self.split_ctype_namespaces(pointer_stripped)
  851. except ValueError:
  852. return self._resolve_type_from_ctype_all_namespaces(typeval, pointer_stripped)
  853. for namespace, name in matches:
  854. target = namespace.get(name)
  855. if not target:
  856. target = namespace.get_by_ctype(pointer_stripped)
  857. if target:
  858. typeval.target_giname = '%s.%s' % (namespace.name, target.name)
  859. return True
  860. return False
  861. def _resolve_type_from_gtype_name(self, typeval):
  862. assert typeval.gtype_name is not None
  863. for ns in self._iter_namespaces():
  864. node = ns.type_names.get(typeval.gtype_name, None)
  865. if node is not None:
  866. typeval.target_giname = '%s.%s' % (ns.name, node.name)
  867. return True
  868. return False
  869. def _resolve_type_internal(self, typeval):
  870. if isinstance(typeval, (ast.Array, ast.List)):
  871. return self.resolve_type(typeval.element_type)
  872. elif isinstance(typeval, ast.Map):
  873. key_resolved = self.resolve_type(typeval.key_type)
  874. value_resolved = self.resolve_type(typeval.value_type)
  875. return key_resolved and value_resolved
  876. elif typeval.resolved:
  877. return True
  878. elif typeval.ctype:
  879. return self._resolve_type_from_ctype(typeval)
  880. elif typeval.gtype_name:
  881. return self._resolve_type_from_gtype_name(typeval)
  882. def resolve_type(self, typeval):
  883. if not self._resolve_type_internal(typeval):
  884. return False
  885. if typeval.target_fundamental or typeval.target_foreign:
  886. return True
  887. assert typeval.target_giname is not None
  888. try:
  889. type_ = self.lookup_giname(typeval.target_giname)
  890. except KeyError:
  891. type_ = None
  892. if type_ is None:
  893. typeval.target_giname = None
  894. return typeval.resolved
  895. def resolve_aliases(self, typenode):
  896. """Removes all aliases from typenode, returns first non-alias
  897. in the typenode alias chain. Returns typenode argument if it
  898. is not an alias."""
  899. while isinstance(typenode, ast.Alias):
  900. if typenode.target.target_giname is not None:
  901. typenode = self.lookup_giname(typenode.target.target_giname)
  902. elif typenode.target.target_fundamental is not None:
  903. typenode = typenode.target
  904. else:
  905. break
  906. return typenode