Skip to content

Commit 3f5bdfb

Browse files
committed
optimized bounds-checks to eliminate some shifts
1 parent e334d49 commit 3f5bdfb

File tree

1 file changed

+34
-41
lines changed

1 file changed

+34
-41
lines changed

middle_end/flambda2/from_lambda/lambda_to_flambda_primitives.ml

+34-41
Original file line numberDiff line numberDiff line change
@@ -604,30 +604,30 @@ let checked_alignment ~dbg ~primitive ~conditions : H.expr_primitive =
604604

605605
let check_bound ~(index_kind : Lambda.array_index_kind) ~(bound_kind : I.t)
606606
~index ~bound : H.expr_primitive =
607-
let (comp_kind : I.t), index, bound =
608-
let convert_bound_to dst =
609-
H.Prim
610-
(Unary (Num_conv { src = I_or_f.of_standard_int bound_kind; dst }, bound))
611-
in
612-
(* The reason why we convert the bound instead of the index value is because
613-
of edge cases around large negative numbers.
614-
615-
Given [-9223372036854775807] as a [Naked_int64] index, its bit
616-
representation is
617-
[0b1000000000000000000000000000000000000000000000000000000000000001]. If
618-
we convert that into a [Tagged_immediate], it becomes [0b11] and the
619-
bounds check would pass in cases that we should reject.
620-
621-
This also has the added benefit of producing better assembly code.
622-
Usually saving one instruction compared to tagging the index value. *)
607+
let index_kind =
623608
match index_kind with
624-
| Ptagged_int_index ->
625-
I.Naked_immediate, untag_int index, convert_bound_to Naked_immediate
626-
| Punboxed_int_index bint ->
627-
( standard_int_of_unboxed_integer bint,
628-
index,
629-
convert_bound_to (standard_int_or_float_of_unboxed_integer bint) )
609+
| Ptagged_int_index -> I.Tagged_immediate
610+
| Punboxed_int_index width -> standard_int_of_unboxed_integer width
611+
in
612+
let comp_kind : I.t =
613+
match index_kind, bound_kind with
614+
| Tagged_immediate, Tagged_immediate -> Tagged_immediate
615+
| Naked_int64, _ | _, Naked_int64 -> Naked_int64
616+
| ( (Naked_nativeint | Tagged_immediate | Naked_immediate | Naked_int32),
617+
(Naked_nativeint | Tagged_immediate | Naked_immediate | Naked_int32) )
618+
->
619+
Naked_nativeint
630620
in
621+
let conv x ~src =
622+
if I.equal src comp_kind
623+
then x
624+
else
625+
let src = I_or_f.of_standard_int src in
626+
let dst = I_or_f.of_standard_int comp_kind in
627+
H.Prim (Unary (Num_conv { src; dst }, x))
628+
in
629+
let index = conv index ~src:index_kind in
630+
let bound = conv bound ~src:bound_kind in
631631
Binary (Int_comp (comp_kind, Yielding_bool (Lt Unsigned)), index, bound)
632632

633633
(* This computes the maximum of a given value [x] with zero, in an optimized
@@ -673,7 +673,14 @@ let actual_max_length_for_string_like_access ~size_int
673673
| Sixty_four -> 7
674674
| One_twenty_eight _ -> 15
675675
in
676-
Targetint_31_63.of_int offset
676+
Targetint_32_64.of_int offset
677+
in
678+
(* We need to convert the length into a naked_nativeint because the optimised
679+
version of the max_with_zero function needs to be on machine-width integers
680+
to work (or at least on an integer number of bytes to work). *)
681+
let length =
682+
H.Prim
683+
(Unary (Num_conv { src = Naked_immediate; dst = Naked_nativeint }, length))
677684
in
678685
match access_size with
679686
| Eight -> length (* micro-optimization *)
@@ -682,31 +689,17 @@ let actual_max_length_for_string_like_access ~size_int
682689
let reduced_length =
683690
H.Prim
684691
(Binary
685-
( Int_arith (Naked_immediate, Sub),
692+
( Int_arith (Naked_nativeint, Sub),
686693
length,
687-
Simple (Simple.const (Reg_width_const.naked_immediate offset)) ))
694+
Simple (Simple.const (Reg_width_const.naked_nativeint offset)) ))
688695
in
689-
(* We need to convert the length into a naked_nativeint because the
690-
optimised version of the max_with_zero function needs to be on
691-
machine-width integers to work (or at least on an integer number of bytes
692-
to work). *)
693-
let reduced_length_nativeint =
694-
H.Prim
695-
(Unary
696-
( Num_conv { src = Naked_immediate; dst = Naked_nativeint },
697-
reduced_length ))
698-
in
699-
let nativeint_res = max_with_zero ~size_int reduced_length_nativeint in
700-
H.Prim
701-
(Unary
702-
( Num_conv { src = Naked_nativeint; dst = Naked_immediate },
703-
nativeint_res ))
696+
max_with_zero ~size_int reduced_length
704697

705698
(* String-like validity conditions *)
706699

707700
let string_like_access_validity_condition ~size_int ~access_size ~length
708701
~index_kind index : H.expr_primitive =
709-
check_bound ~index_kind ~bound_kind:Naked_immediate ~index
702+
check_bound ~index_kind ~bound_kind:Naked_nativeint ~index
710703
~bound:
711704
(actual_max_length_for_string_like_access ~size_int ~access_size length)
712705

0 commit comments

Comments
 (0)