xref: /netbsd-src/external/apache2/llvm/dist/llvm/tools/llvm-exegesis/lib/SnippetFile.cpp (revision 627f7eb200a4419d89b531d55fccd2ee3ffdcde0)
1 //===-- SnippetFile.cpp -----------------------------------------*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "SnippetFile.h"
10 #include "Error.h"
11 #include "llvm/MC/MCContext.h"
12 #include "llvm/MC/MCObjectFileInfo.h"
13 #include "llvm/MC/MCParser/MCAsmParser.h"
14 #include "llvm/MC/MCParser/MCTargetAsmParser.h"
15 #include "llvm/MC/MCRegisterInfo.h"
16 #include "llvm/MC/MCStreamer.h"
17 #include "llvm/Support/Format.h"
18 #include "llvm/Support/Path.h"
19 #include "llvm/Support/SourceMgr.h"
20 #include "llvm/Support/TargetRegistry.h"
21 #include <string>
22 
23 namespace llvm {
24 namespace exegesis {
25 namespace {
26 
27 // An MCStreamer that reads a BenchmarkCode definition from a file.
28 class BenchmarkCodeStreamer : public MCStreamer, public AsmCommentConsumer {
29 public:
30   explicit BenchmarkCodeStreamer(MCContext *Context,
31                                  const MCRegisterInfo *TheRegInfo,
32                                  BenchmarkCode *Result)
33       : MCStreamer(*Context), RegInfo(TheRegInfo), Result(Result) {}
34 
35   // Implementation of the MCStreamer interface. We only care about
36   // instructions.
37   void EmitInstruction(const MCInst &Instruction,
38                        const MCSubtargetInfo &STI) override {
39     Result->Key.Instructions.push_back(Instruction);
40   }
41 
42   // Implementation of the AsmCommentConsumer.
43   void HandleComment(SMLoc Loc, StringRef CommentText) override {
44     CommentText = CommentText.trim();
45     if (!CommentText.consume_front("LLVM-EXEGESIS-"))
46       return;
47     if (CommentText.consume_front("DEFREG")) {
48       // LLVM-EXEGESIS-DEFREF <reg> <hex_value>
49       RegisterValue RegVal;
50       SmallVector<StringRef, 2> Parts;
51       CommentText.split(Parts, ' ', /*unlimited splits*/ -1,
52                         /*do not keep empty strings*/ false);
53       if (Parts.size() != 2) {
54         errs() << "invalid comment 'LLVM-EXEGESIS-DEFREG " << CommentText
55                << "', expected two parameters <REG> <HEX_VALUE>\n";
56         ++InvalidComments;
57         return;
58       }
59       if (!(RegVal.Register = findRegisterByName(Parts[0].trim()))) {
60         errs() << "unknown register '" << Parts[0]
61                << "' in 'LLVM-EXEGESIS-DEFREG " << CommentText << "'\n";
62         ++InvalidComments;
63         return;
64       }
65       const StringRef HexValue = Parts[1].trim();
66       RegVal.Value = APInt(
67           /* each hex digit is 4 bits */ HexValue.size() * 4, HexValue, 16);
68       Result->Key.RegisterInitialValues.push_back(std::move(RegVal));
69       return;
70     }
71     if (CommentText.consume_front("LIVEIN")) {
72       // LLVM-EXEGESIS-LIVEIN <reg>
73       const auto RegName = CommentText.ltrim();
74       if (unsigned Reg = findRegisterByName(RegName))
75         Result->LiveIns.push_back(Reg);
76       else {
77         errs() << "unknown register '" << RegName
78                << "' in 'LLVM-EXEGESIS-LIVEIN " << CommentText << "'\n";
79         ++InvalidComments;
80       }
81       return;
82     }
83   }
84 
85   unsigned numInvalidComments() const { return InvalidComments; }
86 
87 private:
88   // We only care about instructions, we don't implement this part of the API.
89   void EmitCommonSymbol(MCSymbol *Symbol, uint64_t Size,
90                         unsigned ByteAlignment) override {}
91   bool EmitSymbolAttribute(MCSymbol *Symbol, MCSymbolAttr Attribute) override {
92     return false;
93   }
94   void EmitValueToAlignment(unsigned ByteAlignment, int64_t Value,
95                             unsigned ValueSize,
96                             unsigned MaxBytesToEmit) override {}
97   void EmitZerofill(MCSection *Section, MCSymbol *Symbol, uint64_t Size,
98                     unsigned ByteAlignment, SMLoc Loc) override {}
99 
100   unsigned findRegisterByName(const StringRef RegName) const {
101     // FIXME: Can we do better than this ?
102     for (unsigned I = 0, E = RegInfo->getNumRegs(); I < E; ++I) {
103       if (RegName == RegInfo->getName(I))
104         return I;
105     }
106     errs() << "'" << RegName
107            << "' is not a valid register name for the target\n";
108     return 0;
109   }
110 
111   const MCRegisterInfo *const RegInfo;
112   BenchmarkCode *const Result;
113   unsigned InvalidComments = 0;
114 };
115 
116 } // namespace
117 
118 // Reads code snippets from file `Filename`.
119 Expected<std::vector<BenchmarkCode>> readSnippets(const LLVMState &State,
120                                                   StringRef Filename) {
121   ErrorOr<std::unique_ptr<MemoryBuffer>> BufferPtr =
122       MemoryBuffer::getFileOrSTDIN(Filename);
123   if (std::error_code EC = BufferPtr.getError()) {
124     return make_error<Failure>("cannot read snippet: " + Filename + ": " +
125                                EC.message());
126   }
127   SourceMgr SM;
128   SM.AddNewSourceBuffer(std::move(BufferPtr.get()), SMLoc());
129 
130   BenchmarkCode Result;
131 
132   MCObjectFileInfo ObjectFileInfo;
133   const TargetMachine &TM = State.getTargetMachine();
134   MCContext Context(TM.getMCAsmInfo(), TM.getMCRegisterInfo(), &ObjectFileInfo);
135   ObjectFileInfo.InitMCObjectFileInfo(TM.getTargetTriple(), /*PIC*/ false,
136                                       Context);
137   BenchmarkCodeStreamer Streamer(&Context, TM.getMCRegisterInfo(), &Result);
138   const std::unique_ptr<MCAsmParser> AsmParser(
139       createMCAsmParser(SM, Context, Streamer, *TM.getMCAsmInfo()));
140   if (!AsmParser)
141     return make_error<Failure>("cannot create asm parser");
142   AsmParser->getLexer().setCommentConsumer(&Streamer);
143 
144   const std::unique_ptr<MCTargetAsmParser> TargetAsmParser(
145       TM.getTarget().createMCAsmParser(*TM.getMCSubtargetInfo(), *AsmParser,
146                                        *TM.getMCInstrInfo(),
147                                        MCTargetOptions()));
148 
149   if (!TargetAsmParser)
150     return make_error<Failure>("cannot create target asm parser");
151   AsmParser->setTargetParser(*TargetAsmParser);
152 
153   if (AsmParser->Run(false))
154     return make_error<Failure>("cannot parse asm file");
155   if (Streamer.numInvalidComments())
156     return make_error<Failure>(Twine("found ")
157                                    .concat(Twine(Streamer.numInvalidComments()))
158                                    .concat(" invalid LLVM-EXEGESIS comments"));
159   return std::vector<BenchmarkCode>{std::move(Result)};
160 }
161 
162 } // namespace exegesis
163 } // namespace llvm
164