Skip to content

Commit a4fd8b1

Browse files
committed
Fix {AArch64,Arm64EC}InlineAsmReg::emit
1 parent 4eb8d93 commit a4fd8b1

File tree

3 files changed

+122
-101
lines changed

3 files changed

+122
-101
lines changed

compiler/rustc_codegen_llvm/src/asm.rs

+8-95
Original file line numberDiff line numberDiff line change
@@ -542,107 +542,20 @@ fn xmm_reg_index(reg: InlineAsmReg) -> Option<u32> {
542542

543543
/// If the register is an AArch64 integer register then return its index.
544544
fn a64_reg_index(reg: InlineAsmReg) -> Option<u32> {
545-
// Unlike `a64_vreg_index`, we can't subtract `x0` to get the u32 because
546-
// `x19` and `x29` are missing and the integer constants for the
547-
// `x0`..`x30` enum variants don't all match the register number. E.g. the
548-
// integer constant for `x18` is 18, but the constant for `x20` is 19.
549-
Some(match reg {
550-
InlineAsmReg::AArch64(r) => {
551-
use AArch64InlineAsmReg::*;
552-
match r {
553-
x0 => 0,
554-
x1 => 1,
555-
x2 => 2,
556-
x3 => 3,
557-
x4 => 4,
558-
x5 => 5,
559-
x6 => 6,
560-
x7 => 7,
561-
x8 => 8,
562-
x9 => 9,
563-
x10 => 10,
564-
x11 => 11,
565-
x12 => 12,
566-
x13 => 13,
567-
x14 => 14,
568-
x15 => 15,
569-
x16 => 16,
570-
x17 => 17,
571-
x18 => 18,
572-
// x19 is reserved
573-
x20 => 20,
574-
x21 => 21,
575-
x22 => 22,
576-
x23 => 23,
577-
x24 => 24,
578-
x25 => 25,
579-
x26 => 26,
580-
x27 => 27,
581-
x28 => 28,
582-
// x29 is reserved
583-
x30 => 30,
584-
_ => return None,
585-
}
586-
}
587-
InlineAsmReg::Arm64EC(r) => {
588-
use Arm64ECInlineAsmReg::*;
589-
match r {
590-
x0 => 0,
591-
x1 => 1,
592-
x2 => 2,
593-
x3 => 3,
594-
x4 => 4,
595-
x5 => 5,
596-
x6 => 6,
597-
x7 => 7,
598-
x8 => 8,
599-
x9 => 9,
600-
x10 => 10,
601-
x11 => 11,
602-
x12 => 12,
603-
// x13 is reserved
604-
// x14 is reserved
605-
x15 => 15,
606-
x16 => 16,
607-
x17 => 17,
608-
// x18 is reserved
609-
// x19 is reserved
610-
x20 => 20,
611-
x21 => 21,
612-
x22 => 22,
613-
// x23 is reserved
614-
// x24 is reserved
615-
x25 => 25,
616-
x26 => 26,
617-
x27 => 27,
618-
// x28 is reserved
619-
// x29 is reserved
620-
x30 => 30,
621-
_ => return None,
622-
}
623-
}
624-
_ => return None,
625-
})
545+
match reg {
546+
InlineAsmReg::AArch64(r) => r.reg_index(),
547+
InlineAsmReg::Arm64EC(r) => r.reg_index(),
548+
_ => None,
549+
}
626550
}
627551

628552
/// If the register is an AArch64 vector register then return its index.
629553
fn a64_vreg_index(reg: InlineAsmReg) -> Option<u32> {
630554
match reg {
631-
InlineAsmReg::AArch64(reg) => {
632-
use AArch64InlineAsmReg::*;
633-
if reg as u32 >= v0 as u32 && reg as u32 <= v31 as u32 {
634-
return Some(reg as u32 - v0 as u32);
635-
}
636-
}
637-
InlineAsmReg::Arm64EC(reg) => {
638-
use Arm64ECInlineAsmReg::*;
639-
if reg as u32 >= v0 as u32 && reg as u32 <= v15 as u32 {
640-
return Some(reg as u32 - v0 as u32);
641-
}
642-
}
643-
_ => {}
555+
InlineAsmReg::AArch64(reg) => reg.vreg_index(),
556+
InlineAsmReg::Arm64EC(reg) => reg.vreg_index(),
557+
_ => None,
644558
}
645-
None
646559
}
647560

648561
/// Converts a register class to an LLVM constraint code.

compiler/rustc_target/src/asm/aarch64.rs

+57-3
Original file line numberDiff line numberDiff line change
@@ -186,12 +186,66 @@ impl AArch64InlineAsmReg {
186186
_arch: InlineAsmArch,
187187
modifier: Option<char>,
188188
) -> fmt::Result {
189-
let (prefix, index) = if (self as u32) < Self::v0 as u32 {
190-
(modifier.unwrap_or('x'), self as u32 - Self::x0 as u32)
189+
let (prefix, index) = if let Some(index) = self.reg_index() {
190+
(modifier.unwrap_or('x'), index)
191+
} else if let Some(index) = self.vreg_index() {
192+
(modifier.unwrap_or('v'), index)
191193
} else {
192-
(modifier.unwrap_or('v'), self as u32 - Self::v0 as u32)
194+
return out.write_str(self.name());
193195
};
194196
assert!(index < 32);
195197
write!(out, "{prefix}{index}")
196198
}
199+
200+
/// If the register is an integer register then return its index.
201+
pub fn reg_index(self) -> Option<u32> {
202+
// Unlike `vreg_index`, we can't subtract `x0` to get the u32 because
203+
// `x19` and `x29` are missing and the integer constants for the
204+
// `x0`..`x30` enum variants don't all match the register number. E.g. the
205+
// integer constant for `x18` is 18, but the constant for `x20` is 19.
206+
use AArch64InlineAsmReg::*;
207+
Some(match self {
208+
x0 => 0,
209+
x1 => 1,
210+
x2 => 2,
211+
x3 => 3,
212+
x4 => 4,
213+
x5 => 5,
214+
x6 => 6,
215+
x7 => 7,
216+
x8 => 8,
217+
x9 => 9,
218+
x10 => 10,
219+
x11 => 11,
220+
x12 => 12,
221+
x13 => 13,
222+
x14 => 14,
223+
x15 => 15,
224+
x16 => 16,
225+
x17 => 17,
226+
x18 => 18,
227+
// x19 is reserved
228+
x20 => 20,
229+
x21 => 21,
230+
x22 => 22,
231+
x23 => 23,
232+
x24 => 24,
233+
x25 => 25,
234+
x26 => 26,
235+
x27 => 27,
236+
x28 => 28,
237+
// x29 is reserved
238+
x30 => 30,
239+
_ => return None,
240+
})
241+
}
242+
243+
/// If the register is a vector register then return its index.
244+
pub fn vreg_index(self) -> Option<u32> {
245+
use AArch64InlineAsmReg::*;
246+
if self as u32 >= v0 as u32 && self as u32 <= v31 as u32 {
247+
return Some(self as u32 - v0 as u32);
248+
}
249+
None
250+
}
197251
}

compiler/rustc_target/src/asm/arm64ec.rs

+57-3
Original file line numberDiff line numberDiff line change
@@ -99,12 +99,66 @@ impl Arm64ECInlineAsmReg {
9999
_arch: InlineAsmArch,
100100
modifier: Option<char>,
101101
) -> fmt::Result {
102-
let (prefix, index) = if (self as u32) < Self::v0 as u32 {
103-
(modifier.unwrap_or('x'), self as u32 - Self::x0 as u32)
102+
let (prefix, index) = if let Some(index) = self.reg_index() {
103+
(modifier.unwrap_or('x'), index)
104+
} else if let Some(index) = self.vreg_index() {
105+
(modifier.unwrap_or('v'), index)
104106
} else {
105-
(modifier.unwrap_or('v'), self as u32 - Self::v0 as u32)
107+
return out.write_str(self.name());
106108
};
107109
assert!(index < 32);
108110
write!(out, "{prefix}{index}")
109111
}
112+
113+
/// If the register is an integer register then return its index.
114+
pub fn reg_index(self) -> Option<u32> {
115+
// Unlike `vreg_index`, we can't subtract `x0` to get the u32 because
116+
// `x13`, `x19`, `x29`, etc. are missing and the integer constants for the
117+
// `x0`..`x30` enum variants don't all match the register number. E.g. the
118+
// integer constant for `x12` is 12, but the constant for `x15` is 13.
119+
use Arm64ECInlineAsmReg::*;
120+
Some(match self {
121+
x0 => 0,
122+
x1 => 1,
123+
x2 => 2,
124+
x3 => 3,
125+
x4 => 4,
126+
x5 => 5,
127+
x6 => 6,
128+
x7 => 7,
129+
x8 => 8,
130+
x9 => 9,
131+
x10 => 10,
132+
x11 => 11,
133+
x12 => 12,
134+
// x13 is reserved
135+
// x14 is reserved
136+
x15 => 15,
137+
x16 => 16,
138+
x17 => 17,
139+
// x18 is reserved
140+
// x19 is reserved
141+
x20 => 20,
142+
x21 => 21,
143+
x22 => 22,
144+
// x23 is reserved
145+
// x24 is reserved
146+
x25 => 25,
147+
x26 => 26,
148+
x27 => 27,
149+
// x28 is reserved
150+
// x29 is reserved
151+
x30 => 30,
152+
_ => return None,
153+
})
154+
}
155+
156+
/// If the register is a vector register then return its index.
157+
pub fn vreg_index(self) -> Option<u32> {
158+
use Arm64ECInlineAsmReg::*;
159+
if self as u32 >= v0 as u32 && self as u32 <= v15 as u32 {
160+
return Some(self as u32 - v0 as u32);
161+
}
162+
None
163+
}
110164
}

0 commit comments

Comments
 (0)