diff options
| author | Chris Lattner <sabre@nondot.org> | 2010-03-01 18:47:11 +0000 | 
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2010-03-01 18:47:11 +0000 | 
| commit | 7390eebd49b786f3293d24f76fedb02fe183468c (patch) | |
| tree | 54249951ff447cd82835bacb164eff91106c0b80 | |
| parent | d1f94e90fdb5fc4ff57d7544fde017e1223bd9e9 (diff) | |
| download | external_llvm-7390eebd49b786f3293d24f76fedb02fe183468c.zip external_llvm-7390eebd49b786f3293d24f76fedb02fe183468c.tar.gz external_llvm-7390eebd49b786f3293d24f76fedb02fe183468c.tar.bz2 | |
Accelerate isel dispatch for tables that start with a top-level 
OPC_SwitchOpcode to use a table lookup instead of having to go
through the interpreter for this.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@97469 91177308-0d34-0410-b5e6-96231b3b80d8
| -rw-r--r-- | include/llvm/CodeGen/SelectionDAGISel.h | 4 | ||||
| -rw-r--r-- | lib/CodeGen/SelectionDAG/SelectionDAGISel.cpp | 37 | 
2 files changed, 40 insertions, 1 deletions
| diff --git a/include/llvm/CodeGen/SelectionDAGISel.h b/include/llvm/CodeGen/SelectionDAGISel.h index fa3b08f..6ed9f49 100644 --- a/include/llvm/CodeGen/SelectionDAGISel.h +++ b/include/llvm/CodeGen/SelectionDAGISel.h @@ -245,6 +245,10 @@ private:    /// one preferred by the target.    ///    ScheduleDAGSDNodes *CreateScheduler(); +   +  /// OpcodeOffset - This is a cache used to dispatch efficiently into isel +  /// state machines that start with a OPC_SwitchOpcode node. +  std::vector<unsigned> OpcodeOffset;  };  } diff --git a/lib/CodeGen/SelectionDAG/SelectionDAGISel.cpp b/lib/CodeGen/SelectionDAG/SelectionDAGISel.cpp index 03cbe22..3c5e90b 100644 --- a/lib/CodeGen/SelectionDAG/SelectionDAGISel.cpp +++ b/lib/CodeGen/SelectionDAG/SelectionDAGISel.cpp @@ -1665,8 +1665,43 @@ SelectCodeCommon(SDNode *NodeToMatch, const unsigned char *MatcherTable,          NodeToMatch->dump(CurDAG);          errs() << '\n'); -  // Interpreter starts at opcode #0. +  // Determine where to start the interpreter.  Normally we start at opcode #0, +  // but if the state machine starts with an OPC_SwitchOpcode, then we +  // accelerate the first lookup (which is guaranteed to be hot) with the +  // OpcodeOffset table.    unsigned MatcherIndex = 0; +   +  if (!OpcodeOffset.empty()) { +    // Already computed the OpcodeOffset table, just index into it. +    if (N.getOpcode() < OpcodeOffset.size()) +      MatcherIndex = OpcodeOffset[N.getOpcode()]; +    DEBUG(errs() << "  Initial Opcode index to " << MatcherIndex << "\n"); + +  } else if (MatcherTable[0] == OPC_SwitchOpcode) { +    // Otherwise, the table isn't computed, but the state machine does start +    // with an OPC_SwitchOpcode instruction.  Populate the table now, since this +    // is the first time we're selecting an instruction. +    unsigned Idx = 1; +    while (1) { +      // Get the size of this case. +      unsigned CaseSize = MatcherTable[Idx++]; +      if (CaseSize & 128) +        CaseSize = GetVBR(CaseSize, MatcherTable, Idx); +      if (CaseSize == 0) break; + +      // Get the opcode, add the index to the table. +      unsigned Opc = MatcherTable[Idx++]; +      if (Opc >= OpcodeOffset.size()) +        OpcodeOffset.resize((Opc+1)*2); +      OpcodeOffset[Opc] = Idx; +      Idx += CaseSize; +    } + +    // Okay, do the lookup for the first opcode. +    if (N.getOpcode() < OpcodeOffset.size()) +      MatcherIndex = OpcodeOffset[N.getOpcode()]; +  } +      while (1) {      assert(MatcherIndex < TableSize && "Invalid index");      BuiltinOpcodes Opcode = (BuiltinOpcodes)MatcherTable[MatcherIndex++]; | 
