Skip to content

Commit 083addf

Browse files
committed
[llvm-mca] [llvm-mca] Improved error handling and error reporting from class InstrBuilder.
A new class named InstructionError has been added to Support.h in order to improve the error reporting from class InstrBuilder. The llvm-mca driver is responsible for handling InstructionError objects, and printing them out to stderr. The goal of this patch is to remove all the remaining error handling logic from the library code. In particular, this allows us to: - Simplify the logic in InstrBuilder by removing a needless dependency from MCInstrPrinter. - Centralize all the error halding logic in a new function named 'runPipeline' (see llvm-mca.cpp). This is also a first step towards generalizing class InstrBuilder, so that in future, we will be able to reuse its logic to also "lower" MachineInstr to mca::Instruction objects. Differential Revision: https://reviews.llvm.org/D53585 llvm-svn: 345129
1 parent 5a74a9c commit 083addf

File tree

5 files changed

+84
-47
lines changed

5 files changed

+84
-47
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
# RUN: not llvm-mca -march=arm -mcpu=swift -all-views=false 2>&1 < %s | FileCheck %s
2+
3+
add r3, r1, r12, lsl #2
4+
5+
# CHECK: error: unable to resolve scheduling class for write variant.
6+
# CHECK-NEXT: note: instruction: add r3, r1, r12, lsl #2

llvm/tools/llvm-mca/include/InstrBuilder.h

+2-4
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,6 @@
1717

1818
#include "Instruction.h"
1919
#include "Support.h"
20-
#include "llvm/MC/MCInstPrinter.h"
2120
#include "llvm/MC/MCInstrAnalysis.h"
2221
#include "llvm/MC/MCInstrInfo.h"
2322
#include "llvm/MC/MCRegisterInfo.h"
@@ -41,7 +40,6 @@ class InstrBuilder {
4140
const llvm::MCInstrInfo &MCII;
4241
const llvm::MCRegisterInfo &MRI;
4342
const llvm::MCInstrAnalysis &MCIA;
44-
llvm::MCInstPrinter &MCIP;
4543
llvm::SmallVector<uint64_t, 8> ProcResourceMasks;
4644

4745
llvm::DenseMap<unsigned short, std::unique_ptr<const InstrDesc>> Descriptors;
@@ -66,8 +64,8 @@ class InstrBuilder {
6664
public:
6765
InstrBuilder(const llvm::MCSubtargetInfo &sti, const llvm::MCInstrInfo &mcii,
6866
const llvm::MCRegisterInfo &mri,
69-
const llvm::MCInstrAnalysis &mcia, llvm::MCInstPrinter &mcip)
70-
: STI(sti), MCII(mcii), MRI(mri), MCIA(mcia), MCIP(mcip),
67+
const llvm::MCInstrAnalysis &mcia)
68+
: STI(sti), MCII(mcii), MRI(mri), MCIA(mcia),
7169
ProcResourceMasks(STI.getSchedModel().getNumProcResourceKinds()) {
7270
computeProcResourceMasks(STI.getSchedModel(), ProcResourceMasks);
7371
}

llvm/tools/llvm-mca/include/Support.h

+20
Original file line numberDiff line numberDiff line change
@@ -18,9 +18,29 @@
1818
#include "llvm/ADT/ArrayRef.h"
1919
#include "llvm/ADT/SmallVector.h"
2020
#include "llvm/MC/MCSchedule.h"
21+
#include "llvm/Support/Error.h"
2122

2223
namespace mca {
2324

25+
template <typename T>
26+
class InstructionError : public llvm::ErrorInfo<InstructionError<T>> {
27+
public:
28+
static char ID;
29+
std::string Message;
30+
const T &Inst;
31+
32+
InstructionError(std::string M, const T &MCI)
33+
: Message(std::move(M)), Inst(MCI) {}
34+
35+
void log(llvm::raw_ostream &OS) const override { OS << Message; }
36+
37+
std::error_code convertToErrorCode() const override {
38+
return llvm::inconvertibleErrorCode();
39+
}
40+
};
41+
42+
template <typename T> char InstructionError<T>::ID;
43+
2444
/// This class represents the number of cycles per resource (fractions of
2545
/// cycles). That quantity is managed here as a ratio, and accessed via the
2646
/// double cast-operator below. The two quantities, number of cycles and

llvm/tools/llvm-mca/lib/InstrBuilder.cpp

+23-36
Original file line numberDiff line numberDiff line change
@@ -215,9 +215,8 @@ Error InstrBuilder::populateWrites(InstrDesc &ID, const MCInst &MCI,
215215
}
216216

217217
if (CurrentDef != NumExplicitDefs) {
218-
return make_error<StringError>(
219-
"error: Expected more register operand definitions.",
220-
inconvertibleErrorCode());
218+
return make_error<InstructionError<MCInst>>(
219+
"Expected more register operand definitions.", MCI);
221220
}
222221

223222
CurrentDef = 0;
@@ -253,11 +252,12 @@ Error InstrBuilder::populateWrites(InstrDesc &ID, const MCInst &MCI,
253252
// Always assume that the optional definition is the last operand of the
254253
// MCInst sequence.
255254
const MCOperand &Op = MCI.getOperand(MCI.getNumOperands() - 1);
256-
if (i == MCI.getNumOperands() || !Op.isReg())
257-
return make_error<StringError>(
258-
"error: expected a register operand for an optional "
259-
"definition. Instruction has not be correctly analyzed.",
260-
inconvertibleErrorCode());
255+
if (i == MCI.getNumOperands() || !Op.isReg()) {
256+
std::string Message =
257+
"expected a register operand for an optional definition. Instruction "
258+
"has not been correctly analyzed.";
259+
return make_error<InstructionError<MCInst>>(Message, MCI);
260+
}
261261

262262
WriteDescriptor &Write = ID.Writes[TotalDefs - 1];
263263
Write.OpIndex = MCI.getNumOperands() - 1;
@@ -284,9 +284,8 @@ Error InstrBuilder::populateReads(InstrDesc &ID, const MCInst &MCI,
284284
}
285285

286286
if (NumExplicitDefs) {
287-
return make_error<StringError>(
288-
"error: Expected more register operand definitions. ",
289-
inconvertibleErrorCode());
287+
return make_error<InstructionError<MCInst>>(
288+
"Expected more register operand definitions.", MCI);
290289
}
291290

292291
unsigned NumExplicitUses = MCI.getNumOperands() - i;
@@ -332,23 +331,18 @@ Error InstrBuilder::verifyInstrDesc(const InstrDesc &ID,
332331
if (!UsesMemory && !UsesBuffers && !UsesResources)
333332
return ErrorSuccess();
334333

335-
std::string ToString;
336-
raw_string_ostream OS(ToString);
334+
StringRef Message;
337335
if (UsesMemory) {
338-
WithColor::error() << "found an inconsistent instruction that decodes "
339-
<< "into zero opcodes and that consumes load/store "
340-
<< "unit resources.\n";
336+
Message = "found an inconsistent instruction that decodes "
337+
"into zero opcodes and that consumes load/store "
338+
"unit resources.";
341339
} else {
342-
WithColor::error() << "found an inconsistent instruction that decodes"
343-
<< " to zero opcodes and that consumes scheduler "
344-
<< "resources.\n";
340+
Message = "found an inconsistent instruction that decodes "
341+
"to zero opcodes and that consumes scheduler "
342+
"resources.";
345343
}
346344

347-
MCIP.printInst(&MCI, OS, "", STI);
348-
OS.flush();
349-
WithColor::note() << "instruction: " << ToString << '\n';
350-
return make_error<StringError>("Invalid instruction definition found",
351-
inconvertibleErrorCode());
345+
return make_error<InstructionError<MCInst>>(Message, MCI);
352346
}
353347

354348
Expected<const InstrDesc &>
@@ -371,24 +365,17 @@ InstrBuilder::createInstrDescImpl(const MCInst &MCI) {
371365
SchedClassID = STI.resolveVariantSchedClass(SchedClassID, &MCI, CPUID);
372366

373367
if (!SchedClassID) {
374-
return make_error<StringError>("unable to resolve this variant class.",
375-
inconvertibleErrorCode());
368+
return make_error<InstructionError<MCInst>>(
369+
"unable to resolve scheduling class for write variant.", MCI);
376370
}
377371
}
378372

379373
// Check if this instruction is supported. Otherwise, report an error.
380374
const MCSchedClassDesc &SCDesc = *SM.getSchedClassDesc(SchedClassID);
381375
if (SCDesc.NumMicroOps == MCSchedClassDesc::InvalidNumMicroOps) {
382-
std::string ToString;
383-
raw_string_ostream OS(ToString);
384-
WithColor::error() << "found an unsupported instruction in the input"
385-
<< " assembly sequence.\n";
386-
MCIP.printInst(&MCI, OS, "", STI);
387-
OS.flush();
388-
WithColor::note() << "instruction: " << ToString << '\n';
389-
return make_error<StringError>(
390-
"Don't know how to analyze unsupported instructions",
391-
inconvertibleErrorCode());
376+
return make_error<InstructionError<MCInst>>(
377+
"found an unsupported instruction in the input assembly sequence.",
378+
MCI);
392379
}
393380

394381
// Create a new empty descriptor.

llvm/tools/llvm-mca/llvm-mca.cpp

+33-7
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
#include "Views/TimelineView.h"
3636
#include "include/Context.h"
3737
#include "include/Pipeline.h"
38+
#include "include/Support.h"
3839
#include "llvm/MC/MCAsmInfo.h"
3940
#include "llvm/MC/MCContext.h"
4041
#include "llvm/MC/MCObjectFileInfo.h"
@@ -326,6 +327,30 @@ static void processViewOptions() {
326327
processOptionImpl(PrintRetireStats, Default);
327328
}
328329

330+
// Returns true on success.
331+
static bool runPipeline(mca::Pipeline &P, MCInstPrinter &MCIP,
332+
const MCSubtargetInfo &STI) {
333+
// Handle pipeline errors here.
334+
if (auto Err = P.run()) {
335+
if (auto NewE = handleErrors(
336+
std::move(Err),
337+
[&MCIP, &STI](const mca::InstructionError<MCInst> &IE) {
338+
std::string InstructionStr;
339+
raw_string_ostream SS(InstructionStr);
340+
WithColor::error() << IE.Message << '\n';
341+
MCIP.printInst(&IE.Inst, SS, "", STI);
342+
SS.flush();
343+
WithColor::note() << "instruction: " << InstructionStr << '\n';
344+
})) {
345+
// Default case.
346+
WithColor::error() << toString(std::move(NewE));
347+
}
348+
return false;
349+
}
350+
351+
return true;
352+
}
353+
329354
int main(int argc, char **argv) {
330355
InitLLVM X(argc, argv);
331356

@@ -462,7 +487,7 @@ int main(int argc, char **argv) {
462487
Width = DispatchWidth;
463488

464489
// Create an instruction builder.
465-
mca::InstrBuilder IB(*STI, *MCII, *MRI, *MCIA, *IP);
490+
mca::InstrBuilder IB(*STI, *MCII, *MRI, *MCIA);
466491

467492
// Create a context to control ownership of the pipeline hardware.
468493
mca::Context MCA(*MRI, *STI);
@@ -504,9 +529,10 @@ int main(int argc, char **argv) {
504529
}
505530
Printer.addView(
506531
llvm::make_unique<mca::ResourcePressureView>(*STI, *IP, S));
507-
auto Err = P->run();
508-
if (Err)
509-
report_fatal_error(toString(std::move(Err)));
532+
533+
if (!runPipeline(*P, *IP, *STI))
534+
return 1;
535+
510536
Printer.printReport(TOF->os());
511537
continue;
512538
}
@@ -543,9 +569,9 @@ int main(int argc, char **argv) {
543569
*STI, *IP, S, TimelineMaxIterations, TimelineMaxCycles));
544570
}
545571

546-
auto Err = P->run();
547-
if (Err)
548-
report_fatal_error(toString(std::move(Err)));
572+
if (!runPipeline(*P, *IP, *STI))
573+
return 1;
574+
549575
Printer.printReport(TOF->os());
550576

551577
// Clear the InstrBuilder internal state in preparation for another round.

0 commit comments

Comments
 (0)