@@ -604,30 +604,30 @@ let checked_alignment ~dbg ~primitive ~conditions : H.expr_primitive =
604
604
605
605
let check_bound ~(index_kind : Lambda.array_index_kind ) ~(bound_kind : I.t )
606
606
~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 =
623
608
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
630
620
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
631
631
Binary (Int_comp (comp_kind, Yielding_bool (Lt Unsigned )), index, bound)
632
632
633
633
(* 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
673
673
| Sixty_four -> 7
674
674
| One_twenty_eight _ -> 15
675
675
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))
677
684
in
678
685
match access_size with
679
686
| Eight -> length (* micro-optimization *)
@@ -682,31 +689,17 @@ let actual_max_length_for_string_like_access ~size_int
682
689
let reduced_length =
683
690
H. Prim
684
691
(Binary
685
- ( Int_arith (Naked_immediate , Sub ),
692
+ ( Int_arith (Naked_nativeint , Sub ),
686
693
length,
687
- Simple (Simple. const (Reg_width_const. naked_immediate offset)) ))
694
+ Simple (Simple. const (Reg_width_const. naked_nativeint offset)) ))
688
695
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
704
697
705
698
(* String-like validity conditions *)
706
699
707
700
let string_like_access_validity_condition ~size_int ~access_size ~length
708
701
~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
710
703
~bound:
711
704
(actual_max_length_for_string_like_access ~size_int ~access_size length)
712
705
0 commit comments