From a2e819fb092ec6a33438520ce74c3afe193b5359 Mon Sep 17 00:00:00 2001 From: Alex Lorenz Date: Wed, 15 Jul 2015 23:38:35 +0000 Subject: [PATCH] MIR Serialization: Serialize the jump table index operands. Reviewers: Duncan P. N. Exon Smith git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@242358 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/CodeGen/MIRParser/MILexer.cpp | 19 ++++ lib/CodeGen/MIRParser/MILexer.h | 6 +- lib/CodeGen/MIRParser/MIParser.cpp | 17 +++ lib/CodeGen/MIRParser/MIParser.h | 1 + lib/CodeGen/MIRParser/MIRParser.cpp | 8 +- lib/CodeGen/MIRPrinter.cpp | 4 + test/CodeGen/MIR/X86/jump-table-info.mir | 104 +++++++++++++++++- .../MIR/X86/undefined-jump-table-id.mir | 80 ++++++++++++++ 8 files changed, 233 insertions(+), 6 deletions(-) create mode 100644 test/CodeGen/MIR/X86/undefined-jump-table-id.mir diff --git a/lib/CodeGen/MIRParser/MILexer.cpp b/lib/CodeGen/MIRParser/MILexer.cpp index 482c33ae223..9babb3b0a3d 100644 --- a/lib/CodeGen/MIRParser/MILexer.cpp +++ b/lib/CodeGen/MIRParser/MILexer.cpp @@ -115,6 +115,23 @@ static Cursor maybeLexMachineBasicBlock( return C; } +static Cursor maybeLexIndex(Cursor C, MIToken &Token, StringRef Rule, + MIToken::TokenKind Kind) { + if (!C.remaining().startswith(Rule) || !isdigit(C.peek(Rule.size()))) + return None; + auto Range = C; + C.advance(Rule.size()); + auto NumberRange = C; + while (isdigit(C.peek())) + C.advance(); + Token = MIToken(Kind, Range.upto(C), APSInt(NumberRange.upto(C))); + return C; +} + +static Cursor maybeLexJumpTableIndex(Cursor C, MIToken &Token) { + return maybeLexIndex(C, Token, "%jump-table.", MIToken::JumpTableIndex); +} + static Cursor lexVirtualRegister(Cursor C, MIToken &Token) { auto Range = C; C.advance(); // Skip '%' @@ -209,6 +226,8 @@ StringRef llvm::lexMIToken( return R.remaining(); if (Cursor R = maybeLexMachineBasicBlock(C, Token, ErrorCallback)) return R.remaining(); + if (Cursor R = maybeLexJumpTableIndex(C, Token)) + return R.remaining(); if (Cursor R = maybeLexRegister(C, Token)) return R.remaining(); if (Cursor R = maybeLexGlobalValue(C, Token)) diff --git a/lib/CodeGen/MIRParser/MILexer.h b/lib/CodeGen/MIRParser/MILexer.h index 55460b56e7d..b1c05226158 100644 --- a/lib/CodeGen/MIRParser/MILexer.h +++ b/lib/CodeGen/MIRParser/MILexer.h @@ -53,7 +53,8 @@ struct MIToken { // Other tokens IntegerLiteral, - VirtualRegister + VirtualRegister, + JumpTableIndex }; private: @@ -96,7 +97,8 @@ public: bool hasIntegerValue() const { return Kind == IntegerLiteral || Kind == MachineBasicBlock || - Kind == GlobalValue || Kind == VirtualRegister; + Kind == GlobalValue || Kind == VirtualRegister || + Kind == JumpTableIndex; } }; diff --git a/lib/CodeGen/MIRParser/MIParser.cpp b/lib/CodeGen/MIRParser/MIParser.cpp index c00011288a6..62a3fbe1e5f 100644 --- a/lib/CodeGen/MIRParser/MIParser.cpp +++ b/lib/CodeGen/MIRParser/MIParser.cpp @@ -88,6 +88,7 @@ public: bool parseMBBReference(MachineBasicBlock *&MBB); bool parseMBBOperand(MachineOperand &Dest); bool parseGlobalAddressOperand(MachineOperand &Dest); + bool parseJumpTableIndexOperand(MachineOperand &Dest); bool parseMachineOperand(MachineOperand &Dest); private: @@ -468,6 +469,20 @@ bool MIParser::parseGlobalAddressOperand(MachineOperand &Dest) { return false; } +bool MIParser::parseJumpTableIndexOperand(MachineOperand &Dest) { + assert(Token.is(MIToken::JumpTableIndex)); + unsigned ID; + if (getUnsigned(ID)) + return true; + auto JumpTableEntryInfo = PFS.JumpTableSlots.find(ID); + if (JumpTableEntryInfo == PFS.JumpTableSlots.end()) + return error("use of undefined jump table '%jump-table." + Twine(ID) + "'"); + lex(); + // TODO: Parse target flags. + Dest = MachineOperand::CreateJTI(JumpTableEntryInfo->second); + return false; +} + bool MIParser::parseMachineOperand(MachineOperand &Dest) { switch (Token.kind()) { case MIToken::kw_implicit: @@ -486,6 +501,8 @@ bool MIParser::parseMachineOperand(MachineOperand &Dest) { case MIToken::GlobalValue: case MIToken::NamedGlobalValue: return parseGlobalAddressOperand(Dest); + case MIToken::JumpTableIndex: + return parseJumpTableIndexOperand(Dest); case MIToken::Error: return true; case MIToken::Identifier: diff --git a/lib/CodeGen/MIRParser/MIParser.h b/lib/CodeGen/MIRParser/MIParser.h index fca4c4e6f88..405e637267f 100644 --- a/lib/CodeGen/MIRParser/MIParser.h +++ b/lib/CodeGen/MIRParser/MIParser.h @@ -29,6 +29,7 @@ class SourceMgr; struct PerFunctionMIParsingState { DenseMap MBBSlots; DenseMap VirtualRegisterSlots; + DenseMap JumpTableSlots; }; bool parseMachineInstr(MachineInstr *&MI, SourceMgr &SM, MachineFunction &MF, diff --git a/lib/CodeGen/MIRParser/MIRParser.cpp b/lib/CodeGen/MIRParser/MIRParser.cpp index 1ba5db841af..fb1878febda 100644 --- a/lib/CodeGen/MIRParser/MIRParser.cpp +++ b/lib/CodeGen/MIRParser/MIRParser.cpp @@ -113,7 +113,7 @@ public: bool initializeJumpTableInfo(MachineFunction &MF, const yaml::MachineJumpTable &YamlJTI, - const PerFunctionMIParsingState &PFS); + PerFunctionMIParsingState &PFS); private: /// Return a MIR diagnostic converted from an MI string diagnostic. @@ -436,7 +436,7 @@ bool MIRParserImpl::initializeFrameInfo(const Function &F, bool MIRParserImpl::initializeJumpTableInfo( MachineFunction &MF, const yaml::MachineJumpTable &YamlJTI, - const PerFunctionMIParsingState &PFS) { + PerFunctionMIParsingState &PFS) { MachineJumpTableInfo *JTI = MF.getOrCreateJumpTableInfo(YamlJTI.Kind); SMDiagnostic Error; for (const auto &Entry : YamlJTI.Entries) { @@ -447,7 +447,9 @@ bool MIRParserImpl::initializeJumpTableInfo( return error(Error, MBBSource.SourceRange); Blocks.push_back(MBB); } - JTI->createJumpTableIndex(Blocks); + unsigned Index = JTI->createJumpTableIndex(Blocks); + // TODO: Report an error when the same jump table slot ID is redefined. + PFS.JumpTableSlots.insert(std::make_pair(Entry.ID, Index)); } return false; } diff --git a/lib/CodeGen/MIRPrinter.cpp b/lib/CodeGen/MIRPrinter.cpp index 97c323aaba4..50715e9f127 100644 --- a/lib/CodeGen/MIRPrinter.cpp +++ b/lib/CodeGen/MIRPrinter.cpp @@ -350,6 +350,10 @@ void MIPrinter::print(const MachineOperand &Op, const TargetRegisterInfo *TRI) { case MachineOperand::MO_MachineBasicBlock: printMBBReference(*Op.getMBB()); break; + case MachineOperand::MO_JumpTableIndex: + OS << "%jump-table." << Op.getIndex(); + // TODO: Print target flags. + break; case MachineOperand::MO_GlobalAddress: Op.getGlobal()->printAsOperand(OS, /*PrintType=*/false, MST); // TODO: Print offset and target flags. diff --git a/test/CodeGen/MIR/X86/jump-table-info.mir b/test/CodeGen/MIR/X86/jump-table-info.mir index 0246f9f21d1..a52a87257b2 100644 --- a/test/CodeGen/MIR/X86/jump-table-info.mir +++ b/test/CodeGen/MIR/X86/jump-table-info.mir @@ -1,5 +1,6 @@ # RUN: llc -start-after branch-folder -stop-after branch-folder -o /dev/null %s | FileCheck %s -# This test ensures that the MIR parser parses the jump table info correctly. +# This test ensures that the MIR parser parses the jump table info and jump +# table operands correctly. --- | @@ -28,6 +29,31 @@ ret i32 8 } + define i32 @test_jumptable2(i32 %in) { + entry: + switch i32 %in, label %def [ + i32 0, label %lbl1 + i32 1, label %lbl2 + i32 2, label %lbl3 + i32 3, label %lbl4 + ] + + def: + ret i32 0 + + lbl1: + ret i32 1 + + lbl2: + ret i32 2 + + lbl3: + ret i32 4 + + lbl4: + ret i32 8 + } + ... --- name: test_jumptable @@ -46,17 +72,93 @@ body: - id: 0 name: entry successors: [ '%bb.2.def', '%bb.1.entry' ] + instructions: + - '%eax = MOV32rr %edi, implicit-def %rax' + - 'CMP32ri8 %edi, 3, implicit-def %eflags' + - 'JA_1 %bb.2.def, implicit %eflags' - id: 1 name: entry successors: [ '%bb.3.lbl1', '%bb.4.lbl2', '%bb.5.lbl3', '%bb.6.lbl4' ] + instructions: + # CHECK: %rcx = LEA64r %rip, 1, _, %jump-table.0, _ + - '%rcx = LEA64r %rip, 1, _, %jump-table.0, _' + - '%rax = MOVSX64rm32 %rcx, 4, %rax, 0, _' + - '%rax = ADD64rr %rax, %rcx, implicit-def %eflags' + - 'JMP64r %rax' - id: 2 name: def + instructions: + - '%eax = MOV32r0 implicit-def %eflags' + - 'RETQ %eax' - id: 3 name: lbl1 + instructions: + - '%eax = MOV32ri 1' + - 'RETQ %eax' - id: 4 name: lbl2 + instructions: + - '%eax = MOV32ri 2' + - 'RETQ %eax' - id: 5 name: lbl3 + instructions: + - '%eax = MOV32ri 4' + - 'RETQ %eax' - id: 6 name: lbl4 + instructions: + - '%eax = MOV32ri 8' + - 'RETQ %eax' +... +--- +name: test_jumptable2 +jumpTable: + kind: label-difference32 + entries: + - id: 1 + blocks: [ '%bb.3.lbl1', '%bb.4.lbl2', '%bb.5.lbl3', '%bb.6.lbl4' ] +body: + - id: 0 + name: entry + successors: [ '%bb.2.def', '%bb.1.entry' ] + instructions: + - '%eax = MOV32rr %edi, implicit-def %rax' + - 'CMP32ri8 %edi, 3, implicit-def %eflags' + - 'JA_1 %bb.2.def, implicit %eflags' + - id: 1 + name: entry + successors: [ '%bb.3.lbl1', '%bb.4.lbl2', '%bb.5.lbl3', '%bb.6.lbl4' ] + instructions: + # Verify that the printer will use an id of 0 for this jump table: + # CHECK: %rcx = LEA64r %rip, 1, _, %jump-table.0, _ + - '%rcx = LEA64r %rip, 1, _, %jump-table.1, _' + - '%rax = MOVSX64rm32 %rcx, 4, %rax, 0, _' + - '%rax = ADD64rr %rax, %rcx, implicit-def %eflags' + - 'JMP64r %rax' + - id: 2 + name: def + instructions: + - '%eax = MOV32r0 implicit-def %eflags' + - 'RETQ %eax' + - id: 3 + name: lbl1 + instructions: + - '%eax = MOV32ri 1' + - 'RETQ %eax' + - id: 4 + name: lbl2 + instructions: + - '%eax = MOV32ri 2' + - 'RETQ %eax' + - id: 5 + name: lbl3 + instructions: + - '%eax = MOV32ri 4' + - 'RETQ %eax' + - id: 6 + name: lbl4 + instructions: + - '%eax = MOV32ri 8' + - 'RETQ %eax' ... diff --git a/test/CodeGen/MIR/X86/undefined-jump-table-id.mir b/test/CodeGen/MIR/X86/undefined-jump-table-id.mir new file mode 100644 index 00000000000..f4bc41cd8f7 --- /dev/null +++ b/test/CodeGen/MIR/X86/undefined-jump-table-id.mir @@ -0,0 +1,80 @@ +# RUN: not llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s 2>&1 | FileCheck %s + +--- | + + define i32 @test_jumptable(i32 %in) { + entry: + switch i32 %in, label %def [ + i32 0, label %lbl1 + i32 1, label %lbl2 + i32 2, label %lbl3 + i32 3, label %lbl4 + ] + + def: + ret i32 0 + + lbl1: + ret i32 1 + + lbl2: + ret i32 2 + + lbl3: + ret i32 4 + + lbl4: + ret i32 8 + } + +... +--- +name: test_jumptable +jumpTable: + kind: label-difference32 + entries: + - id: 0 + blocks: [ '%bb.3.lbl1', '%bb.4.lbl2', '%bb.5.lbl3', '%bb.6.lbl4' ] +body: + - id: 0 + name: entry + successors: [ '%bb.2.def', '%bb.1.entry' ] + instructions: + - '%eax = MOV32rr %edi, implicit-def %rax' + - 'CMP32ri8 %edi, 3, implicit-def %eflags' + - 'JA_1 %bb.2.def, implicit %eflags' + - id: 1 + name: entry + successors: [ '%bb.3.lbl1', '%bb.4.lbl2', '%bb.5.lbl3', '%bb.6.lbl4' ] + instructions: + # CHECK: [[@LINE+1]]:36: use of undefined jump table '%jump-table.2' + - '%rcx = LEA64r %rip, 1, _, %jump-table.2, _' + - '%rax = MOVSX64rm32 %rcx, 4, %rax, 0, _' + - '%rax = ADD64rr %rax, %rcx, implicit-def %eflags' + - 'JMP64r %rax' + - id: 2 + name: def + instructions: + - '%eax = MOV32r0 implicit-def %eflags' + - 'RETQ %eax' + - id: 3 + name: lbl1 + instructions: + - '%eax = MOV32ri 1' + - 'RETQ %eax' + - id: 4 + name: lbl2 + instructions: + - '%eax = MOV32ri 2' + - 'RETQ %eax' + - id: 5 + name: lbl3 + instructions: + - '%eax = MOV32ri 4' + - 'RETQ %eax' + - id: 6 + name: lbl4 + instructions: + - '%eax = MOV32ri 8' + - 'RETQ %eax' +...