-- | Functions to implement the @FR@ (as in "free regs") type class. -- -- For LLVM GHC calling convention (used registers), see -- https://github.com/llvm/llvm-project/blob/6ab900f8746e7d8e24afafb5886a40801f6799f4/llvm/lib/Target/RISCV/RISCVISelLowering.cpp#L13638-L13685 module GHC.CmmToAsm.Reg.Linear.RV64 ( allocateReg, getFreeRegs, initFreeRegs, releaseReg, FreeRegs (..), ) where import Data.Word import GHC.CmmToAsm.RV64.Regs import GHC.Platform import GHC.Platform.Reg import GHC.Platform.Reg.Class.Separate import GHC.Prelude import GHC.Stack import GHC.Utils.Outputable import GHC.Utils.Panic -- | Bitmaps to indicate which registers are free (currently unused) -- -- The bit index represents the `RegNo`, in case of floating point registers -- with an offset of 32. The register is free when the bit is set. data FreeRegs = FreeRegs -- | integer/general purpose registers (`RcInteger`) !Word32 -- | floating point registers (`RcDouble`) !Word32 instance Show FreeRegs where show (FreeRegs g f) = "FreeRegs 0b" ++ showBits g ++ " 0b" ++ showBits f -- | Show bits as a `String` of @1@s and @0@s showBits :: Word32 -> String showBits w = map (\i -> if testBit w i then '1' else '0') [0 .. 31] instance Outputable FreeRegs where ppr (FreeRegs g f) = text " " <+> foldr (\i x -> pad_int i <+> x) (text "") [0 .. 31] $$ text "GPR" <+> foldr (\i x -> show_bit g i <+> x) (text "") [0 .. 31] $$ text "FPR" <+> foldr (\i x -> show_bit f i <+> x) (text "") [0 .. 31] where pad_int i | i < 10 = char ' ' <> int i pad_int i = int i -- remember bit = 1 means it's available. show_bit bits bit | testBit bits bit = text " " show_bit _ _ = text " x" -- | Set bits of all allocatable registers to 1 initFreeRegs :: Platform -> FreeRegs initFreeRegs platform = foldl' (flip releaseReg) noFreeRegs (allocatableRegs platform) where noFreeRegs :: FreeRegs noFreeRegs = FreeRegs 0 0 -- | Get all free `RealReg`s (i.e. those where the corresponding bit is 1) getFreeRegs :: RegClass -> FreeRegs -> [RealReg] getFreeRegs cls (FreeRegs g f) = case cls of RcInteger -> go 0 g allocatableIntRegs RcFloat -> go 32 f allocatableDoubleRegs RcVector -> sorry "Linear.RV64.getFreeRegs: vector registers are not supported" where go _ _ [] = [] go off x (i : is) | testBit x i = RealRegSingle (off + i) : (go off x $! is) | otherwise = go off x $! is -- The lists of allocatable registers are manually crafted: Register -- allocation is pretty hot code. We don't want to iterate and map like -- `initFreeRegs` all the time! (The register mappings aren't supposed to -- change often.) allocatableIntRegs = [5 .. 7] ++ [10 .. 17] ++ [28 .. 30] allocatableDoubleRegs = [0 .. 7] ++ [10 .. 17] ++ [28 .. 31] -- | Set corresponding register bit to 0 allocateReg :: (HasCallStack) => RealReg -> FreeRegs -> FreeRegs allocateReg (RealRegSingle r) (FreeRegs g f) | r > 31 && testBit f (r - 32) = FreeRegs g (clearBit f (r - 32)) | r < 32 && testBit g r = FreeRegs (clearBit g r) f | r > 31 = panic $ "Linear.RV64.allocReg: double allocation of float reg v" ++ show (r - 32) ++ "; " ++ showBits f | otherwise = pprPanic "Linear.RV64.allocReg" $ text ("double allocation of gp reg x" ++ show r ++ "; " ++ showBits g) -- | Set corresponding register bit to 1 releaseReg :: (HasCallStack) => RealReg -> FreeRegs -> FreeRegs releaseReg (RealRegSingle r) (FreeRegs g f) | r > 31 && testBit f (r - 32) = pprPanic "Linear.RV64.releaseReg" (text "can't release non-allocated reg v" <> int (r - 32)) | r < 32 && testBit g r = pprPanic "Linear.RV64.releaseReg" (text "can't release non-allocated reg x" <> int r) | r > 31 = FreeRegs g (setBit f (r - 32)) | otherwise = FreeRegs (setBit g r) f