forked from AU-COBRA/ConCert
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathEIP20LiquidityExtraction.v
195 lines (167 loc) · 6.81 KB
/
EIP20LiquidityExtraction.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
(** * Extraction of a simple counter contract *)
From MetaCoq.Template Require Import All.
From ConCert.Embedding Require Import Notations.
From ConCert.Embedding.Extraction Require Import PreludeExt.
From ConCert.Extraction Require LiquidityPretty.
From ConCert.Extraction Require Import LiquidityExtract.
From ConCert.Extraction Require Import Common.
From ConCert.Execution Require Import Monad.
From ConCert.Execution Require OptionMonad.
From ConCert.Execution Require Import ResultMonad.
From ConCert.Execution Require Import Blockchain.
From ConCert.Examples.EIP20 Require EIP20Token.
From ConCert.Execution Require Import ContractCommon.
From Coq Require Import String.
From Coq Require Import ZArith.
Import MCMonadNotation.
Local Coercion bytestring.String.of_string : string >-> bytestring.string.
Local Open Scope string_scope.
Definition PREFIX := "".
Definition TT_remap_default : list (kername * string) :=
[ (* types *)
remap <%% Z %%> "tez"
; remap <%% N %%> "int"
; remap <%% nat %%> "nat"
; remap <%% bool %%> "bool"
; remap <%% unit %%> "unit"
; remap <%% list %%> "list"
; remap <%% @fst %%> "fst"
; remap <%% @snd %%> "snd"
; remap <%% option %%> "option"
; remap <%% ConCert.Execution.ResultMonad.result %%> "result"
; remap <%% positive %%> "nat"
; remap <%% Amount %%> "tez"
; remap <%% @Address %%> "address"
(* operations *)
(* Note: N is mapped to Int instead of nat because Liquidity's minus
operator on nats returns an int, which is not compatible with
N.sub since it returns an N *)
; remap <%% List.fold_left %%> "List.fold"
; remap <%% Pos.add %%> "addNat"
; remap <%% Pos.sub %%> "subNat"
; remap <%% Pos.leb %%> "leNat"
; remap <%% Pos.eqb %%> "eqNat"
; remap <%% Z.add %%> "addTez"
; remap <%% Z.sub %%> "subTez"
; remap <%% Z.leb %%> "leTez"
; remap <%% Z.ltb %%> "ltTez"
; remap <%% Z.eqb %%> "eqTez"
; remap <%% Z.gtb %%> "gtbTez"
; remap <%% N.add %%> "addInt"
; remap <%% N.sub %%> "subInt"
; remap <%% N.leb %%> "leInt"
; remap <%% N.ltb %%> "ltInt"
; remap <%% N.eqb %%> "eqInt"
; remap <%% andb %%> "andb"
; remap <%% negb %%> "not"
; remap <%% orb %%> "orb"
; remap <%% eqb_addr %%> "eq_addr"
].
Section EIP20TokenExtraction.
Import EIP20Token.
Notation params := (ContractCallContext × option EIP20Token.Msg).
Open Scope N_scope.
Open Scope bool.
(* We define a version of [receive] that has the right signature. *)
(* TODO: remove, once the [LiquidityMod] is fixed. *)
Definition test_receive
(ctx : ContractCallContext)
(state : EIP20Token.State)
(maybe_msg : option EIP20Token.Msg)
: result (list ActionBody * EIP20Token.State) Error :=
let sender := ctx.(ctx_from) in
let without_actions x := x >>= (fun new_state => Ok ([], new_state)) in
match maybe_msg with
| Some (transfer to_addr amountt) =>
without_actions (try_transfer sender to_addr amountt state)
| Some (transfer_from from to_addr amountt) =>
without_actions (try_transfer_from sender from to_addr amountt state)
| Some (approve delegate amount) =>
without_actions (try_approve sender delegate amount state)
| _ => Err default_error
end.
Definition receive_wrapper
(params : params)
(st : State)
: result (list ActionBody × State) Error :=
test_receive params.1 st params.2.
(* The same as for [test_receive].
TODO: remove, once the [LiquidityMod] is fixed. *)
Definition init (ctx : ContractCallContext)
(setup : EIP20Token.Setup)
: result EIP20Token.State Error :=
(* ensure extraction does not optimize unused ctx away
NOTE: can be dealt with in a better way using
the mask-override mechanism of dearging *)
let ctx_ := ctx in
Ok {| total_supply := setup.(init_amount);
balances := AddressMap.add (EIP20Token.owner setup)
(init_amount setup)
AddressMap.empty;
allowances := AddressMap.empty |}.
Open Scope Z_scope.
Definition printERC20Wrapper (contract : string) : string :=
"let wrapper param (st : storage)"
++ " = "
++ "match " ++ contract ++ " (" ++ liquidity_call_ctx ++ ", param) st" ++ " with"
++ "| Ok v -> v"
++ "| Err e -> failwith e".
Definition EIP20Token_MODULE : LiquidityMod params ContractCallContext EIP20Token.Setup EIP20Token.State ActionBody Error :=
{| (* a name for the definition with the extracted code *)
lmd_module_name := "liquidity_eip20token" ;
(* definitions of operations on pairs and ints *)
lmd_prelude := LiquidityPretty.LiquidityPrelude;
(* initial storage *)
lmd_init := init ;
lmd_init_prelude := "";
(* the main functionality *)
lmd_receive := receive_wrapper ;
(* code for the entry point *)
lmd_entry_point := "type storage = state"
++ nl
++ printERC20Wrapper (PREFIX ++ "receive_wrapper")
++ nl
++ LiquidityPretty.printMain
|}.
Definition TT_remap_eip20token : list (kername * string) :=
TT_remap_default ++ [
remap <%% @ContractCallContext %%> "(timestamp * (address * (tez * tez)))"
(* small hack, but valid since ContractCallContext is mapped to a tuple *)
; remap <%% @ctx_from %%> "(fun x -> x.(1).(0))"
; remap <%% gmap.gmap %%> "map"
; remap <%% @AddressMap.add %%> "Map.add"
; remap <%% @AddressMap.find %%> "Map.find"
; remap <%% @AddressMap.empty %%> "(Map [])"
].
Definition TT_rename_eip20token :=
[ ("Z0" ,"0tez")
; ("N0", "0")
; ("N1", "1")
; ("O", "0")
; ("Ok", "Ok")
; ("Err", "Err")
; ("nil", "[]")
; ("tt", "()") ].
Definition TT_inlines_eip20token : list kername :=
[ <%% OptionMonad.Monad_option %%>
; <%% @ConCert.Execution.ResultMonad.Monad_result %%>
; <%% @Monad.bind %%>
; <%% @Monad.ret %%>
; <%% bool_rect %%>
; <%% bool_rec %%>
; <%% option_map %%>
; <%% @Extras.with_default %%>
; <%% @setter_from_getter_State_balances %%>
; <%% @setter_from_getter_State_total_supply %%>
; <%% @setter_from_getter_State_allowances %%>
; <%% @set_State_balances %%>
; <%% @set_State_allowances%%>
].
Time MetaCoq Run
(liquidity_prepare_extraction PREFIX TT_remap_eip20token
TT_rename_eip20token TT_inlines_eip20token EIP20Token_MODULE).
Time Definition liquidity_eip20token := Eval vm_compute in liquidity_eip20token_prepared.
(** We redirect the extraction result for later processing and compiling with the Liquidity compiler *)
Redirect "../extraction/tests/extracted-code/liquidity-extract/liquidity_eip20token.liq"
MetaCoq Run (tmMsg (bytestring.String.of_string liquidity_eip20token)).
End EIP20TokenExtraction.