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