maintransformer.py 63 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475
  1. # -*- Mode: Python -*-
  2. # Copyright (C) 2010 Red Hat, Inc.
  3. #
  4. # This library is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU Lesser General Public
  6. # License as published by the Free Software Foundation; either
  7. # version 2 of the License, or (at your option) any later version.
  8. #
  9. # This library is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. # Lesser General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU Lesser General Public
  15. # License along with this library; if not, write to the
  16. # Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  17. # Boston, MA 02111-1307, USA.
  18. #
  19. from __future__ import absolute_import
  20. from __future__ import division
  21. from __future__ import print_function
  22. from __future__ import unicode_literals
  23. import re
  24. from . import ast
  25. from . import message
  26. from .annotationparser import (TAG_DEPRECATED, TAG_SINCE, TAG_STABILITY, TAG_RETURNS)
  27. from .annotationparser import (ANN_ALLOW_NONE, ANN_ARRAY, ANN_ATTRIBUTES, ANN_CLOSURE,
  28. ANN_CONSTRUCTOR, ANN_DESTROY, ANN_ELEMENT_TYPE, ANN_FOREIGN,
  29. ANN_GET_VALUE_FUNC, ANN_IN, ANN_INOUT, ANN_METHOD, ANN_OUT,
  30. ANN_REF_FUNC, ANN_RENAME_TO, ANN_SCOPE, ANN_SET_VALUE_FUNC,
  31. ANN_SKIP, ANN_TRANSFER, ANN_TYPE, ANN_UNREF_FUNC, ANN_VALUE,
  32. ANN_VFUNC, ANN_NULLABLE, ANN_OPTIONAL, ANN_NOT)
  33. from .annotationparser import (OPT_ARRAY_FIXED_SIZE, OPT_ARRAY_LENGTH, OPT_ARRAY_ZERO_TERMINATED,
  34. OPT_OUT_CALLEE_ALLOCATES, OPT_OUT_CALLER_ALLOCATES,
  35. OPT_TRANSFER_CONTAINER, OPT_TRANSFER_FLOATING, OPT_TRANSFER_NONE,
  36. OPT_NOT_NULLABLE)
  37. from .utils import to_underscores_noprefix
  38. class MainTransformer(object):
  39. def __init__(self, transformer, blocks):
  40. self._transformer = transformer
  41. self._blocks = blocks
  42. self._namespace = transformer.namespace
  43. self._uscore_type_names = {}
  44. # Public API
  45. def transform(self):
  46. if not self._namespace.names:
  47. message.fatal('Namespace is empty; likely causes are:\n'
  48. '* Not including .h files to be scanned\n'
  49. '* Broken --identifier-prefix')
  50. # Some initial namespace surgery
  51. self._namespace.walk(self._pass_fixup_hidden_fields)
  52. # We have a rough tree which should have most of of the types
  53. # we know about. Let's attempt closure; walk over all of the
  54. # Type() types and see if they match up with something.
  55. self._namespace.walk(self._pass_type_resolution)
  56. # Read in annotations needed early
  57. self._namespace.walk(self._pass_read_annotations_early)
  58. # Determine some default values for transfer etc.
  59. # based on the current tree.
  60. self._namespace.walk(self._pass_callable_defaults)
  61. # Read in most annotations now.
  62. self._namespace.walk(self._pass_read_annotations)
  63. # Now that we've possibly seen more types from annotations,
  64. # do another type resolution pass.
  65. self._namespace.walk(self._pass_type_resolution)
  66. # Generate a reverse mapping "bar_baz" -> BarBaz
  67. for node in self._namespace.values():
  68. if isinstance(node, ast.Registered) and node.get_type is not None:
  69. self._uscore_type_names[node.c_symbol_prefix] = node
  70. elif isinstance(node, (ast.Record, ast.Union)):
  71. uscored = to_underscores_noprefix(node.name).lower()
  72. self._uscore_type_names[uscored] = node
  73. for node in list(self._namespace.values()):
  74. if isinstance(node, ast.Function):
  75. # Discover which toplevel functions are actually methods
  76. self._pair_function(node)
  77. if isinstance(node, (ast.Class, ast.Interface)):
  78. self._pair_class_virtuals(node)
  79. # Some annotations need to be post function pairing
  80. self._namespace.walk(self._pass_read_annotations2)
  81. # Another type resolution pass after we've parsed virtuals, etc.
  82. self._namespace.walk(self._pass_type_resolution)
  83. self._namespace.walk(self._pass3)
  84. # TODO - merge into pass3
  85. self._pair_quarks_with_enums()
  86. # Private
  87. def _pass_fixup_hidden_fields(self, node, chain):
  88. """Hide all callbacks starting with _; the typical
  89. usage is void (*_gtk_reserved1)(void);"""
  90. if isinstance(node, (ast.Class, ast.Interface, ast.Record, ast.Union)):
  91. for field in node.fields:
  92. if (field
  93. and field.name is not None
  94. and field.name.startswith('_')
  95. and field.anonymous_node is not None
  96. and isinstance(field.anonymous_node, ast.Callback)):
  97. field.introspectable = False
  98. return True
  99. def _get_validate_parameter_name(self, parent, param_name, origin):
  100. try:
  101. param = parent.get_parameter(param_name)
  102. except ValueError:
  103. param = None
  104. if param is None:
  105. if isinstance(origin, ast.Parameter):
  106. origin_name = 'parameter %s' % (origin.argname, )
  107. else:
  108. origin_name = 'return value'
  109. message.log_node(
  110. message.FATAL, parent,
  111. "can't find parameter %s referenced by %s of '%s'"
  112. % (param_name, origin_name, parent.name))
  113. return param.argname
  114. def _get_validate_field_name(self, parent, field_name, origin):
  115. try:
  116. field = parent.get_field(field_name)
  117. except ValueError:
  118. field = None
  119. if field is None:
  120. origin_name = 'field %s' % (origin.name, )
  121. message.log_node(
  122. message.FATAL, parent,
  123. "can't find field %s referenced by %s of '%s'"
  124. % (field_name, origin_name, parent.name))
  125. return field.name
  126. def _apply_annotation_rename_to(self, node, chain, block):
  127. if not block:
  128. return
  129. rename_to = block.annotations.get(ANN_RENAME_TO)
  130. if not rename_to:
  131. return
  132. rename_to = rename_to[0]
  133. target = self._namespace.get_by_symbol(rename_to)
  134. if not target:
  135. message.warn_node(node,
  136. "Can't find symbol '%s' referenced by \"rename-to\" annotation" % (rename_to, ))
  137. elif target.shadowed_by:
  138. message.warn_node(node,
  139. "Function '%s' already shadowed by '%s', can't overwrite "
  140. "with '%s'" % (target.symbol,
  141. target.shadowed_by,
  142. rename_to))
  143. elif target.shadows:
  144. message.warn_node(node,
  145. "Function '%s' already shadows '%s', can't multiply shadow "
  146. "with '%s'" % (target.symbol,
  147. target.shadows,
  148. rename_to))
  149. else:
  150. target.shadowed_by = node.name
  151. node.shadows = target.name
  152. def _apply_annotations_function(self, node, chain):
  153. block = self._blocks.get(node.symbol)
  154. self._apply_annotations_callable(node, chain, block)
  155. def _pass_read_annotations_early(self, node, chain):
  156. if isinstance(node, ast.Record):
  157. if node.ctype is not None:
  158. block = self._blocks.get(node.ctype)
  159. else:
  160. block = self._blocks.get(node.c_name)
  161. self._apply_annotations_annotated(node, block)
  162. return True
  163. def _pass_callable_defaults(self, node, chain):
  164. if isinstance(node, (ast.Callable, ast.Signal)):
  165. for param in node.parameters:
  166. if param.transfer is None:
  167. param.transfer = self._get_transfer_default(node, param)
  168. if node.retval.transfer is None:
  169. node.retval.transfer = self._get_transfer_default(node, node.retval)
  170. return True
  171. def _get_annotation_name(self, node):
  172. if isinstance(node, (ast.Class, ast.Interface, ast.Record,
  173. ast.Union, ast.Enum, ast.Bitfield,
  174. ast.Callback, ast.Alias, ast.Constant)):
  175. if node.ctype is not None:
  176. return node.ctype
  177. elif isinstance(node, ast.Registered) and node.gtype_name is not None:
  178. return node.gtype_name
  179. return node.c_name
  180. raise AssertionError("Unhandled node '%s'" % (node, ))
  181. def _get_block(self, node):
  182. return self._blocks.get(self._get_annotation_name(node))
  183. def _pass_read_annotations(self, node, chain):
  184. if not node.namespace:
  185. return False
  186. if isinstance(node, ast.Alias):
  187. self._apply_annotations_alias(node, chain)
  188. if isinstance(node, ast.Function):
  189. self._apply_annotations_function(node, chain)
  190. if isinstance(node, ast.Callback):
  191. self._apply_annotations_callable(node, chain, block=self._get_block(node))
  192. if isinstance(node, (ast.Class, ast.Interface, ast.Union, ast.Enum,
  193. ast.Bitfield, ast.Callback)):
  194. self._apply_annotations_annotated(node, self._get_block(node))
  195. if isinstance(node, (ast.Enum, ast.Bitfield)):
  196. self._apply_annotations_enum_members(node, self._get_block(node))
  197. if isinstance(node, (ast.Class, ast.Interface, ast.Record, ast.Union)):
  198. block = self._get_block(node)
  199. for field in node.fields:
  200. self._apply_annotations_field(node, block, field)
  201. name = self._get_annotation_name(node)
  202. section_name = 'SECTION:%s' % (name.lower(), )
  203. block = self._blocks.get(section_name)
  204. if block and block.description:
  205. node.doc = block.description
  206. if isinstance(node, (ast.Class, ast.Interface)):
  207. for prop in node.properties:
  208. self._apply_annotations_property(node, prop)
  209. for sig in node.signals:
  210. self._apply_annotations_signal(node, sig)
  211. if isinstance(node, ast.Class):
  212. block = self._get_block(node)
  213. if block:
  214. annotation = block.annotations.get(ANN_UNREF_FUNC)
  215. node.unref_func = annotation[0] if annotation else None
  216. annotation = block.annotations.get(ANN_REF_FUNC)
  217. node.ref_func = annotation[0] if annotation else None
  218. annotation = block.annotations.get(ANN_SET_VALUE_FUNC)
  219. node.set_value_func = annotation[0] if annotation else None
  220. annotation = block.annotations.get(ANN_GET_VALUE_FUNC)
  221. node.get_value_func = annotation[0] if annotation else None
  222. if isinstance(node, ast.Constant):
  223. self._apply_annotations_constant(node)
  224. return True
  225. def _adjust_container_type(self, parent, node, annotations):
  226. if ANN_ARRAY in annotations:
  227. self._apply_annotations_array(parent, node, annotations)
  228. elif ANN_ELEMENT_TYPE in annotations:
  229. self._apply_annotations_element_type(parent, node, annotations)
  230. if isinstance(node.type, ast.Array):
  231. self._check_array_element_type(node.type, annotations)
  232. def _resolve(self, type_str, type_node=None, node=None, parent=None):
  233. def grab_one(type_str, resolver, top_combiner, combiner):
  234. """Return a complete type, and the trailing string part after it.
  235. Use resolver() on each identifier, and combiner() on the parts of
  236. each complete type. (top_combiner is used on the top-most type.)"""
  237. bits = re.split(r'([,<>()])', type_str, 1)
  238. first, sep, rest = [bits[0], '', ''] if (len(bits) == 1) else bits
  239. args = [resolver(first)]
  240. if sep == '<' or sep == '(':
  241. lastsep = '>' if (sep == '<') else ')'
  242. while sep != lastsep:
  243. next, rest = grab_one(rest, resolver, combiner, combiner)
  244. args.append(next)
  245. sep, rest = rest[0], rest[1:]
  246. else:
  247. rest = sep + rest
  248. return top_combiner(*args), rest
  249. def resolver(ident):
  250. res = self._transformer.create_type_from_user_string(ident)
  251. return res
  252. def combiner(base, *rest):
  253. if not rest:
  254. return base
  255. if isinstance(base, ast.List) and len(rest) == 1:
  256. return ast.List(base.name, *rest)
  257. elif isinstance(base, ast.Array) and len(rest) == 1:
  258. base.element_type = rest[0]
  259. return base
  260. elif isinstance(base, ast.Map) and len(rest) == 2:
  261. return ast.Map(*rest)
  262. message.warn(
  263. "Too many parameters in type specification '%s'" % (type_str, ))
  264. return base
  265. def top_combiner(base, *rest):
  266. if type_node is not None and isinstance(type_node, ast.Type):
  267. base.is_const = type_node.is_const
  268. return combiner(base, *rest)
  269. result, rest = grab_one(type_str, resolver, top_combiner, combiner)
  270. if rest:
  271. message.warn("Trailing components in type specification '%s'" % (
  272. type_str, ))
  273. if not result.resolved:
  274. position = None
  275. if parent is not None and isinstance(parent, ast.Function):
  276. text = parent.symbol
  277. position = self._get_position(parent, node)
  278. else:
  279. text = type_str
  280. message.warn_node(parent, "%s: Unknown type: '%s'" %
  281. (text, type_str), positions=position)
  282. return result
  283. def _resolve_toplevel(self, type_str, type_node=None, node=None, parent=None):
  284. """Like _resolve(), but attempt to preserve more attributes of original type."""
  285. result = self._resolve(type_str, type_node=type_node, node=node, parent=parent)
  286. # If we replace a node with a new type (such as an annotated) we
  287. # might lose the ctype from the original node.
  288. if type_node is not None:
  289. result.ctype = type_node.ctype
  290. return result
  291. def _get_position(self, func, param):
  292. block = self._blocks.get(func.symbol)
  293. if block:
  294. if isinstance(param, ast.Parameter):
  295. part = block.params.get(param.argname)
  296. elif isinstance(param, ast.Return):
  297. part = block.tags.get(TAG_RETURNS)
  298. else:
  299. part = None
  300. if part.position:
  301. return part.position
  302. return block.position
  303. def _check_array_element_type(self, array, annotations):
  304. array_type = array.array_type
  305. element_type = array.element_type
  306. # GPtrArrays are allowed to contain non basic types
  307. # (except enums and flags) or basic types that are
  308. # as big as a gpointer
  309. if array_type == ast.Array.GLIB_PTRARRAY:
  310. if ((element_type in ast.BASIC_GIR_TYPES and element_type not in ast.POINTER_TYPES)
  311. or isinstance(element_type, (ast.Enum, ast.Bitfield))):
  312. message.warn("invalid (element-type) for a GPtrArray, "
  313. "must be a pointer", annotations.position)
  314. # GByteArrays have (element-type) guint8 by default
  315. if array_type == ast.Array.GLIB_BYTEARRAY:
  316. if element_type == ast.TYPE_ANY:
  317. array.element_type = ast.TYPE_UINT8
  318. elif element_type not in [ast.TYPE_UINT8, ast.TYPE_INT8, ast.TYPE_CHAR]:
  319. message.warn("invalid (element-type) for a GByteArray, "
  320. "must be one of guint8, gint8 or gchar",
  321. annotations.position)
  322. def _apply_annotations_array(self, parent, node, annotations):
  323. element_type_options = annotations.get(ANN_ELEMENT_TYPE)
  324. if element_type_options:
  325. element_type_node = self._resolve(element_type_options[0],
  326. node.type, node, parent)
  327. elif isinstance(node.type, ast.Array):
  328. element_type_node = node.type.element_type
  329. else:
  330. # We're assuming here that Foo* with an (array) annotation
  331. # and no (element-type) means array of Foo
  332. element_type_node = node.type.clone()
  333. # The element's ctype is the array's dereferenced
  334. if element_type_node.ctype is not None and element_type_node.ctype.endswith('*'):
  335. element_type_node.ctype = element_type_node.ctype[:-1]
  336. if isinstance(node.type, ast.Array):
  337. array_type = node.type.array_type
  338. else:
  339. array_type = None
  340. array_options = annotations.get(ANN_ARRAY)
  341. container_type = ast.Array(array_type, element_type_node, ctype=node.type.ctype,
  342. is_const=node.type.is_const)
  343. if array_options.get(OPT_ARRAY_ZERO_TERMINATED, '0') == '0':
  344. container_type.zeroterminated = False
  345. else:
  346. if (OPT_ARRAY_ZERO_TERMINATED in array_options
  347. or array_options.get(OPT_ARRAY_ZERO_TERMINATED) == '1'):
  348. container_type.zeroterminated = True
  349. else:
  350. container_type.zeroterminated = False
  351. length = array_options.get(OPT_ARRAY_LENGTH)
  352. if length:
  353. if isinstance(parent, ast.Compound):
  354. paramname = self._get_validate_field_name(parent, length, node)
  355. else:
  356. paramname = self._get_validate_parameter_name(parent, length, node)
  357. if paramname:
  358. param = parent.get_parameter(paramname)
  359. param.direction = node.direction
  360. if param.direction == ast.PARAM_DIRECTION_OUT:
  361. param.transfer = ast.PARAM_TRANSFER_FULL
  362. if paramname:
  363. container_type.length_param_name = paramname
  364. fixed = array_options.get(OPT_ARRAY_FIXED_SIZE)
  365. if fixed:
  366. try:
  367. container_type.size = int(fixed)
  368. except (TypeError, ValueError):
  369. # Already warned in annotationparser.py
  370. return
  371. node.type = container_type
  372. def _apply_annotations_element_type(self, parent, node, annotations):
  373. element_type_options = annotations.get(ANN_ELEMENT_TYPE)
  374. if element_type_options is None:
  375. return
  376. if isinstance(node.type, ast.List):
  377. if len(element_type_options) != 1:
  378. message.warn(
  379. '"element-type" annotation for a list must have exactly '
  380. 'one option, not %d options' % (len(element_type_options), ),
  381. annotations.position)
  382. return
  383. node.type.element_type = self._resolve(element_type_options[0],
  384. node.type, node, parent)
  385. elif isinstance(node.type, ast.Map):
  386. if len(element_type_options) != 2:
  387. message.warn(
  388. '"element-type" annotation for a hash table must have exactly '
  389. 'two options, not %d option(s)' % (len(element_type_options), ),
  390. annotations.position)
  391. return
  392. node.type.key_type = self._resolve(element_type_options[0],
  393. node.type, node, parent)
  394. node.type.value_type = self._resolve(element_type_options[1],
  395. node.type, node, parent)
  396. elif isinstance(node.type, ast.Array):
  397. if len(element_type_options) != 1:
  398. message.warn(
  399. '"element-type" annotation for an array must have exactly '
  400. 'one option, not %d options' % (len(element_type_options), ),
  401. annotations.position)
  402. return
  403. node.type.element_type = self._resolve(element_type_options[0],
  404. node.type, node, parent)
  405. else:
  406. message.warn(
  407. "Unknown container %r for element-type annotation" % (node.type, ),
  408. annotations.position)
  409. def _get_transfer_default_param(self, parent, node):
  410. if node.direction in [ast.PARAM_DIRECTION_INOUT,
  411. ast.PARAM_DIRECTION_OUT]:
  412. if node.caller_allocates:
  413. return ast.PARAM_TRANSFER_NONE
  414. return ast.PARAM_TRANSFER_FULL
  415. return ast.PARAM_TRANSFER_NONE
  416. def _get_transfer_default_returntype_basic(self, typeval):
  417. if (typeval.is_equiv(ast.BASIC_GIR_TYPES)
  418. or typeval.is_const
  419. or typeval.is_equiv((ast.TYPE_ANY, ast.TYPE_NONE))):
  420. return ast.PARAM_TRANSFER_NONE
  421. elif typeval.is_equiv(ast.TYPE_STRING):
  422. # Non-const strings default to FULL
  423. return ast.PARAM_TRANSFER_FULL
  424. elif typeval.target_fundamental:
  425. # This looks like just GType right now
  426. return None
  427. return None
  428. def _is_gi_subclass(self, typeval, supercls_type):
  429. cls = self._transformer.lookup_typenode(typeval)
  430. assert cls, str(typeval)
  431. supercls = self._transformer.lookup_typenode(supercls_type)
  432. assert supercls
  433. if cls is supercls:
  434. return True
  435. if cls.parent_type and cls.parent_type.target_giname != 'GObject.Object':
  436. return self._is_gi_subclass(cls.parent_type, supercls_type)
  437. return False
  438. def _get_transfer_default_return(self, parent, node):
  439. typeval = node.type
  440. basic = self._get_transfer_default_returntype_basic(typeval)
  441. if basic:
  442. return basic
  443. if not typeval.target_giname:
  444. return None
  445. target = self._transformer.lookup_typenode(typeval)
  446. if isinstance(target, ast.Alias):
  447. return self._get_transfer_default_returntype_basic(target.target)
  448. elif (isinstance(target, ast.Boxed)
  449. or (isinstance(target, (ast.Record, ast.Union))
  450. and (target.gtype_name is not None or target.foreign))):
  451. return ast.PARAM_TRANSFER_FULL
  452. elif isinstance(target, (ast.Enum, ast.Bitfield)):
  453. return ast.PARAM_TRANSFER_NONE
  454. # Handle constructors specially here
  455. elif isinstance(parent, ast.Function) and parent.is_constructor:
  456. if isinstance(target, ast.Class):
  457. initially_unowned_type = ast.Type(target_giname='GObject.InitiallyUnowned')
  458. try:
  459. initially_unowned = self._transformer.lookup_typenode(initially_unowned_type)
  460. except KeyError as e:
  461. message.error_node(node, "constructor found but GObject is not in includes")
  462. return None
  463. if initially_unowned and self._is_gi_subclass(typeval, initially_unowned_type):
  464. return ast.PARAM_TRANSFER_NONE
  465. else:
  466. return ast.PARAM_TRANSFER_FULL
  467. elif isinstance(target, (ast.Record, ast.Union)):
  468. return ast.PARAM_TRANSFER_FULL
  469. else:
  470. raise AssertionError("Invalid constructor")
  471. elif isinstance(target, (ast.Class, ast.Record, ast.Union)):
  472. # Explicitly no default for these
  473. return None
  474. else:
  475. return None
  476. def _get_transfer_default(self, parent, node):
  477. if node.type.is_equiv(ast.TYPE_NONE) or isinstance(node.type, ast.Varargs):
  478. return ast.PARAM_TRANSFER_NONE
  479. elif isinstance(node, ast.Parameter):
  480. return self._get_transfer_default_param(parent, node)
  481. elif isinstance(node, ast.Return):
  482. return self._get_transfer_default_return(parent, node)
  483. elif isinstance(node, ast.Field):
  484. return ast.PARAM_TRANSFER_NONE
  485. elif isinstance(node, ast.Property):
  486. return ast.PARAM_TRANSFER_NONE
  487. else:
  488. raise AssertionError(node)
  489. def _is_pointer_type(self, node, annotations):
  490. if (not isinstance(node, ast.Return) and
  491. node.direction in (ast.PARAM_DIRECTION_OUT,
  492. ast.PARAM_DIRECTION_INOUT)):
  493. return True
  494. target = self._transformer.lookup_typenode(node.type)
  495. target = self._transformer.resolve_aliases(target)
  496. target = node.type if target is None else target
  497. return (not isinstance(target, ast.Type) or
  498. target not in ast.BASIC_TYPES or
  499. target.ctype.endswith('*'))
  500. def _apply_transfer_annotation(self, parent, node, annotations):
  501. transfer_annotation = annotations.get(ANN_TRANSFER)
  502. if not transfer_annotation or len(transfer_annotation) != 1:
  503. return
  504. transfer = transfer_annotation[0]
  505. target = self._transformer.lookup_typenode(node.type)
  506. target = self._transformer.resolve_aliases(target)
  507. target = node.type if target is None else target
  508. node_type = target if isinstance(target, ast.Type) else node.type
  509. if transfer == OPT_TRANSFER_FLOATING:
  510. transfer = OPT_TRANSFER_NONE
  511. if (not isinstance(target, (ast.Class, ast.Interface))
  512. and node_type.target_giname != 'GLib.Variant'):
  513. message.warn('invalid "transfer" annotation for {0}: '
  514. 'only valid for object and GVariant types'.format(target),
  515. annotations.position)
  516. return
  517. elif transfer == OPT_TRANSFER_CONTAINER:
  518. if (ANN_ARRAY not in annotations and
  519. not isinstance(target, (ast.Array, ast.List, ast.Map))):
  520. message.warn('invalid "transfer" annotation for {0}: '
  521. 'only valid for container types'.format(target),
  522. annotations.position)
  523. return
  524. elif (not self._is_pointer_type(node, annotations) and
  525. node_type not in (ast.TYPE_STRING, ast.TYPE_FILENAME) and
  526. not isinstance(target, (ast.Array, ast.List, ast.Map,
  527. ast.Record, ast.Compound, ast.Boxed,
  528. ast.Class, ast.Interface))):
  529. message.warn('invalid "transfer" annotation for {0}: '
  530. 'only valid for array, struct, union, boxed, '
  531. 'object and interface types'.format(target),
  532. annotations.position)
  533. return
  534. node.transfer = transfer
  535. def _apply_annotations_param_ret_common(self, parent, node, tag):
  536. annotations = tag.annotations if tag else {}
  537. type_annotation = annotations.get(ANN_TYPE)
  538. if type_annotation:
  539. node.type = self._resolve_toplevel(type_annotation[0],
  540. node.type, node, parent)
  541. caller_allocates = False
  542. annotated_direction = None
  543. if ANN_INOUT in annotations:
  544. annotated_direction = ast.PARAM_DIRECTION_INOUT
  545. elif ANN_OUT in annotations:
  546. annotated_direction = ast.PARAM_DIRECTION_OUT
  547. options = annotations[ANN_OUT]
  548. if len(options) == 0:
  549. if node.type.target_giname and node.type.ctype:
  550. target = self._transformer.lookup_giname(node.type.target_giname)
  551. target = self._transformer.resolve_aliases(target)
  552. has_double_indirection = '**' in node.type.ctype
  553. is_structure_or_union = isinstance(target, (ast.Record, ast.Union))
  554. caller_allocates = (not has_double_indirection and is_structure_or_union)
  555. else:
  556. caller_allocates = False
  557. else:
  558. option = options[0]
  559. if option == OPT_OUT_CALLER_ALLOCATES:
  560. caller_allocates = True
  561. elif option == OPT_OUT_CALLEE_ALLOCATES:
  562. caller_allocates = False
  563. elif ANN_IN in annotations:
  564. annotated_direction = ast.PARAM_DIRECTION_IN
  565. if (annotated_direction is not None) and (annotated_direction != node.direction):
  566. node.direction = annotated_direction
  567. node.caller_allocates = caller_allocates
  568. # Also reset the transfer default if we're toggling direction
  569. node.transfer = self._get_transfer_default(parent, node)
  570. self._apply_transfer_annotation(parent, node, annotations)
  571. self._adjust_container_type(parent, node, annotations)
  572. # gpointer parameters and return values are always nullable unless:
  573. # - annotated with (type) and not also with (nullable); or
  574. # - annotated with (element-type) and not also with (nullable); or
  575. # - annotated (not nullable)
  576. # See: https://bugzilla.gnome.org/show_bug.cgi?id=719966#c22
  577. if node.type.is_equiv(ast.TYPE_ANY):
  578. node.nullable = True
  579. if ANN_NULLABLE in annotations:
  580. if self._is_pointer_type(node, annotations):
  581. node.nullable = True
  582. node.not_nullable = False
  583. else:
  584. message.warn('invalid "nullable" annotation: '
  585. 'only valid for pointer types and out parameters',
  586. annotations.position)
  587. if ANN_OPTIONAL in annotations:
  588. if (not isinstance(node, ast.Return) and
  589. node.direction == ast.PARAM_DIRECTION_OUT):
  590. node.optional = True
  591. else:
  592. message.warn('invalid "optional" annotation: '
  593. 'only valid for out parameters',
  594. annotations.position)
  595. if ANN_ALLOW_NONE in annotations:
  596. if (node.direction == ast.PARAM_DIRECTION_OUT and
  597. not isinstance(node, ast.Return)):
  598. node.optional = True
  599. elif self._is_pointer_type(node, annotations):
  600. node.nullable = True
  601. else:
  602. message.warn('invalid "allow-none" annotation: '
  603. 'only valid for pointer types and out parameters',
  604. annotations.position)
  605. if (node.direction != ast.PARAM_DIRECTION_OUT and
  606. (node.type.target_giname == 'Gio.AsyncReadyCallback' or
  607. node.type.target_giname == 'Gio.Cancellable')):
  608. node.nullable = True
  609. # Final override for nullability
  610. if ANN_NOT in annotations:
  611. node.nullable = False
  612. node.not_nullable = True
  613. if tag and tag.description:
  614. node.doc = tag.description
  615. if ANN_SKIP in annotations:
  616. node.skip = True
  617. if annotations:
  618. attributes_annotation = annotations.get(ANN_ATTRIBUTES)
  619. if attributes_annotation is not None:
  620. for key, value in attributes_annotation.items():
  621. if value:
  622. node.attributes[key] = value
  623. def _apply_annotations_annotated(self, node, block):
  624. if block is None:
  625. return
  626. if block.description:
  627. node.doc = block.description
  628. since_tag = block.tags.get(TAG_SINCE)
  629. if since_tag is not None:
  630. if since_tag.value:
  631. node.version = since_tag.value
  632. if since_tag.description:
  633. node.version_doc = since_tag.description
  634. deprecated_tag = block.tags.get(TAG_DEPRECATED)
  635. if deprecated_tag is not None:
  636. if deprecated_tag.value:
  637. node.deprecated = deprecated_tag.value
  638. if deprecated_tag.description:
  639. node.deprecated_doc = deprecated_tag.description
  640. stability_tag = block.tags.get(TAG_STABILITY)
  641. if stability_tag is not None:
  642. if stability_tag.value:
  643. node.stability = stability_tag.value
  644. if stability_tag.description:
  645. node.stability_doc = stability_tag.description
  646. attributes_annotation = block.annotations.get(ANN_ATTRIBUTES)
  647. if attributes_annotation is not None:
  648. for key, value in attributes_annotation.items():
  649. if value:
  650. node.attributes[key] = value
  651. if ANN_SKIP in block.annotations:
  652. node.skip = True
  653. if ANN_FOREIGN in block.annotations:
  654. node.foreign = True
  655. if ANN_CONSTRUCTOR in block.annotations and isinstance(node, ast.Function):
  656. node.is_constructor = True
  657. if ANN_METHOD in block.annotations:
  658. node.is_method = True
  659. def _apply_annotations_alias(self, node, chain):
  660. block = self._get_block(node)
  661. self._apply_annotations_annotated(node, block)
  662. def _apply_annotations_param(self, parent, param, tag):
  663. annotations = tag.annotations if tag else {}
  664. if isinstance(parent, (ast.Function, ast.VFunction)):
  665. scope_annotation = annotations.get(ANN_SCOPE)
  666. if scope_annotation and len(scope_annotation) == 1:
  667. param.scope = scope_annotation[0]
  668. param.transfer = ast.PARAM_TRANSFER_NONE
  669. destroy_annotation = annotations.get(ANN_DESTROY)
  670. if destroy_annotation:
  671. param.destroy_name = self._get_validate_parameter_name(parent,
  672. destroy_annotation[0],
  673. param)
  674. if param.destroy_name is not None:
  675. param.scope = ast.PARAM_SCOPE_NOTIFIED
  676. destroy_param = parent.get_parameter(param.destroy_name)
  677. # This is technically bogus; we're setting the scope on the destroy
  678. # itself. But this helps avoid tripping a warning from finaltransformer,
  679. # since we don't have a way right now to flag this callback a destroy.
  680. destroy_param.scope = ast.PARAM_SCOPE_NOTIFIED
  681. closure_annotation = annotations.get(ANN_CLOSURE)
  682. if closure_annotation and len(closure_annotation) == 1:
  683. param.closure_name = self._get_validate_parameter_name(parent,
  684. closure_annotation[0],
  685. param)
  686. elif isinstance(parent, ast.Callback):
  687. if ANN_CLOSURE in annotations:
  688. # For callbacks, (closure) appears without an
  689. # argument, and tags a parameter that is a closure. We
  690. # represent it (weirdly) in the gir and typelib by
  691. # setting param.closure_name to itself.
  692. param.closure_name = param.argname
  693. self._apply_annotations_param_ret_common(parent, param, tag)
  694. def _apply_annotations_return(self, parent, return_, block):
  695. if block:
  696. tag = block.tags.get(TAG_RETURNS)
  697. else:
  698. tag = None
  699. if tag is not None and return_.type == ast.TYPE_NONE:
  700. message.warn('%s: invalid return annotation' % (block.name,),
  701. tag.position)
  702. tag = None
  703. self._apply_annotations_param_ret_common(parent, return_, tag)
  704. def _apply_annotations_params(self, parent, params, block):
  705. declparams = set([])
  706. if parent.instance_parameter:
  707. if block:
  708. doc_param = block.params.get(parent.instance_parameter.argname)
  709. else:
  710. doc_param = None
  711. self._apply_annotations_param(parent, parent.instance_parameter, doc_param)
  712. declparams.add(parent.instance_parameter.argname)
  713. for param in params:
  714. if block:
  715. doc_param = block.params.get(param.argname)
  716. else:
  717. doc_param = None
  718. self._apply_annotations_param(parent, param, doc_param)
  719. declparams.add(param.argname)
  720. if not block:
  721. return
  722. docparams = set(block.params)
  723. unknown = docparams - declparams
  724. unused = declparams - docparams
  725. for doc_name in unknown:
  726. if len(unused) == 0:
  727. text = ''
  728. elif len(unused) == 1:
  729. (param, ) = unused
  730. text = ", should be '%s'" % (param, )
  731. else:
  732. text = ", should be one of %s" % \
  733. (', '.join("'%s'" % p for p in sorted(unused)), )
  734. param = block.params.get(doc_name)
  735. message.warn("%s: unknown parameter '%s' in documentation "
  736. "comment%s" % (block.name, doc_name, text),
  737. param.position)
  738. def _apply_annotations_callable(self, node, chain, block):
  739. self._apply_annotations_annotated(node, block)
  740. self._apply_annotations_params(node, node.parameters, block)
  741. self._apply_annotations_return(node, node.retval, block)
  742. def _apply_annotations_field(self, parent, block, field):
  743. if not block:
  744. return
  745. tag = block.params.get(field.name)
  746. if not tag:
  747. return
  748. type_annotation = tag.annotations.get(ANN_TYPE)
  749. if type_annotation:
  750. field.type = self._transformer.create_type_from_user_string(type_annotation[0])
  751. field.doc = tag.description
  752. try:
  753. self._adjust_container_type(parent, field, tag.annotations)
  754. except AttributeError as ex:
  755. print(ex)
  756. def _apply_annotations_property(self, parent, prop):
  757. prefix = self._get_annotation_name(parent)
  758. block = self._blocks.get('%s:%s' % (prefix, prop.name))
  759. self._apply_annotations_annotated(prop, block)
  760. if not block:
  761. return
  762. transfer_annotation = block.annotations.get(ANN_TRANSFER)
  763. if transfer_annotation is not None:
  764. transfer = transfer_annotation[0]
  765. if transfer == OPT_TRANSFER_FLOATING:
  766. transfer = OPT_TRANSFER_NONE
  767. prop.transfer = transfer
  768. else:
  769. prop.transfer = self._get_transfer_default(parent, prop)
  770. type_annotation = block.annotations.get(ANN_TYPE)
  771. if type_annotation:
  772. prop.type = self._resolve_toplevel(type_annotation[0], prop.type, prop, parent)
  773. def _apply_annotations_signal(self, parent, signal):
  774. names = []
  775. prefix = self._get_annotation_name(parent)
  776. block = self._blocks.get('%s::%s' % (prefix, signal.name))
  777. if block:
  778. self._apply_annotations_annotated(signal, block)
  779. # We're only attempting to name the signal parameters if
  780. # the number of parameters (@foo) is the same or greater
  781. # than the number of signal parameters
  782. if len(block.params) > len(signal.parameters):
  783. names = block.params.items()
  784. # Resolve real parameter names early, so that in later
  785. # phase we can refer to them while resolving annotations.
  786. for i, param in enumerate(signal.parameters):
  787. param.argname, tag = names[i + 1]
  788. elif len(signal.parameters) != 0:
  789. # Only warn about missing params if there are actually parameters
  790. # besides implicit self.
  791. message.warn("incorrect number of parameters in comment block, "
  792. "parameter annotations will be ignored.", block.position)
  793. for i, param in enumerate(signal.parameters):
  794. if names:
  795. name, tag = names[i + 1]
  796. if tag:
  797. type_annotation = tag.annotations.get(ANN_TYPE)
  798. if type_annotation:
  799. param.type = self._resolve_toplevel(type_annotation[0], param.type,
  800. param, parent)
  801. else:
  802. tag = None
  803. self._apply_annotations_param(signal, param, tag)
  804. self._apply_annotations_return(signal, signal.retval, block)
  805. def _apply_annotations_constant(self, node):
  806. block = self._get_block(node)
  807. if block is None:
  808. return
  809. self._apply_annotations_annotated(node, block)
  810. value_annotation = block.annotations.get(ANN_VALUE)
  811. if value_annotation:
  812. node.value = value_annotation[0]
  813. def _apply_annotations_enum_members(self, node, block):
  814. if block is None:
  815. return
  816. for m in node.members:
  817. param = block.params.get(m.symbol, None)
  818. if param and param.description:
  819. m.doc = param.description
  820. def _pass_read_annotations2(self, node, chain):
  821. if isinstance(node, ast.Function):
  822. block = self._blocks.get(node.symbol)
  823. self._apply_annotation_rename_to(node, chain, block)
  824. # Handle virtual invokers
  825. parent = chain[-1] if chain else None
  826. if (block and parent):
  827. virtual_annotation = block.annotations.get(ANN_VFUNC)
  828. if virtual_annotation:
  829. invoker_name = virtual_annotation[0]
  830. matched = False
  831. for vfunc in parent.virtual_methods:
  832. if vfunc.name == invoker_name:
  833. matched = True
  834. vfunc.invoker = node.name
  835. # Also merge in annotations
  836. self._apply_annotations_callable(vfunc, [parent], block)
  837. break
  838. if not matched:
  839. message.warn_node(node,
  840. "Virtual slot '%s' not found for '%s' annotation" % (invoker_name,
  841. ANN_VFUNC))
  842. return True
  843. def _resolve_and_filter_type_list(self, typelist):
  844. """Given a list of Type instances, return a new list of types with
  845. the ones that failed to resolve removed."""
  846. # Create a copy we'll modify
  847. new_typelist = list(typelist)
  848. for typeval in typelist:
  849. resolved = self._transformer.resolve_type(typeval)
  850. if not resolved:
  851. new_typelist.remove(typeval)
  852. return new_typelist
  853. def _pass_type_resolution(self, node, chain):
  854. if isinstance(node, ast.Alias):
  855. self._transformer.resolve_type(node.target)
  856. if isinstance(node, ast.Callable):
  857. for parameter in node.parameters:
  858. self._transformer.resolve_type(parameter.type)
  859. self._transformer.resolve_type(node.retval.type)
  860. if isinstance(node, ast.Constant):
  861. self._transformer.resolve_type(node.value_type)
  862. if isinstance(node, (ast.Class, ast.Interface, ast.Record, ast.Union)):
  863. for field in node.fields:
  864. if field.anonymous_node:
  865. pass
  866. else:
  867. self._transformer.resolve_type(field.type)
  868. if isinstance(node, (ast.Class, ast.Interface)):
  869. for parent in node.parent_chain:
  870. try:
  871. self._transformer.resolve_type(parent)
  872. except ValueError:
  873. continue
  874. target = self._transformer.lookup_typenode(parent)
  875. if target:
  876. node.parent_type = parent
  877. break
  878. else:
  879. if isinstance(node, ast.Interface):
  880. node.parent_type = ast.Type(target_giname='GObject.Object')
  881. for prop in node.properties:
  882. self._transformer.resolve_type(prop.type)
  883. for sig in node.signals:
  884. for param in sig.parameters:
  885. self._transformer.resolve_type(param.type)
  886. if isinstance(node, ast.Class):
  887. node.interfaces = self._resolve_and_filter_type_list(node.interfaces)
  888. if isinstance(node, ast.Interface):
  889. node.prerequisites = self._resolve_and_filter_type_list(node.prerequisites)
  890. return True
  891. def _pair_quarks_with_enums(self):
  892. # self._uscore_type_names is an authoritative mapping of types
  893. # to underscored versions, since it is based on get_type() methods;
  894. # but only covers enums that are registered as GObject enums.
  895. # Create a fallback mapping based on all known enums in this module.
  896. uscore_enums = {}
  897. for enum in self._namespace.values():
  898. if not isinstance(enum, ast.Enum):
  899. continue
  900. uscored = to_underscores_noprefix(enum.name).lower()
  901. uscore_enums[uscored] = enum
  902. uscore_enums[enum.name] = enum
  903. for node in self._namespace.values():
  904. if not isinstance(node, ast.ErrorQuarkFunction):
  905. continue
  906. full = node.symbol[:-len('_quark')]
  907. ns, short = self._transformer.split_csymbol(node.symbol)
  908. short = short[:-len('_quark')]
  909. if full == "g_io_error":
  910. # Special case; GIOError was already taken forcing GIOErrorEnum
  911. assert self._namespace.name == 'Gio'
  912. enum = self._namespace.get('IOErrorEnum')
  913. else:
  914. enum = self._uscore_type_names.get(short)
  915. if enum is None:
  916. enum = uscore_enums.get(short)
  917. if enum is not None:
  918. enum.error_domain = node.error_domain
  919. else:
  920. message.warn_node(node,
  921. """%s: Couldn't find corresponding enumeration""" % (node.symbol, ))
  922. def _split_uscored_by_type(self, uscored):
  923. """'uscored' should be an un-prefixed uscore string. This
  924. function searches through the namespace for the longest type which
  925. prefixes uscored, and returns (type, suffix). Example, assuming
  926. namespace Gtk, type is TextBuffer:
  927. _split_uscored_by_type(text_buffer_try_new) -> (ast.Class(TextBuffer), 'try_new')"""
  928. node = None
  929. count = 0
  930. prev_split_count = -1
  931. while True:
  932. components = uscored.rsplit('_', count)
  933. if len(components) == prev_split_count:
  934. return None
  935. prev_split_count = len(components)
  936. type_string = components[0]
  937. node = self._uscore_type_names.get(type_string)
  938. if node:
  939. return (node, '_'.join(components[1:]))
  940. count += 1
  941. def _pair_function(self, func):
  942. """Check to see whether a toplevel function should be a
  943. method or constructor of some type."""
  944. # Ignore internal symbols and type metadata functions
  945. if func.symbol.startswith('_') or func.is_type_meta_function():
  946. return
  947. (ns, subsymbol) = self._transformer.split_csymbol(func.symbol)
  948. assert ns == self._namespace
  949. if self._is_constructor(func, subsymbol):
  950. self._set_up_constructor(func, subsymbol)
  951. return
  952. elif self._is_method(func, subsymbol):
  953. self._setup_method(func, subsymbol)
  954. return
  955. elif self._pair_static_method(func, subsymbol):
  956. return
  957. def _uscored_identifier_for_type(self, typeval):
  958. """Given a Type(target_giname='Foo.BarBaz'), return 'bar_baz'."""
  959. name = typeval.get_giname()
  960. return to_underscores_noprefix(name).lower()
  961. def _is_method(self, func, subsymbol):
  962. if not func.parameters:
  963. if func.is_method:
  964. message.warn_node(func,
  965. '%s: Methods must have parameters' % (func.symbol, ))
  966. return False
  967. first = func.parameters[0]
  968. target = self._transformer.lookup_typenode(first.type)
  969. if not isinstance(target, (ast.Class, ast.Interface,
  970. ast.Record, ast.Union,
  971. ast.Boxed)):
  972. if func.is_method:
  973. message.warn_node(func,
  974. '%s: Methods must have a pointer as their first '
  975. 'parameter' % (func.symbol, ))
  976. return False
  977. if target.namespace != self._namespace:
  978. if func.is_method:
  979. message.warn_node(func,
  980. '%s: Methods must belong to the same namespace as the '
  981. 'class they belong to' % (func.symbol, ))
  982. return False
  983. if first.direction == ast.PARAM_DIRECTION_OUT:
  984. if func.is_method:
  985. message.warn_node(func,
  986. '%s: The first argument of methods cannot be an '
  987. 'out-argument' % (func.symbol, ))
  988. return False
  989. # A quick hack here...in the future we should catch C signature/GI signature
  990. # mismatches in a general way in finaltransformer
  991. if first.type.ctype is not None and first.type.ctype.count('*') > 1:
  992. return False
  993. if not func.is_method:
  994. uscored_prefix = self._get_uscored_prefix(func, subsymbol)
  995. if not subsymbol.startswith(uscored_prefix):
  996. return False
  997. return True
  998. def _setup_method(self, func, subsymbol):
  999. uscored_prefix = self._get_uscored_prefix(func, subsymbol)
  1000. target = self._transformer.lookup_typenode(func.parameters[0].type)
  1001. if not func.is_method and not subsymbol.startswith(uscored_prefix + '_'):
  1002. # Uh oh! This function starts with uscored_prefix, but not
  1003. # uscored_prefix + '_', so if we split, we're splitting on something
  1004. # which is not _
  1005. # Examples of this are g_resources_register() (splits as
  1006. # g_resource + _register) and gdk_events_get_angle() (splits as
  1007. # gdk_event + _get_angle).
  1008. # As the C name suggests, these are not methods, but for backward
  1009. # compatibility reasons we need to create a method with the old
  1010. # name, and a moved-to annotation pointing to the new variant.
  1011. newfunc = func.clone()
  1012. newfunc.moved_to = func.name
  1013. newfunc.instance_parameter = newfunc.parameters.pop(0)
  1014. subsym_idx = func.symbol.find(subsymbol)
  1015. newfunc.name = func.symbol[(subsym_idx + len(uscored_prefix) + 1):]
  1016. newfunc.is_method = True
  1017. target.methods.append(newfunc)
  1018. else:
  1019. func.instance_parameter = func.parameters.pop(0)
  1020. self._namespace.float(func)
  1021. if not func.is_method:
  1022. subsym_idx = func.symbol.find(subsymbol)
  1023. func.name = func.symbol[(subsym_idx + len(uscored_prefix) + 1):]
  1024. func.is_method = True
  1025. target.methods.append(func)
  1026. def _get_uscored_prefix(self, func, subsymbol):
  1027. # Here we check both the c_symbol_prefix and (if that fails),
  1028. # attempt to do a default uscoring of the type. The reason we
  1029. # look at a default underscore transformation is for
  1030. # gdk_window_object_get_type(), which says to us that the
  1031. # prefix is "gdk_window_object", when really it's just
  1032. # "gdk_window". Possibly need an annotation to override this.
  1033. prefix_matches = False
  1034. uscored_prefix = None
  1035. first_arg = func.parameters[0]
  1036. target = self._transformer.lookup_typenode(first_arg.type)
  1037. if hasattr(target, 'c_symbol_prefix') and target.c_symbol_prefix is not None:
  1038. prefix_matches = subsymbol.startswith(target.c_symbol_prefix)
  1039. if prefix_matches:
  1040. uscored_prefix = target.c_symbol_prefix
  1041. if not prefix_matches:
  1042. uscored_prefix = self._uscored_identifier_for_type(first_arg.type)
  1043. return uscored_prefix
  1044. def _pair_static_method(self, func, subsymbol):
  1045. split = self._split_uscored_by_type(subsymbol)
  1046. if split is None:
  1047. return False
  1048. (node, funcname) = split
  1049. if funcname == '':
  1050. return False
  1051. if isinstance(node, ast.Class):
  1052. self._namespace.float(func)
  1053. func.name = funcname
  1054. node.static_methods.append(func)
  1055. return True
  1056. elif isinstance(node, (ast.Interface, ast.Record, ast.Union,
  1057. ast.Boxed, ast.Enum, ast.Bitfield)):
  1058. # prior to the introduction of this part of the code, only
  1059. # ast.Class could have static methods. so for backwards
  1060. # compatibility, instead of removing the func from the namespace,
  1061. # leave it there and get a copy instead. modify the copy and push
  1062. # it onto static_methods. we need to copy the parameters list
  1063. # separately, because in the third pass functions are flagged as
  1064. # 'throws' depending on the presence of a GError parameter which is
  1065. # then removed from the parameters list. without the explicit
  1066. # copy, only one of the two functions would thus get flagged as
  1067. # 'throws'. clone() does this for us.
  1068. new_func = func.clone()
  1069. new_func.name = funcname
  1070. node.static_methods.append(new_func)
  1071. # flag the func as a backwards-comptability kludge (thus it will
  1072. # get pruned in the introspectable pass if introspectable=0).
  1073. func.moved_to = node.name + '.' + new_func.name
  1074. return True
  1075. return False
  1076. def _set_up_constructor(self, func, subsymbol):
  1077. self._namespace.float(func)
  1078. func.name = self._get_constructor_name(func, subsymbol)
  1079. origin_node = self._get_constructor_class(func, subsymbol)
  1080. origin_node.constructors.append(func)
  1081. func.is_constructor = True
  1082. # Constructors have default return semantics
  1083. if not func.retval.transfer:
  1084. func.retval.transfer = self._get_transfer_default_return(func,
  1085. func.retval)
  1086. def _get_constructor_class(self, func, subsymbol):
  1087. origin_node = None
  1088. split = self._split_uscored_by_type(subsymbol)
  1089. if split is None:
  1090. if func.is_constructor:
  1091. origin_node = self._transformer.lookup_typenode(func.retval.type)
  1092. else:
  1093. origin_node, _ = split
  1094. return origin_node
  1095. def _get_constructor_name(self, func, subsymbol):
  1096. name = None
  1097. split = self._split_uscored_by_type(subsymbol)
  1098. if split is None:
  1099. if func.is_constructor:
  1100. name = func.name
  1101. else:
  1102. _, name = split
  1103. return name
  1104. def _guess_constructor_by_name(self, symbol):
  1105. # Normal constructors, gtk_button_new etc
  1106. if symbol.endswith('_new'):
  1107. return True
  1108. # Alternative constructor, gtk_button_new_with_label
  1109. if '_new_' in symbol:
  1110. return True
  1111. # gtk_list_store_newv,gtk_tree_store_newv etc
  1112. if symbol.endswith('_newv'):
  1113. return True
  1114. return False
  1115. def _is_constructor(self, func, subsymbol):
  1116. # func.is_constructor will be True if we have a (constructor) annotation
  1117. if not func.is_constructor:
  1118. if not self._guess_constructor_by_name(func.symbol):
  1119. return False
  1120. target = self._transformer.lookup_typenode(func.retval.type)
  1121. if not (isinstance(target, ast.Class)
  1122. or (isinstance(target, (ast.Record, ast.Union, ast.Boxed))
  1123. and (target.get_type is not None or target.foreign))):
  1124. if func.is_constructor:
  1125. message.warn_node(func,
  1126. '%s: Constructors must return an instance of their class'
  1127. % (func.symbol, ))
  1128. return False
  1129. origin_node = self._get_constructor_class(func, subsymbol)
  1130. if origin_node is None:
  1131. if func.is_constructor:
  1132. message.warn_node(
  1133. func,
  1134. "Can't find matching type for constructor; symbol='%s'" % (func.symbol, ))
  1135. return False
  1136. # Some sanity checks; only objects and boxeds can have ctors
  1137. if not (isinstance(origin_node, ast.Class)
  1138. or (isinstance(origin_node, (ast.Record, ast.Union, ast.Boxed))
  1139. and (origin_node.get_type is not None or origin_node.foreign))):
  1140. return False
  1141. # Verify the namespace - don't want to append to foreign namespaces!
  1142. if origin_node.namespace != self._namespace:
  1143. if func.is_constructor:
  1144. message.warn_node(func,
  1145. '%s: Constructors must belong to the same namespace as the '
  1146. 'class they belong to' % (func.symbol, ))
  1147. return False
  1148. # If it takes the object as a first arg, guess it's not a constructor
  1149. if not func.is_constructor and len(func.parameters) > 0:
  1150. first_arg = self._transformer.lookup_typenode(func.parameters[0].type)
  1151. if (first_arg is not None) and first_arg.gi_name == origin_node.gi_name:
  1152. return False
  1153. if isinstance(target, ast.Class):
  1154. parent = origin_node
  1155. while parent and (not parent.gi_name == 'GObject.Object'):
  1156. if parent == target:
  1157. break
  1158. if parent.parent_type:
  1159. parent = self._transformer.lookup_typenode(parent.parent_type)
  1160. else:
  1161. parent = None
  1162. if parent is None:
  1163. message.warn_node(func,
  1164. "Return value is not superclass for constructor; "
  1165. "symbol='%s' constructed='%s' return='%s'" %
  1166. (func.symbol,
  1167. str(origin_node.create_type()),
  1168. str(func.retval.type)))
  1169. return False
  1170. else:
  1171. if origin_node != target:
  1172. message.warn_node(func,
  1173. "Constructor return type mismatch symbol='%s' "
  1174. "constructed='%s' return='%s'" %
  1175. (func.symbol,
  1176. str(origin_node.create_type()),
  1177. str(func.retval.type)))
  1178. return False
  1179. return True
  1180. def _pair_class_virtuals(self, node):
  1181. """Look for virtual methods from the class structure."""
  1182. if not node.glib_type_struct:
  1183. # https://bugzilla.gnome.org/show_bug.cgi?id=629080
  1184. # message.warn_node(node,
  1185. # "Failed to find class structure for '%s'" % (node.name, ))
  1186. return
  1187. node_type = node.create_type()
  1188. class_struct = self._transformer.lookup_typenode(node.glib_type_struct)
  1189. # Object class fields are assumed to be read-only
  1190. # (see also _introspect_object and transformer.py)
  1191. for field in class_struct.fields:
  1192. if isinstance(field, ast.Field):
  1193. field.writable = False
  1194. for field in class_struct.fields:
  1195. callback = None
  1196. if isinstance(field.anonymous_node, ast.Callback):
  1197. callback = field.anonymous_node
  1198. elif field.type is not None:
  1199. callback = self._transformer.lookup_typenode(field.type)
  1200. if not isinstance(callback, ast.Callback):
  1201. continue
  1202. else:
  1203. continue
  1204. # Check the first parameter is the object
  1205. if len(callback.parameters) == 0:
  1206. continue
  1207. firstparam_type = callback.parameters[0].type
  1208. if firstparam_type != node_type:
  1209. continue
  1210. vfunc = ast.VFunction.from_callback(field.name, callback)
  1211. vfunc.instance_parameter = callback.parameters[0]
  1212. vfunc.inherit_file_positions(callback)
  1213. prefix = self._get_annotation_name(class_struct)
  1214. block = self._blocks.get('%s::%s' % (prefix, vfunc.name))
  1215. self._apply_annotations_callable(vfunc, [node], block)
  1216. node.virtual_methods.append(vfunc)
  1217. # Take the set of virtual methods we found, and try
  1218. # to pair up with any matching methods using the
  1219. # name+signature.
  1220. for vfunc in node.virtual_methods:
  1221. for method in node.methods:
  1222. if method.name != vfunc.name:
  1223. continue
  1224. if method.retval.type != vfunc.retval.type:
  1225. continue
  1226. if len(method.parameters) != len(vfunc.parameters):
  1227. continue
  1228. for i in range(len(method.parameters)):
  1229. m_type = method.parameters[i].type
  1230. v_type = vfunc.parameters[i].type
  1231. if m_type != v_type:
  1232. continue
  1233. vfunc.invoker = method.name
  1234. # Apply any annotations we have from the invoker to
  1235. # the vfunc
  1236. block = self._blocks.get(method.symbol)
  1237. self._apply_annotations_callable(vfunc, [], block)
  1238. break
  1239. def _pass3(self, node, chain):
  1240. """Pass 3 is after we've loaded GType data and performed type
  1241. closure."""
  1242. if isinstance(node, ast.Callable):
  1243. self._pass3_callable_callbacks(node)
  1244. self._pass3_callable_throws(node)
  1245. return True
  1246. def _pass3_callable_callbacks(self, node):
  1247. """Check to see if we have anything that looks like a
  1248. callback+user_data+GDestroyNotify set."""
  1249. params = node.parameters
  1250. # First, do defaults for well-known callback types
  1251. for param in params:
  1252. argnode = self._transformer.lookup_typenode(param.type)
  1253. if isinstance(argnode, ast.Callback):
  1254. if param.type.target_giname in ('Gio.AsyncReadyCallback',
  1255. 'GLib.DestroyNotify'):
  1256. param.scope = ast.PARAM_SCOPE_ASYNC
  1257. param.transfer = ast.PARAM_TRANSFER_NONE
  1258. callback_param = None
  1259. for param in params:
  1260. argnode = self._transformer.lookup_typenode(param.type)
  1261. is_destroynotify = False
  1262. if isinstance(argnode, ast.Callback):
  1263. if param.type.target_giname == 'GLib.DestroyNotify':
  1264. is_destroynotify = True
  1265. else:
  1266. callback_param = param
  1267. continue
  1268. if callback_param is None:
  1269. continue
  1270. if is_destroynotify:
  1271. callback_param.destroy_name = param.argname
  1272. callback_param.scope = ast.PARAM_SCOPE_NOTIFIED
  1273. callback_param.transfer = ast.PARAM_TRANSFER_NONE
  1274. elif (param.type.is_equiv(ast.TYPE_ANY) and
  1275. param.argname is not None and
  1276. param.argname.endswith('data')):
  1277. callback_param.closure_name = param.argname
  1278. for param in params:
  1279. # By convention, closure user_data parameters are always nullable.
  1280. if param.closure_name is not None:
  1281. idx = node.get_parameter_index(param.closure_name)
  1282. assert idx >= 0
  1283. closure_param = params[idx]
  1284. if not closure_param.not_nullable:
  1285. closure_param.nullable = True
  1286. def _pass3_callable_throws(self, node):
  1287. """Check to see if we have anything that looks like a
  1288. callback+user_data+GDestroyNotify set."""
  1289. if not node.parameters:
  1290. return
  1291. last_param = node.parameters[-1]
  1292. # Checking type.name=='GLib.Error' generates false positives
  1293. # on methods that take a 'GError *'
  1294. if last_param.type.ctype == 'GError**':
  1295. node.parameters.pop()
  1296. node.throws = True