codegen.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. # -*- Mode: Python -*-
  2. # GObject-Introspection - a framework for introspecting GObject libraries
  3. # Copyright (C) 2010 Red Hat, Inc.
  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 with_statement
  21. from __future__ import absolute_import
  22. from __future__ import division
  23. from __future__ import print_function
  24. from __future__ import unicode_literals
  25. from contextlib import contextmanager
  26. from . import ast
  27. class CCodeGenerator(object):
  28. def __init__(self, namespace,
  29. out_h_filename,
  30. out_c_filename,
  31. function_decoration=[],
  32. include_first_header=[],
  33. include_last_header=[],
  34. include_first_src=[],
  35. include_last_src=[]):
  36. self.out_h_filename = out_h_filename
  37. self.out_c_filename = out_c_filename
  38. self.function_decoration = function_decoration
  39. self.include_first_header = include_first_header
  40. self.include_last_header = include_last_header
  41. self.include_first_src = include_first_src
  42. self.include_last_src = include_last_src
  43. self._function_bodies = {}
  44. self.namespace = namespace
  45. def gen_symbol(self, name):
  46. name = name.replace(' ', '_')
  47. return '%s_%s' % (self.namespace.symbol_prefixes[0], name)
  48. def _typecontainer_to_ctype(self, param):
  49. if (isinstance(param, ast.Parameter)
  50. and param.direction in (ast.PARAM_DIRECTION_OUT, ast.PARAM_DIRECTION_INOUT)):
  51. suffix = '*'
  52. else:
  53. suffix = ''
  54. if (param.type.is_equiv((ast.TYPE_STRING, ast.TYPE_FILENAME))
  55. and param.transfer == ast.PARAM_TRANSFER_NONE):
  56. return "const gchar*" + suffix
  57. return param.type.ctype + suffix
  58. def _write_prelude(self, out, func):
  59. if self.function_decoration:
  60. out.write("""
  61. %s""" % " ".join(self.function_decoration))
  62. out.write("""
  63. %s
  64. %s (""" % (self._typecontainer_to_ctype(func.retval), func.symbol))
  65. l = len(func.parameters)
  66. if func.parameters:
  67. for i, param in enumerate(func.parameters):
  68. ctype = self._typecontainer_to_ctype(param)
  69. out.write('%s %s' % (ctype, param.argname))
  70. if i < l - 1:
  71. out.write(", ")
  72. else:
  73. out.write('void')
  74. out.write(")")
  75. def _write_prototype(self, func):
  76. self._write_prelude(self.out_h, func)
  77. self.out_h.write(";\n\n")
  78. def _write_annotation_transfer(self, node):
  79. if (node.type not in ast.BASIC_TYPES or
  80. node.type.ctype.endswith('*')):
  81. self.out_c.write(" (transfer %s)" % (node.transfer, ))
  82. def _write_docs(self, func):
  83. self.out_c.write("/**\n * %s:\n" % (func.symbol, ))
  84. for param in func.parameters:
  85. self.out_c.write(" * @%s" % (param.argname, ))
  86. if param.direction in (ast.PARAM_DIRECTION_OUT,
  87. ast.PARAM_DIRECTION_INOUT):
  88. if param.caller_allocates:
  89. allocate_string = ' caller-allocates'
  90. else:
  91. allocate_string = ''
  92. self.out_c.write(": (%s%s) " % (param.direction,
  93. allocate_string))
  94. self._write_annotation_transfer(param)
  95. self.out_c.write(":\n")
  96. self.out_c.write(' *\n')
  97. self.out_c.write(' * Undocumented.')
  98. if func.retval.type != ast.TYPE_NONE:
  99. self.out_c.write('\n *\n')
  100. self.out_c.write(' * Returns: ')
  101. self._write_annotation_transfer(func.retval)
  102. self.out_c.write('\n */')
  103. @contextmanager
  104. def _function(self, func):
  105. self._write_prototype(func)
  106. self._write_docs(func)
  107. self._write_prelude(self.out_c, func)
  108. self.out_c.write("\n{\n")
  109. yield
  110. self.out_c.write("}\n\n")
  111. def _codegen_start(self):
  112. warning = '/* GENERATED BY testcodegen.py; DO NOT EDIT */\n\n'
  113. self.out_h.write(warning)
  114. nsupper = self.namespace.name.upper()
  115. for header in self.include_first_header:
  116. self.out_h.write("""#include "%s"\n""" % header)
  117. self.out_h.write("""
  118. #ifndef __%s_H__
  119. #define __%s_H__
  120. #include <glib-object.h>
  121. """ % (nsupper, nsupper))
  122. for header in self.include_last_header:
  123. self.out_h.write("""#include "%s"\n""" % header)
  124. self.out_c.write(warning)
  125. for header in self.include_first_src:
  126. self.out_c.write("""#include "%s"\n""" % header)
  127. self.out_c.write("""#include "%s"\n\n""" % (self.out_h_filename, ))
  128. for header in self.include_last_src:
  129. self.out_c.write("""#include "%s"\n""" % header)
  130. def _codegen_end(self):
  131. self.out_h.write("""#endif\n""")
  132. self.out_h.close()
  133. self.out_c.close()
  134. def set_function_body(self, node, body):
  135. assert isinstance(node, ast.Function)
  136. self._function_bodies[node] = body
  137. def codegen(self):
  138. self.out_h = open(self.out_h_filename, 'w')
  139. self.out_c = open(self.out_c_filename, 'w')
  140. self._codegen_start()
  141. for node in self.namespace.values():
  142. if isinstance(node, ast.Function):
  143. with self._function(node):
  144. body = self._function_bodies.get(node)
  145. if not body:
  146. body = ''
  147. self.out_c.write(body)
  148. self._codegen_end()