1
- // FIXME: The assumes we're using the non-vector ABI, i.e., compiling
2
- // for a pre-z13 machine or using -mno-vx.
1
+ use rustc_abi:: { BackendRepr , HasDataLayout , Size , TyAbiInterface , TyAndLayout } ;
3
2
4
- use crate :: abi:: call:: { ArgAbi , FnAbi , Reg } ;
5
- use crate :: abi:: { HasDataLayout , TyAbiInterface } ;
3
+ use crate :: abi:: call:: { ArgAbi , FnAbi , Reg , RegKind } ;
6
4
use crate :: spec:: HasTargetSpec ;
7
5
6
+ fn contains_vector < ' a , Ty , C > ( cx : & C , layout : TyAndLayout < ' a , Ty > , expected_size : Size ) -> bool
7
+ where
8
+ Ty : TyAbiInterface < ' a , C > + Copy ,
9
+ {
10
+ match layout. backend_repr {
11
+ BackendRepr :: Uninhabited | BackendRepr :: Scalar ( _) | BackendRepr :: ScalarPair ( ..) => false ,
12
+ BackendRepr :: Vector { .. } => layout. size == expected_size,
13
+ BackendRepr :: Memory { .. } => {
14
+ for i in 0 ..layout. fields . count ( ) {
15
+ if contains_vector ( cx, layout. field ( cx, i) , expected_size) {
16
+ return true ;
17
+ }
18
+ }
19
+ false
20
+ }
21
+ }
22
+ }
23
+
8
24
fn classify_ret < Ty > ( ret : & mut ArgAbi < ' _ , Ty > ) {
9
- if !ret. layout . is_aggregate ( ) && ret. layout . size . bits ( ) <= 64 {
25
+ let size = ret. layout . size ;
26
+ if size. bits ( ) <= 128 && matches ! ( ret. layout. backend_repr, BackendRepr :: Vector { .. } ) {
27
+ return ;
28
+ }
29
+ if !ret. layout . is_aggregate ( ) && size. bits ( ) <= 64 {
10
30
ret. extend_integer_width_to ( 64 ) ;
11
- } else {
12
- ret. make_indirect ( ) ;
31
+ return ;
13
32
}
33
+ ret. make_indirect ( ) ;
14
34
}
15
35
16
36
fn classify_arg < ' a , Ty , C > ( cx : & C , arg : & mut ArgAbi < ' a , Ty > )
@@ -32,19 +52,25 @@ where
32
52
}
33
53
return ;
34
54
}
35
- if !arg. layout . is_aggregate ( ) && arg. layout . size . bits ( ) <= 64 {
55
+
56
+ let size = arg. layout . size ;
57
+ if size. bits ( ) <= 128 && contains_vector ( cx, arg. layout , size) {
58
+ arg. cast_to ( Reg { kind : RegKind :: Vector , size } ) ;
59
+ return ;
60
+ }
61
+ if !arg. layout . is_aggregate ( ) && size. bits ( ) <= 64 {
36
62
arg. extend_integer_width_to ( 64 ) ;
37
63
return ;
38
64
}
39
65
40
66
if arg. layout . is_single_fp_element ( cx) {
41
- match arg . layout . size . bytes ( ) {
67
+ match size. bytes ( ) {
42
68
4 => arg. cast_to ( Reg :: f32 ( ) ) ,
43
69
8 => arg. cast_to ( Reg :: f64 ( ) ) ,
44
70
_ => arg. make_indirect ( ) ,
45
71
}
46
72
} else {
47
- match arg . layout . size . bytes ( ) {
73
+ match size. bytes ( ) {
48
74
1 => arg. cast_to ( Reg :: i8 ( ) ) ,
49
75
2 => arg. cast_to ( Reg :: i16 ( ) ) ,
50
76
4 => arg. cast_to ( Reg :: i32 ( ) ) ,
0 commit comments