introspectablepass.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  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. from . import ast
  24. from . import message
  25. from .annotationparser import TAG_RETURNS
  26. class IntrospectablePass(object):
  27. def __init__(self, transformer, blocks):
  28. self._transformer = transformer
  29. self._namespace = transformer.namespace
  30. self._blocks = blocks
  31. # Public API
  32. def validate(self):
  33. self._namespace.walk(self._introspectable_alias_analysis)
  34. self._namespace.walk(self._propagate_callable_skips)
  35. self._namespace.walk(self._analyze_node)
  36. self._namespace.walk(self._introspectable_callable_analysis)
  37. self._namespace.walk(self._introspectable_callable_analysis)
  38. self._namespace.walk(self._introspectable_pass3)
  39. self._namespace.walk(self._remove_non_reachable_backcompat_copies)
  40. def _parameter_warning(self, parent, param, text, position=None):
  41. # Suppress VFunctions and Callbacks warnings for now
  42. # they cause more problems then they are worth
  43. if isinstance(parent, (ast.VFunction, ast.Callback)):
  44. return
  45. block = None
  46. if hasattr(parent, 'symbol'):
  47. prefix = '%s: ' % (parent.symbol, )
  48. block = self._blocks.get(parent.symbol)
  49. if block:
  50. position = block.position
  51. else:
  52. prefix = ''
  53. if isinstance(param, ast.Parameter):
  54. context = "argument %s: " % (param.argname, )
  55. else:
  56. context = "return value: "
  57. if block:
  58. return_tag = block.tags.get(TAG_RETURNS)
  59. if return_tag:
  60. position = return_tag.position
  61. message.warn_node(parent, prefix + context + text,
  62. positions=position)
  63. def _introspectable_param_analysis(self, parent, node):
  64. is_return = isinstance(node, ast.Return)
  65. is_parameter = isinstance(node, ast.Parameter)
  66. assert is_return or is_parameter
  67. if node.type.target_giname is not None:
  68. target = self._transformer.lookup_typenode(node.type)
  69. else:
  70. target = None
  71. if node.skip:
  72. return
  73. if not node.type.resolved:
  74. self._parameter_warning(parent, node,
  75. "Unresolved type: %r" % (node.type.unresolved_string, ))
  76. parent.introspectable = False
  77. return
  78. if isinstance(node.type, ast.Varargs):
  79. parent.introspectable = False
  80. return
  81. if (isinstance(node.type, (ast.List, ast.Array))
  82. and node.type.element_type == ast.TYPE_ANY):
  83. self._parameter_warning(parent, node, "Missing (element-type) annotation")
  84. parent.introspectable = False
  85. return
  86. if (is_parameter
  87. and isinstance(target, ast.Callback)
  88. and node.type.target_giname not in ('GLib.DestroyNotify', 'Gio.AsyncReadyCallback')
  89. and node.scope is None):
  90. self._parameter_warning(
  91. parent,
  92. node,
  93. "Missing (scope) annotation for callback without "
  94. "GDestroyNotify (valid: %s, %s)" % (ast.PARAM_SCOPE_CALL, ast.PARAM_SCOPE_ASYNC))
  95. parent.introspectable = False
  96. return
  97. if is_return and isinstance(target, ast.Callback):
  98. self._parameter_warning(parent, node, "Callbacks cannot be return values; use (skip)")
  99. parent.introspectable = False
  100. return
  101. if (is_return
  102. and isinstance(target, (ast.Record, ast.Union))
  103. and target.get_type is None
  104. and not target.foreign):
  105. if node.transfer != ast.PARAM_TRANSFER_NONE:
  106. self._parameter_warning(
  107. parent, node,
  108. "Invalid non-constant return of bare structure or union; "
  109. "register as boxed type or (skip)")
  110. parent.introspectable = False
  111. return
  112. if node.transfer is None:
  113. self._parameter_warning(parent, node, "Missing (transfer) annotation")
  114. parent.introspectable = False
  115. return
  116. def _type_is_introspectable(self, typeval, warn=False):
  117. if not typeval.resolved:
  118. return False
  119. if isinstance(typeval, ast.TypeUnknown):
  120. return False
  121. if isinstance(typeval, (ast.Array, ast.List)):
  122. return self._type_is_introspectable(typeval.element_type)
  123. elif isinstance(typeval, ast.Map):
  124. return (self._type_is_introspectable(typeval.key_type)
  125. and self._type_is_introspectable(typeval.value_type))
  126. if typeval.target_foreign:
  127. return True
  128. if typeval.target_fundamental:
  129. if typeval.is_equiv(ast.TYPE_VALIST):
  130. return False
  131. # These are not introspectable pending us adding
  132. # larger type tags to the typelib (in theory these could
  133. # be 128 bit or larger)
  134. elif typeval.is_equiv((ast.TYPE_LONG_LONG, ast.TYPE_LONG_ULONG, ast.TYPE_LONG_DOUBLE)):
  135. return False
  136. else:
  137. return True
  138. target = self._transformer.lookup_typenode(typeval)
  139. if not target:
  140. return False
  141. return target.introspectable and (not target.skip)
  142. def _propagate_parameter_skip(self, parent, node):
  143. if node.type.target_giname is not None:
  144. target = self._transformer.lookup_typenode(node.type)
  145. if target is None:
  146. return
  147. else:
  148. return
  149. if target.skip:
  150. parent.skip = True
  151. def _introspectable_alias_analysis(self, obj, stack):
  152. if isinstance(obj, ast.Alias):
  153. if not self._type_is_introspectable(obj.target):
  154. obj.introspectable = False
  155. return True
  156. def _propagate_callable_skips(self, obj, stack):
  157. if isinstance(obj, ast.Callable):
  158. for param in obj.parameters:
  159. self._propagate_parameter_skip(obj, param)
  160. self._propagate_parameter_skip(obj, obj.retval)
  161. return True
  162. def _analyze_node(self, obj, stack):
  163. if obj.skip:
  164. return False
  165. # Our first pass for scriptability
  166. if isinstance(obj, ast.Callable):
  167. for param in obj.parameters:
  168. self._introspectable_param_analysis(obj, param)
  169. self._introspectable_param_analysis(obj, obj.retval)
  170. if isinstance(obj, (ast.Class, ast.Interface, ast.Record, ast.Union)):
  171. for field in obj.fields:
  172. if field.type:
  173. if not self._type_is_introspectable(field.type):
  174. field.introspectable = False
  175. return True
  176. def _introspectable_callable_analysis(self, obj, stack):
  177. if obj.skip:
  178. return False
  179. # Propagate introspectability of parameters to entire functions
  180. if isinstance(obj, ast.Callable):
  181. for param in obj.parameters:
  182. if not self._type_is_introspectable(param.type):
  183. obj.introspectable = False
  184. return True
  185. if not self._type_is_introspectable(obj.retval.type):
  186. obj.introspectable = False
  187. return True
  188. return True
  189. def _introspectable_pass3(self, obj, stack):
  190. if obj.skip:
  191. return False
  192. # Propagate introspectability for fields
  193. if isinstance(obj, (ast.Class, ast.Interface, ast.Record, ast.Union)):
  194. for field in obj.fields:
  195. if field.anonymous_node:
  196. if not field.anonymous_node.introspectable:
  197. field.introspectable = False
  198. else:
  199. if not self._type_is_introspectable(field.type):
  200. field.introspectable = False
  201. # Propagate introspectability for properties
  202. if isinstance(obj, (ast.Class, ast.Interface)):
  203. for prop in obj.properties:
  204. if not self._type_is_introspectable(prop.type):
  205. prop.introspectable = False
  206. for sig in obj.signals:
  207. self._introspectable_callable_analysis(sig, [obj])
  208. return True
  209. def _remove_non_reachable_backcompat_copies(self, obj, stack):
  210. if obj.skip:
  211. return False
  212. if (isinstance(obj, ast.Function) and obj.moved_to is not None):
  213. # remove functions that are not introspectable
  214. if not obj.introspectable:
  215. obj.internal_skipped = True
  216. return True