Skip to content

Commit 8585a1c

Browse files
committed
Split large memory accesses in stack alloc
1 parent 2d72a31 commit 8585a1c

31 files changed

+1646
-239
lines changed

compiler/config/tests.config

+5-6
Original file line numberDiff line numberDiff line change
@@ -40,18 +40,18 @@ exclude = !tests/success/noextract
4040

4141
[test-arm-m4]
4242
bin = ./scripts/check-arm-m4
43-
okdirs = examples/**/arm-m4 tests/success/**/arm-m4 tests/success/**/common
43+
okdirs = examples/**/arm-m4 tests/success/**/arm-m4 tests/success/**/common tests/success/arm-m4/large_memory_immediates
4444
kodirs = tests/fail/**/arm-m4
4545

4646
[test-arm-m4-print]
4747
bin = ./scripts/parse-print-parse
4848
args = arm
49-
okdirs = tests/success/**/arm-m4 tests/success/**/common
49+
okdirs = tests/success/**/arm-m4 tests/success/**/common tests/success/arm-m4/large_memory_immediates
5050

5151
[test-arm-m4-extraction]
5252
bin = ./scripts/extract-and-check
5353
args = arm
54-
okdirs = examples/**/arm-m4 tests/success/**/arm-m4 tests/success/**/common
54+
okdirs = examples/**/arm-m4 tests/success/**/arm-m4 tests/success/**/common tests/success/arm-m4/large_memory_immediates
5555
exclude = !tests/success/noextract
5656

5757
[test-x86-64-stack-zero-loop]
@@ -71,12 +71,11 @@ exclude = !tests/fail/warning
7171
[test-arm-m4-stack-zero-loop]
7272
bin = ./scripts/check-arm-m4
7373
args = -stack-zero=loop
74-
okdirs = examples/**/arm-m4 tests/success/**/arm-m4
74+
okdirs = examples/**/arm-m4 tests/success/**/arm-m4 tests/success/arm-m4/large_memory_immediates
7575
kodirs = tests/fail/**/arm-m4
7676

7777
[test-arm-m4-stack-zero-unrolled]
7878
bin = ./scripts/check-arm-m4
7979
args = -stack-zero=unrolled
8080
okdirs = examples/**/arm-m4 tests/success/**/arm-m4
81-
kodirs = tests/fail/**/arm-m4
82-
exclude = !tests/success/arm-m4/large_stack
81+
kodirs = tests/fail/**/arm-m4 test/success/arm-m4/large_memory_immediates

compiler/src/compile.ml

+16-4
Original file line numberDiff line numberDiff line change
@@ -183,10 +183,22 @@ let compile (type reg regx xreg rflag cond asm_op extra_op)
183183
(fn, f) |> Conv.fdef_of_cufdef |> refresh_i_loc_f |> Conv.cufdef_of_fdef |> snd
184184
in
185185

186-
let warning ii msg =
187-
(if not !Glob_options.lea then
188-
let loc, _ = ii in
189-
warning UseLea loc "%a" Printer.pp_warning_msg msg);
186+
let warning ii w =
187+
let o =
188+
match w with
189+
| Compiler_util.Use_lea ->
190+
if not !Glob_options.lea
191+
then Some (UseLea, "LEA instruction is used")
192+
else None
193+
| Split_memory_access ->
194+
let msg =
195+
"This memory immediate does not fit in one instruction, several \
196+
instructions were issued."
197+
in
198+
Some (SplitMemoryAccess, msg)
199+
in
200+
let loc, _ = ii in
201+
Option.may (fun (w, msg) -> warning w loc "%s" msg) o;
190202
ii
191203
in
192204

compiler/src/printer.ml

-3
Original file line numberDiff line numberDiff line change
@@ -405,9 +405,6 @@ let pp_sprog ~debug pd asmOp fmt ((funcs, p_extra):('info, 'asm) Prog.sprog) =
405405

406406
(* ----------------------------------------------------------------------- *)
407407

408-
let pp_warning_msg fmt = function
409-
| Compiler_util.Use_lea -> Format.fprintf fmt "LEA instruction is used"
410-
411408
let pp_err ~debug fmt (pp_e : Compiler_util.pp_error) =
412409
let pp_var fmt v =
413410
let v = Conv.var_of_cvar v in

compiler/src/printer.mli

-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
open Prog
22

3-
val pp_warning_msg : Format.formatter -> Compiler_util.warning_msg -> unit
43
val pp_err : debug:bool -> Format.formatter -> Compiler_util.pp_error -> unit
54

65
val pp_print_X : Format.formatter -> Z.t -> unit

compiler/src/stackAlloc.ml

+1
Original file line numberDiff line numberDiff line change
@@ -190,6 +190,7 @@ let memory_analysis pp_err ~debug up =
190190
gao.gao_data
191191
cglobs
192192
get_sao
193+
(fun ii _ -> ii)
193194
up
194195
with
195196
| Utils0.Ok sp -> sp

compiler/src/utils.ml

+1
Original file line numberDiff line numberDiff line change
@@ -404,6 +404,7 @@ type warning =
404404
| SimplifyVectorSuffix
405405
| DuplicateVar
406406
| UnusedVar
407+
| SplitMemoryAccess
407408
| SCTchecker
408409
| Deprecated
409410
| Experimental

compiler/src/utils.mli

+1
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,7 @@ type warning =
176176
| SimplifyVectorSuffix
177177
| DuplicateVar
178178
| UnusedVar
179+
| SplitMemoryAccess
179180
| SCTchecker
180181
| Deprecated
181182
| Experimental
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// Stack allocation needs to split the array set and get operations into several
2+
// instructions to that the immediates fit in one instruction.
3+
4+
// The LDR/STR instructions take an immediate of up to 12 bits.
5+
// Since elements are 4 bytes, we the size limit for an array is 2^12 / 4.
6+
param int LIMIT = 1024;
7+
8+
export
9+
fn large_array() -> reg u32 {
10+
reg u32 r;
11+
stack u32[LIMIT + 1] a;
12+
13+
r = 0;
14+
a[LIMIT] = r;
15+
r = a[LIMIT];
16+
17+
return r;
18+
}

compiler/tests/success/arm-m4/large_memory_immediates/too_many_stack_variables.jazz

+508
Large diffs are not rendered by default.

proofs/compiler/arm_decl.v

+9-3
Original file line numberDiff line numberDiff line change
@@ -381,6 +381,12 @@ Definition is_expandable_or_shift (n : Z) : bool :=
381381
| EI_byte | EI_pattern | EI_shift => true
382382
| EI_none => false
383383
end.
384-
385-
Definition is_w12_encoding (z : Z) : bool := (z <? Z.pow 2 12)%Z.
386-
Definition is_w16_encoding (z : Z) : bool := (z <? Z.pow 2 16)%Z.
384+
385+
Definition is_w12_encoding (z : Z) : bool := [&& 0 <=? z & z <? Z.pow 2 12 ]%Z.
386+
Definition is_w16_encoding (z : Z) : bool := [&& 0 <=? z & z <? Z.pow 2 16 ]%Z.
387+
388+
(* The LDR/STR families can take 12-bit unsigned immediates or 8-bit signed
389+
immediates. *)
390+
Definition is_mem_immediate (z : Z) : bool :=
391+
let u := (if z <? 0 then z + wbase reg_size else z)%Z in
392+
is_w12_encoding u || [&& -256 <? z & z <? 256 ]%Z.

proofs/compiler/arm_params.v

+35
Original file line numberDiff line numberDiff line change
@@ -77,11 +77,46 @@ Definition arm_immediate (x: var_i) z :=
7777
Definition arm_swap t (x y z w : var_i) :=
7878
Copn [:: Lvar x; Lvar y] t (Oasm (ExtOp (Oarm_swap reg_size))) [:: Plvar z; Plvar w].
7979

80+
(* Build the the immediate [eoff] if it does not fit in a single LDR/STR
81+
instruction. *)
82+
Definition lower_mem_off (tmp : var_i) (eoff : pexpr) : seq copn_args * pexpr :=
83+
if expr.is_wconst reg_size eoff is Some woff
84+
then ARMCopn.load_mem_imm tmp woff
85+
else ([::], eoff).
86+
87+
Definition split_mem_opn_match_lvs lvs :=
88+
if lvs is [:: Lmem ws vbase eoff ] then Some (ws, vbase, eoff) else None.
89+
90+
Definition split_mem_opn_match_es es :=
91+
if es is [:: Pload ws vbase eoff ] then Some (ws, vbase, eoff) else None.
92+
93+
(* Call [lower_mem_off] on memory accesses. *)
94+
Definition split_mem_opn
95+
(tmp : var_i)
96+
(lvs : seq lval)
97+
(op : sopn)
98+
(es : seq pexpr) :
99+
cexec (seq copn_args) :=
100+
if split_mem_opn_match_lvs lvs is Some (ws, vbase, eoff)
101+
then
102+
(* STR *)
103+
let '(pre, eoff) := lower_mem_off tmp eoff in
104+
ok (rcons pre ([:: Lmem ws vbase eoff ], op, es))
105+
else
106+
if split_mem_opn_match_es es is Some (ws, vbase, eoff)
107+
then
108+
(* LDR *)
109+
let '(pre, eoff) := lower_mem_off tmp eoff in
110+
ok (rcons pre (lvs, op, [:: Pload ws vbase eoff ]))
111+
else
112+
ok [:: (lvs, op, es) ].
113+
80114
Definition arm_saparams : stack_alloc_params :=
81115
{|
82116
sap_mov_ofs := arm_mov_ofs;
83117
sap_immediate := arm_immediate;
84118
sap_swap := arm_swap;
119+
sap_split_mem_opn := split_mem_opn;
85120
|}.
86121

87122
(* ------------------------------------------------------------------------ *)

proofs/compiler/arm_params_common.v

+6
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,12 @@ Module ARMOpn (Args : OpnArgs).
8484
Precondition: if [imm] is large, [x <> tmp]. *)
8585
Definition smart_subi_tmp x tmp imm := map to_opn (Core.smart_subi_tmp x tmp imm).
8686

87+
Definition load_mem_imm
88+
{_ : PointerData} (tmp : var_i) (woff : word Uptr) : seq opn_args * rval :=
89+
let z := wsigned woff in
90+
if is_mem_immediate z
91+
then ([::], rconst Uptr z)
92+
else (li tmp (wunsigned woff), rvar tmp).
8793

8894
End WITH_PARAMS.
8995

proofs/compiler/arm_params_common_proof.v

+81
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,87 @@ Ltac t_arm_op :=
4545
rewrite ?zero_extend_u ?addn1;
4646
t_simpl_rewrites.
4747

48+
Module ARMCopnP.
49+
50+
Section WITH_PARAMS.
51+
52+
Context
53+
{wsw : WithSubWord}
54+
{dc : DirectCall}
55+
{syscall_state : Type}
56+
{sc_sem : syscall_sem syscall_state}
57+
{atoI : arch_toIdent}
58+
{pT : progT}
59+
{sCP : semCallParams}
60+
(p : prog)
61+
(evt : extra_val_t)
62+
.
63+
64+
Let mkv xname vi :=
65+
let: x := {| vname := xname; vtype := sword arm_reg_size; |} in
66+
let: xi := {| v_var := x; v_info := vi; |} in
67+
(xi, x).
68+
69+
Lemma sem_copn_equiv gd args s :
70+
ARMCopn_coreP.sem_copn_args gd args s
71+
= sem_copn_args gd (ARMCopn.to_opn args) s.
72+
Proof.
73+
rewrite /sem_copn_args /sem_sopn /exec_sopn /=.
74+
case: args => -[lvs o] es /=.
75+
case: sem_pexprs => //= ?.
76+
by case: app_sopn.
77+
Qed.
78+
79+
Lemma sem_copns_equiv gd args s :
80+
ARMCopn_coreP.sem_copns_args gd s args
81+
= sem_copns_args gd s (map ARMCopn.to_opn args).
82+
Proof.
83+
elim: args s => //= o os ih s.
84+
rewrite sem_copn_equiv.
85+
by case: sem_copn_args.
86+
Qed.
87+
88+
Lemma li_sem gd s xname vi imm :
89+
let: (xi, x) := mkv xname vi in
90+
let: lcmd := ARMCopn.li xi imm in
91+
exists vm',
92+
[/\ sem_copns_args gd s lcmd = ok (with_vm s vm')
93+
, vm' =[\ Sv.singleton x ] evm s
94+
& get_var true vm' x = ok (Vword (wrepr reg_size imm))
95+
].
96+
Proof.
97+
have [vm' []] := ARMCopn_coreP.li_sem_copns_args gd s xname vi imm.
98+
rewrite sem_copns_equiv => h1 h2 h3.
99+
by exists vm'.
100+
Qed.
101+
Opaque ARMCopn.li.
102+
103+
Lemma load_mem_immP pre eimm xname vi imm ii tag s :
104+
let: x := {| vname := xname; vtype := sword arm_reg_size; |} in
105+
let: xi := {| v_var := x; v_info := vi; |} in
106+
let: c := [seq i_of_copn_args ii tag a | a <- pre ] in
107+
ARMCopn.load_mem_imm xi imm = (pre, eimm) ->
108+
exists vm,
109+
let: s' := with_vm s vm in
110+
[/\ sem p evt s c s'
111+
, vm =[\ Sv.singleton x ] evm s
112+
& sem_pexpr true (p_globs p) s' eimm = ok (Vword imm)
113+
].
114+
Proof.
115+
set x := {| vname := _; |}; set xi := {| v_var := _; |}.
116+
rewrite /ARMCopn.load_mem_imm.
117+
case: is_mem_immediate => -[??]; subst pre eimm.
118+
- exists (evm s); split => //. + rewrite with_vm_same /=. by econstructor.
119+
by rewrite /= /sem_sop1 /= wrepr_signed.
120+
have [vm [hsem hvm hget]] := li_sem (p_globs p) s xname vi (wunsigned imm).
121+
exists vm; split=> //; first exact: sem_copns_args_sem.
122+
by rewrite /sem_pexpr /= /get_gvar /= hget wrepr_unsigned.
123+
Qed.
124+
125+
End WITH_PARAMS.
126+
127+
End ARMCopnP.
128+
48129
Module ARMFopnP.
49130

50131
Section WITH_PARAMS.

proofs/compiler/arm_params_core.v

+1
Original file line numberDiff line numberDiff line change
@@ -105,4 +105,5 @@ Module ARMOpn_core (Args : OpnArgs).
105105

106106
End ARMOpn_core.
107107

108+
Module ARMCopn_core := ARMOpn_core(CopnArgs).
108109
Module ARMFopn_core := ARMOpn_core(FopnArgs).

0 commit comments

Comments
 (0)