diff options
Diffstat (limited to 'utils/TableGen/FastISelEmitter.cpp')
| -rw-r--r-- | utils/TableGen/FastISelEmitter.cpp | 243 | 
1 files changed, 127 insertions, 116 deletions
| diff --git a/utils/TableGen/FastISelEmitter.cpp b/utils/TableGen/FastISelEmitter.cpp index 154f96d3d023..748c92347783 100644 --- a/utils/TableGen/FastISelEmitter.cpp +++ b/utils/TableGen/FastISelEmitter.cpp @@ -19,6 +19,7 @@  #include "CodeGenDAGPatterns.h"  #include "llvm/ADT/SmallString.h" +#include "llvm/ADT/StringSwitch.h"  #include "llvm/Support/Debug.h"  #include "llvm/Support/ErrorHandling.h"  #include "llvm/TableGen/Error.h" @@ -36,6 +37,7 @@ struct InstructionMemo {    const CodeGenRegisterClass *RC;    std::string SubRegNo;    std::vector<std::string>* PhysRegs; +  std::string PredicateCheck;  };  } // End anonymous namespace @@ -347,7 +349,7 @@ struct OperandsSignature {          // Implicit physical register operand. e.g. Instruction::Mul expect to          // select to a binary op. On x86, mul may take a single operand with          // the other operand being implicit. We must emit something that looks -        // like a binary instruction except for the very inner FastEmitInst_* +        // like a binary instruction except for the very inner fastEmitInst_*          // call.          continue;        Operands[i].printManglingSuffix(OS, ImmPredicates, StripImmCodes); @@ -364,7 +366,9 @@ struct OperandsSignature {  namespace {  class FastISelMap { -  typedef std::map<std::string, InstructionMemo> PredMap; +  // A multimap is needed instead of a "plain" map because the key is  +  // the instruction's complexity (an int) and they are not unique. +  typedef std::multimap<int, InstructionMemo> PredMap;    typedef std::map<MVT::SimpleValueType, PredMap> RetPredMap;    typedef std::map<MVT::SimpleValueType, RetPredMap> TypeRetPredMap;    typedef std::map<std::string, TypeRetPredMap> OpcodeTypeRetPredMap; @@ -373,6 +377,16 @@ class FastISelMap {    OperandsOpcodeTypeRetPredMap SimplePatterns; +  // This is used to check that there are no duplicate predicates             +  typedef std::multimap<std::string, bool> PredCheckMap; +  typedef std::map<MVT::SimpleValueType, PredCheckMap> RetPredCheckMap; +  typedef std::map<MVT::SimpleValueType, RetPredCheckMap> TypeRetPredCheckMap; +  typedef std::map<std::string, TypeRetPredCheckMap> OpcodeTypeRetPredCheckMap; +  typedef std::map<OperandsSignature, OpcodeTypeRetPredCheckMap> +            OperandsOpcodeTypeRetPredCheckMap; + +  OperandsOpcodeTypeRetPredCheckMap SimplePatternsCheck; +    std::map<OperandsSignature, std::vector<OperandsSignature> >      SignaturesWithConstantForms; @@ -384,6 +398,11 @@ public:    void collectPatterns(CodeGenDAGPatterns &CGP);    void printImmediatePredicates(raw_ostream &OS);    void printFunctionDefinitions(raw_ostream &OS); +private:   +  void emitInstructionCode(raw_ostream &OS,  +                           const OperandsSignature &Operands, +                           const PredMap &PM,  +                           const std::string &RetVTName);  };  } // End anonymous namespace @@ -541,6 +560,17 @@ void FastISelMap::collectPatterns(CodeGenDAGPatterns &CGP) {          continue;      } +    // Check if the operands match one of the patterns handled by FastISel. +    std::string ManglingSuffix; +    raw_string_ostream SuffixOS(ManglingSuffix); +    Operands.PrintManglingSuffix(SuffixOS, ImmediatePredicates, true); +    SuffixOS.flush(); +    if (!StringSwitch<bool>(ManglingSuffix) +        .Cases("", "r", "rr", "ri", "rf", true) +        .Cases("rri", "i", "f", true) +        .Default(false)) +      continue; +      // Get the predicate that guards this pattern.      std::string PredicateCheck = Pattern.getPredicateCheck(); @@ -549,14 +579,24 @@ void FastISelMap::collectPatterns(CodeGenDAGPatterns &CGP) {        Pattern.getDstPattern()->getOperator()->getName(),        DstRC,        SubRegNo, -      PhysRegInputs +      PhysRegInputs, +      PredicateCheck      }; +     +    int complexity = Pattern.getPatternComplexity(CGP); -    if (SimplePatterns[Operands][OpcodeName][VT][RetVT].count(PredicateCheck)) +    if (SimplePatternsCheck[Operands][OpcodeName][VT] +         [RetVT].count(PredicateCheck)) {        PrintFatalError(Pattern.getSrcRecord()->getLoc(), -                    "Duplicate record in FastISel table!"); +                    "Duplicate predicate in FastISel table!"); +    } +    SimplePatternsCheck[Operands][OpcodeName][VT][RetVT].insert( +            std::make_pair(PredicateCheck, true)); -    SimplePatterns[Operands][OpcodeName][VT][RetVT][PredicateCheck] = Memo; +       // Note: Instructions with the same complexity will appear in the order +          // that they are encountered. +    SimplePatterns[Operands][OpcodeName][VT][RetVT].insert( +      std::make_pair(complexity, Memo));      // If any of the operands were immediates with predicates on them, strip      // them down to a signature that doesn't have predicates so that we can @@ -582,6 +622,72 @@ void FastISelMap::printImmediatePredicates(raw_ostream &OS) {    OS << "\n\n";  } +void FastISelMap::emitInstructionCode(raw_ostream &OS,  +                                      const OperandsSignature &Operands, +                                      const PredMap &PM,  +                                      const std::string &RetVTName) { +  // Emit code for each possible instruction. There may be +  // multiple if there are subtarget concerns.  A reverse iterator +  // is used to produce the ones with highest complexity first. + +  bool OneHadNoPredicate = false; +  for (PredMap::const_reverse_iterator PI = PM.rbegin(), PE = PM.rend(); +       PI != PE; ++PI) { +    const InstructionMemo &Memo = PI->second; +    std::string PredicateCheck = Memo.PredicateCheck; + +    if (PredicateCheck.empty()) { +      assert(!OneHadNoPredicate && +             "Multiple instructions match and more than one had " +             "no predicate!"); +      OneHadNoPredicate = true; +    } else { +      if (OneHadNoPredicate) { +        // FIXME: This should be a PrintError once the x86 target +        // fixes PR21575. +        PrintWarning("Multiple instructions match and one with no " +                     "predicate came before one with a predicate!  " +                     "name:" + Memo.Name + "  predicate: " +  +                     PredicateCheck); +      } +      OS << "  if (" + PredicateCheck + ") {\n"; +      OS << "  "; +    } + +    for (unsigned i = 0; i < Memo.PhysRegs->size(); ++i) { +      if ((*Memo.PhysRegs)[i] != "") +        OS << "  BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, " +           << "TII.get(TargetOpcode::COPY), " +           << (*Memo.PhysRegs)[i] << ").addReg(Op" << i << ");\n"; +    } + +    OS << "  return fastEmitInst_"; +    if (Memo.SubRegNo.empty()) { +      Operands.PrintManglingSuffix(OS, *Memo.PhysRegs, +     ImmediatePredicates, true); +      OS << "(" << InstNS << Memo.Name << ", "; +      OS << "&" << InstNS << Memo.RC->getName() << "RegClass"; +      if (!Operands.empty()) +        OS << ", "; +      Operands.PrintArguments(OS, *Memo.PhysRegs); +      OS << ");\n"; +    } else { +      OS << "extractsubreg(" << RetVTName +         << ", Op0, Op0IsKill, " << Memo.SubRegNo << ");\n"; +    } + +    if (!PredicateCheck.empty()) { +      OS << "  }\n"; +    } +  } +  // Return 0 if all of the possibilities had predicates but none +  // were satisfied. +  if (!OneHadNoPredicate) +    OS << "  return 0;\n"; +  OS << "}\n"; +  OS << "\n"; +} +  void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {    // Now emit code for all the patterns that we collected. @@ -608,9 +714,8 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {                 RI != RE; ++RI) {              MVT::SimpleValueType RetVT = RI->first;              const PredMap &PM = RI->second; -            bool HasPred = false; -            OS << "unsigned FastEmit_" +            OS << "unsigned fastEmit_"                 << getLegalCName(Opcode)                 << "_" << getLegalCName(getName(VT))                 << "_" << getLegalCName(getName(RetVT)) << "_"; @@ -619,58 +724,11 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {              Operands.PrintParameters(OS);              OS << ") {\n"; -            // Emit code for each possible instruction. There may be -            // multiple if there are subtarget concerns. -            for (PredMap::const_iterator PI = PM.begin(), PE = PM.end(); -                 PI != PE; ++PI) { -              std::string PredicateCheck = PI->first; -              const InstructionMemo &Memo = PI->second; - -              if (PredicateCheck.empty()) { -                assert(!HasPred && -                       "Multiple instructions match, at least one has " -                       "a predicate and at least one doesn't!"); -              } else { -                OS << "  if (" + PredicateCheck + ") {\n"; -                OS << "  "; -                HasPred = true; -              } - -              for (unsigned i = 0; i < Memo.PhysRegs->size(); ++i) { -                if ((*Memo.PhysRegs)[i] != "") -                  OS << "  BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, " -                     << "TII.get(TargetOpcode::COPY), " -                     << (*Memo.PhysRegs)[i] << ").addReg(Op" << i << ");\n"; -              } - -              OS << "  return FastEmitInst_"; -              if (Memo.SubRegNo.empty()) { -                Operands.PrintManglingSuffix(OS, *Memo.PhysRegs, -                                             ImmediatePredicates, true); -                OS << "(" << InstNS << Memo.Name << ", "; -                OS << "&" << InstNS << Memo.RC->getName() << "RegClass"; -                if (!Operands.empty()) -                  OS << ", "; -                Operands.PrintArguments(OS, *Memo.PhysRegs); -                OS << ");\n"; -              } else { -                OS << "extractsubreg(" << getName(RetVT); -                OS << ", Op0, Op0IsKill, " << Memo.SubRegNo << ");\n"; -              } - -              if (HasPred) -                OS << "  }\n"; - -            } -            // Return 0 if none of the predicates were satisfied. -            if (HasPred) -              OS << "  return 0;\n"; -            OS << "}\n"; -            OS << "\n"; +            emitInstructionCode(OS, Operands, PM, getName(RetVT));            }            // Emit one function for the type that demultiplexes on return type. -          OS << "unsigned FastEmit_" +          OS << "unsigned fastEmit_"               << getLegalCName(Opcode) << "_"               << getLegalCName(getName(VT)) << "_";            Operands.PrintManglingSuffix(OS, ImmediatePredicates); @@ -682,7 +740,7 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {            for (RetPredMap::const_iterator RI = RM.begin(), RE = RM.end();                 RI != RE; ++RI) {              MVT::SimpleValueType RetVT = RI->first; -            OS << "  case " << getName(RetVT) << ": return FastEmit_" +            OS << "  case " << getName(RetVT) << ": return fastEmit_"                 << getLegalCName(Opcode) << "_" << getLegalCName(getName(VT))                 << "_" << getLegalCName(getName(RetVT)) << "_";              Operands.PrintManglingSuffix(OS, ImmediatePredicates); @@ -694,7 +752,7 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {          } else {            // Non-variadic return type. -          OS << "unsigned FastEmit_" +          OS << "unsigned fastEmit_"               << getLegalCName(Opcode) << "_"               << getLegalCName(getName(VT)) << "_";            Operands.PrintManglingSuffix(OS, ImmediatePredicates); @@ -708,63 +766,13 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {               << ")\n    return 0;\n";            const PredMap &PM = RM.begin()->second; -          bool HasPred = false; - -          // Emit code for each possible instruction. There may be -          // multiple if there are subtarget concerns. -          for (PredMap::const_iterator PI = PM.begin(), PE = PM.end(); PI != PE; -               ++PI) { -            std::string PredicateCheck = PI->first; -            const InstructionMemo &Memo = PI->second; - -            if (PredicateCheck.empty()) { -              assert(!HasPred && -                     "Multiple instructions match, at least one has " -                     "a predicate and at least one doesn't!"); -            } else { -              OS << "  if (" + PredicateCheck + ") {\n"; -              OS << "  "; -              HasPred = true; -            } - -            for (unsigned i = 0; i < Memo.PhysRegs->size(); ++i) { -              if ((*Memo.PhysRegs)[i] != "") -                OS << "  BuildMI(*FuncInfo.MBB, FuncInfo.InsertPt, DbgLoc, " -                   << "TII.get(TargetOpcode::COPY), " -                   << (*Memo.PhysRegs)[i] << ").addReg(Op" << i << ");\n"; -            } - -            OS << "  return FastEmitInst_"; - -            if (Memo.SubRegNo.empty()) { -              Operands.PrintManglingSuffix(OS, *Memo.PhysRegs, -                                           ImmediatePredicates, true); -              OS << "(" << InstNS << Memo.Name << ", "; -              OS << "&" << InstNS << Memo.RC->getName() << "RegClass"; -              if (!Operands.empty()) -                OS << ", "; -              Operands.PrintArguments(OS, *Memo.PhysRegs); -              OS << ");\n"; -            } else { -              OS << "extractsubreg(RetVT, Op0, Op0IsKill, "; -              OS << Memo.SubRegNo; -              OS << ");\n"; -            } - -             if (HasPred) -               OS << "  }\n"; -          } -          // Return 0 if none of the predicates were satisfied. -          if (HasPred) -            OS << "  return 0;\n"; -          OS << "}\n"; -          OS << "\n"; +          emitInstructionCode(OS, Operands, PM, "RetVT");          }        }        // Emit one function for the opcode that demultiplexes based on the type. -      OS << "unsigned FastEmit_" +      OS << "unsigned fastEmit_"           << getLegalCName(Opcode) << "_";        Operands.PrintManglingSuffix(OS, ImmediatePredicates);        OS << "(MVT VT, MVT RetVT"; @@ -777,7 +785,7 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {             TI != TE; ++TI) {          MVT::SimpleValueType VT = TI->first;          std::string TypeName = getName(VT); -        OS << "  case " << TypeName << ": return FastEmit_" +        OS << "  case " << TypeName << ": return fastEmit_"             << getLegalCName(Opcode) << "_" << getLegalCName(TypeName) << "_";          Operands.PrintManglingSuffix(OS, ImmediatePredicates);          OS << "(RetVT"; @@ -797,13 +805,16 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {      // Emit one function for the operand signature that demultiplexes based      // on opcode and type. -    OS << "unsigned FastEmit_"; +    OS << "unsigned fastEmit_";      Operands.PrintManglingSuffix(OS, ImmediatePredicates);      OS << "(MVT VT, MVT RetVT, unsigned Opcode";      if (!Operands.empty())        OS << ", ";      Operands.PrintParameters(OS); -    OS << ") {\n"; +    OS << ") "; +    if (!Operands.hasAnyImmediateCodes()) +      OS << "override "; +    OS << "{\n";      // If there are any forms of this signature available that operate on      // constrained forms of the immediate (e.g., 32-bit sext immediate in a @@ -823,7 +834,7 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {        for (unsigned i = 0, e = MI->second.size(); i != e; ++i) {          OS << "  if (";          MI->second[i].emitImmediatePredicate(OS, ImmediatePredicates); -        OS << ")\n    if (unsigned Reg = FastEmit_"; +        OS << ")\n    if (unsigned Reg = fastEmit_";          MI->second[i].PrintManglingSuffix(OS, ImmediatePredicates);          OS << "(VT, RetVT, Opcode";          if (!MI->second[i].empty()) @@ -841,7 +852,7 @@ void FastISelMap::printFunctionDefinitions(raw_ostream &OS) {           I != E; ++I) {        const std::string &Opcode = I->first; -      OS << "  case " << Opcode << ": return FastEmit_" +      OS << "  case " << Opcode << ": return fastEmit_"           << getLegalCName(Opcode) << "_";        Operands.PrintManglingSuffix(OS, ImmediatePredicates);        OS << "(VT, RetVT"; | 
