blob: 63df5c634537b7d0a8948c75cd6b659eb9c333ee [file] [log] [blame]
Clement Courbet3e138162019-09-30 12:50:25 +00001//===-- 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"
Clement Courbet03a3d292019-09-30 13:53:50 +000010#include "Error.h"
Clement Courbet3e138162019-09-30 12:50:25 +000011#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
23namespace llvm {
24namespace exegesis {
25namespace {
26
27// An MCStreamer that reads a BenchmarkCode definition from a file.
28class BenchmarkCodeStreamer : public MCStreamer, public AsmCommentConsumer {
29public:
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 {
Clement Courbet49195342019-10-08 09:06:48 +000039 Result->Key.Instructions.push_back(Instruction);
Clement Courbet3e138162019-09-30 12:50:25 +000040 }
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);
Clement Courbet49195342019-10-08 09:06:48 +000068 Result->Key.RegisterInitialValues.push_back(std::move(RegVal));
Clement Courbet3e138162019-09-30 12:50:25 +000069 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
87private:
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`.
119Expected<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()) {
Clement Courbet03a3d292019-09-30 13:53:50 +0000124 return make_error<Failure>("cannot read snippet: " + Filename + ": " +
125 EC.message());
Clement Courbet3e138162019-09-30 12:50:25 +0000126 }
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)
Clement Courbet03a3d292019-09-30 13:53:50 +0000141 return make_error<Failure>("cannot create asm parser");
Clement Courbet3e138162019-09-30 12:50:25 +0000142 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)
Clement Courbet03a3d292019-09-30 13:53:50 +0000150 return make_error<Failure>("cannot create target asm parser");
Clement Courbet3e138162019-09-30 12:50:25 +0000151 AsmParser->setTargetParser(*TargetAsmParser);
152
153 if (AsmParser->Run(false))
Clement Courbet03a3d292019-09-30 13:53:50 +0000154 return make_error<Failure>("cannot parse asm file");
Clement Courbet3e138162019-09-30 12:50:25 +0000155 if (Streamer.numInvalidComments())
Clement Courbet03a3d292019-09-30 13:53:50 +0000156 return make_error<Failure>(Twine("found ")
157 .concat(Twine(Streamer.numInvalidComments()))
158 .concat(" invalid LLVM-EXEGESIS comments"));
Clement Courbet3e138162019-09-30 12:50:25 +0000159 return std::vector<BenchmarkCode>{std::move(Result)};
160}
161
162} // namespace exegesis
163} // namespace llvm