forked from mit-plv/koika
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathParsing.v
389 lines (321 loc) · 23.8 KB
/
Parsing.v
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
(* Frontend | Parser for the (untyped) Kôika EDSL
*
* This file contains all notations which are used
* to adjust coq's parser and enable it to parse
* koika.
*
* The notations in this file are ordered by their
* level, beginning with the lowest precedence (highest
* level).
*
* Note:
* The notations in this file produce untyped koika
* syntax trees. There exist very similar notations
* for directly parsing the typed koika syntax in
* the file TypedParsing.v
*)
Require Import
Koika.Common
Koika.Syntax
Koika.IdentParsing
Koika.TypedSyntaxMacros.
Export Koika.Types.SigNotations.
Export Koika.Primitives.PrimUntyped.
Export TypedSyntaxMacros.
Export Coq.Strings.String.
Export Coq.Lists.List.ListNotations.
Declare Custom Entry koika.
(* This is the entry point to transition from normal
Coq (constr) to koika *)
Notation "'{{' e '}}'" := (e) (e custom koika, format "'{{' '[v' '/' e '/' ']' '}}'").
(* koika_var - utility grammar
*
* This grammar is used throughout the parser
* to parse identifiers as strings.
*)
Declare Custom Entry koika_var.
Notation "a" := (ident_to_string a) (in custom koika_var at level 0, a ident, only parsing).
Notation "a" := (a) (in custom koika_var at level 0, a ident, format "'[' a ']'", only printing).
Declare Custom Entry koika_binder.
(* koika_binder - utility grammar
*
* This grammar defines the structure of an
* arguments list
*)
(* TODO improve arg list to be more consistent on nil case *)
Notation "'(' x ':' y ')'" := (x%string, y) (in custom koika_binder at level 0, x custom koika_var, y constr).
Notation "a .. b" := (cons a .. (cons b nil) ..) (in custom koika_binder at level 1, a custom koika_binder at next level, b custom koika_binder at next level).
Notation "'fun' nm args ':' ret '=>' body" :=
(Build_UInternalFunction' nm%string args ret body)
(in custom koika at level 200, nm custom koika_var, args custom koika_binder, ret constr at level 0, right associativity, format "'[v' 'fun' nm args ':' ret '=>' '/' body ']'").
Notation "'fun' nm '()' ':' ret '=>' body" :=
(Build_UInternalFunction' nm%string nil ret body)
(in custom koika at level 200, nm custom koika_var, ret constr at level 0, right associativity, format "'[v' 'fun' nm '()' ':' ret '=>' '/' body ']'").
Notation "'assert' a 'in' c" := (UIf a c (USugar (UConstBits Ob))) (in custom koika at level 200, right associativity, format "'[v' 'assert' a '/' 'in' c ']'").
Notation "'assert' a 'else' b 'in' c" := (UIf a c b ) (in custom koika at level 200, right associativity, format "'[v' 'assert' a '/' 'else' b '/' 'in' c ']'").
Notation "'when' a 'do' t " := (UIf a t (USugar (UConstBits Ob))) (in custom koika at level 200, right associativity, format "'[v' 'when' a '/' 'do' t '/' ']'").
Notation "'let' a ':=' b 'in' c" := (UBind a b c) (in custom koika at level 200, a custom koika_var, right associativity, format "'[v' 'let' a ':=' b 'in' '/' c ']'").
Notation "a ';' b" := (USeq a b) (in custom koika at level 90, b at level 200, format "'[v' a ';' '/' b ']'" ).
Notation "a ';'" := (USeq a (USugar (UConstBits Ob))) (in custom koika at level 90). (* trailing comma *)
Notation "'set' a ':=' b" := (UAssign a b) (in custom koika at level 89, a custom koika_var, format "'set' a ':=' b").
Notation "'if' a 'then' t 'else' f" := (UIf a t f) (in custom koika at level 89, right associativity, format "'[v' 'if' a '/' 'then' t '/' 'else' f ']'").
Notation "'guard' '(' a ')' " := (UIf (UUnop (UBits1 UNot) a) (UFail (bits_t 0)) (USugar (UConstBits Ob))) (in custom koika at level 89, right associativity, format "'guard' '(' a ')'").
(* Inspired by cpp the precedence *)
(* https://en.cppreference.com/w/cpp/language/operator_precedence *)
(* TODO id really prefer to use || and && for logical operations and & | for bitwise *)
(* Bit operations *)
Notation "a '||' b" := (UBinop (UBits2 UOr) a b) (in custom koika at level 85).
Notation "a '^' b" := (UBinop (UBits2 UXor) a b) (in custom koika at level 84).
Notation "a '&&' b" := (UBinop (UBits2 UAnd) a b) (in custom koika at level 83).
(* Comparisons *)
Notation "a '!=' b" := (UBinop (UEq true) a b) (in custom koika at level 80).
Notation "a '==' b" := (UBinop (UEq false) a b) (in custom koika at level 80).
Notation "a '<' b" := (UBinop (UBits2 (UCompare false cLt)) a b) (in custom koika at level 79).
Notation "a '<s' b" := (UBinop (UBits2 (UCompare true cLt)) a b) (in custom koika at level 79).
Notation "a '<=' b" := (UBinop (UBits2 (UCompare false cLe)) a b) (in custom koika at level 79).
Notation "a '<s=' b" := (UBinop (UBits2 (UCompare true cLe)) a b) (in custom koika at level 79).
Notation "a '>' b" := (UBinop (UBits2 (UCompare false cGt)) a b) (in custom koika at level 79).
Notation "a '>s' b" := (UBinop (UBits2 (UCompare true cGt)) a b) (in custom koika at level 79).
Notation "a '>=' b" := (UBinop (UBits2 (UCompare false cGe)) a b) (in custom koika at level 79).
Notation "a '>s=' b" := (UBinop (UBits2 (UCompare true cGe)) a b) (in custom koika at level 79).
(* Bit concatenation / shifts *)
Notation "a '++' b" := (UBinop (UBits2 UConcat) a b) (in custom koika at level 75).
Notation "a '>>' b" := (UBinop (UBits2 ULsr) a b) (in custom koika at level 74).
Notation "a '>>>' b" := (UBinop (UBits2 UAsr) a b) (in custom koika at level 74).
Notation "a '<<' b" := (UBinop (UBits2 ULsl) a b) (in custom koika at level 74).
(* Arithmetic *)
Notation "a '+' b" := (UBinop (UBits2 UPlus) a b) (in custom koika at level 70).
Notation "a '-' b" := (UBinop (UBits2 UMinus) a b) (in custom koika at level 70).
Notation "a '*' b" := (UBinop (UBits2 UMul) a b) (in custom koika at level 69).
(* Unary operators *)
Notation "'!' a" := (UUnop (UBits1 UNot) a) (in custom koika at level 65).
Notation "'><' a" := (UUnop (UBits1 URev) a) (in custom koika at level 65).
Notation "a '[' b ']'" := (UBinop (UBits2 USel) a b) (in custom koika at level 60, format "'[' a [ b ] ']'").
Notation "a '[' b ':+' c ']'" := (UBinop (UBits2 (UIndexedSlice c)) a b) (in custom koika at level 60, c constr at level 0, format "'[' a [ b ':+' c ] ']'").
Notation "'`' a '`'" := (a) (in custom koika, a constr).
Notation "'(' a ')'" := (a) (in custom koika, format "'[v' '(' a ')' ']'").
Notation "'#' s" := (USugar (UConstBits s)) (in custom koika at level 0, s constr at level 0, only parsing).
(* ========================================================================= *)
(* Notations beginning with an identifier *)
(* ========================================================================= *)
(*
Note:
All these notations need to be on the same level (here 0)
else the parser would match on the highest level first and
never even consider notations on a lower level.
Likewise, all these notations need to start with a variable
on the same level and in the same grammar (here a constr on level 0).
Only so the parser can parse this variable first and then decide
(depending on the following tokens) which notation matches.
Note:
Some of the literal notations also start with an identifier.
Thus, the same restrictions apply.
*)
Notation "a" := (UVar (ident_to_string a)) (in custom koika at level 0, a constr at level 0, only parsing).
Notation "a" := (UVar a) (in custom koika at level 0, a constr at level 0, only printing).
Declare Custom Entry koika_args.
Notation "'(' x ',' .. ',' y ')'" := (cons x .. (cons y nil) ..) (in custom koika_args, x custom koika, y custom koika).
Notation "'(' ')'" := (nil) (in custom koika_args).
Notation "'()'" := (nil) (in custom koika_args).
Notation "fn args" := (USugar (UCallModule id _ fn args))
(in custom koika at level 0, fn constr at level 0 , args custom koika_args, only parsing).
Notation "instance '.(' fn ')' args" := (USugar (UCallModule instance _ fn args))
(in custom koika at level 0, fn constr, instance constr at level 0, args custom koika_args).
Notation "'{' fn '}' args" := (USugar (UCallModule id _ fn args))
(in custom koika at level 0, fn constr, args custom koika_args, only parsing).
(* ========================================================================= *)
(* Constructors *)
(* ========================================================================= *)
Declare Custom Entry koika_struct_field.
Declare Custom Entry koika_struct_init.
(* expr at level at level 89 to stay below koika's sequence (a ; b), because it
* uses the same separation character *)
Notation "f ':=' expr" := (f, expr) (in custom koika_struct_field at level 0, f custom koika_var, expr custom koika at level 89).
Notation "a ';' b" := (cons a b) (in custom koika_struct_init at level 0, a custom koika_struct_field, right associativity).
Notation "a ';'" := (cons a nil) (in custom koika_struct_init at level 0, a custom koika_struct_field). (* trailing comma *)
Notation "a" := (cons a nil) (in custom koika_struct_init at level 0, a custom koika_struct_field).
(* deprecated *)
Notation "'struct' sig '{' fields '}'" :=
(USugar (UStructInit sig fields)) (in custom koika, sig constr at level 0, fields custom koika_struct_init).
Notation "'struct' sig '{' '}'" :=
(USugar (UStructInit sig [])) (in custom koika, sig constr at level 0).
Notation "'struct' sig '::{' fields '}'" :=
(USugar (UStructInit sig fields)) (in custom koika, sig constr at level 0, fields custom koika_struct_init).
Notation "'struct' sig '::{' '}'" :=
(USugar (UStructInit sig [])) (in custom koika, sig constr at level 0).
Notation "sig '::{' fields '}'" :=
(USugar (UStructInit sig fields)) (in custom koika at level 0, sig constr at level 0, fields custom koika_struct_init).
Notation "sig '::{' '}'" :=
(USugar (UStructInit sig [])) (in custom koika at level 0, sig constr at level 0).
(* deprecated *)
Notation "'enum' sig '{' f '}'" :=
(USugar (UConstEnum sig f))
(in custom koika, sig constr at level 0, f custom koika_var at level 1).
Notation "'enum' sig '::<' f '>'" :=
(USugar (UConstEnum sig f))
(in custom koika, sig constr at level 0, f custom koika_var at level 1).
Notation "sig '::<' f '>'" :=
(USugar (UConstEnum sig f))
(in custom koika at level 0, sig constr at level 0, f custom koika_var at level 1).
(* ========================================================================= *)
(* Literals *)
(* ========================================================================= *)
Declare Custom Entry koika_consts.
Notation "'1'" := (Ob~1) (in custom koika_consts at level 0).
Notation "'0'" := (Ob~0) (in custom koika_consts at level 0).
Notation "bs '~' '0'" := (Bits.cons false bs) (in custom koika_consts at level 1, left associativity, format "bs '~' '0'").
Notation "bs '~' '1'" := (Bits.cons true bs) (in custom koika_consts at level 1, left associativity, format "bs '~' '1'").
Notation "'Ob' '~' number" := (USugar (UConstBits number))
(in custom koika at level 0, number custom koika_consts, format "'Ob' '~' number").
Notation "'Ob'" := (USugar (UConstBits Bits.nil))
(in custom koika at level 0).
Local Definition len := String.length.
Notation "num ':b' sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (bin_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, only parsing).
Notation "num ':b'" := (USugar (UConstBits (Bits.of_N ((len num) * 1) (bin_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0b' num sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (bin_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, format "'0b' num sz").
Notation "'0b' num" := (USugar (UConstBits (Bits.of_N ((len num) * 1) (bin_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0b' num" := (USugar (UConstBits (Bits.of_N _ (bin_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only printing, format "'0b' num").
Notation "num ':o' sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (oct_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, only parsing).
Notation "num ':o'" := (USugar (UConstBits (Bits.of_N ((len num) * 3) (oct_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0o' num sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (oct_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, format "'0o' num sz").
Notation "'0o' num" := (USugar (UConstBits (Bits.of_N ((len num) * 3) (oct_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0o' num" := (USugar (UConstBits (Bits.of_N _ (oct_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only printing, format "'0o' num").
Notation "num ':d' sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (dec_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, only parsing).
Notation "num ':d'" := (USugar (UConstBits (Bits.of_N (1 + (N.to_nat (N.log2 (dec_string_to_N num))))
(dec_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0d' num sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (dec_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, format "'0d' num sz").
Notation "'0d' num" := (USugar (UConstBits (Bits.of_N (1 + (N.to_nat (N.log2 (dec_string_to_N num))))
(dec_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0d' num" := (USugar (UConstBits (Bits.of_N _ (dec_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only printing, format "'0d' num").
Notation "num ':h' sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (hex_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, only parsing).
Notation "num ':h'" := (USugar (UConstBits (Bits.of_N ((len num) * 4) (hex_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0x' num sz" := (USugar (UConstBits (Bits.of_N (sz <: nat) (hex_string_to_N num)))) (in custom koika at level 0, num constr at level 0, sz constr at level 0, format "'0x' num sz").
Notation "'0x' num" := (USugar (UConstBits (Bits.of_N ((len num) * 4) (hex_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only parsing).
Notation "'0x' num" := (USugar (UConstBits (Bits.of_N _ (hex_string_to_N num)))) (in custom koika at level 0, num constr at level 0, only printing, format "'0x' num").
Notation "'|' a '`d' b '|'" :=
(USugar (UConstBits (Bits.of_N (a<:nat) b%N)))
(in custom koika, a constr at level 0 , b constr at level 0).
(* ========================================================================= *)
(* Closed Notations on level 0 *)
(* ========================================================================= *)
Notation "'pass'" := (USugar (UConstBits Ob)) (in custom koika).
Notation "'magic'" := (USugar UErrorInAst) (in custom koika).
Notation "'fail'" := (UFail (bits_t 0)) (in custom koika, format "'fail'").
Notation "'fail' '(' t ')'" := (UFail (bits_t t)) (in custom koika, t constr, format "'fail' '(' t ')'").
Notation "'fail' '@(' t ')'" := (UFail t ) (in custom koika, t constr, format "'fail' '@(' t ')'").
Notation "'read0' '(' reg ')'" := (URead P0 reg ) (in custom koika, reg constr, format "'read0' '(' reg ')'").
Notation "'read1' '(' reg ')'" := (URead P1 reg ) (in custom koika, reg constr, format "'read1' '(' reg ')'").
Notation "'write0' '(' reg ',' value ')'" := (UWrite P0 reg value) (in custom koika, reg constr, format "'write0' '(' reg ',' value ')'").
Notation "'write1' '(' reg ',' value ')'" := (UWrite P1 reg value) (in custom koika, reg constr, format "'write1' '(' reg ',' value ')'").
Notation "'zeroExtend(' a ',' b ')'" := (UUnop (UBits1 (UZExtL b)) a) (in custom koika, b constr, format "'zeroExtend(' a ',' b ')'").
Notation "'sext(' a ',' b ')'" := (UUnop (UBits1 (USExt b)) a) (in custom koika, b constr, format "'sext(' a ',' b ')'").
Notation "'ignore(' a ')'" := (UUnop (UConv UIgnore) a) (in custom koika).
Notation "'pack(' a ')'" := (UUnop (UConv UPack) a) (in custom koika).
Notation "'unpack(' t ',' v ')'" := (UUnop (UConv (UUnpack t)) v) (in custom koika, t constr).
Notation "'extcall' method '(' arg ')'" := (UExternalCall method arg) (in custom koika, method constr at level 0).
Notation "'get' '(' v ',' f ')'" := (UUnop (UStruct1 (UGetField f)) v ) (in custom koika, f custom koika_var, format "'get' '(' v ',' f ')'").
Notation "'getbits' '(' t ',' v ',' f ')'" := (UUnop (UStruct1 (UGetFieldBits t f)) v ) (in custom koika, t constr, f custom koika_var, format "'getbits' '(' t ',' v ',' f ')'").
Notation "'subst' '(' v ',' f ',' a ')'" := (UBinop (UStruct2 (USubstField f)) v a) (in custom koika, f custom koika_var, format "'subst' '(' v ',' f ',' a ')'").
Notation "'substbits' '(' t ',' v ',' f ',' a ')'" := (UBinop (UStruct2 (USubstFieldBits t f)) v a) (in custom koika, t constr, f custom koika_var, format "'substbits' '(' t ',' v ',' f ',' a ')'").
Notation "'aref' '(' v ',' f ')'" := (UUnop (UArray1 (UGetElement f)) v ) (in custom koika, f constr, format "'aref' '(' v ',' f ')'").
Notation "'arefbits' '(' t ',' v ',' f ')'" := (UUnop (UArray1 (UGetElementBits t f)) v ) (in custom koika, t constr, f constr, format "'arefbits' '(' t ',' v ',' f ')'").
Notation "'asubst' '(' v ',' f ',' a ')'" := (UBinop (UArray2 (USubstElement f)) v a) (in custom koika, f constr, format "'asubst' '(' v ',' f ',' a ')'").
Notation "'asubstbits' '(' t ',' v ',' f ',' a ')'" := (UBinop (UArray2 (USubstElementBits t f)) v a) (in custom koika, t constr, f constr, format "'asubstbits' '(' t ',' v ',' f ',' a ')'").
(* koika_branches - utility
*
* This grammar is used for the syntax
* of match branches
*)
Declare Custom Entry koika_branches.
Notation "x '=>' a " := [(x,a)] (in custom koika_branches at level 0, x custom koika at level 200, a custom koika at level 200).
Notation "arg1 '|' arg2" := (arg1 ++ arg2) (in custom koika_branches at level 1, format "'[v' arg1 ']' '/' '|' '[v' arg2 ']'").
Notation "'match' var 'with' '|' branches 'return' 'default' ':' default 'end'" :=
(USugar (USwitch var default branches))
(in custom koika, branches custom koika_branches,
format "'match' var 'with' '/' '[v' '|' branches '/' 'return' 'default' ':' default ']' 'end'").
(* ========================================================================= *)
(* scheduler notations *)
(* ========================================================================= *)
(* at level 60, simply because it's the level of the `cons` notation (::) *)
Notation "r '|>' s" :=
(Cons r s)
(at level 60, right associativity).
Notation "'done'" :=
Done.
Module Type Tests.
Parameter pos_t : Type.
Parameter fn_name_t : Type.
Parameter reg_t : Type.
Parameter ext_fn_t : Type.
Notation uaction reg_t := (uaction pos_t string fn_name_t reg_t ext_fn_t).
Definition test_2 : uaction reg_t := {{ yo; yoyo }}.
Definition test_3 : uaction reg_t := {{ set yoyo := `UVar "magic" : uaction reg_t` }}.
Definition test_1 : uaction reg_t := {{ let yoyo := fail(2) in magic }}.
Definition test_1' : uaction reg_t := {{ let yoyo := fail(2) in yoyo }}.
Definition test_2' : uaction reg_t := {{ magic; magic }}.
Definition test_3' : uaction reg_t := {{ set yoyo := magic ; magic }}.
Definition test_4 : uaction reg_t := {{ magic ; set yoyo := magic }}.
Definition test_5 : uaction reg_t := {{ let yoyo := set yoyo := magic in magic }}.
Definition test_6 : uaction reg_t := {{ let yoyo := set yoyo := magic; magic in magic;magic }}.
Definition test_7 : uaction reg_t := {{ (let yoyo := (set yoyo := magic); magic in magic;magic) }}.
Definition test_8 : uaction reg_t := {{ (let yoyo := set yoyo := magic; magic in magic);magic }}.
Inductive test : Set := |rData (n:nat).
Axiom data0 : reg_t.
Axiom data1 : reg_t.
Definition test_9 : uaction reg_t := {{ read0(data0) }}.
Definition test_10 : nat -> uaction test := (fun idx => {{ read0(rData idx) }}).
Definition test_11 : uaction reg_t := {{ (let yoyo := read0(data0) in write0(data0, magic)); fail }}.
Definition test_12 : uaction reg_t := {{ (let yoyo := if fail then read0(data0) else fail in write0(data0,magic));fail }}.
Definition test_13 : uaction reg_t := {{ yoyo; pass; }}.
Definition test_14 : uaction reg_t := {{ !yoyo && yoyo }}.
Definition test_14' : uaction reg_t := {{ !(yoyo && yoyo) }}.
Goal test_14 <> test_14'. compute; congruence. Qed.
Definition test_15 : uaction reg_t := {{ yoyo && read0(data0) }}.
Definition test_16 : uaction reg_t := {{ !read0(data1) && !read0(data1) }}.
Definition test_17 : uaction reg_t := {{ !read0(data1) && magic}}.
Definition test_18 : uaction reg_t := {{ !read0(data1) && Yoyo }}.
Definition test_19 : uaction reg_t := {{ yoy [ oio && ab ] && Ob~1~0 }}.
Definition test_20 : uaction reg_t := {{ yoyo [ magic :+ 3 ] && yoyo }}.
Definition test_20' : uaction reg_t := {{ (yoyo [ magic :+ 3] ++ yoyo) && yoyo }}.
Definition test_20'' : uaction reg_t := {{ ( yoyo [ magic :+ 3 ] ++ yoyo ++bla) && yoyo; }}.
Definition test_27 : uaction reg_t := {{
if !read0(data0) then
write0(data0, Ob~1);
let yo := if (dlk == Ob~1 ) then bla else yoyo || foo
in
write0(data0, Ob~1)
else
read0(data0);
fail
}}.
Notation UInternalFunction :=
(UInternalFunction pos_t string string reg_t ext_fn_t).
Definition test_23 : UInternalFunction := {{ fun test (arg1 : (bits_t 3)) (arg2 : bits_t 2) : bits_t 4 => magic }}.
Definition test_24 : nat -> UInternalFunction := (fun sz => {{ fun test (arg1 : bits_t sz) (arg1 : bits_t sz) : bits_t sz => magic}}).
Definition test_25 : nat -> UInternalFunction := (fun sz => {{fun test (arg1 : bits_t sz ) : bits_t sz => let oo := magic >> magic in magic}}).
Definition test_26 : nat -> UInternalFunction := (fun sz => {{ fun test () : bits_t sz => magic }}).
Definition test_28 : uaction reg_t := {{
match var with
| magic => magic
return default: magic
end
}}.
Definition mem_req :=
{| struct_name := "mem_req";
struct_fields := cons ("type", bits_t 1) nil |}.
Definition test_30'' : uaction reg_t :=
{{
struct mem_req::{ foo := upu[#(Bits.of_nat 3 0) :+ 2] ;
bar := |32`d98| }
}}.
Definition test_31'' : uaction reg_t :=
{{
pack(a)
}}.
Definition test_31' : uaction reg_t :=
{{
let a := struct mem_req::{ foo := upu[#(Bits.of_nat 3 0) :+ 2] ;
bar := |32`d98| } in
unpack(struct_t mem_req, pack(a))
}}.
End Tests.