xref: /linux/tools/net/sunrpc/xdrgen/generators/union.py (revision 1fd1dc41724319406b0aff221a352a400b0ddfc5)
1#!/usr/bin/env python3
2# ex: set filetype=python:
3
4"""Generate code to handle XDR unions"""
5
6from jinja2 import Environment
7
8from generators import SourceGenerator
9from generators import create_jinja2_environment, get_jinja2_template
10
11from xdr_ast import _XdrBasic, _XdrUnion, _XdrVoid, _XdrString, get_header_name
12from xdr_ast import _XdrDeclaration, _XdrCaseSpec, public_apis, big_endian
13
14
15def emit_union_declaration(environment: Environment, node: _XdrUnion) -> None:
16    """Emit one declaration pair for an XDR union type"""
17    if node.name in public_apis:
18        template = get_jinja2_template(environment, "declaration", "close")
19        print(template.render(name=node.name))
20
21
22def emit_union_switch_spec_definition(
23    environment: Environment, node: _XdrDeclaration
24) -> None:
25    """Emit a definition for an XDR union's discriminant"""
26    assert isinstance(node, _XdrBasic)
27    template = get_jinja2_template(environment, "definition", "switch_spec")
28    print(
29        template.render(
30            name=node.name,
31            type=node.spec.type_name,
32            classifier=node.spec.c_classifier,
33        )
34    )
35
36
37def emit_union_case_spec_definition(
38    environment: Environment, node: _XdrDeclaration
39) -> None:
40    """Emit a definition for an XDR union's case arm"""
41    if isinstance(node.arm, _XdrVoid):
42        return
43    if isinstance(node.arm, _XdrString):
44        type_name = "char *"
45        classifier = ""
46    else:
47        type_name = node.arm.spec.type_name
48        classifier = node.arm.spec.c_classifier
49
50    assert isinstance(node.arm, (_XdrBasic, _XdrString))
51    template = get_jinja2_template(environment, "definition", "case_spec")
52    print(
53        template.render(
54            name=node.arm.name,
55            type=type_name,
56            classifier=classifier,
57        )
58    )
59
60
61def emit_union_definition(environment: Environment, node: _XdrUnion) -> None:
62    """Emit one XDR union definition"""
63    template = get_jinja2_template(environment, "definition", "open")
64    print(template.render(name=node.name))
65
66    emit_union_switch_spec_definition(environment, node.discriminant)
67
68    for case in node.cases:
69        emit_union_case_spec_definition(environment, case)
70
71    if node.default is not None:
72        emit_union_case_spec_definition(environment, node.default)
73
74    template = get_jinja2_template(environment, "definition", "close")
75    print(template.render(name=node.name))
76
77
78def emit_union_switch_spec_decoder(
79    environment: Environment, node: _XdrDeclaration
80) -> None:
81    """Emit a decoder for an XDR union's discriminant"""
82    assert isinstance(node, _XdrBasic)
83    template = get_jinja2_template(environment, "decoder", "switch_spec")
84    print(template.render(name=node.name, type=node.spec.type_name))
85
86
87def emit_union_arm_decoder(
88    environment: Environment, node: _XdrCaseSpec
89) -> None:
90    """Emit decoder for an XDR union's arm (data only, no case/break)"""
91
92    if isinstance(node.arm, _XdrVoid):
93        return
94    if isinstance(node.arm, _XdrString):
95        type_name = "char *"
96        classifier = ""
97    else:
98        type_name = node.arm.spec.type_name
99        classifier = node.arm.spec.c_classifier
100
101    assert isinstance(node.arm, (_XdrBasic, _XdrString))
102    template = get_jinja2_template(environment, "decoder", node.arm.template)
103    print(
104        template.render(
105            name=node.arm.name,
106            type=type_name,
107            classifier=classifier,
108        )
109    )
110
111
112def emit_union_case_spec_decoder(
113    environment: Environment, node: _XdrCaseSpec, big_endian_discriminant: bool
114) -> None:
115    """Emit decoder functions for an XDR union's case arm"""
116
117    if isinstance(node.arm, _XdrVoid):
118        return
119    if isinstance(node.arm, _XdrString):
120        type_name = "char *"
121        classifier = ""
122    else:
123        type_name = node.arm.spec.type_name
124        classifier = node.arm.spec.c_classifier
125
126    if big_endian_discriminant:
127        template = get_jinja2_template(environment, "decoder", "case_spec_be")
128    else:
129        template = get_jinja2_template(environment, "decoder", "case_spec")
130    for case in node.values:
131        print(template.render(case=case))
132
133    assert isinstance(node.arm, (_XdrBasic, _XdrString))
134    template = get_jinja2_template(environment, "decoder", node.arm.template)
135    print(
136        template.render(
137            name=node.arm.name,
138            type=type_name,
139            classifier=classifier,
140        )
141    )
142
143    template = get_jinja2_template(environment, "decoder", "break")
144    print(template.render())
145
146
147def emit_union_default_spec_decoder(environment: Environment, node: _XdrUnion) -> None:
148    """Emit a decoder function for an XDR union's default arm"""
149    default_case = node.default
150
151    # Avoid a gcc warning about a default case with boolean discriminant
152    if default_case is None and node.discriminant.spec.type_name == "bool":
153        return
154
155    template = get_jinja2_template(environment, "decoder", "default_spec")
156    print(template.render())
157
158    if default_case is None or isinstance(default_case.arm, _XdrVoid):
159        template = get_jinja2_template(environment, "decoder", "break")
160        print(template.render())
161        return
162
163    assert isinstance(default_case.arm, _XdrBasic)
164    template = get_jinja2_template(environment, "decoder", default_case.arm.template)
165    print(
166        template.render(
167            name=default_case.arm.name,
168            type=default_case.arm.spec.type_name,
169            classifier=default_case.arm.spec.c_classifier,
170        )
171    )
172
173
174def emit_union_decoder(environment: Environment, node: _XdrUnion) -> None:
175    """Emit one XDR union decoder"""
176    template = get_jinja2_template(environment, "decoder", "open")
177    print(template.render(name=node.name))
178
179    # For boolean discriminants, use if statement instead of switch
180    if node.discriminant.spec.type_name == "bool":
181        template = get_jinja2_template(environment, "decoder", "bool_spec")
182        print(template.render(name=node.discriminant.name, type=node.discriminant.spec.type_name))
183
184        # Find and emit the TRUE case
185        for case in node.cases:
186            if case.values and case.values[0] == "TRUE":
187                emit_union_arm_decoder(environment, case)
188                break
189
190        template = get_jinja2_template(environment, "decoder", "close")
191        print(template.render())
192    else:
193        emit_union_switch_spec_decoder(environment, node.discriminant)
194
195        for case in node.cases:
196            emit_union_case_spec_decoder(
197                environment,
198                case,
199                node.discriminant.spec.type_name in big_endian,
200            )
201
202        emit_union_default_spec_decoder(environment, node)
203
204        template = get_jinja2_template(environment, "decoder", "close")
205        print(template.render())
206
207
208def emit_union_switch_spec_encoder(
209    environment: Environment, node: _XdrDeclaration
210) -> None:
211    """Emit an encoder for an XDR union's discriminant"""
212    assert isinstance(node, _XdrBasic)
213    template = get_jinja2_template(environment, "encoder", "switch_spec")
214    print(template.render(name=node.name, type=node.spec.type_name))
215
216
217def emit_union_arm_encoder(
218    environment: Environment, node: _XdrCaseSpec
219) -> None:
220    """Emit encoder for an XDR union's arm (data only, no case/break)"""
221
222    if isinstance(node.arm, _XdrVoid):
223        return
224    if isinstance(node.arm, _XdrString):
225        type_name = "char *"
226    else:
227        type_name = node.arm.spec.type_name
228
229    assert isinstance(node.arm, (_XdrBasic, _XdrString))
230    template = get_jinja2_template(environment, "encoder", node.arm.template)
231    print(
232        template.render(
233            name=node.arm.name,
234            type=type_name,
235        )
236    )
237
238
239def emit_union_case_spec_encoder(
240    environment: Environment, node: _XdrCaseSpec, big_endian_discriminant: bool
241) -> None:
242    """Emit encoder functions for an XDR union's case arm"""
243
244    if isinstance(node.arm, _XdrVoid):
245        return
246    if isinstance(node.arm, _XdrString):
247        type_name = "char *"
248    else:
249        type_name = node.arm.spec.type_name
250    if big_endian_discriminant:
251        template = get_jinja2_template(environment, "encoder", "case_spec_be")
252    else:
253        template = get_jinja2_template(environment, "encoder", "case_spec")
254    for case in node.values:
255        print(template.render(case=case))
256
257    template = get_jinja2_template(environment, "encoder", node.arm.template)
258    print(
259        template.render(
260            name=node.arm.name,
261            type=type_name,
262        )
263    )
264
265    template = get_jinja2_template(environment, "encoder", "break")
266    print(template.render())
267
268
269def emit_union_default_spec_encoder(environment: Environment, node: _XdrUnion) -> None:
270    """Emit an encoder function for an XDR union's default arm"""
271    default_case = node.default
272
273    # Avoid a gcc warning about a default case with boolean discriminant
274    if default_case is None and node.discriminant.spec.type_name == "bool":
275        return
276
277    template = get_jinja2_template(environment, "encoder", "default_spec")
278    print(template.render())
279
280    if default_case is None or isinstance(default_case.arm, _XdrVoid):
281        template = get_jinja2_template(environment, "encoder", "break")
282        print(template.render())
283        return
284
285    template = get_jinja2_template(environment, "encoder", default_case.arm.template)
286    print(
287        template.render(
288            name=default_case.arm.name,
289            type=default_case.arm.spec.type_name,
290        )
291    )
292
293
294def emit_union_encoder(environment, node: _XdrUnion) -> None:
295    """Emit one XDR union encoder"""
296    template = get_jinja2_template(environment, "encoder", "open")
297    print(template.render(name=node.name))
298
299    # For boolean discriminants, use if statement instead of switch
300    if node.discriminant.spec.type_name == "bool":
301        template = get_jinja2_template(environment, "encoder", "bool_spec")
302        print(template.render(name=node.discriminant.name, type=node.discriminant.spec.type_name))
303
304        # Find and emit the TRUE case
305        for case in node.cases:
306            if case.values and case.values[0] == "TRUE":
307                emit_union_arm_encoder(environment, case)
308                break
309
310        template = get_jinja2_template(environment, "encoder", "close")
311        print(template.render())
312    else:
313        emit_union_switch_spec_encoder(environment, node.discriminant)
314
315        for case in node.cases:
316            emit_union_case_spec_encoder(
317                environment,
318                case,
319                node.discriminant.spec.type_name in big_endian,
320            )
321
322        emit_union_default_spec_encoder(environment, node)
323
324        template = get_jinja2_template(environment, "encoder", "close")
325        print(template.render())
326
327
328def emit_union_maxsize(environment: Environment, node: _XdrUnion) -> None:
329    """Emit one maxsize macro for an XDR union type"""
330    macro_name = get_header_name().upper() + "_" + node.name + "_sz"
331    template = get_jinja2_template(environment, "maxsize", "union")
332    print(
333        template.render(
334            macro=macro_name,
335            width=" + ".join(node.symbolic_width()),
336        )
337    )
338
339
340class XdrUnionGenerator(SourceGenerator):
341    """Generate source code for XDR unions"""
342
343    def __init__(self, language: str, peer: str):
344        """Initialize an instance of this class"""
345        self.environment = create_jinja2_environment(language, "union")
346        self.peer = peer
347
348    def emit_declaration(self, node: _XdrUnion) -> None:
349        """Emit one declaration pair for an XDR union"""
350        emit_union_declaration(self.environment, node)
351
352    def emit_definition(self, node: _XdrUnion) -> None:
353        """Emit one definition for an XDR union"""
354        emit_union_definition(self.environment, node)
355
356    def emit_decoder(self, node: _XdrUnion) -> None:
357        """Emit one decoder function for an XDR union"""
358        emit_union_decoder(self.environment, node)
359
360    def emit_encoder(self, node: _XdrUnion) -> None:
361        """Emit one encoder function for an XDR union"""
362        emit_union_encoder(self.environment, node)
363
364    def emit_maxsize(self, node: _XdrUnion) -> None:
365        """Emit one maxsize macro for an XDR union"""
366        emit_union_maxsize(self.environment, node)
367