1# Copyright 2016 The Gemmlowp Authors. All rights reserved.
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7#    http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14"""CC code emitter.
15
16Used by generators to programatically prepare C++ code. Contains some simple
17tools that allow generating nicely indented code and do basic correctness
18checking.
19"""
20
21
22class Error(Exception):
23  """Module level error."""
24
25
26class NamespaceError(Error):
27  """Invalid namespace operation."""
28
29
30class HeaderError(Error):
31  """Invalid cc header structure."""
32
33
34class ClassError(Error):
35  """Invalid class syntax."""
36
37
38class CCEmitter(object):
39  """Emits c++ code."""
40
41  def __init__(self, debug=False):
42    self.indent = ''
43    self.debug = debug
44    self.namespaces = []
45    self.classes = []
46    self.header_name = None
47
48  def PushIndent(self):
49    self.indent += '  '
50
51  def PopIndent(self):
52    self.indent = self.indent[:-2]
53
54  def EmitIndented(self, what):
55    print self.indent + what
56
57  def EmitNewline(self):
58    print ''
59
60  def EmitPreprocessor1(self, op, param):
61    print '#%s %s' % (op, param)
62
63  def EmitPreprocessor(self, op):
64    print '#%s' % op
65
66  def EmitInclude(self, include):
67    self.EmitPreprocessor1('include', include)
68
69  def EmitAssign(self, variable, value):
70    self.EmitBinaryOp(variable, '=', value)
71
72  def EmitAssignIncrement(self, variable, value):
73    self.EmitBinaryOp(variable, '+=', value)
74
75  def EmitBinaryOp(self, operand_1, op, operand_2):
76    self.EmitCode('%s %s %s' % (operand_1, op, operand_2))
77
78  def EmitCall(self, function, params=None):
79    if not params:
80      params = []
81    self.EmitCode('%s(%s)' % (function, ', '.join(map(str, params))))
82
83  def EmitCode(self, code):
84    self.EmitIndented('%s;' % code)
85
86  def EmitCodeNoSemicolon(self, code):
87    self.EmitIndented('%s' % code)
88
89  def EmitDeclare(self, decl_type, name, value):
90    self.EmitAssign('%s %s' % (decl_type, name), value)
91
92  def EmitAssert(self, assert_expression):
93    if self.debug:
94      self.EmitCall1('assert', assert_expression)
95
96  def EmitHeaderBegin(self, header_name, includes=None):
97    if includes is None:
98      includes = []
99    if self.header_name:
100      raise HeaderError('Header already defined.')
101    self.EmitPreprocessor1('ifndef', (header_name + '_H_').upper())
102    self.EmitPreprocessor1('define', (header_name + '_H_').upper())
103    self.EmitNewline()
104    if includes:
105      for include in includes:
106        self.EmitInclude(include)
107      self.EmitNewline()
108    self.header_name = header_name
109
110  def EmitHeaderEnd(self):
111    if not self.header_name:
112      raise HeaderError('Header undefined.')
113    self.EmitPreprocessor1('endif',
114                           ' // %s' % (self.header_name + '_H_').upper())
115    self.header_name = None
116
117  def EmitMemberFunctionBegin(self, class_name, class_template_params,
118                              class_specializations, function_name,
119                              function_params, return_type):
120    """Emit member function of a template/specialized class."""
121    if class_template_params or class_specializations:
122      self.EmitIndented('template<%s>' % ', '.join(class_template_params))
123
124    if class_specializations:
125      class_name += '<%s>' % ', '.join(map(str, class_specializations))
126
127    self.EmitIndented('%s %s::%s(%s) {' % (
128        return_type, class_name, function_name,
129        ', '.join(['%s %s' % (t, n) for (t, n) in function_params])))
130    self.PushIndent()
131
132  def EmitFunctionBegin(self, function_name, params, return_type):
133    self.EmitIndented('%s %s(%s) {' %
134                      (return_type, function_name,
135                       ', '.join(['%s %s' % (t, n) for (t, n) in params])))
136    self.PushIndent()
137
138  def EmitFunctionEnd(self):
139    self.PopIndent()
140    self.EmitIndented('}')
141    self.EmitNewline()
142
143  def EmitClassBegin(self, class_name, template_params, specializations,
144                     base_classes):
145    """Emit class block header."""
146    self.classes.append(class_name)
147    if template_params or specializations:
148      self.EmitIndented('template<%s>' % ', '.join(template_params))
149
150    class_name_extended = class_name
151    if specializations:
152      class_name_extended += '<%s>' % ', '.join(map(str, specializations))
153    if base_classes:
154      class_name_extended += ' : ' + ', '.join(base_classes)
155    self.EmitIndented('class %s {' % class_name_extended)
156    self.PushIndent()
157
158  def EmitClassEnd(self):
159    if not self.classes:
160      raise ClassError('No class on stack.')
161    self.classes.pop()
162    self.PopIndent()
163    self.EmitIndented('};')
164    self.EmitNewline()
165
166  def EmitAccessModifier(self, modifier):
167    if not self.classes:
168      raise ClassError('No class on stack.')
169    self.PopIndent()
170    self.EmitIndented(' %s:' % modifier)
171    self.PushIndent()
172
173  def EmitNamespaceBegin(self, namespace):
174    self.EmitCodeNoSemicolon('namespace %s {' % namespace)
175    self.namespaces.append(namespace)
176
177  def EmitNamespaceEnd(self):
178    if not self.namespaces:
179      raise NamespaceError('No namespace on stack.')
180    self.EmitCodeNoSemicolon('}  // namespace %s' % self.namespaces.pop())
181
182  def EmitComment(self, comment):
183    self.EmitIndented('// ' + comment)
184
185  def EmitOpenBracket(self, pre_bracket=None):
186    if pre_bracket:
187      self.EmitIndented('%s {' % pre_bracket)
188    else:
189      self.EmitIndented('{')
190    self.PushIndent()
191
192  def EmitCloseBracket(self):
193    self.PopIndent()
194    self.EmitIndented('}')
195
196  def EmitSwitch(self, switch):
197    self.EmitOpenBracket('switch (%s)' % switch)
198
199  def EmitSwitchEnd(self):
200    self.EmitCloseBracket()
201
202  def EmitCase(self, value):
203    self.EmitCodeNoSemicolon('case %s:' % value)
204
205  def EmitBreak(self):
206    self.EmitCode('break')
207
208  def EmitIf(self, condition):
209    self.EmitOpenBracket('if (%s)' % condition)
210
211  def EmitElse(self):
212    self.PopIndent()
213    self.EmitCodeNoSemicolon('} else {')
214    self.PushIndent()
215
216  def EmitEndif(self):
217    self.EmitCloseBracket()
218
219  def Scope(self, scope, value):
220    return '%s::%s' % (scope, value)
221