You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

429 regels
14 KiB

  1. #------------------------------------------------------------------------------
  2. # pycparser: c_generator.py
  3. #
  4. # C code generator from pycparser AST nodes.
  5. #
  6. # Eli Bendersky [https://eli.thegreenplace.net/]
  7. # License: BSD
  8. #------------------------------------------------------------------------------
  9. from . import c_ast
  10. class CGenerator(object):
  11. """ Uses the same visitor pattern as c_ast.NodeVisitor, but modified to
  12. return a value from each visit method, using string accumulation in
  13. generic_visit.
  14. """
  15. def __init__(self):
  16. # Statements start with indentation of self.indent_level spaces, using
  17. # the _make_indent method
  18. #
  19. self.indent_level = 0
  20. def _make_indent(self):
  21. return ' ' * self.indent_level
  22. def visit(self, node):
  23. method = 'visit_' + node.__class__.__name__
  24. return getattr(self, method, self.generic_visit)(node)
  25. def generic_visit(self, node):
  26. #~ print('generic:', type(node))
  27. if node is None:
  28. return ''
  29. else:
  30. return ''.join(self.visit(c) for c_name, c in node.children())
  31. def visit_Constant(self, n):
  32. return n.value
  33. def visit_ID(self, n):
  34. return n.name
  35. def visit_Pragma(self, n):
  36. ret = '#pragma'
  37. if n.string:
  38. ret += ' ' + n.string
  39. return ret
  40. def visit_ArrayRef(self, n):
  41. arrref = self._parenthesize_unless_simple(n.name)
  42. return arrref + '[' + self.visit(n.subscript) + ']'
  43. def visit_StructRef(self, n):
  44. sref = self._parenthesize_unless_simple(n.name)
  45. return sref + n.type + self.visit(n.field)
  46. def visit_FuncCall(self, n):
  47. fref = self._parenthesize_unless_simple(n.name)
  48. return fref + '(' + self.visit(n.args) + ')'
  49. def visit_UnaryOp(self, n):
  50. operand = self._parenthesize_unless_simple(n.expr)
  51. if n.op == 'p++':
  52. return '%s++' % operand
  53. elif n.op == 'p--':
  54. return '%s--' % operand
  55. elif n.op == 'sizeof':
  56. # Always parenthesize the argument of sizeof since it can be
  57. # a name.
  58. return 'sizeof(%s)' % self.visit(n.expr)
  59. else:
  60. return '%s%s' % (n.op, operand)
  61. def visit_BinaryOp(self, n):
  62. lval_str = self._parenthesize_if(n.left,
  63. lambda d: not self._is_simple_node(d))
  64. rval_str = self._parenthesize_if(n.right,
  65. lambda d: not self._is_simple_node(d))
  66. return '%s %s %s' % (lval_str, n.op, rval_str)
  67. def visit_Assignment(self, n):
  68. rval_str = self._parenthesize_if(
  69. n.rvalue,
  70. lambda n: isinstance(n, c_ast.Assignment))
  71. return '%s %s %s' % (self.visit(n.lvalue), n.op, rval_str)
  72. def visit_IdentifierType(self, n):
  73. return ' '.join(n.names)
  74. def _visit_expr(self, n):
  75. if isinstance(n, c_ast.InitList):
  76. return '{' + self.visit(n) + '}'
  77. elif isinstance(n, c_ast.ExprList):
  78. return '(' + self.visit(n) + ')'
  79. else:
  80. return self.visit(n)
  81. def visit_Decl(self, n, no_type=False):
  82. # no_type is used when a Decl is part of a DeclList, where the type is
  83. # explicitly only for the first declaration in a list.
  84. #
  85. s = n.name if no_type else self._generate_decl(n)
  86. if n.bitsize: s += ' : ' + self.visit(n.bitsize)
  87. if n.init:
  88. s += ' = ' + self._visit_expr(n.init)
  89. return s
  90. def visit_DeclList(self, n):
  91. s = self.visit(n.decls[0])
  92. if len(n.decls) > 1:
  93. s += ', ' + ', '.join(self.visit_Decl(decl, no_type=True)
  94. for decl in n.decls[1:])
  95. return s
  96. def visit_Typedef(self, n):
  97. s = ''
  98. if n.storage: s += ' '.join(n.storage) + ' '
  99. s += self._generate_type(n.type)
  100. return s
  101. def visit_Cast(self, n):
  102. s = '(' + self._generate_type(n.to_type) + ')'
  103. return s + ' ' + self._parenthesize_unless_simple(n.expr)
  104. def visit_ExprList(self, n):
  105. visited_subexprs = []
  106. for expr in n.exprs:
  107. visited_subexprs.append(self._visit_expr(expr))
  108. return ', '.join(visited_subexprs)
  109. def visit_InitList(self, n):
  110. visited_subexprs = []
  111. for expr in n.exprs:
  112. visited_subexprs.append(self._visit_expr(expr))
  113. return ', '.join(visited_subexprs)
  114. def visit_Enum(self, n):
  115. return self._generate_struct_union_enum(n, name='enum')
  116. def visit_Enumerator(self, n):
  117. if not n.value:
  118. return '{indent}{name},\n'.format(
  119. indent=self._make_indent(),
  120. name=n.name,
  121. )
  122. else:
  123. return '{indent}{name} = {value},\n'.format(
  124. indent=self._make_indent(),
  125. name=n.name,
  126. value=self.visit(n.value),
  127. )
  128. def visit_FuncDef(self, n):
  129. decl = self.visit(n.decl)
  130. self.indent_level = 0
  131. body = self.visit(n.body)
  132. if n.param_decls:
  133. knrdecls = ';\n'.join(self.visit(p) for p in n.param_decls)
  134. return decl + '\n' + knrdecls + ';\n' + body + '\n'
  135. else:
  136. return decl + '\n' + body + '\n'
  137. def visit_FileAST(self, n):
  138. s = ''
  139. for ext in n.ext:
  140. if isinstance(ext, c_ast.FuncDef):
  141. s += self.visit(ext)
  142. elif isinstance(ext, c_ast.Pragma):
  143. s += self.visit(ext) + '\n'
  144. else:
  145. s += self.visit(ext) + ';\n'
  146. return s
  147. def visit_Compound(self, n):
  148. s = self._make_indent() + '{\n'
  149. self.indent_level += 2
  150. if n.block_items:
  151. s += ''.join(self._generate_stmt(stmt) for stmt in n.block_items)
  152. self.indent_level -= 2
  153. s += self._make_indent() + '}\n'
  154. return s
  155. def visit_CompoundLiteral(self, n):
  156. return '(' + self.visit(n.type) + '){' + self.visit(n.init) + '}'
  157. def visit_EmptyStatement(self, n):
  158. return ';'
  159. def visit_ParamList(self, n):
  160. return ', '.join(self.visit(param) for param in n.params)
  161. def visit_Return(self, n):
  162. s = 'return'
  163. if n.expr: s += ' ' + self.visit(n.expr)
  164. return s + ';'
  165. def visit_Break(self, n):
  166. return 'break;'
  167. def visit_Continue(self, n):
  168. return 'continue;'
  169. def visit_TernaryOp(self, n):
  170. s = '(' + self._visit_expr(n.cond) + ') ? '
  171. s += '(' + self._visit_expr(n.iftrue) + ') : '
  172. s += '(' + self._visit_expr(n.iffalse) + ')'
  173. return s
  174. def visit_If(self, n):
  175. s = 'if ('
  176. if n.cond: s += self.visit(n.cond)
  177. s += ')\n'
  178. s += self._generate_stmt(n.iftrue, add_indent=True)
  179. if n.iffalse:
  180. s += self._make_indent() + 'else\n'
  181. s += self._generate_stmt(n.iffalse, add_indent=True)
  182. return s
  183. def visit_For(self, n):
  184. s = 'for ('
  185. if n.init: s += self.visit(n.init)
  186. s += ';'
  187. if n.cond: s += ' ' + self.visit(n.cond)
  188. s += ';'
  189. if n.next: s += ' ' + self.visit(n.next)
  190. s += ')\n'
  191. s += self._generate_stmt(n.stmt, add_indent=True)
  192. return s
  193. def visit_While(self, n):
  194. s = 'while ('
  195. if n.cond: s += self.visit(n.cond)
  196. s += ')\n'
  197. s += self._generate_stmt(n.stmt, add_indent=True)
  198. return s
  199. def visit_DoWhile(self, n):
  200. s = 'do\n'
  201. s += self._generate_stmt(n.stmt, add_indent=True)
  202. s += self._make_indent() + 'while ('
  203. if n.cond: s += self.visit(n.cond)
  204. s += ');'
  205. return s
  206. def visit_Switch(self, n):
  207. s = 'switch (' + self.visit(n.cond) + ')\n'
  208. s += self._generate_stmt(n.stmt, add_indent=True)
  209. return s
  210. def visit_Case(self, n):
  211. s = 'case ' + self.visit(n.expr) + ':\n'
  212. for stmt in n.stmts:
  213. s += self._generate_stmt(stmt, add_indent=True)
  214. return s
  215. def visit_Default(self, n):
  216. s = 'default:\n'
  217. for stmt in n.stmts:
  218. s += self._generate_stmt(stmt, add_indent=True)
  219. return s
  220. def visit_Label(self, n):
  221. return n.name + ':\n' + self._generate_stmt(n.stmt)
  222. def visit_Goto(self, n):
  223. return 'goto ' + n.name + ';'
  224. def visit_EllipsisParam(self, n):
  225. return '...'
  226. def visit_Struct(self, n):
  227. return self._generate_struct_union_enum(n, 'struct')
  228. def visit_Typename(self, n):
  229. return self._generate_type(n.type)
  230. def visit_Union(self, n):
  231. return self._generate_struct_union_enum(n, 'union')
  232. def visit_NamedInitializer(self, n):
  233. s = ''
  234. for name in n.name:
  235. if isinstance(name, c_ast.ID):
  236. s += '.' + name.name
  237. else:
  238. s += '[' + self.visit(name) + ']'
  239. s += ' = ' + self._visit_expr(n.expr)
  240. return s
  241. def visit_FuncDecl(self, n):
  242. return self._generate_type(n)
  243. def _generate_struct_union_enum(self, n, name):
  244. """ Generates code for structs, unions, and enums. name should be
  245. 'struct', 'union', or 'enum'.
  246. """
  247. if name in ('struct', 'union'):
  248. members = n.decls
  249. body_function = self._generate_struct_union_body
  250. else:
  251. assert name == 'enum'
  252. members = None if n.values is None else n.values.enumerators
  253. body_function = self._generate_enum_body
  254. s = name + ' ' + (n.name or '')
  255. if members is not None:
  256. # None means no members
  257. # Empty sequence means an empty list of members
  258. s += '\n'
  259. s += self._make_indent()
  260. self.indent_level += 2
  261. s += '{\n'
  262. s += body_function(members)
  263. self.indent_level -= 2
  264. s += self._make_indent() + '}'
  265. return s
  266. def _generate_struct_union_body(self, members):
  267. return ''.join(self._generate_stmt(decl) for decl in members)
  268. def _generate_enum_body(self, members):
  269. # `[:-2] + '\n'` removes the final `,` from the enumerator list
  270. return ''.join(self.visit(value) for value in members)[:-2] + '\n'
  271. def _generate_stmt(self, n, add_indent=False):
  272. """ Generation from a statement node. This method exists as a wrapper
  273. for individual visit_* methods to handle different treatment of
  274. some statements in this context.
  275. """
  276. typ = type(n)
  277. if add_indent: self.indent_level += 2
  278. indent = self._make_indent()
  279. if add_indent: self.indent_level -= 2
  280. if typ in (
  281. c_ast.Decl, c_ast.Assignment, c_ast.Cast, c_ast.UnaryOp,
  282. c_ast.BinaryOp, c_ast.TernaryOp, c_ast.FuncCall, c_ast.ArrayRef,
  283. c_ast.StructRef, c_ast.Constant, c_ast.ID, c_ast.Typedef,
  284. c_ast.ExprList):
  285. # These can also appear in an expression context so no semicolon
  286. # is added to them automatically
  287. #
  288. return indent + self.visit(n) + ';\n'
  289. elif typ in (c_ast.Compound,):
  290. # No extra indentation required before the opening brace of a
  291. # compound - because it consists of multiple lines it has to
  292. # compute its own indentation.
  293. #
  294. return self.visit(n)
  295. else:
  296. return indent + self.visit(n) + '\n'
  297. def _generate_decl(self, n):
  298. """ Generation from a Decl node.
  299. """
  300. s = ''
  301. if n.funcspec: s = ' '.join(n.funcspec) + ' '
  302. if n.storage: s += ' '.join(n.storage) + ' '
  303. s += self._generate_type(n.type)
  304. return s
  305. def _generate_type(self, n, modifiers=[]):
  306. """ Recursive generation from a type node. n is the type node.
  307. modifiers collects the PtrDecl, ArrayDecl and FuncDecl modifiers
  308. encountered on the way down to a TypeDecl, to allow proper
  309. generation from it.
  310. """
  311. typ = type(n)
  312. #~ print(n, modifiers)
  313. if typ == c_ast.TypeDecl:
  314. s = ''
  315. if n.quals: s += ' '.join(n.quals) + ' '
  316. s += self.visit(n.type)
  317. nstr = n.declname if n.declname else ''
  318. # Resolve modifiers.
  319. # Wrap in parens to distinguish pointer to array and pointer to
  320. # function syntax.
  321. #
  322. for i, modifier in enumerate(modifiers):
  323. if isinstance(modifier, c_ast.ArrayDecl):
  324. if (i != 0 and isinstance(modifiers[i - 1], c_ast.PtrDecl)):
  325. nstr = '(' + nstr + ')'
  326. nstr += '[' + self.visit(modifier.dim) + ']'
  327. elif isinstance(modifier, c_ast.FuncDecl):
  328. if (i != 0 and isinstance(modifiers[i - 1], c_ast.PtrDecl)):
  329. nstr = '(' + nstr + ')'
  330. nstr += '(' + self.visit(modifier.args) + ')'
  331. elif isinstance(modifier, c_ast.PtrDecl):
  332. if modifier.quals:
  333. nstr = '* %s %s' % (' '.join(modifier.quals), nstr)
  334. else:
  335. nstr = '*' + nstr
  336. if nstr: s += ' ' + nstr
  337. return s
  338. elif typ == c_ast.Decl:
  339. return self._generate_decl(n.type)
  340. elif typ == c_ast.Typename:
  341. return self._generate_type(n.type)
  342. elif typ == c_ast.IdentifierType:
  343. return ' '.join(n.names) + ' '
  344. elif typ in (c_ast.ArrayDecl, c_ast.PtrDecl, c_ast.FuncDecl):
  345. return self._generate_type(n.type, modifiers + [n])
  346. else:
  347. return self.visit(n)
  348. def _parenthesize_if(self, n, condition):
  349. """ Visits 'n' and returns its string representation, parenthesized
  350. if the condition function applied to the node returns True.
  351. """
  352. s = self._visit_expr(n)
  353. if condition(n):
  354. return '(' + s + ')'
  355. else:
  356. return s
  357. def _parenthesize_unless_simple(self, n):
  358. """ Common use case for _parenthesize_if
  359. """
  360. return self._parenthesize_if(n, lambda d: not self._is_simple_node(d))
  361. def _is_simple_node(self, n):
  362. """ Returns True for nodes that are "simple" - i.e. nodes that always
  363. have higher precedence than operators.
  364. """
  365. return isinstance(n, (c_ast.Constant, c_ast.ID, c_ast.ArrayRef,
  366. c_ast.StructRef, c_ast.FuncCall))