| //===------ LeonPasses.cpp - Define passes specific to LEON ---------------===// | 
 | // | 
 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
 | // See https://llvm.org/LICENSE.txt for license information. | 
 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 | // | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | #include "LeonPasses.h" | 
 | #include "SparcSubtarget.h" | 
 | #include "llvm/CodeGen/MachineBasicBlock.h" | 
 | #include "llvm/CodeGen/MachineFunction.h" | 
 | #include "llvm/CodeGen/MachineInstr.h" | 
 | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
 | #include "llvm/Support/raw_ostream.h" | 
 |  | 
 | using namespace llvm; | 
 |  | 
 | char ErrataWorkaround::ID = 0; | 
 |  | 
 | ErrataWorkaround::ErrataWorkaround() : MachineFunctionPass(ID) { | 
 |   initializeErrataWorkaroundPass(*PassRegistry::getPassRegistry()); | 
 | } | 
 |  | 
 | INITIALIZE_PASS(ErrataWorkaround, "errata-workaround", "Errata workaround pass", | 
 |                 false, false) | 
 |  | 
 | // Move iterator to the next instruction in the function, ignoring | 
 | // meta instructions and inline assembly. Returns false when reaching | 
 | // the end of the function. | 
 | bool ErrataWorkaround::moveNext(MachineBasicBlock::iterator &I) { | 
 |  | 
 |   MachineBasicBlock *MBB = I->getParent(); | 
 |  | 
 |   do { | 
 |     I++; | 
 |  | 
 |     while (I == MBB->end()) { | 
 |       if (MBB->getFallThrough() == nullptr) | 
 |         return false; | 
 |       MBB = MBB->getFallThrough(); | 
 |       I = MBB->begin(); | 
 |     } | 
 |   } while (I->isMetaInstruction() || I->isInlineAsm()); | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | void ErrataWorkaround::insertNop(MachineBasicBlock::iterator I) { | 
 |   BuildMI(*I->getParent(), I, I->getDebugLoc(), TII->get(SP::NOP)); | 
 | } | 
 |  | 
 | bool ErrataWorkaround::isFloat(MachineBasicBlock::iterator I) { | 
 |   if (I->getNumOperands() == 0) | 
 |     return false; | 
 |  | 
 |   if (!I->getOperand(0).isReg()) | 
 |     return false; | 
 |  | 
 |   unsigned reg = I->getOperand(0).getReg(); | 
 |  | 
 |   if (!SP::FPRegsRegClass.contains(reg) && !SP::DFPRegsRegClass.contains(reg)) | 
 |     return false; | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | bool ErrataWorkaround::isDivSqrt(MachineBasicBlock::iterator I) { | 
 |   switch (I->getOpcode()) { | 
 |   case SP::FDIVS: | 
 |   case SP::FDIVD: | 
 |   case SP::FSQRTS: | 
 |   case SP::FSQRTD: | 
 |     return true; | 
 |   } | 
 |   return false; | 
 | } | 
 |  | 
 | // Prevents the following code sequence from being generated: | 
 | // (stb/sth/st/stf) -> (single non-store/load instruction) -> (any store) | 
 | // If the sequence is detected a NOP instruction is inserted after | 
 | // the first store instruction. | 
 | bool ErrataWorkaround::checkSeqTN0009A(MachineBasicBlock::iterator I) { | 
 |   switch (I->getOpcode()) { | 
 |   case SP::STrr: | 
 |   case SP::STri: | 
 |   case SP::STBrr: | 
 |   case SP::STBri: | 
 |   case SP::STHrr: | 
 |   case SP::STHri: | 
 |   case SP::STFrr: | 
 |   case SP::STFri: | 
 |     break; | 
 |   default: | 
 |     return false; | 
 |   } | 
 |  | 
 |   MachineBasicBlock::iterator MI = I; | 
 |   if (!moveNext(MI)) | 
 |     return false; | 
 |  | 
 |   if (MI->mayStore() || MI->mayLoad()) | 
 |     return false; | 
 |  | 
 |   MachineBasicBlock::iterator PatchHere = MI; | 
 |  | 
 |   if (!moveNext(MI)) | 
 |     return false; | 
 |  | 
 |   if (!MI->mayStore()) | 
 |     return false; | 
 |  | 
 |   insertNop(PatchHere); | 
 |   return true; | 
 | } | 
 |  | 
 | // Prevents the following code sequence from being generated: | 
 | // (std/stdf) -> (any store) | 
 | // If the sequence is detected a NOP instruction is inserted after | 
 | // the first store instruction. | 
 | bool ErrataWorkaround::checkSeqTN0009B(MachineBasicBlock::iterator I) { | 
 |  | 
 |   switch (I->getOpcode()) { | 
 |   case SP::STDrr: | 
 |   case SP::STDri: | 
 |   case SP::STDFrr: | 
 |   case SP::STDFri: | 
 |     break; | 
 |   default: | 
 |     return false; | 
 |   } | 
 |  | 
 |   MachineBasicBlock::iterator MI = I; | 
 |  | 
 |   if (!moveNext(MI)) | 
 |     return false; | 
 |  | 
 |   if (!MI->mayStore()) | 
 |     return false; | 
 |  | 
 |   insertNop(MI); | 
 |   return true; | 
 | } | 
 |  | 
 | // Insert a NOP at branch target if load in delay slot and atomic | 
 | // instruction at branch target. Also insert a NOP between load | 
 | // instruction and atomic instruction (swap or casa). | 
 | bool ErrataWorkaround::checkSeqTN0010(MachineBasicBlock::iterator I) { | 
 |  | 
 |   // Check for load instruction or branch bundled with load instruction | 
 |   if (!I->mayLoad()) | 
 |     return false; | 
 |  | 
 |   // Check for branch to atomic instruction with load in delay slot | 
 |   if (I->isBranch()) { | 
 |     MachineBasicBlock *TargetMBB = I->getOperand(0).getMBB(); | 
 |     MachineBasicBlock::iterator MI = TargetMBB->begin(); | 
 |  | 
 |     while (MI != TargetMBB->end() && MI->isMetaInstruction()) | 
 |       MI++; | 
 |  | 
 |     if (MI == TargetMBB->end()) | 
 |       return false; | 
 |  | 
 |     switch (MI->getOpcode()) { | 
 |     case SP::SWAPrr: | 
 |     case SP::SWAPri: | 
 |     case SP::CASArr: | 
 |       insertNop(MI); | 
 |       break; | 
 |     default: | 
 |       break; | 
 |     } | 
 |   } | 
 |  | 
 |   // Check for load followed by atomic instruction | 
 |   MachineBasicBlock::iterator MI = I; | 
 |   if (!moveNext(MI)) | 
 |     return false; | 
 |  | 
 |   switch (MI->getOpcode()) { | 
 |   case SP::SWAPrr: | 
 |   case SP::SWAPri: | 
 |   case SP::CASArr: | 
 |     break; | 
 |   default: | 
 |     return false; | 
 |   } | 
 |   insertNop(MI); | 
 |   return true; | 
 | } | 
 |  | 
 | // Do not allow functions to begin with an atomic instruction | 
 | bool ErrataWorkaround::checkSeqTN0010First(MachineBasicBlock &MBB) { | 
 |   MachineBasicBlock::iterator I = MBB.begin(); | 
 |   while (I != MBB.end() && I->isMetaInstruction()) | 
 |     I++; | 
 |   switch (I->getOpcode()) { | 
 |   case SP::SWAPrr: | 
 |   case SP::SWAPri: | 
 |   case SP::CASArr: | 
 |     break; | 
 |   default: | 
 |     return false; | 
 |   } | 
 |   insertNop(I); | 
 |   return true; | 
 | } | 
 |  | 
 | // Inserts a NOP instruction at the target of an integer branch if the | 
 | // target is a floating-point instruction or floating-point branch. | 
 | bool ErrataWorkaround::checkSeqTN0012(MachineBasicBlock::iterator I) { | 
 |  | 
 |   if (I->getOpcode() != SP::BCOND && I->getOpcode() != SP::BCONDA) | 
 |     return false; | 
 |  | 
 |   MachineBasicBlock *TargetMBB = I->getOperand(0).getMBB(); | 
 |   MachineBasicBlock::iterator MI = TargetMBB->begin(); | 
 |  | 
 |   while (MI != TargetMBB->end() && MI->isMetaInstruction()) | 
 |     MI++; | 
 |  | 
 |   if (MI == TargetMBB->end()) | 
 |     return false; | 
 |  | 
 |   if (!isFloat(MI) && MI->getOpcode() != SP::FBCOND) | 
 |     return false; | 
 |  | 
 |   insertNop(MI); | 
 |   return true; | 
 | } | 
 |  | 
 | // Prevents the following code sequence from being generated: | 
 | // (div/sqrt) -> (2 to 3 floating-point operations or loads) -> (div/sqrt) | 
 | // If the sequence is detected one or two NOP instruction are inserted after | 
 | // the first div/sqrt instruction. No NOPs are inserted if one of the floating- | 
 | // point instructions in the middle of the sequence is a (div/sqrt), or if | 
 | // they have dependency on the destination register of the first (div/sqrt). | 
 | // | 
 | // The function also prevents the following code sequence from being generated, | 
 | // (div/sqrt) -> (branch), by inserting a NOP instruction after the (div/sqrt). | 
 | bool ErrataWorkaround::checkSeqTN0013(MachineBasicBlock::iterator I) { | 
 |  | 
 |   if (!isDivSqrt(I)) | 
 |     return false; | 
 |  | 
 |   unsigned dstReg = I->getOperand(0).getReg(); | 
 |  | 
 |   MachineBasicBlock::iterator MI = I; | 
 |   if (!moveNext(MI)) | 
 |     return false; | 
 |  | 
 |   if (MI->isBranch()) { | 
 |     insertNop(MI); | 
 |     return true; | 
 |   } | 
 |  | 
 |   MachineBasicBlock::iterator PatchHere = MI; | 
 |  | 
 |   unsigned fpFound = 0; | 
 |   for (unsigned i = 0; i < 4; i++) { | 
 |  | 
 |     if (!isFloat(MI)) { | 
 |       if (!moveNext(MI)) | 
 |         return false; | 
 |       continue; | 
 |     } | 
 |  | 
 |     if (MI->readsRegister(dstReg, TRI)) | 
 |       return false; | 
 |  | 
 |     if (isDivSqrt(MI)) { | 
 |       if (i < 2) | 
 |         return false; | 
 |       if (fpFound < 2) | 
 |         return false; | 
 |  | 
 |       insertNop(PatchHere); | 
 |       if (i == 2) | 
 |         insertNop(PatchHere); | 
 |       return true; | 
 |     } | 
 |  | 
 |     fpFound++; | 
 |     if (!moveNext(MI)) | 
 |       return false; | 
 |   } | 
 |  | 
 |   return false; | 
 | } | 
 |  | 
 | bool ErrataWorkaround::runOnMachineFunction(MachineFunction &MF) { | 
 |   bool Changed = false; | 
 |   ST = &MF.getSubtarget<SparcSubtarget>(); | 
 |  | 
 |   if (!(ST->fixTN0009() || ST->fixTN0010() || ST->fixTN0012() || | 
 |         ST->fixTN0013())) | 
 |     return false; | 
 |  | 
 |   TII = ST->getInstrInfo(); | 
 |   TRI = ST->getRegisterInfo(); | 
 |  | 
 |   if (ST->fixTN0010()) | 
 |     Changed |= checkSeqTN0010First(MF.front()); | 
 |  | 
 |   for (auto &MBB : MF) { | 
 |     for (auto &I : MBB) { | 
 |       if (ST->fixTN0009()) { | 
 |         Changed |= checkSeqTN0009A(I); | 
 |         Changed |= checkSeqTN0009B(I); | 
 |       } | 
 |       if (ST->fixTN0010()) | 
 |         Changed |= checkSeqTN0010(I); | 
 |       if (ST->fixTN0012()) | 
 |         Changed |= checkSeqTN0012(I); | 
 |       if (ST->fixTN0013()) | 
 |         Changed |= checkSeqTN0013(I); | 
 |     } | 
 |   } | 
 |   return Changed; | 
 | } | 
 |  | 
 | LEONMachineFunctionPass::LEONMachineFunctionPass(char &ID) | 
 |     : MachineFunctionPass(ID) {} | 
 |  | 
 | //***************************************************************************** | 
 | //**** InsertNOPLoad pass | 
 | //***************************************************************************** | 
 | // This pass fixes the incorrectly working Load instructions that exists for | 
 | // some earlier versions of the LEON processor line. NOP instructions must | 
 | // be inserted after the load instruction to ensure that the Load instruction | 
 | // behaves as expected for these processors. | 
 | // | 
 | // This pass inserts a NOP after any LD or LDF instruction. | 
 | // | 
 | char InsertNOPLoad::ID = 0; | 
 |  | 
 | InsertNOPLoad::InsertNOPLoad() : LEONMachineFunctionPass(ID) {} | 
 |  | 
 | bool InsertNOPLoad::runOnMachineFunction(MachineFunction &MF) { | 
 |   Subtarget = &MF.getSubtarget<SparcSubtarget>(); | 
 |   if (!Subtarget->insertNOPLoad()) | 
 |     return false; | 
 |  | 
 |   const TargetInstrInfo &TII = *Subtarget->getInstrInfo(); | 
 |   DebugLoc DL = DebugLoc(); | 
 |  | 
 |   bool Modified = false; | 
 |   for (MachineBasicBlock &MBB : MF) { | 
 |     for (auto MBBI = MBB.begin(), E = MBB.end(); MBBI != E; ++MBBI) { | 
 |       MachineInstr &MI = *MBBI; | 
 |       unsigned Opcode = MI.getOpcode(); | 
 |       if (Opcode >= SP::LDDArr && Opcode <= SP::LDrr) { | 
 |         MachineBasicBlock::iterator NMBBI = std::next(MBBI); | 
 |         BuildMI(MBB, NMBBI, DL, TII.get(SP::NOP)); | 
 |         Modified = true; | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   return Modified; | 
 | } | 
 |  | 
 |  | 
 |  | 
 | //***************************************************************************** | 
 | //**** DetectRoundChange pass | 
 | //***************************************************************************** | 
 | // To prevent any explicit change of the default rounding mode, this pass | 
 | // detects any call of the fesetround function. | 
 | // A warning is generated to ensure the user knows this has happened. | 
 | // | 
 | // Detects an erratum in UT699 LEON 3 processor | 
 |  | 
 | char DetectRoundChange::ID = 0; | 
 |  | 
 | DetectRoundChange::DetectRoundChange() : LEONMachineFunctionPass(ID) {} | 
 |  | 
 | bool DetectRoundChange::runOnMachineFunction(MachineFunction &MF) { | 
 |   Subtarget = &MF.getSubtarget<SparcSubtarget>(); | 
 |   if (!Subtarget->detectRoundChange()) | 
 |     return false; | 
 |  | 
 |   bool Modified = false; | 
 |   for (MachineBasicBlock &MBB : MF) { | 
 |     for (MachineInstr &MI : MBB) { | 
 |       unsigned Opcode = MI.getOpcode(); | 
 |       if (Opcode == SP::CALL && MI.getNumOperands() > 0) { | 
 |         MachineOperand &MO = MI.getOperand(0); | 
 |  | 
 |         if (MO.isGlobal()) { | 
 |           StringRef FuncName = MO.getGlobal()->getName(); | 
 |           if (FuncName.compare_insensitive("fesetround") == 0) { | 
 |             errs() << "Error: You are using the detectroundchange " | 
 |                       "option to detect rounding changes that will " | 
 |                       "cause LEON errata. The only way to fix this " | 
 |                       "is to remove the call to fesetround from " | 
 |                       "the source code.\n"; | 
 |           } | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   return Modified; | 
 | } | 
 |  | 
 | //***************************************************************************** | 
 | //**** FixAllFDIVSQRT pass | 
 | //***************************************************************************** | 
 | // This pass fixes the incorrectly working FDIVx and FSQRTx instructions that | 
 | // exist for some earlier versions of the LEON processor line. Five NOP | 
 | // instructions need to be inserted after these instructions to ensure the | 
 | // correct result is placed in the destination registers before they are used. | 
 | // | 
 | // This pass implements two fixes: | 
 | //  1) fixing the FSQRTS and FSQRTD instructions. | 
 | //  2) fixing the FDIVS and FDIVD instructions. | 
 | // | 
 | // FSQRTS and FDIVS are converted to FDIVD and FSQRTD respectively earlier in | 
 | // the pipeline when this option is enabled, so this pass needs only to deal | 
 | // with the changes that still need implementing for the "double" versions | 
 | // of these instructions. | 
 | // | 
 | char FixAllFDIVSQRT::ID = 0; | 
 |  | 
 | FixAllFDIVSQRT::FixAllFDIVSQRT() : LEONMachineFunctionPass(ID) {} | 
 |  | 
 | bool FixAllFDIVSQRT::runOnMachineFunction(MachineFunction &MF) { | 
 |   Subtarget = &MF.getSubtarget<SparcSubtarget>(); | 
 |   if (!Subtarget->fixAllFDIVSQRT()) | 
 |     return false; | 
 |  | 
 |   const TargetInstrInfo &TII = *Subtarget->getInstrInfo(); | 
 |   DebugLoc DL = DebugLoc(); | 
 |  | 
 |   bool Modified = false; | 
 |   for (MachineBasicBlock &MBB : MF) { | 
 |     for (auto MBBI = MBB.begin(), E = MBB.end(); MBBI != E; ++MBBI) { | 
 |       MachineInstr &MI = *MBBI; | 
 |       unsigned Opcode = MI.getOpcode(); | 
 |  | 
 |       // Note: FDIVS and FSQRTS cannot be generated when this erratum fix is | 
 |       // switched on so we don't need to check for them here. They will | 
 |       // already have been converted to FSQRTD or FDIVD earlier in the | 
 |       // pipeline. | 
 |       if (Opcode == SP::FSQRTD || Opcode == SP::FDIVD) { | 
 |         for (int InsertedCount = 0; InsertedCount < 5; InsertedCount++) | 
 |           BuildMI(MBB, MBBI, DL, TII.get(SP::NOP)); | 
 |  | 
 |         MachineBasicBlock::iterator NMBBI = std::next(MBBI); | 
 |         for (int InsertedCount = 0; InsertedCount < 28; InsertedCount++) | 
 |           BuildMI(MBB, NMBBI, DL, TII.get(SP::NOP)); | 
 |  | 
 |         Modified = true; | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   return Modified; | 
 | } |