16ac3f739SEugene Zelenko //===- DFAPacketizerEmitter.cpp - Packetization DFA for a VLIW machine ----===//
208ebdc1eSAnshuman Dasgupta //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
608ebdc1eSAnshuman Dasgupta //
708ebdc1eSAnshuman Dasgupta //===----------------------------------------------------------------------===//
808ebdc1eSAnshuman Dasgupta //
908ebdc1eSAnshuman Dasgupta // This class parses the Schedule.td file and produces an API that can be used
1008ebdc1eSAnshuman Dasgupta // to reason about whether an instruction can be added to a packet on a VLIW
1108ebdc1eSAnshuman Dasgupta // architecture. The class internally generates a deterministic finite
1208ebdc1eSAnshuman Dasgupta // automaton (DFA) that models all possible mappings of machine instructions
1308ebdc1eSAnshuman Dasgupta // to functional units as instructions are added to a packet.
1408ebdc1eSAnshuman Dasgupta //
1508ebdc1eSAnshuman Dasgupta //===----------------------------------------------------------------------===//
1608ebdc1eSAnshuman Dasgupta 
1739525a67Sjmolloy #include "CodeGenSchedule.h"
1808ebdc1eSAnshuman Dasgupta #include "CodeGenTarget.h"
1912092a96SJames Molloy #include "DFAEmitter.h"
20a3fe70d2SEugene Zelenko #include "llvm/ADT/SmallVector.h"
21b4655729SKrzysztof Parzyszek #include "llvm/Support/Debug.h"
22a3fe70d2SEugene Zelenko #include "llvm/Support/raw_ostream.h"
2339525a67Sjmolloy #include "llvm/TableGen/Record.h"
2439525a67Sjmolloy #include "llvm/TableGen/TableGenBackend.h"
25a3fe70d2SEugene Zelenko #include <cassert>
26a3fe70d2SEugene Zelenko #include <cstdint>
27e6aed139SJakob Stoklund Olesen #include <map>
28a3fe70d2SEugene Zelenko #include <set>
29e6aed139SJakob Stoklund Olesen #include <string>
306ccd6732SJames Molloy #include <unordered_map>
31a3fe70d2SEugene Zelenko #include <vector>
326ac3f739SEugene Zelenko 
33*71acce68SMindong Chen #define DEBUG_TYPE "dfa-emitter"
34*71acce68SMindong Chen 
3508ebdc1eSAnshuman Dasgupta using namespace llvm;
3608ebdc1eSAnshuman Dasgupta 
3739525a67Sjmolloy // We use a uint64_t to represent a resource bitmask.
3839525a67Sjmolloy #define DFA_MAX_RESOURCES 64
396753f333SKrzysztof Parzyszek 
406753f333SKrzysztof Parzyszek namespace {
4139525a67Sjmolloy using ResourceVector = SmallVector<uint64_t, 4>;
42a3fe70d2SEugene Zelenko 
4339525a67Sjmolloy struct ScheduleClass {
4439525a67Sjmolloy   /// The parent itinerary index (processor model ID).
4539525a67Sjmolloy   unsigned ItineraryID;
466753f333SKrzysztof Parzyszek 
4739525a67Sjmolloy   /// Index within this itinerary of the schedule class.
4839525a67Sjmolloy   unsigned Idx;
49a3fe70d2SEugene Zelenko 
5039525a67Sjmolloy   /// The index within the uniqued set of required resources of Resources.
5139525a67Sjmolloy   unsigned ResourcesIdx;
526ac3f739SEugene Zelenko 
5339525a67Sjmolloy   /// Conjunctive list of resource requirements:
5439525a67Sjmolloy   ///   {a|b, b|c} => (a OR b) AND (b or c).
5539525a67Sjmolloy   /// Resources are unique across all itineraries.
5639525a67Sjmolloy   ResourceVector Resources;
5739525a67Sjmolloy };
586753f333SKrzysztof Parzyszek 
5939525a67Sjmolloy // Generates and prints out the DFA for resource tracking.
60e6aed139SJakob Stoklund Olesen class DFAPacketizerEmitter {
61e6aed139SJakob Stoklund Olesen private:
62e6aed139SJakob Stoklund Olesen   std::string TargetName;
63e6aed139SJakob Stoklund Olesen   RecordKeeper &Records;
64e6aed139SJakob Stoklund Olesen 
6539525a67Sjmolloy   UniqueVector<ResourceVector> UniqueResources;
6639525a67Sjmolloy   std::vector<ScheduleClass> ScheduleClasses;
6739525a67Sjmolloy   std::map<std::string, uint64_t> FUNameToBitsMap;
6839525a67Sjmolloy   std::map<unsigned, uint64_t> ComboBitToBitsMap;
6939525a67Sjmolloy 
70e6aed139SJakob Stoklund Olesen public:
71e6aed139SJakob Stoklund Olesen   DFAPacketizerEmitter(RecordKeeper &R);
72e6aed139SJakob Stoklund Olesen 
7339525a67Sjmolloy   // Construct a map of function unit names to bits.
7439525a67Sjmolloy   int collectAllFuncUnits(
7539525a67Sjmolloy       ArrayRef<const CodeGenProcModel *> ProcModels);
76b4655729SKrzysztof Parzyszek 
7739525a67Sjmolloy   // Construct a map from a combo function unit bit to the bits of all included
7839525a67Sjmolloy   // functional units.
7939525a67Sjmolloy   int collectAllComboFuncs(ArrayRef<Record *> ComboFuncList);
80b4655729SKrzysztof Parzyszek 
8139525a67Sjmolloy   ResourceVector getResourcesForItinerary(Record *Itinerary);
8239525a67Sjmolloy   void createScheduleClasses(unsigned ItineraryIdx, const RecVec &Itineraries);
83e6aed139SJakob Stoklund Olesen 
846ccd6732SJames Molloy   // Emit code for a subset of itineraries.
856ccd6732SJames Molloy   void emitForItineraries(raw_ostream &OS,
8639525a67Sjmolloy                           std::vector<const CodeGenProcModel *> &ProcItinList,
876ccd6732SJames Molloy                           std::string DFAName);
886ccd6732SJames Molloy 
89e6aed139SJakob Stoklund Olesen   void run(raw_ostream &OS);
90e6aed139SJakob Stoklund Olesen };
916ac3f739SEugene Zelenko } // end anonymous namespace
9208ebdc1eSAnshuman Dasgupta 
DFAPacketizerEmitter(RecordKeeper & R)9339525a67Sjmolloy DFAPacketizerEmitter::DFAPacketizerEmitter(RecordKeeper &R)
94adcd0268SBenjamin Kramer     : TargetName(std::string(CodeGenTarget(R).getName())), Records(R) {}
95b4655729SKrzysztof Parzyszek 
collectAllFuncUnits(ArrayRef<const CodeGenProcModel * > ProcModels)96b4655729SKrzysztof Parzyszek int DFAPacketizerEmitter::collectAllFuncUnits(
9739525a67Sjmolloy     ArrayRef<const CodeGenProcModel *> ProcModels) {
98d34e60caSNicola Zaghen   LLVM_DEBUG(dbgs() << "-------------------------------------------------------"
99d34e60caSNicola Zaghen                        "----------------------\n");
100d34e60caSNicola Zaghen   LLVM_DEBUG(dbgs() << "collectAllFuncUnits");
10139525a67Sjmolloy   LLVM_DEBUG(dbgs() << " (" << ProcModels.size() << " itineraries)\n");
10239525a67Sjmolloy 
10339525a67Sjmolloy   std::set<Record *> ProcItinList;
10439525a67Sjmolloy   for (const CodeGenProcModel *Model : ProcModels)
10539525a67Sjmolloy     ProcItinList.insert(Model->ItinsDef);
10608ebdc1eSAnshuman Dasgupta 
107b4655729SKrzysztof Parzyszek   int totalFUs = 0;
10808ebdc1eSAnshuman Dasgupta   // Parse functional units for all the itineraries.
10939525a67Sjmolloy   for (Record *Proc : ProcItinList) {
11008ebdc1eSAnshuman Dasgupta     std::vector<Record *> FUs = Proc->getValueAsListOfDefs("FU");
11108ebdc1eSAnshuman Dasgupta 
11239525a67Sjmolloy     LLVM_DEBUG(dbgs() << "    FU:"
11339525a67Sjmolloy                       << " (" << FUs.size() << " FUs) " << Proc->getName());
114b4655729SKrzysztof Parzyszek 
1159aa6137dSSebastian Pop     // Convert macros to bits for each stage.
116b4655729SKrzysztof Parzyszek     unsigned numFUs = FUs.size();
117b4655729SKrzysztof Parzyszek     for (unsigned j = 0; j < numFUs; ++j) {
118b4655729SKrzysztof Parzyszek       assert((j < DFA_MAX_RESOURCES) &&
119b4655729SKrzysztof Parzyszek              "Exceeded maximum number of representable resources");
12039525a67Sjmolloy       uint64_t FuncResources = 1ULL << j;
121adcd0268SBenjamin Kramer       FUNameToBitsMap[std::string(FUs[j]->getName())] = FuncResources;
122d34e60caSNicola Zaghen       LLVM_DEBUG(dbgs() << " " << FUs[j]->getName() << ":0x"
1233a13ed60SBenjamin Kramer                         << Twine::utohexstr(FuncResources));
12408ebdc1eSAnshuman Dasgupta     }
125b4655729SKrzysztof Parzyszek     totalFUs += numFUs;
126d34e60caSNicola Zaghen     LLVM_DEBUG(dbgs() << "\n");
127b4655729SKrzysztof Parzyszek   }
128b4655729SKrzysztof Parzyszek   return totalFUs;
129b4655729SKrzysztof Parzyszek }
130b4655729SKrzysztof Parzyszek 
collectAllComboFuncs(ArrayRef<Record * > ComboFuncList)13139525a67Sjmolloy int DFAPacketizerEmitter::collectAllComboFuncs(ArrayRef<Record *> ComboFuncList) {
132d34e60caSNicola Zaghen   LLVM_DEBUG(dbgs() << "-------------------------------------------------------"
133d34e60caSNicola Zaghen                        "----------------------\n");
134d34e60caSNicola Zaghen   LLVM_DEBUG(dbgs() << "collectAllComboFuncs");
135d34e60caSNicola Zaghen   LLVM_DEBUG(dbgs() << " (" << ComboFuncList.size() << " sets)\n");
136b4655729SKrzysztof Parzyszek 
137b4655729SKrzysztof Parzyszek   int numCombos = 0;
138b4655729SKrzysztof Parzyszek   for (unsigned i = 0, N = ComboFuncList.size(); i < N; ++i) {
139b4655729SKrzysztof Parzyszek     Record *Func = ComboFuncList[i];
140b4655729SKrzysztof Parzyszek     std::vector<Record *> FUs = Func->getValueAsListOfDefs("CFD");
141b4655729SKrzysztof Parzyszek 
142d34e60caSNicola Zaghen     LLVM_DEBUG(dbgs() << "    CFD:" << i << " (" << FUs.size() << " combo FUs) "
1438dd552dbSKrzysztof Parzyszek                       << Func->getName() << "\n");
144b4655729SKrzysztof Parzyszek 
145b4655729SKrzysztof Parzyszek     // Convert macros to bits for each stage.
146b4655729SKrzysztof Parzyszek     for (unsigned j = 0, N = FUs.size(); j < N; ++j) {
147b4655729SKrzysztof Parzyszek       assert((j < DFA_MAX_RESOURCES) &&
148b4655729SKrzysztof Parzyszek              "Exceeded maximum number of DFA resources");
149b4655729SKrzysztof Parzyszek       Record *FuncData = FUs[j];
150b4655729SKrzysztof Parzyszek       Record *ComboFunc = FuncData->getValueAsDef("TheComboFunc");
151b4655729SKrzysztof Parzyszek       const std::vector<Record *> &FuncList =
152b4655729SKrzysztof Parzyszek           FuncData->getValueAsListOfDefs("FuncList");
153adcd0268SBenjamin Kramer       const std::string &ComboFuncName = std::string(ComboFunc->getName());
15439525a67Sjmolloy       uint64_t ComboBit = FUNameToBitsMap[ComboFuncName];
15539525a67Sjmolloy       uint64_t ComboResources = ComboBit;
156d34e60caSNicola Zaghen       LLVM_DEBUG(dbgs() << "      combo: " << ComboFuncName << ":0x"
1573a13ed60SBenjamin Kramer                         << Twine::utohexstr(ComboResources) << "\n");
158e6cf3d64SCoelacanthus       for (auto *K : FuncList) {
159e6cf3d64SCoelacanthus         std::string FuncName = std::string(K->getName());
16039525a67Sjmolloy         uint64_t FuncResources = FUNameToBitsMap[FuncName];
161d34e60caSNicola Zaghen         LLVM_DEBUG(dbgs() << "        " << FuncName << ":0x"
1623a13ed60SBenjamin Kramer                           << Twine::utohexstr(FuncResources) << "\n");
163b4655729SKrzysztof Parzyszek         ComboResources |= FuncResources;
164b4655729SKrzysztof Parzyszek       }
165b4655729SKrzysztof Parzyszek       ComboBitToBitsMap[ComboBit] = ComboResources;
166b4655729SKrzysztof Parzyszek       numCombos++;
167d34e60caSNicola Zaghen       LLVM_DEBUG(dbgs() << "          => combo bits: " << ComboFuncName << ":0x"
1683a13ed60SBenjamin Kramer                         << Twine::utohexstr(ComboBit) << " = 0x"
1693a13ed60SBenjamin Kramer                         << Twine::utohexstr(ComboResources) << "\n");
170b4655729SKrzysztof Parzyszek     }
171b4655729SKrzysztof Parzyszek   }
172b4655729SKrzysztof Parzyszek   return numCombos;
173b4655729SKrzysztof Parzyszek }
174b4655729SKrzysztof Parzyszek 
17539525a67Sjmolloy ResourceVector
getResourcesForItinerary(Record * Itinerary)17639525a67Sjmolloy DFAPacketizerEmitter::getResourcesForItinerary(Record *Itinerary) {
17739525a67Sjmolloy   ResourceVector Resources;
17839525a67Sjmolloy   assert(Itinerary);
17939525a67Sjmolloy   for (Record *StageDef : Itinerary->getValueAsListOfDefs("Stages")) {
18039525a67Sjmolloy     uint64_t StageResources = 0;
18139525a67Sjmolloy     for (Record *Unit : StageDef->getValueAsListOfDefs("Units")) {
182adcd0268SBenjamin Kramer       StageResources |= FUNameToBitsMap[std::string(Unit->getName())];
18339525a67Sjmolloy     }
18439525a67Sjmolloy     if (StageResources != 0)
18539525a67Sjmolloy       Resources.push_back(StageResources);
18639525a67Sjmolloy   }
18739525a67Sjmolloy   return Resources;
18808ebdc1eSAnshuman Dasgupta }
18908ebdc1eSAnshuman Dasgupta 
createScheduleClasses(unsigned ItineraryIdx,const RecVec & Itineraries)19039525a67Sjmolloy void DFAPacketizerEmitter::createScheduleClasses(unsigned ItineraryIdx,
19139525a67Sjmolloy                                                  const RecVec &Itineraries) {
19239525a67Sjmolloy   unsigned Idx = 0;
19339525a67Sjmolloy   for (Record *Itinerary : Itineraries) {
19439525a67Sjmolloy     if (!Itinerary) {
19539525a67Sjmolloy       ScheduleClasses.push_back({ItineraryIdx, Idx++, 0, ResourceVector{}});
19639525a67Sjmolloy       continue;
19708ebdc1eSAnshuman Dasgupta     }
19839525a67Sjmolloy     ResourceVector Resources = getResourcesForItinerary(Itinerary);
19939525a67Sjmolloy     ScheduleClasses.push_back(
20039525a67Sjmolloy         {ItineraryIdx, Idx++, UniqueResources.insert(Resources), Resources});
20108ebdc1eSAnshuman Dasgupta   }
202b4655729SKrzysztof Parzyszek }
20308ebdc1eSAnshuman Dasgupta 
20408ebdc1eSAnshuman Dasgupta //
2059aa6137dSSebastian Pop // Run the worklist algorithm to generate the DFA.
20608ebdc1eSAnshuman Dasgupta //
run(raw_ostream & OS)207e6aed139SJakob Stoklund Olesen void DFAPacketizerEmitter::run(raw_ostream &OS) {
2086ccd6732SJames Molloy   OS << "\n"
2096ccd6732SJames Molloy      << "#include \"llvm/CodeGen/DFAPacketizer.h\"\n";
2106ccd6732SJames Molloy   OS << "namespace llvm {\n";
2116ccd6732SJames Molloy 
21239525a67Sjmolloy   CodeGenTarget CGT(Records);
21339525a67Sjmolloy   CodeGenSchedModels CGS(Records, CGT);
2146ccd6732SJames Molloy 
21539525a67Sjmolloy   std::unordered_map<std::string, std::vector<const CodeGenProcModel *>>
21639525a67Sjmolloy       ItinsByNamespace;
21739525a67Sjmolloy   for (const CodeGenProcModel &ProcModel : CGS.procModels()) {
21839525a67Sjmolloy     if (ProcModel.hasItineraries()) {
21939525a67Sjmolloy       auto NS = ProcModel.ItinsDef->getValueAsString("PacketizerNamespace");
220adcd0268SBenjamin Kramer       ItinsByNamespace[std::string(NS)].push_back(&ProcModel);
22139525a67Sjmolloy     }
22239525a67Sjmolloy   }
2236ccd6732SJames Molloy 
2246ccd6732SJames Molloy   for (auto &KV : ItinsByNamespace)
2256ccd6732SJames Molloy     emitForItineraries(OS, KV.second, KV.first);
2266ccd6732SJames Molloy   OS << "} // end namespace llvm\n";
2276ccd6732SJames Molloy }
2286ccd6732SJames Molloy 
emitForItineraries(raw_ostream & OS,std::vector<const CodeGenProcModel * > & ProcModels,std::string DFAName)2296ccd6732SJames Molloy void DFAPacketizerEmitter::emitForItineraries(
23039525a67Sjmolloy     raw_ostream &OS, std::vector<const CodeGenProcModel *> &ProcModels,
2316ccd6732SJames Molloy     std::string DFAName) {
23239525a67Sjmolloy   OS << "} // end namespace llvm\n\n";
23339525a67Sjmolloy   OS << "namespace {\n";
23439525a67Sjmolloy   collectAllFuncUnits(ProcModels);
23539525a67Sjmolloy   collectAllComboFuncs(Records.getAllDerivedDefinitions("ComboFuncUnits"));
236b4655729SKrzysztof Parzyszek 
237b4655729SKrzysztof Parzyszek   // Collect the itineraries.
23839525a67Sjmolloy   DenseMap<const CodeGenProcModel *, unsigned> ProcModelStartIdx;
23939525a67Sjmolloy   for (const CodeGenProcModel *Model : ProcModels) {
24039525a67Sjmolloy     assert(Model->hasItineraries());
24139525a67Sjmolloy     ProcModelStartIdx[Model] = ScheduleClasses.size();
24239525a67Sjmolloy     createScheduleClasses(Model->Index, Model->ItinDefList);
24308ebdc1eSAnshuman Dasgupta   }
24408ebdc1eSAnshuman Dasgupta 
24539525a67Sjmolloy   // Output the mapping from ScheduleClass to ResourcesIdx.
24639525a67Sjmolloy   unsigned Idx = 0;
2479e4b761aSBenjamin Kramer   OS << "constexpr unsigned " << TargetName << DFAName
2489e4b761aSBenjamin Kramer      << "ResourceIndices[] = {";
24939525a67Sjmolloy   for (const ScheduleClass &SC : ScheduleClasses) {
25039525a67Sjmolloy     if (Idx++ % 32 == 0)
25139525a67Sjmolloy       OS << "\n  ";
25239525a67Sjmolloy     OS << SC.ResourcesIdx << ", ";
25339525a67Sjmolloy   }
25439525a67Sjmolloy   OS << "\n};\n\n";
25539525a67Sjmolloy 
25639525a67Sjmolloy   // And the mapping from Itinerary index into the previous table.
2579e4b761aSBenjamin Kramer   OS << "constexpr unsigned " << TargetName << DFAName
25839525a67Sjmolloy      << "ProcResourceIndexStart[] = {\n";
25939525a67Sjmolloy   OS << "  0, // NoSchedModel\n";
26039525a67Sjmolloy   for (const CodeGenProcModel *Model : ProcModels) {
26139525a67Sjmolloy     OS << "  " << ProcModelStartIdx[Model] << ", // " << Model->ModelName
26239525a67Sjmolloy        << "\n";
26339525a67Sjmolloy   }
264d0b8810fSJay Foad   OS << "  " << ScheduleClasses.size() << "\n};\n\n";
26539525a67Sjmolloy 
26612092a96SJames Molloy   // The type of a state in the nondeterministic automaton we're defining.
26739525a67Sjmolloy   using NfaStateTy = uint64_t;
26808ebdc1eSAnshuman Dasgupta 
26912092a96SJames Molloy   // Given a resource state, return all resource states by applying
27012092a96SJames Molloy   // InsnClass.
27139525a67Sjmolloy   auto applyInsnClass = [&](const ResourceVector &InsnClass,
27239525a67Sjmolloy                             NfaStateTy State) -> std::deque<NfaStateTy> {
27339525a67Sjmolloy     std::deque<NfaStateTy> V(1, State);
27412092a96SJames Molloy     // Apply every stage in the class individually.
27539525a67Sjmolloy     for (NfaStateTy Stage : InsnClass) {
27612092a96SJames Molloy       // Apply this stage to every existing member of V in turn.
27712092a96SJames Molloy       size_t Sz = V.size();
27812092a96SJames Molloy       for (unsigned I = 0; I < Sz; ++I) {
27939525a67Sjmolloy         NfaStateTy S = V.front();
28012092a96SJames Molloy         V.pop_front();
28108ebdc1eSAnshuman Dasgupta 
28212092a96SJames Molloy         // For this stage, state combination, try all possible resources.
28312092a96SJames Molloy         for (unsigned J = 0; J < DFA_MAX_RESOURCES; ++J) {
28439525a67Sjmolloy           NfaStateTy ResourceMask = 1ULL << J;
28512092a96SJames Molloy           if ((ResourceMask & Stage) == 0)
28612092a96SJames Molloy             // This resource isn't required by this stage.
287b4655729SKrzysztof Parzyszek             continue;
28839525a67Sjmolloy           NfaStateTy Combo = ComboBitToBitsMap[ResourceMask];
28912092a96SJames Molloy           if (Combo && ((~S & Combo) != Combo))
29012092a96SJames Molloy             // This combo units bits are not available.
29112092a96SJames Molloy             continue;
29239525a67Sjmolloy           NfaStateTy ResultingResourceState = S | ResourceMask | Combo;
29312092a96SJames Molloy           if (ResultingResourceState == S)
29412092a96SJames Molloy             continue;
29512092a96SJames Molloy           V.push_back(ResultingResourceState);
296b4655729SKrzysztof Parzyszek         }
29708ebdc1eSAnshuman Dasgupta       }
29812092a96SJames Molloy     }
29912092a96SJames Molloy     return V;
30012092a96SJames Molloy   };
30108ebdc1eSAnshuman Dasgupta 
30212092a96SJames Molloy   // Given a resource state, return a quick (conservative) guess as to whether
30312092a96SJames Molloy   // InsnClass can be applied. This is a filter for the more heavyweight
30412092a96SJames Molloy   // applyInsnClass.
30539525a67Sjmolloy   auto canApplyInsnClass = [](const ResourceVector &InsnClass,
30612092a96SJames Molloy                               NfaStateTy State) -> bool {
30739525a67Sjmolloy     for (NfaStateTy Resources : InsnClass) {
30812092a96SJames Molloy       if ((State | Resources) == State)
30912092a96SJames Molloy         return false;
31012092a96SJames Molloy     }
31112092a96SJames Molloy     return true;
31212092a96SJames Molloy   };
31312092a96SJames Molloy 
31412092a96SJames Molloy   DfaEmitter Emitter;
31512092a96SJames Molloy   std::deque<NfaStateTy> Worklist(1, 0);
31612092a96SJames Molloy   std::set<NfaStateTy> SeenStates;
31712092a96SJames Molloy   SeenStates.insert(Worklist.front());
31812092a96SJames Molloy   while (!Worklist.empty()) {
31912092a96SJames Molloy     NfaStateTy State = Worklist.front();
32012092a96SJames Molloy     Worklist.pop_front();
32139525a67Sjmolloy     for (const ResourceVector &Resources : UniqueResources) {
32239525a67Sjmolloy       if (!canApplyInsnClass(Resources, State))
32312092a96SJames Molloy         continue;
32439525a67Sjmolloy       unsigned ResourcesID = UniqueResources.idFor(Resources);
32539525a67Sjmolloy       for (uint64_t NewState : applyInsnClass(Resources, State)) {
32612092a96SJames Molloy         if (SeenStates.emplace(NewState).second)
32712092a96SJames Molloy           Worklist.emplace_back(NewState);
32839525a67Sjmolloy         Emitter.addTransition(State, NewState, ResourcesID);
32908ebdc1eSAnshuman Dasgupta       }
33008ebdc1eSAnshuman Dasgupta     }
33108ebdc1eSAnshuman Dasgupta   }
33208ebdc1eSAnshuman Dasgupta 
33312092a96SJames Molloy   std::string TargetAndDFAName = TargetName + DFAName;
33412092a96SJames Molloy   Emitter.emit(TargetAndDFAName, OS);
33512092a96SJames Molloy   OS << "} // end anonymous namespace\n\n";
3366ccd6732SJames Molloy 
3376ccd6732SJames Molloy   std::string SubTargetClassName = TargetName + "GenSubtargetInfo";
3386ccd6732SJames Molloy   OS << "namespace llvm {\n";
3396ccd6732SJames Molloy   OS << "DFAPacketizer *" << SubTargetClassName << "::"
3406ccd6732SJames Molloy      << "create" << DFAName
3416ccd6732SJames Molloy      << "DFAPacketizer(const InstrItineraryData *IID) const {\n"
342d5afdbe5SJames Molloy      << "  static Automaton<uint64_t> A(ArrayRef<" << TargetAndDFAName
34312092a96SJames Molloy      << "Transition>(" << TargetAndDFAName << "Transitions), "
34412092a96SJames Molloy      << TargetAndDFAName << "TransitionInfo);\n"
34539525a67Sjmolloy      << "  unsigned ProcResIdxStart = " << TargetAndDFAName
34639525a67Sjmolloy      << "ProcResourceIndexStart[IID->SchedModel.ProcID];\n"
34739525a67Sjmolloy      << "  unsigned ProcResIdxNum = " << TargetAndDFAName
34839525a67Sjmolloy      << "ProcResourceIndexStart[IID->SchedModel.ProcID + 1] - "
34939525a67Sjmolloy         "ProcResIdxStart;\n"
35039525a67Sjmolloy      << "  return new DFAPacketizer(IID, A, {&" << TargetAndDFAName
35139525a67Sjmolloy      << "ResourceIndices[ProcResIdxStart], ProcResIdxNum});\n"
35212092a96SJames Molloy      << "\n}\n\n";
35308ebdc1eSAnshuman Dasgupta }
354e6aed139SJakob Stoklund Olesen 
355e6aed139SJakob Stoklund Olesen namespace llvm {
356e6aed139SJakob Stoklund Olesen 
EmitDFAPacketizer(RecordKeeper & RK,raw_ostream & OS)357e6aed139SJakob Stoklund Olesen void EmitDFAPacketizer(RecordKeeper &RK, raw_ostream &OS) {
358e6aed139SJakob Stoklund Olesen   emitSourceFileHeader("Target DFA Packetizer Tables", OS);
359e6aed139SJakob Stoklund Olesen   DFAPacketizerEmitter(RK).run(OS);
360e6aed139SJakob Stoklund Olesen }
361e6aed139SJakob Stoklund Olesen 
362a3fe70d2SEugene Zelenko } // end namespace llvm
363