-
Notifications
You must be signed in to change notification settings - Fork 47
/
Copy pathcodegen.py
4086 lines (3631 loc) · 160 KB
/
codegen.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
# © 2021-2023 Intel Corporation
# SPDX-License-Identifier: MPL-2.0
import re
from abc import ABC, abstractmethod, abstractproperty
import operator
import contextlib
from functools import reduce
import itertools
import os
import math
from . import objects, crep, ctree, ast, int_register, logging, serialize
from . import dmlparse, output
from .logging import *
from .expr import *
from .ctree import *
from .expr_util import *
from .symtab import *
from .messages import *
from .output import out
from .types import *
from .set import Set
from .slotsmeta import auto_init
import dml.globals
__all__ = (
'mark_method_exported',
'mark_method_referenced',
'exported_methods',
'statically_exported_methods',
'method_queue',
'saved_method_variables',
'get_type_sequence_info',
'eval_arraylen',
'eval_type',
'eval_method_inp',
'eval_method_outp',
'eval_initializer',
'get_initializer',
'codegen_expression',
'codegen_expression_maybe_nonvalue',
'NoFailure',
'InitFailure',
'LogFailure',
'CatchFailure',
'ReturnFailure',
'IgnoreFailure',
'c_rettype',
'c_inargs',
'method_instance',
'require_fully_typed',
'codegen_method_func',
'codegen_method',
'mkcall_method',
'codegen_call',
'codegen_call_byname',
'codegen_call_expr',
'codegen_call_traitmethod',
'codegen_inline',
'codegen_inline_byname',
#'compound',
'declarations',
)
class UnknownMethod(Exception):
def __init__(self, obj, method):
Exception.__init__(self)
self.obj = obj
self.method = method
def __str__(self):
return 'Unknown method in %s : %s' % (repr(self.obj), self.method)
gensym_counter = 0
def gensym(prefix = '_gensym'):
global gensym_counter
gensym_counter += 1
return prefix + str(gensym_counter)
# Keep track of which methods are referenced, thus needing generated code.
referenced_methods = {}
method_queue = []
exported_methods = {}
statically_exported_methods = Set()
# Saved variables in methods, method->list of symbols
saved_method_variables = {}
class LoopContext:
'''Class representing loop contexts within methods.
LoopContext.current is the nearest enclosing loop context at any given
point during code generation.'''
current = None
def __enter__(self):
self.prev = LoopContext.current
LoopContext.current = self
def __exit__(self, exc_type, exc_val, exc_tb):
assert LoopContext.current is self
LoopContext.current = self.prev
class CLoopContext(LoopContext):
'''DML loop context corresponding to a C loop'''
def break_(self, site):
return [mkBreak(site)]
class NoLoopContext(LoopContext):
'''DML loop context corresponding to an inlined method call.
This is used to delimit the loop stack between the method containing
the inline method call and the method called.'''
def break_(self, site):
raise EBREAK(site)
class GotoLoopContext(LoopContext):
'''DML loop context not directly corresponding to a single C loop
statement. Uses of `break` is codegen:d as a goto past the loop.'''
count = 0
def __init__(self):
self.used = False
self.id = GotoLoopContext.count
self.label = f'_abstractloopbreak{self.id}'
GotoLoopContext.count += 1
def break_(self, site):
self.used = True
return [mkGotoBreak(site, self.label)]
class Failure(ABC):
'''Handle exceptions failure handling is supposed to handle the various kind of
functions that are generated, with different ways of signaling
success/failure.'''
allowed = True
fail_stack = []
def __init__(self, site):
self.site = site
def __enter__(self):
self.fail_stack.append(self)
def __exit__(self, exc_type, exc_val, exc_tb):
top = self.fail_stack.pop()
assert top is self
@abstractmethod
def fail(self, site):
'''Return code that is used to handle an exception'''
class NoFailure(Failure):
'''Disallow exceptions to be thrown in compile time'''
allowed = False
def fail(self, site):
raise ICE(site, "fail not allowed here")
class InitFailure(Failure):
'''Returns NULL from init_object on exception'''
def fail(self, site):
return mkReturn(site, mkIntegerLiteral(site, 0))
class LogFailure(Failure):
'''Log exceptions as errors, without aborting execution'''
def __init__(self, site, node, indices):
super(LogFailure, self).__init__(site)
self.node = node
self.indices = indices
def fail(self, site):
return log_statement(site, self.node, self.indices, "error",
mkIntegerLiteral(site, 1), None,
"Uncaught DML exception")
class ReturnFailure(Failure):
'''Generate boolean return statements to signal success. False
means success.'''
def fail(self, site):
return mkReturn(site, mkBoolConstant(site, True))
def nofail(self):
'''Return code that is used to leave the method successfully'''
return mkReturn(self.site, mkBoolConstant(self.site, False))
class CatchFailure(Failure):
'''Handle exceptions by re-throwing them, which in C means jumping to
a label.'''
def __init__(self, site, method_node):
Failure.__init__(self, site)
self.label = None
self.method = method_node
def fail(self, site):
if not self.label:
self.label = gensym('throw')
return mkThrow(site, self.label)
class IgnoreFailure(Failure):
'''Ignore exceptions'''
def fail(self, site):
return mkNull(site)
class ExitHandler(ABC):
current = None
def __enter__(self):
self.prev = ExitHandler.current
ExitHandler.current = self
def __exit__(self, exc_type, exc_val, exc_tb):
assert ExitHandler.current is self
ExitHandler.current = self.prev
@abstractmethod
def codegen_exit(self, site, retvals):
'''Return a statement for returning from the current method. retvals
is None in DML 1.2, and a list of return values in DML 1.4.'''
pass
def codegen_exit(site, retvals):
return ExitHandler.current.codegen_exit(site, retvals)
class GotoExit(ExitHandler):
count = 0
def __init__(self):
self.used = False
GotoExit.count += 1
self.label = 'exit%d' % (self.count,)
class GotoExit_dml12(GotoExit):
def codegen_exit(self, site, retvals):
assert retvals is None
self.used = True
return mkGoto(site, self.label)
class GotoExit_dml14(GotoExit):
def __init__(self, outvars):
self.outvars = outvars
super(GotoExit_dml14, self).__init__()
def codegen_exit(self, site, retvals):
assert retvals is not None
assert len(retvals) == len(self.outvars)
self.used = True
return mkCompound(
site,
[mkCopyData(site, val, out)
for (out, val) in zip(self.outvars, retvals)]
+ [mkReturnFromInline(site, self.label)])
class ReturnExit(ExitHandler):
def __init__(self, outp, throws):
self.outp = outp
self.throws = throws
def codegen_exit(self, site, retvals):
assert retvals is not None, 'dml 1.2/1.4 mixup'
return codegen_return(site, self.outp, self.throws, retvals)
def memoized_return_failure_leave(site, make_ref, failed):
ran = make_ref('ran', TInt(8, True))
threw = make_ref('threw', TBool())
stmts = [mkCopyData(site, mkIntegerLiteral(site, 1), ran),
mkCopyData(site, mkBoolConstant(site, failed), threw),
mkReturn(site, mkBoolConstant(site, failed))]
return stmts
class MemoizedReturnFailure(Failure):
'''Generate boolean return statements to signal success. False
means success.'''
def __init__(self, site, make_ref):
self.site = site
self.make_ref = make_ref
def fail(self, site):
return mkCompound(
site,
memoized_return_failure_leave(site, self.make_ref, True))
def nofail(self):
'''Return code that is used to leave the method successfully'''
return mkCompound(
self.site,
memoized_return_failure_leave(self.site, self.make_ref, False))
class MemoizedReturnExit(ExitHandler):
def __init__(self, site, outp, throws, make_ref):
self.site = site
self.outp = outp
self.throws = throws
self.make_ref = make_ref
def codegen_exit(self, site, retvals):
ran = self.make_ref('ran', TInt(8, True))
stmts = [mkCopyData(site, mkIntegerLiteral(site, 1), ran)]
if self.throws:
threw = self.make_ref('threw', TBool())
stmts.append(mkCopyData(site, mkBoolConstant(site, False), threw))
targets = []
for ((name, typ), val) in zip(self.outp, retvals):
target = self.make_ref(f'p_{name}', typ)
stmts.append(mkCopyData(site, val, target))
targets.append(target)
stmts.append(codegen_return(site, self.outp, self.throws, targets))
return mkCompound(site, stmts)
class Memoization(ABC):
@abstractmethod
def prelude(self):
pass
@abstractmethod
def exit_handler(self):
pass
@abstractmethod
def fail_handler(self):
pass
class IndependentMemoized(Memoization):
def __init__(self, func):
assert func.method.independent
self.func = func
self.method = func.method
def make_ref(self, ref, typ):
indexing = ''.join([f'[_idx{i}]'
for i in range(self.method.dimensions)])
return mkLit(self.method.site, f'_memo{indexing}.{ref}', typ)
def prelude(self):
struct_body = ''.join(
f'{typ.declaration(name)}; '
for (name, typ) in (
[('ran', TInt(8, True))]
+ ([('threw', TBool())] if self.func.throws else [])
+ [(f'p_{name}', typ) for (name, typ) in self.func.outp]))
array_defs = ''.join([f'[{i}]' for i in self.method.dimsizes])
struct_var_def = mkInline(
self.method.site,
f'static struct {{{struct_body}}} _memo{array_defs};')
return [struct_var_def] + memoization_common_prelude(
self.method.logname_anonymized(), self.method.site, self.func.outp,
self.func.throws, self.make_ref)
def exit_handler(self):
return MemoizedReturnExit(self.method.site, self.func.outp,
self.func.throws, self.make_ref)
def fail_handler(self):
return (MemoizedReturnFailure(self.method.rbrace_site, self.make_ref)
if self.func.throws else NoFailure(self.method.site))
class SharedIndependentMemoized(Memoization):
def __init__(self, method):
assert method.independent
self.method = method
def make_ref(self, ref, typ):
traitname = cident(self.method.trait.name)
return mkLit(self.method.site,
f'((struct _{traitname} *) _{traitname}.trait)'
+ f'->_memo_outs_{self.method.name}'
+ f'[_{traitname}.id.encoded_index].{ref}', typ)
def prelude(self):
return memoization_common_prelude(
self.method.name, self.method.site, self.method.outp,
self.method.throws, self.make_ref)
def exit_handler(self):
return MemoizedReturnExit(self.method.site, self.method.outp,
self.method.throws, self.make_ref)
def fail_handler(self):
return (MemoizedReturnFailure(self.method.rbrace_site, self.make_ref)
if self.method.throws else NoFailure(self.method.site))
def memoization_common_prelude(name, site, outp, throws, make_ref):
has_run_stmts = []
# Throwing is treated as a special kind of output parameter, stored through
# 'threw'. When 'ran' indicates the method has been called before to
# completion, 'threw' is retrieved to check whether the method threw or
# not. If it did, then the call completes by throwing again; otherwise, the
# cached return values are retrieved and returned.
if throws:
has_run_stmts.append(
mkIf(site, make_ref('threw', TBool()),
mkReturn(site, mkBoolConstant(site, True))))
has_run_stmts.append(
codegen_return(site, outp, throws,
[make_ref(f'p_{pname}', ptype)
for (pname, ptype) in outp]))
# 'ran' is used to check whether the function has been called or not:
# - 0: never been called before. Set to -1, and execute the body.
# Before any return, cache the results, and set 'ran' to 1.
# - 1: called to completion before. Fetch the cached results and return
# immediately without executing the rest of the body.
# - -1: called before, but not to completion. This only happens due to an
# (indirect) recursive call. Raise critical error, and then proceed
# as though ran == 0. Hopefully things will turn out ok.
ran = make_ref('ran', TInt(8, True))
unrun = [mkCase(site, mkIntegerLiteral(site, 0)),
mkCopyData(site, mkIntegerConstant(site, -1, True), ran),
mkBreak(site)]
has_run = [mkCase(site, mkIntegerLiteral(site, 1))] + has_run_stmts
running = [mkDefault(site),
mkInline(site, f'_memoized_recursion("{name}");')]
return [mkSwitch(site,
make_ref('ran', TInt(8, True)),
mkCompound(site, unrun + has_run + running))]
class TypeSequenceInfo:
"""Bookkeeping surrounding a realtyped unique type sequence"""
def __init__(self, types, uniq):
self.types = types
self.uniq = uniq
self.after_on_hooks = {}
self.struct = (TStruct({f'comp{i}': typ
for (i, typ) in enumerate(types)},
label=f'_typeseq_{self.uniq}')
if types else None)
type_keys = ', '.join(f"'{typ.key()}'" for typ in types)
self.string_key = f'({type_keys})'
def get_after_on_hook(self, aoh_prim_key, param_to_msg_comp, no_params,
create_new=False):
param_to_msg_comp_key = tuple(param_to_msg_comp.get(i)
for i in range(no_params))
try:
return self.after_on_hooks[(aoh_prim_key, param_to_msg_comp_key)]
except KeyError:
if not create_new:
raise
info = after_on_hook_info_constructors[type(aoh_prim_key)](
self, aoh_prim_key, param_to_msg_comp)
self.after_on_hooks[(aoh_prim_key, param_to_msg_comp_key)] = info
return info
def get_type_sequence_info(index, create_new=False):
typeseq = TypeSequence(index)
try:
return dml.globals.type_sequence_infos[typeseq]
except KeyError:
if create_new:
info = TypeSequenceInfo(typeseq.types,
len(dml.globals.type_sequence_infos))
dml.globals.type_sequence_infos[typeseq] = info
return info
else:
return None
class AfterInfo(ABC):
'''Information used to generate artifact corresponding to a unique usage of
the after statement. After statements are considered unique in respect to
the artifacts they need: two after statements that can share the same
generated artifacts -- thus the same AfterInfo -- are considered identical
usages of after.'''
def __init__(self, key, dimsizes):
self.key = key
self.dimsizes = dimsizes
@abstractproperty
def cident_prefix(self):
'''A prefix for C identifiers used for the naming of artifacts
generated for the unique usage of 'after' '''
@abstractproperty
def args_type(self):
'''A type that represents all information needed to execute the
callback of the 'after', excluding indices and after domains.
If no additional information is needed this should return None.
'''
@abstractproperty
def types_to_declare(self):
'''An iterable of the novel types used by the generated artifacts of
the 'after', and which must be declared.'''
@property
def cident_callback(self):
return self.cident_prefix + 'callback'
@property
def dimensions(self):
return len(self.dimsizes)
class CheckpointedAfterInfo(AfterInfo):
@abstractproperty
def string_key(self):
'''A key for the AfterInfo -- thus a key for the set of artifacts
generated for the unique usage of 'after'. This must be suitable for
use as a string in generated code -- in particular anonymization of
confidential names must be done.'''
class AfterDelayInfo(CheckpointedAfterInfo):
def __init__(self, key, dimsizes, uniq):
self.uniq = uniq
super().__init__(key, dimsizes)
@abstractmethod
def generate_callback_call(self, indices_lit, args_lit): pass
@property
def cident_prefix(self):
return f'_simple_event_{self.uniq}_'
@property
def cident_get_value(self):
if self.dimensions or self.args_type is not None:
return self.cident_prefix + 'get_value'
else:
return '_simple_event_only_domains_get_value'
@property
def cident_set_value(self):
if self.dimensions or self.args_type is not None:
return self.cident_prefix + 'set_value'
else:
return '_simple_event_only_domains_set_value'
class AfterOnHookInfo(CheckpointedAfterInfo):
def __init__(self, dimsizes, parent, typeseq_info, prim_key,
param_to_msg_comp, inp, has_serialized_args):
self.typeseq_info = typeseq_info
self.parent = parent
# TODO drop?
self.inp = inp
self.has_serialized_args = has_serialized_args
self.prim_key = prim_key
self.param_to_msg_comp = param_to_msg_comp
self.uniq = len(dml.globals.after_on_hook_infos)
dml.globals.after_on_hook_infos.append(self)
super().__init__((prim_key, self.param_to_msg_comp_key), dimsizes)
@abstractmethod
def generate_callback_call(self, indices_lit, args_lit, msg_lit): pass
@abstractmethod
def generate_args_serializer(self, site, args_expr, out_expr): pass
@abstractmethod
def generate_args_deserializer(self, site, val_expr, out_expr, error_out):
pass
@abstractproperty
def string_prim_key(self):
'''The AfterOnHookInfo key for the primary component -- the target
callback -- of the hook-based after'''
@property
def param_to_msg_comp_key(self):
return tuple(i if i in self.param_to_msg_comp else None
for i in range(len(self.inp)))
@property
def string_key(self):
return str((self.string_prim_key, self.param_to_msg_comp_key))
@property
def cident_prefix(self):
return f'_after_on_hook_{self.uniq}_'
@property
def cident_args_serializer(self):
assert self.has_serialized_args
return self.cident_prefix + 'args_serializer'
@property
def cident_args_deserializer(self):
assert self.has_serialized_args
return self.cident_prefix + 'args_deserializer'
class ImmediateAfterInfo(AfterInfo):
def __init__(self, key, dimsizes, uniq):
self.uniq = uniq
super().__init__(key, dimsizes)
@abstractmethod
def generate_callback_call(self, indices_lit, args_lit): pass
@property
def cident_prefix(self):
return f'_immediate_after_{self.uniq}_'
class AfterDelayIntoMethodInfo(AfterDelayInfo):
def __init__(self, method, uniq):
self.method = method
super().__init__(method, method.dimsizes, uniq)
self._args_type = (TStruct(dict(method.inp),
label=f'_simple_event_{self.uniq}_args')
if method.inp else None)
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return (self._args_type,) * (self._args_type is not None)
@property
def string_key(self):
return self.method.logname_anonymized()
def generate_callback_call(self, indices_lit, args_lit):
site = self.method.site
indices = tuple(mkLit(site, f'{indices_lit}[{i}]', TInt(32, False))
for i in range(self.method.dimensions))
args = tuple(mkLit(site, f'{args_lit}->{pname}', ptype)
for (pname, ptype) in self.method.inp)
with LogFailure(site, self.method, indices), \
crep.DeviceInstanceContext():
code = codegen_call(site, self.method, indices, args, ())
code = mkCompound(site, [code])
code.toc()
class AfterDelayIntoSendNowInfo(AfterDelayInfo):
def __init__(self, typeseq_info, uniq):
super().__init__(typeseq_info, [], uniq)
self .typeseq_info = typeseq_info
hookref_type = THook(typeseq_info.types, validated=True)
self._args_type = (
TStruct({'hookref': hookref_type,
'args': typeseq_info.struct},
label=f'_simple_event_{self.uniq}_args')
if typeseq_info.types else hookref_type)
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return (self._args_type,) * bool(self.typeseq_info.types)
@property
def string_key(self):
return self.typeseq_info.string_key
def generate_callback_call(self, indices_lit, args_lit):
assert indices_lit is None
has_args = bool(self.typeseq_info.types)
hookref = f'{args_lit}->hookref' if has_args else f'*{args_lit}'
args = f'&{args_lit}->args' if has_args else 'NULL'
out('_DML_send_hook(&_dev->obj, &_dev->_detached_hook_queue_stack, '
+ f'_DML_resolve_hookref(_dev, _hook_aux_infos, {hookref}), '
+ f'{args});\n')
def get_after_delay(key):
try:
return dml.globals.after_delay_infos[key]
except:
uniq = len(dml.globals.after_delay_infos)
info = after_delay_info_constructors[type(key)](key, uniq)
dml.globals.after_delay_infos[key] = info
return info
class AfterOnHookIntoMethodInfo(AfterOnHookInfo):
def __init__(self, typeseq_info, method, param_to_msg_comp):
self.method = method
super().__init__(method.dimsizes, method.parent, typeseq_info, method,
param_to_msg_comp, method.inp, bool(self.method.inp))
self._args_type = (
TStruct({name: typ
for (i, (name, typ)) in enumerate(method.inp)
if i not in param_to_msg_comp},
label=f'_after_on_hook_{self.uniq}_args')
if len(self.method.inp) > len(param_to_msg_comp) else None)
def generate_callback_call(self, indices_lit, args_lit, msg_lit):
site = self.method.site
indices = tuple(mkLit(site, f'{indices_lit}[{i}]', TInt(32, False))
for i in range(self.method.dimensions))
args = tuple(
mkLit(site,
f'{msg_lit}->comp{self.param_to_msg_comp[i]}'
if i in self.param_to_msg_comp else f'{args_lit}->{pname}',
ptype)
for (i, (pname, ptype)) in enumerate(self.method.inp))
with LogFailure(site, self.method, indices), \
crep.DeviceInstanceContext():
code = codegen_call(site, self.method, indices, args, ())
code = mkCompound(site, [code])
code.toc()
def generate_args_serializer(self, site, args_expr, out_expr):
sources = tuple((ctree.mkSubRef(site, args_expr, name, "."),
safe_realtype(typ))
if i not in self.param_to_msg_comp else None
for (i, (name, typ)) in enumerate(self.method.inp))
serialize.serialize_sources_to_list(site, sources, out_expr)
def generate_args_deserializer(self, site, val_expr, out_expr, error_out):
if self.args_type:
tmp_out_decl, tmp_out_ref = serialize.declare_variable(
site, '_tmp_out', self.args_type)
tmp_out_decl.toc()
else:
tmp_out_ref = None
targets = tuple((ctree.mkSubRef(site, tmp_out_ref, name, "."),
safe_realtype(typ))
if i not in self.param_to_msg_comp else None
for (i, (name, typ)) in enumerate(self.method.inp))
def error_out_at_index(_i, exc, msg):
return error_out(exc, msg)
serialize.deserialize_list_to_targets(
site, val_expr, targets, error_out_at_index,
f'deserialization of arguments to {self.method.name}')
if self.args_type:
ctree.mkAssignStatement(site, out_expr,
ctree.ExpressionInitializer(
tmp_out_ref)).toc()
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return (self._args_type,) * (self._args_type is not None)
@property
def string_prim_key(self):
return self.method.logname_anonymized(("%u",) * self.method.dimensions)
class AfterOnHookIntoSendNowInfo(AfterOnHookInfo):
def __init__(self, typeseq_info, sendnow_typeseq_info, param_to_msg_comp):
self.sendnow_typeseq_info = sendnow_typeseq_info
inp = [(f'comp{i}', typ)
for (i, typ) in enumerate(sendnow_typeseq_info.types)]
has_inner_args = len(inp) > len(param_to_msg_comp)
super().__init__([], dml.globals.device, typeseq_info,
sendnow_typeseq_info, param_to_msg_comp, inp, True)
sendnow_hookref_type = THook(sendnow_typeseq_info.types,
validated=True)
self.inner_args_type = (
TStruct({name: typ
for (i, (name, typ)) in enumerate(inp)
if i not in param_to_msg_comp},
label=f'_after_on_hook_{self.uniq}_inner_args')
if has_inner_args else None)
self._args_type = (
TStruct({'hookref': sendnow_hookref_type,
'args': self.inner_args_type},
label=f'_after_on_hook_{self.uniq}_args')
if has_inner_args else sendnow_hookref_type)
def generate_callback_call(self, indices_lit, args_lit, msg_lit):
assert indices_lit is None
has_inner_args = (len(self.sendnow_typeseq_info.types)
> len(self.param_to_msg_comp))
hookref = f'{args_lit}->hookref' if has_inner_args else f'*{args_lit}'
sendnow_msg_struct = self.sendnow_typeseq_info.struct
args = (('&(%s_t) {%s}'
% (sendnow_msg_struct.label,
', '.join(
f'{msg_lit}->comp{self.param_to_msg_comp[i]}'
if i in self.param_to_msg_comp else
f'{args_lit}->args.comp{i}'
for i in range(len(self.sendnow_typeseq_info.types)))))
if self.sendnow_typeseq_info.types else 'NULL')
out('_DML_send_hook(&_dev->obj, &_dev->_detached_hook_queue_stack, '
+ f'_DML_resolve_hookref(_dev, _hook_aux_infos, {hookref}), '
+ f'{args});\n')
def generate_args_serializer(self, site, args_expr, out_expr):
has_inner_args = (len(self.sendnow_typeseq_info.types)
> len(self.param_to_msg_comp))
hookref = (ctree.mkSubRef(site, args_expr, "hookref", ".")
if has_inner_args else args_expr)
sources = [(hookref,
THook(self.sendnow_typeseq_info.types, validated=True))]
if has_inner_args:
inner_args_val_decl, inner_args_val = serialize.declare_variable(
site, 'inner_args_val', TNamed('attr_value_t'))
inner_args_val_decl.toc()
inner_args = ctree.mkSubRef(site, args_expr, 'args', '.')
inner_args_sources = (
(ctree.mkSubRef(site, inner_args, f'comp{i}', '.'), typ)
if i not in self.param_to_msg_comp else None
for (i, typ) in enumerate(self.sendnow_typeseq_info.types))
serialize.serialize_sources_to_list(site, inner_args_sources,
inner_args_val)
sources.append((inner_args_val, None))
serialize.serialize_sources_to_list(site, sources, out_expr)
def generate_args_deserializer(self, site, val_expr, out_expr, error_out):
has_inner_args = (len(self.sendnow_typeseq_info.types)
> len(self.param_to_msg_comp))
tmp_out_decl, tmp_out_ref = serialize.declare_variable(
site, '_tmp_out', self.args_type)
tmp_out_decl.toc()
hookref = (ctree.mkSubRef(site, tmp_out_ref, 'hookref', '.')
if has_inner_args else tmp_out_ref)
targets = [(hookref,
safe_realtype(THook(self.sendnow_typeseq_info.types,
validated=True)))]
if has_inner_args:
inner_args_val_decl, inner_args_val = serialize.declare_variable(
site, '_inner_args_val', TNamed('attr_value_t'))
inner_args_val_decl.toc()
targets.append((inner_args_val, None))
def error_out_at_index(_i, exc, msg):
return error_out(exc, msg)
serialize.deserialize_list_to_targets(
site, val_expr, targets, error_out_at_index,
'deserialization of arguments to a send_now')
if has_inner_args:
inner_args = ctree.mkSubRef(site, tmp_out_ref, 'args', '.')
inner_args_targets = (
(ctree.mkSubRef(site, inner_args, f'comp{i}', '.'),
safe_realtype(typ))
if i not in self.param_to_msg_comp else None
for (i, typ) in enumerate(self.sendnow_typeseq_info.types))
serialize.deserialize_list_to_targets(
site, inner_args_val, inner_args_targets, error_out_at_index,
'deserialization of arguments to a send_now')
ctree.mkAssignStatement(site, out_expr,
ctree.ExpressionInitializer(tmp_out_ref)).toc()
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return ((self.inner_args_type, self._args_type)
* (self.inner_args_type is not None))
@property
def string_prim_key(self):
return self.sendnow_typeseq_info.string_key
class ImmediateAfterIntoMethodInfo(ImmediateAfterInfo):
def __init__(self, method, uniq):
self.method = method
super().__init__(method, method.dimsizes, uniq)
self._args_type = (TStruct(dict(method.inp),
label=f'_immediate_after_{self.uniq}_args')
if method.inp else None)
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return (self._args_type,) * (self._args_type is not None)
def generate_callback_call(self, indices_lit, args_lit):
site = self.method.site
indices = tuple(mkLit(site, f'{indices_lit}[{i}]', TInt(32, False))
for i in range(self.method.dimensions))
args = tuple(mkLit(site, f'{args_lit}->{pname}', ptype)
for (pname, ptype) in self.method.inp)
with LogFailure(site, self.method, indices), \
crep.DeviceInstanceContext():
code = codegen_call(site, self.method, indices, args, ())
code = mkCompound(site, [code])
code.toc()
class ImmediateAfterIntoSendNowInfo(ImmediateAfterInfo):
def __init__(self, typeseq_info, uniq):
super().__init__(typeseq_info, [], uniq)
self.typeseq_info = typeseq_info
hookref_type = THook(typeseq_info.types, validated=True)
self._args_type = (
TStruct({'hookref': hookref_type,
'args': typeseq_info.struct},
label=f'_immediate_after_{self.uniq}_args')
if typeseq_info.types else hookref_type)
@property
def args_type(self):
return self._args_type
@property
def types_to_declare(self):
return (self._args_type,) * bool(self.typeseq_info.types)
def generate_callback_call(self, indices_lit, args_lit):
assert indices_lit is None
has_args = bool(self.typeseq_info.types)
hookref = f'{args_lit}->hookref' if has_args else f'*{args_lit}'
args = f'&{args_lit}->args' if has_args else 'NULL'
out('_DML_send_hook(&_dev->obj, &_dev->_detached_hook_queue_stack, '
+ f'_DML_resolve_hookref(_dev, _hook_aux_infos, {hookref}), '
+ f'{args});\n')
def get_immediate_after(key):
try:
return dml.globals.immediate_after_infos[key]
except:
uniq = len(dml.globals.immediate_after_infos)
info = immediate_after_info_constructors[type(key)](key, uniq)
dml.globals.immediate_after_infos[key] = info
return info
after_delay_info_constructors = {
objects.Method: AfterDelayIntoMethodInfo,
TypeSequenceInfo: AfterDelayIntoSendNowInfo
}
after_on_hook_info_constructors = {
objects.Method: AfterOnHookIntoMethodInfo,
TypeSequenceInfo: AfterOnHookIntoSendNowInfo
}
immediate_after_info_constructors = {
objects.Method: ImmediateAfterIntoMethodInfo,
TypeSequenceInfo: ImmediateAfterIntoSendNowInfo
}
class AfterArgsInit:
@abstractmethod
def args_init(self): pass
class AfterIntoSendNowArgsInit(AfterArgsInit):
def __init__(self, inargs, hookref):
self.inargs = inargs
self.hookref = hookref
def args_init(self):
if self.inargs:
return ('{ %s, { %s } }'
% (self.hookref.read(),
', '.join(inarg.read() for inarg in self.inargs)))
else:
return self.hookref.read()
class AfterIntoMethodArgsInit(AfterArgsInit):
def __init__(self, inargs):
self.inargs = inargs
def args_init(self):
assert self.inargs
return f'{{ {", ".join(inarg.read() for inarg in self.inargs)} }}'
def declarations(scope):
"Get all local declarations in a scope as a list of Declaration objects"
decls = []
for sym in scope.symbols():
if sym.pseudo:
# dbg("declarations(%s): skipping %r" % (scope.id, sym))
continue
if sym.stmt:
continue
decl = sym_declaration(sym)
if decl:
decls.append(decl)
return decls
# Expression dispatch
expression_dispatcher = ast.astdispatcher('expr_')
codegen_expression_maybe_nonvalue = expression_dispatcher.dispatch
def codegen_expression(ast, location, scope):
expr = codegen_expression_maybe_nonvalue(ast, location, scope)
if isinstance(expr, NonValue):
raise expr.exc()
return expr
@expression_dispatcher
def expr_set(tree, location, scope):
[target, source] = tree.args
return mkAssignOp(tree.site,
codegen_expression(target, location, scope),
codegen_expression(source, location, scope))
@expression_dispatcher
def expr_conditional(tree, location, scope):
[cond, texpr, fexpr] = tree.args
cond = codegen_expression(cond, location, scope)
if cond.constant and dml.globals.dml_version == (1, 2):
# Constant propagate
live_ast = texpr if cond.value else fexpr
live_expr = codegen_expression_maybe_nonvalue(live_ast, location, scope)
# Skip code generation for dead branch, but only in 1.2
if (logging.show_porting
and not tree.site.filename().endswith('dml-builtins.dml')):
# If any branch contains an error or a non-value, then
# it must be converted to '#?'.
with logging.suppress_errors() as errors:
dead_ast = fexpr if cond.value else texpr
try:
codegen_expression(dead_ast, location, scope)
except DMLError as e:
errors.append(e)
if errors or isinstance(live_expr, NonValue):
report(PHASH(tree.site))
report(PHASHELSE(dmlparse.end_site(texpr.site), ':'))
return live_expr
return mkIfExpr(tree.site,
cond,
codegen_expression(texpr, location, scope),
codegen_expression(fexpr, location, scope))