diff options
Diffstat (limited to 'llvm/lib/Target/Mips/MipsMCInstLower.cpp')
| -rw-r--r-- | llvm/lib/Target/Mips/MipsMCInstLower.cpp | 75 | 
1 files changed, 75 insertions, 0 deletions
| diff --git a/llvm/lib/Target/Mips/MipsMCInstLower.cpp b/llvm/lib/Target/Mips/MipsMCInstLower.cpp index 7c9a9ed8312..85f78674661 100644 --- a/llvm/lib/Target/Mips/MipsMCInstLower.cpp +++ b/llvm/lib/Target/Mips/MipsMCInstLower.cpp @@ -151,7 +151,82 @@ MCOperand MipsMCInstLower::LowerOperand(const MachineOperand &MO,    return MCOperand();  } +MCOperand MipsMCInstLower::createSub(MachineBasicBlock *BB1, +                                     MachineBasicBlock *BB2, +                                     MCSymbolRefExpr::VariantKind Kind) const { +  const MCSymbolRefExpr *Sym1 = MCSymbolRefExpr::Create(BB1->getSymbol(), *Ctx); +  const MCSymbolRefExpr *Sym2 = MCSymbolRefExpr::Create(BB2->getSymbol(), *Ctx); +  const MCBinaryExpr *Sub = MCBinaryExpr::CreateSub(Sym1, Sym2, *Ctx); + +  return MCOperand::CreateExpr(MipsMCExpr::Create(Kind, Sub, *Ctx)); +} + +void MipsMCInstLower:: +lowerLongBranchLUi(const MachineInstr *MI, MCInst &OutMI, int Opcode, +                   MCSymbolRefExpr::VariantKind Kind) const { +  OutMI.setOpcode(Opcode); + +  // Lower register operand. +  OutMI.addOperand(LowerOperand(MI->getOperand(0))); + +  // Create %hi($tgt-$baltgt) or %highest($tgt-$baltgt). +  OutMI.addOperand(createSub(MI->getOperand(1).getMBB(), +                             MI->getOperand(2).getMBB(), Kind)); +} + +void MipsMCInstLower:: +lowerLongBranchADDiu(const MachineInstr *MI, MCInst &OutMI, int Opcode, +                     MCSymbolRefExpr::VariantKind Kind) const { +  OutMI.setOpcode(Opcode); + +  // Lower two register operands. +  for (unsigned I = 0, E = 2; I != E; ++I) { +    const MachineOperand &MO = MI->getOperand(I); +    OutMI.addOperand(LowerOperand(MO)); +  } + +  // Create %lo($tgt-$baltgt), %hi($tgt-$baltgt) or %higher($tgt-$baltgt). +  OutMI.addOperand(createSub(MI->getOperand(2).getMBB(), +                             MI->getOperand(3).getMBB(), Kind)); +} + +bool MipsMCInstLower::lowerLongBranch(const MachineInstr *MI, +                                      MCInst &OutMI) const { +  switch (MI->getOpcode()) { +  default: +    return false; +  case Mips::LONG_BRANCH_LUi: +    lowerLongBranchLUi(MI, OutMI, Mips::LUi, MCSymbolRefExpr::VK_Mips_ABS_HI); +    return true; +  case Mips::LONG_BRANCH_LUi64: +    lowerLongBranchLUi(MI, OutMI, Mips::LUi64, +                       MCSymbolRefExpr::VK_Mips_HIGHEST); +    return true; +  case Mips::LONG_BRANCH_ADDiu: +    lowerLongBranchADDiu(MI, OutMI, Mips::ADDiu, +                         MCSymbolRefExpr::VK_Mips_ABS_LO); +    return true; +  case Mips::LONG_BRANCH_DADDiu: +    unsigned TargetFlags = MI->getOperand(2).getTargetFlags(); +    if (TargetFlags == MipsII::MO_HIGHER) +      lowerLongBranchADDiu(MI, OutMI, Mips::DADDiu, +                           MCSymbolRefExpr::VK_Mips_HIGHER); +    else if (TargetFlags == MipsII::MO_ABS_HI) +      lowerLongBranchADDiu(MI, OutMI, Mips::DADDiu, +                           MCSymbolRefExpr::VK_Mips_ABS_HI); +    else if (TargetFlags == MipsII::MO_ABS_LO) +      lowerLongBranchADDiu(MI, OutMI, Mips::DADDiu, +                           MCSymbolRefExpr::VK_Mips_ABS_LO); +    else +      report_fatal_error("Unexpected flags for LONG_BRANCH_DADDiu"); +    return true; +  } +} +  void MipsMCInstLower::Lower(const MachineInstr *MI, MCInst &OutMI) const { +  if (lowerLongBranch(MI, OutMI)) +    return; +    OutMI.setOpcode(MI->getOpcode());    for (unsigned i = 0, e = MI->getNumOperands(); i != e; ++i) { | 

