1
1
// reference: https://github.com/espressif/clang-xtensa/commit/6fb488d2553f06029e6611cf81c6efbd45b56e47#diff-aa74ae1e1ab6b7149789237edb78e688R8450
2
2
3
-
4
3
use crate :: abi:: call:: { ArgType , FnType , Reg , Uniform } ;
5
4
6
5
const NUM_ARG_GPR : u64 = 6 ;
7
- const MAX_ARG_IN_REGS_SIZE : u64 = 4 * 32 ;
6
+ const MAX_ARG_IN_REGS_SIZE : u64 = 4 * 32 ;
8
7
// const MAX_ARG_DIRECT_SIZE: u64 = MAX_ARG_IN_REGS_SIZE;
9
- const MAX_RET_IN_REGS_SIZE : u64 = 2 * 32 ;
8
+ const MAX_RET_IN_REGS_SIZE : u64 = 2 * 32 ;
10
9
11
10
fn classify_ret_ty < Ty > ( arg : & mut ArgType < ' _ , Ty > , xlen : u64 ) {
12
11
// The rules for return and argument types are the same, so defer to
13
12
// classifyArgumentType.
14
13
classify_arg_ty ( arg, xlen, & mut 2 ) ; // two as max return size
15
14
}
16
15
17
-
18
16
fn classify_arg_ty < Ty > ( arg : & mut ArgType < ' _ , Ty > , xlen : u64 , remaining_gpr : & mut u64 ) {
19
17
// Determine the number of GPRs needed to pass the current argument
20
18
// according to the ABI. 2*XLen-aligned varargs are passed in "aligned"
21
19
// register pairs, so may consume 3 registers.
22
-
20
+
23
21
let arg_size = arg. layout . size ;
24
22
if arg_size. bits ( ) > MAX_ARG_IN_REGS_SIZE {
25
23
arg. make_indirect ( ) ;
@@ -31,7 +29,7 @@ fn classify_arg_ty<Ty>(arg: &mut ArgType<'_, Ty>, xlen: u64, remaining_gpr: &mut
31
29
32
30
if alignment. bits ( ) == 2 * xlen {
33
31
required_gpr = 2 + ( * remaining_gpr % 2 ) ;
34
- } else if arg_size. bits ( ) > xlen && arg_size. bits ( ) <= MAX_ARG_IN_REGS_SIZE {
32
+ } else if arg_size. bits ( ) > xlen && arg_size. bits ( ) <= MAX_ARG_IN_REGS_SIZE {
35
33
required_gpr = ( arg_size. bits ( ) + ( xlen - 1 ) ) / xlen;
36
34
}
37
35
@@ -44,14 +42,16 @@ fn classify_arg_ty<Ty>(arg: &mut ArgType<'_, Ty>, xlen: u64, remaining_gpr: &mut
44
42
45
43
// if a value can fit in a reg and the
46
44
// stack is not required, extend
47
- if !arg. layout . is_aggregate ( ) { // non-aggregate types
45
+ if !arg. layout . is_aggregate ( ) {
46
+ // non-aggregate types
48
47
if arg_size. bits ( ) < xlen && !stack_required {
49
48
arg. extend_integer_width_to ( xlen) ;
50
49
}
51
- } else if arg_size. bits ( ) as u64 <= MAX_ARG_IN_REGS_SIZE { // aggregate types
50
+ } else if arg_size. bits ( ) as u64 <= MAX_ARG_IN_REGS_SIZE {
51
+ // aggregate types
52
52
// Aggregates which are <= 4*32 will be passed in registers if possible,
53
53
// so coerce to integers.
54
-
54
+
55
55
// Use a single XLen int if possible, 2*XLen if 2*XLen alignment is
56
56
// required, and a 2-element XLen array if only XLen alignment is
57
57
// required.
@@ -61,48 +61,37 @@ fn classify_arg_ty<Ty>(arg: &mut ArgType<'_, Ty>, xlen: u64, remaining_gpr: &mut
61
61
// arg.extend_integer_width_to(arg_size + (xlen - 1) / xlen);
62
62
// }
63
63
if alignment. bits ( ) == 2 * xlen {
64
- arg. cast_to ( Uniform {
65
- unit : Reg :: i64 ( ) ,
66
- total : arg_size
67
- } ) ;
64
+ arg. cast_to ( Uniform { unit : Reg :: i64 ( ) , total : arg_size } ) ;
68
65
} else {
69
- //TODO array type - this should be a homogenous array type
66
+ //FIXME array type - this should be a homogenous array type
70
67
// arg.extend_integer_width_to(arg_size + (xlen - 1) / xlen);
71
68
}
72
-
73
69
} else {
74
70
// if we get here the stack is required
75
71
assert ! ( stack_required) ;
76
72
arg. make_indirect ( ) ;
77
73
}
78
74
79
-
80
- // if arg_size as u64 <= MAX_ARG_IN_REGS_SIZE {
81
- // let align = arg.layout.align.abi.bytes();
82
- // let total = arg.layout.size;
83
- // arg.cast_to(Uniform {
84
- // unit: if align <= 4 { Reg::i32() } else { Reg::i64() },
85
- // total
86
- // });
87
- // return;
88
- // }
89
-
90
-
75
+ // if arg_size as u64 <= MAX_ARG_IN_REGS_SIZE {
76
+ // let align = arg.layout.align.abi.bytes();
77
+ // let total = arg.layout.size;
78
+ // arg.cast_to(Uniform {
79
+ // unit: if align <= 4 { Reg::i32() } else { Reg::i64() },
80
+ // total
81
+ // });
82
+ // return;
83
+ // }
91
84
}
92
85
93
86
pub fn compute_abi_info < Ty > ( fty : & mut FnType < ' _ , Ty > , xlen : u64 ) {
94
87
if !fty. ret . is_ignore ( ) {
95
88
classify_ret_ty ( & mut fty. ret , xlen) ;
96
89
}
97
90
98
- let return_indirect = fty . ret . layout . size . bits ( ) > MAX_RET_IN_REGS_SIZE ||
99
- fty. ret . is_indirect ( ) ;
91
+ let return_indirect =
92
+ fty . ret . layout . size . bits ( ) > MAX_RET_IN_REGS_SIZE || fty. ret . is_indirect ( ) ;
100
93
101
- let mut remaining_gpr = if return_indirect {
102
- NUM_ARG_GPR - 1
103
- } else {
104
- NUM_ARG_GPR
105
- } ;
94
+ let mut remaining_gpr = if return_indirect { NUM_ARG_GPR - 1 } else { NUM_ARG_GPR } ;
106
95
107
96
for arg in & mut fty. args {
108
97
if arg. is_ignore ( ) {
0 commit comments