When lowering switch statement, if bit tests are used then LLVM will always generates a jump to the default statement in the last bit test. However, this is not necessary when all cases in bit tests cover a contiguous range. This is because when generating the bit tests header MBB, there is a range check that guarantees cases in bit tests won't go outside of [low, high], where low and high are minimum and maximum case values in the bit tests. This patch checks if this is the case and then doesn't emit jump to default statement and hence saves a bit test and a branch.
For example, given the following IR:
define void @foo(i32 %x) {
entry:
switch i32 %x, label %return [
i32 0, label %bb0
i32 2, label %bb0
i32 4, label %bb0
i32 1, label %bb1
i32 3, label %bb1
i32 5, label %bb1
]
bb0: tail call void @a() br label %return
bb1: tail call void @b() br label %return
return: ret void
}
declare void @a()
declare void @b()Previously LLVM generates the following MC:
BB#0: derived from LLVM BB %entry
Live Ins: %EDI
CMP32ri8 %EDI, 5, %EFLAGS<imp-def>
JA_1 <BB#3>, %EFLAGS<imp-use>
Successors according to CFG: BB#3(16) BB#1(16)
BB#1: derived from LLVM BB %entry
Live Ins: %EDI
Predecessors according to CFG: BB#0
%EAX<def> = MOV32ri 21
BT32rr %EAX<kill>, %EDI, %EFLAGS<imp-def>
JAE_1 <BB#2>, %EFLAGS<imp-use>
Successors according to CFG: BB#4(48) BB#2(48)
BB#4: derived from LLVM BB %bb0
Predecessors according to CFG: BB#1
TAILJMPd64 <ga:@a>, <regmask>, %RSP<imp-use>, %RSP<imp-use>
BB#2: derived from LLVM BB %entry
Live Ins: %EDI
Predecessors according to CFG: BB#1
%EAX<def> = MOV32ri 42
BT32rr %EAX<kill>, %EDI<kill>, %EFLAGS<imp-def>
JAE_1 <BB#3>, %EFLAGS<imp-use>
Successors according to CFG: BB#5(48) BB#3(16)
BB#5: derived from LLVM BB %bb1
Predecessors according to CFG: BB#2
TAILJMPd64 <ga:@b>, <regmask>, %RSP<imp-use>, %RSP<imp-use>
BB#3: derived from LLVM BB %return
Predecessors according to CFG: BB#0 BB#2
RETQWith this patch the MC generated is:
BB#0: derived from LLVM BB %entry
Live Ins: %EDI
CMP32ri8 %EDI, 5, %EFLAGS<imp-def>
JBE_1 <BB#1>, %EFLAGS<imp-use>
Successors according to CFG: BB#3(16) BB#1(16)
BB#3: derived from LLVM BB %return
Predecessors according to CFG: BB#0
RETQ
BB#1: derived from LLVM BB %entry
Live Ins: %EDI
Predecessors according to CFG: BB#0
%EAX<def> = MOV32ri 21
BT32rr %EAX<kill>, %EDI<kill>, %EFLAGS<imp-def>
JAE_1 <BB#2>, %EFLAGS<imp-use>
Successors according to CFG: BB#4(48) BB#2(48)
BB#4: derived from LLVM BB %bb0
Predecessors according to CFG: BB#1
TAILJMPd64 <ga:@a>, <regmask>, %RSP<imp-use>, %RSP<imp-use>
BB#2: derived from LLVM BB %bb1
Predecessors according to CFG: BB#1
TAILJMPd64 <ga:@b>, <regmask>, %RSP<imp-use>, %RSP<imp-use>We can save one basic block with 3 instructions.