use std::fmt; use rustc_data_structures::fx::FxIndexSet; use rustc_span::Symbol; use super::{InlineAsmArch, InlineAsmType, ModifierInfo}; use crate::spec::{RelocModel, Target}; def_reg_class! { Sparc SparcInlineAsmRegClass { reg, yreg, } } impl SparcInlineAsmRegClass { pub fn valid_modifiers(self, _arch: super::InlineAsmArch) -> &'static [char] { &[] } pub fn suggest_class(self, _arch: InlineAsmArch, _ty: InlineAsmType) -> Option { None } pub fn suggest_modifier( self, _arch: InlineAsmArch, _ty: InlineAsmType, ) -> Option { None } pub fn default_modifier(self, _arch: InlineAsmArch) -> Option { None } pub fn supported_types( self, arch: InlineAsmArch, ) -> &'static [(InlineAsmType, Option)] { match self { Self::reg => { if arch == InlineAsmArch::Sparc { types! { _: I8, I16, I32; // FIXME: i64 is ok for g*/o* registers on SPARC-V8+ ("h" constraint in GCC), // but not yet supported in LLVM. // v8plus: I64; } } else { types! { _: I8, I16, I32, I64; } } } Self::yreg => &[], } } } fn reserved_g5( arch: InlineAsmArch, _reloc_model: RelocModel, _target_features: &FxIndexSet, _target: &Target, _is_clobber: bool, ) -> Result<(), &'static str> { if arch == InlineAsmArch::Sparc { // FIXME: Section 2.1.5 "Function Registers with Unassigned Roles" of the V8+ Technical // Specification says "%g5; no longer reserved for system software" [1], but LLVM always // reserves it on SPARC32 [2]. // [1]: https://temlib.org/pub/SparcStation/Standards/V8plus.pdf // [2]: https://github.com/llvm/llvm-project/blob/llvmorg-19.1.0/llvm/lib/Target/Sparc/SparcRegisterInfo.cpp#L64-L66 Err("g5 is reserved for system on SPARC32") } else { Ok(()) } } def_regs! { Sparc SparcInlineAsmReg SparcInlineAsmRegClass { // FIXME: // - LLVM has reserve-{g,o,l,i}N feature to reserve each general-purpose registers. // - g2-g4 are reserved for application (optional in both LLVM and GCC, and GCC has -mno-app-regs option to reserve them). // There are currently no builtin targets that use them, but in the future they may need to // be supported via options similar to AArch64's -Z fixed-x18. r2: reg = ["r2", "g2"], // % reserved_g2 r3: reg = ["r3", "g3"], // % reserved_g3 r4: reg = ["r4", "g4"], // % reserved_g4 r5: reg = ["r5", "g5"] % reserved_g5, r8: reg = ["r8", "o0"], // % reserved_o0 r9: reg = ["r9", "o1"], // % reserved_o1 r10: reg = ["r10", "o2"], // % reserved_o2 r11: reg = ["r11", "o3"], // % reserved_o3 r12: reg = ["r12", "o4"], // % reserved_o4 r13: reg = ["r13", "o5"], // % reserved_o5 r15: reg = ["r15", "o7"], // % reserved_o7 r16: reg = ["r16", "l0"], // % reserved_l0 r17: reg = ["r17", "l1"], // % reserved_l1 r18: reg = ["r18", "l2"], // % reserved_l2 r19: reg = ["r19", "l3"], // % reserved_l3 r20: reg = ["r20", "l4"], // % reserved_l4 r21: reg = ["r21", "l5"], // % reserved_l5 r22: reg = ["r22", "l6"], // % reserved_l6 r23: reg = ["r23", "l7"], // % reserved_l7 r24: reg = ["r24", "i0"], // % reserved_i0 r25: reg = ["r25", "i1"], // % reserved_i1 r26: reg = ["r26", "i2"], // % reserved_i2 r27: reg = ["r27", "i3"], // % reserved_i3 r28: reg = ["r28", "i4"], // % reserved_i4 r29: reg = ["r29", "i5"], // % reserved_i5 y: yreg = ["y"], #error = ["r0", "g0"] => "g0 is always zero and cannot be used as an operand for inline asm", // FIXME: %g1 is volatile in ABI, but used internally by LLVM. // https://github.com/llvm/llvm-project/blob/llvmorg-19.1.0/llvm/lib/Target/Sparc/SparcRegisterInfo.cpp#L55-L56 // > FIXME: G1 reserved for now for large imm generation by frame code. #error = ["r1", "g1"] => "reserved by LLVM and cannot be used as an operand for inline asm", #error = ["r6", "g6", "r7", "g7"] => "reserved for system and cannot be used as an operand for inline asm", #error = ["sp", "r14", "o6"] => "the stack pointer cannot be used as an operand for inline asm", #error = ["fp", "r30", "i6"] => "the frame pointer cannot be used as an operand for inline asm", #error = ["r31", "i7"] => "the return address register cannot be used as an operand for inline asm", } } impl SparcInlineAsmReg { pub fn emit( self, out: &mut dyn fmt::Write, _arch: InlineAsmArch, _modifier: Option, ) -> fmt::Result { write!(out, "%{}", self.name()) } }