Skip to content

Commit e34801c

Browse files
author
Sjoerd Meijer
committed
[ARM][MVE] VPT Blocks: findVCMPToFoldIntoVPS
This is a recommit of D71330, but with a few things fixed and changed: 1) ReachingDefAnalysis: this was not running with optnone as it was checking skipFunction(), which other analysis passes don't do. I guess this is a copy-paste from a codegen pass. 2) VPTBlockPass: here I've added skipFunction(), because like most/all optimisations, we don't want to run this with optnone. This fixes the issues with the initial/previous commit: the VPTBlockPass was running with optnone, but ReachingDefAnalysis wasn't, and so VPTBlockPass was crashing querying ReachingDefAnalysis. I've added test case mve-vpt-block-optnone.mir to check that we don't run VPTBlock with optnone. Differential Revision: https://reviews.llvm.org/D71470
1 parent c0365aa commit e34801c

File tree

5 files changed

+245
-33
lines changed

5 files changed

+245
-33
lines changed

llvm/lib/CodeGen/ReachingDefAnalysis.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -133,8 +133,6 @@ void ReachingDefAnalysis::processBasicBlock(
133133
}
134134

135135
bool ReachingDefAnalysis::runOnMachineFunction(MachineFunction &mf) {
136-
if (skipFunction(mf.getFunction()))
137-
return false;
138136
MF = &mf;
139137
TRI = MF->getSubtarget().getRegisterInfo();
140138

llvm/lib/Target/ARM/MVEVPTBlockPass.cpp

+41-31
Original file line numberDiff line numberDiff line change
@@ -22,9 +22,9 @@
2222
#include "llvm/CodeGen/MachineInstrBuilder.h"
2323
#include "llvm/CodeGen/MachineInstrBundle.h"
2424
#include "llvm/CodeGen/MachineOperand.h"
25+
#include "llvm/CodeGen/ReachingDefAnalysis.h"
2526
#include "llvm/IR/DebugLoc.h"
2627
#include "llvm/MC/MCInstrDesc.h"
27-
#include "llvm/MC/MCRegisterInfo.h"
2828
#include "llvm/Support/Debug.h"
2929
#include <cassert>
3030
#include <new>
@@ -37,16 +37,21 @@ namespace {
3737
class MVEVPTBlock : public MachineFunctionPass {
3838
public:
3939
static char ID;
40-
const Thumb2InstrInfo *TII;
41-
const TargetRegisterInfo *TRI;
4240

4341
MVEVPTBlock() : MachineFunctionPass(ID) {}
4442

4543
bool runOnMachineFunction(MachineFunction &Fn) override;
4644

45+
void getAnalysisUsage(AnalysisUsage &AU) const override {
46+
AU.setPreservesCFG();
47+
AU.addRequired<ReachingDefAnalysis>();
48+
MachineFunctionPass::getAnalysisUsage(AU);
49+
}
50+
4751
MachineFunctionProperties getRequiredProperties() const override {
4852
return MachineFunctionProperties().set(
49-
MachineFunctionProperties::Property::NoVRegs);
53+
MachineFunctionProperties::Property::NoVRegs).set(
54+
MachineFunctionProperties::Property::TracksLiveness);
5055
}
5156

5257
StringRef getPassName() const override {
@@ -55,6 +60,9 @@ namespace {
5560

5661
private:
5762
bool InsertVPTBlocks(MachineBasicBlock &MBB);
63+
64+
const Thumb2InstrInfo *TII = nullptr;
65+
ReachingDefAnalysis *RDA = nullptr;
5866
};
5967

6068
char MVEVPTBlock::ID = 0;
@@ -63,41 +71,32 @@ namespace {
6371

6472
INITIALIZE_PASS(MVEVPTBlock, DEBUG_TYPE, "ARM MVE VPT block pass", false, false)
6573

66-
static MachineInstr *findVCMPToFoldIntoVPST(MachineBasicBlock::iterator MI,
67-
const TargetRegisterInfo *TRI,
74+
static MachineInstr *findVCMPToFoldIntoVPST(MachineInstr *MI,
75+
ReachingDefAnalysis *RDA,
6876
unsigned &NewOpcode) {
69-
// Search backwards to the instruction that defines VPR. This may or not
70-
// be a VCMP, we check that after this loop. If we find another instruction
71-
// that reads cpsr, we return nullptr.
72-
MachineBasicBlock::iterator CmpMI = MI;
73-
while (CmpMI != MI->getParent()->begin()) {
74-
--CmpMI;
75-
if (CmpMI->modifiesRegister(ARM::VPR, TRI))
76-
break;
77-
if (CmpMI->readsRegister(ARM::VPR, TRI))
78-
break;
79-
}
80-
81-
if (CmpMI == MI)
82-
return nullptr;
83-
NewOpcode = VCMPOpcodeToVPT(CmpMI->getOpcode());
84-
if (NewOpcode == 0)
77+
// First, search backwards to the instruction that defines VPR
78+
auto *Def = RDA->getReachingMIDef(MI, ARM::VPR);
79+
if (!Def)
8580
return nullptr;
8681

87-
// Search forward from CmpMI to MI, checking if either register was def'd
88-
if (registerDefinedBetween(CmpMI->getOperand(1).getReg(), std::next(CmpMI),
89-
MI, TRI))
82+
// Now check that Def is a VCMP
83+
if (!(NewOpcode = VCMPOpcodeToVPT(Def->getOpcode())))
9084
return nullptr;
91-
if (registerDefinedBetween(CmpMI->getOperand(2).getReg(), std::next(CmpMI),
92-
MI, TRI))
85+
86+
// Check that Def's operands are not defined between the VCMP and MI, i.e.
87+
// check that they have the same reaching def.
88+
if (!RDA->hasSameReachingDef(Def, MI, Def->getOperand(1).getReg()) ||
89+
!RDA->hasSameReachingDef(Def, MI, Def->getOperand(2).getReg()))
9390
return nullptr;
94-
return &*CmpMI;
91+
92+
return Def;
9593
}
9694

9795
bool MVEVPTBlock::InsertVPTBlocks(MachineBasicBlock &Block) {
9896
bool Modified = false;
9997
MachineBasicBlock::instr_iterator MBIter = Block.instr_begin();
10098
MachineBasicBlock::instr_iterator EndIter = Block.instr_end();
99+
SmallVector<MachineInstr *, 4> RemovedVCMPs;
101100

102101
while (MBIter != EndIter) {
103102
MachineInstr *MI = &*MBIter;
@@ -143,15 +142,19 @@ bool MVEVPTBlock::InsertVPTBlocks(MachineBasicBlock &Block) {
143142
// a VPST directly
144143
MachineInstrBuilder MIBuilder;
145144
unsigned NewOpcode;
146-
MachineInstr *VCMP = findVCMPToFoldIntoVPST(MI, TRI, NewOpcode);
145+
MachineInstr *VCMP = findVCMPToFoldIntoVPST(MI, RDA, NewOpcode);
147146
if (VCMP) {
148147
LLVM_DEBUG(dbgs() << " folding VCMP into VPST: "; VCMP->dump());
149148
MIBuilder = BuildMI(Block, MI, dl, TII->get(NewOpcode));
150149
MIBuilder.addImm(BlockMask);
151150
MIBuilder.add(VCMP->getOperand(1));
152151
MIBuilder.add(VCMP->getOperand(2));
153152
MIBuilder.add(VCMP->getOperand(3));
154-
VCMP->eraseFromParent();
153+
// We delay removing the actual VCMP instruction by saving it to a list
154+
// and deleting all instructions in this list in one go after we have
155+
// created the VPT blocks. We do this in order not to invalidate the
156+
// ReachingDefAnalysis that is queried by 'findVCMPToFoldIntoVPST'.
157+
RemovedVCMPs.push_back(VCMP);
155158
} else {
156159
MIBuilder = BuildMI(Block, MI, dl, TII->get(ARM::MVE_VPST));
157160
MIBuilder.addImm(BlockMask);
@@ -162,18 +165,25 @@ bool MVEVPTBlock::InsertVPTBlocks(MachineBasicBlock &Block) {
162165

163166
Modified = true;
164167
}
168+
169+
for (auto *I : RemovedVCMPs)
170+
I->eraseFromParent();
171+
165172
return Modified;
166173
}
167174

168175
bool MVEVPTBlock::runOnMachineFunction(MachineFunction &Fn) {
176+
if (skipFunction(Fn.getFunction()))
177+
return false;
178+
169179
const ARMSubtarget &STI =
170180
static_cast<const ARMSubtarget &>(Fn.getSubtarget());
171181

172182
if (!STI.isThumb2() || !STI.hasMVEIntegerOps())
173183
return false;
174184

175185
TII = static_cast<const Thumb2InstrInfo *>(STI.getInstrInfo());
176-
TRI = STI.getRegisterInfo();
186+
RDA = &getAnalysis<ReachingDefAnalysis>();
177187

178188
LLVM_DEBUG(dbgs() << "********** ARM MVE VPT BLOCKS **********\n"
179189
<< "********** Function: " << Fn.getName() << '\n');

llvm/test/CodeGen/ARM/O3-pipeline.ll

+1
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,7 @@
144144
; CHECK-NEXT: Machine Natural Loop Construction
145145
; CHECK-NEXT: Machine Block Frequency Analysis
146146
; CHECK-NEXT: If Converter
147+
; CHECK-NEXT: ReachingDefAnalysis
147148
; CHECK-NEXT: MVE VPT block insertion pass
148149
; CHECK-NEXT: Thumb IT blocks insertion pass
149150
; CHECK-NEXT: MachineDominator Tree Construction
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
# RUN: llc -run-pass arm-mve-vpt %s -o - | FileCheck %s
2+
3+
--- |
4+
target datalayout = "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
5+
target triple = "thumbv8.1m.main-arm-unknown-eabihf"
6+
define dso_local <4 x i32> @foo(<4 x i32>* %src, <4 x i32>* %src2, <4 x i32>* %src3, <4 x i32>* %dest, <4 x i32>* %dest2, <4 x i32>* %dest3, <4 x float> %a1) local_unnamed_addr #0 {
7+
entry:
8+
%c = fcmp one <4 x float> %a1, zeroinitializer
9+
%w = call <4 x i32> @llvm.masked.load.v4i32.p0v4i32(<4 x i32>* %src, i32 4, <4 x i1> %c, <4 x i32> undef)
10+
tail call void @llvm.masked.store.v4i32.p0v4i32(<4 x i32> %w, <4 x i32>* %dest, i32 4, <4 x i1> %c)
11+
%w2 = call <4 x i32> @llvm.masked.load.v4i32.p0v4i32(<4 x i32>* %src2, i32 4, <4 x i1> %c, <4 x i32> undef)
12+
tail call void @llvm.masked.store.v4i32.p0v4i32(<4 x i32> %w2, <4 x i32>* %dest2, i32 4, <4 x i1> %c)
13+
%w3 = call <4 x i32> @llvm.masked.load.v4i32.p0v4i32(<4 x i32>* %src3, i32 4, <4 x i1> %c, <4 x i32> undef)
14+
tail call void @llvm.masked.store.v4i32.p0v4i32(<4 x i32> %w3, <4 x i32>* %dest3, i32 4, <4 x i1> %c)
15+
ret <4 x i32> %w3
16+
}
17+
declare <4 x i32> @llvm.masked.load.v4i32.p0v4i32(<4 x i32>*, i32 immarg, <4 x i1>, <4 x i32>) #2
18+
declare void @llvm.masked.store.v4i32.p0v4i32(<4 x i32>, <4 x i32>*, i32 immarg, <4 x i1>) #3
19+
20+
attributes #0 = { nounwind readnone "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "frame-pointer"="all" "less-precise-fpmad"="false" "min-legal-vector-width"="128" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="generic" "target-features"="+armv8.1-m.main,+fp-armv8d16sp,+fp16,+fpregs,+fullfp16,+hwdiv,+lob,+mve.fp,+ras,+strict-align,+thumb-mode,+vfp2sp,+vfp3d16sp,+vfp4d16sp" "unsafe-fp-math"="false" "use-soft-float"="false" }
21+
attributes #1 = { nounwind readnone }
22+
attributes #2 = { argmemonly nounwind readonly willreturn }
23+
attributes #3 = { argmemonly nounwind willreturn }
24+
attributes #4 = { noduplicate nounwind }
25+
attributes #5 = { nounwind }
26+
27+
!llvm.module.flags = !{!0, !1}
28+
!llvm.ident = !{!2}
29+
30+
!0 = !{i32 1, !"wchar_size", i32 4}
31+
!1 = !{i32 1, !"min_enum_size", i32 4}
32+
!2 = !{!"clang version 10.0.0 (http://github.com/llvm/llvm-project 90450197deaf91160a22825e6746d998aad05704)"}
33+
34+
...
35+
---
36+
name: foo
37+
alignment: 2
38+
exposesReturnsTwice: false
39+
legalized: false
40+
regBankSelected: false
41+
selected: false
42+
failedISel: false
43+
tracksRegLiveness: true
44+
hasWinCFI: false
45+
registers: []
46+
liveins:
47+
- { reg: '$r0', virtual-reg: '' }
48+
- { reg: '$r1', virtual-reg: '' }
49+
- { reg: '$r2', virtual-reg: '' }
50+
- { reg: '$q0', virtual-reg: '' }
51+
frameInfo:
52+
isFrameAddressTaken: false
53+
isReturnAddressTaken: false
54+
hasStackMap: false
55+
hasPatchPoint: false
56+
stackSize: 8
57+
offsetAdjustment: 0
58+
maxAlignment: 4
59+
adjustsStack: false
60+
hasCalls: false
61+
stackProtector: ''
62+
maxCallFrameSize: 0
63+
cvBytesOfCalleeSavedRegisters: 0
64+
hasOpaqueSPAdjustment: false
65+
hasVAStart: false
66+
hasMustTailInVarArgFunc: false
67+
localFrameSize: 0
68+
savePoint: ''
69+
restorePoint: ''
70+
fixedStack:
71+
- { id: 0, type: default, offset: 12, size: 4, alignment: 4, stack-id: default,
72+
isImmutable: true, isAliased: false, callee-saved-register: '', callee-saved-restored: true,
73+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
74+
- { id: 1, type: default, offset: 8, size: 4, alignment: 8, stack-id: default,
75+
isImmutable: true, isAliased: false, callee-saved-register: '', callee-saved-restored: true,
76+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
77+
- { id: 2, type: default, offset: 4, size: 4, alignment: 4, stack-id: default,
78+
isImmutable: true, isAliased: false, callee-saved-register: '', callee-saved-restored: true,
79+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
80+
- { id: 3, type: default, offset: 0, size: 4, alignment: 8, stack-id: default,
81+
isImmutable: true, isAliased: false, callee-saved-register: '', callee-saved-restored: true,
82+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
83+
stack:
84+
- { id: 0, name: '', type: spill-slot, offset: -4, size: 4, alignment: 4,
85+
stack-id: default, callee-saved-register: '$lr', callee-saved-restored: false,
86+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
87+
- { id: 1, name: '', type: spill-slot, offset: -8, size: 4, alignment: 4,
88+
stack-id: default, callee-saved-register: '$r7', callee-saved-restored: true,
89+
debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
90+
callSites: []
91+
constants: []
92+
machineFunctionInfo: {}
93+
body: |
94+
bb.0.entry:
95+
liveins: $q0, $r0, $r1, $r2, $lr
96+
97+
; CHECK: BUNDLE implicit-def $vpr, implicit-def dead $q0, implicit-def $d0, implicit-def $s0, implicit-def $s1, implicit-def $d1, implicit-def $s2, implicit-def $s3, implicit $q0, implicit $zr, implicit killed $r0, implicit killed $r3, implicit killed $r1, implicit killed $lr {
98+
; CHECK: MVE_VPTv4f32r 1, renamable $q0, $zr, 10, implicit-def $vpr
99+
; CHECK: renamable $q0 = MVE_VLDRWU32 killed renamable $r0, 0, 1, internal renamable $vpr :: (load 16 from %ir.src, align 4)
100+
; CHECK: MVE_VSTRWU32 internal killed renamable $q0, killed renamable $r3, 0, 1, internal renamable $vpr :: (store 16 into %ir.dest, align 4)
101+
; CHECK: renamable $q0 = MVE_VLDRWU32 killed renamable $r1, 0, 1, internal renamable $vpr :: (load 16 from %ir.src2, align 4)
102+
; CHECK: MVE_VSTRWU32 internal killed renamable $q0, killed renamable $lr, 0, 1, internal renamable $vpr :: (store 16 into %ir.dest2, align 4)
103+
; CHECK: }
104+
; CHECK: BUNDLE implicit-def $q0, implicit-def $d0, implicit-def $s0, implicit-def $s1, implicit-def $d1, implicit-def $s2, implicit-def $s3, implicit killed $vpr, implicit killed $r2, implicit killed $r12 {
105+
; CHECK: MVE_VPST 4, implicit $vpr
106+
; CHECK: renamable $q0 = MVE_VLDRWU32 killed renamable $r2, 0, 1, renamable $vpr :: (load 16 from %ir.src3, align 4)
107+
; CHECK: MVE_VSTRWU32 internal renamable $q0, killed renamable $r12, 0, 1, killed renamable $vpr :: (store 16 into %ir.dest3, align 4)
108+
; CHECK: }
109+
110+
$sp = frame-setup t2STMDB_UPD $sp, 14, $noreg, killed $r7, killed $lr
111+
frame-setup CFI_INSTRUCTION def_cfa_offset 8
112+
frame-setup CFI_INSTRUCTION offset $lr, -4
113+
frame-setup CFI_INSTRUCTION offset $r7, -8
114+
$r7 = frame-setup tMOVr killed $sp, 14, $noreg
115+
frame-setup CFI_INSTRUCTION def_cfa_register $r7
116+
renamable $r12 = t2LDRi12 $r7, 16, 14, $noreg :: (load 4 from %fixed-stack.1)
117+
renamable $lr = t2LDRi12 $r7, 12, 14, $noreg :: (load 4 from %fixed-stack.2)
118+
renamable $r3 = t2LDRi12 $r7, 8, 14, $noreg :: (load 4 from %fixed-stack.3)
119+
renamable $vpr = MVE_VCMPf32r renamable $q0, $zr, 10, 0, $noreg
120+
renamable $q0 = MVE_VLDRWU32 killed renamable $r0, 0, 1, renamable $vpr :: (load 16 from %ir.src, align 4)
121+
MVE_VSTRWU32 killed renamable $q0, killed renamable $r3, 0, 1, renamable $vpr :: (store 16 into %ir.dest, align 4)
122+
renamable $q0 = MVE_VLDRWU32 killed renamable $r1, 0, 1, renamable $vpr :: (load 16 from %ir.src2, align 4)
123+
MVE_VSTRWU32 killed renamable $q0, killed renamable $lr, 0, 1, renamable $vpr :: (store 16 into %ir.dest2, align 4)
124+
renamable $q0 = MVE_VLDRWU32 killed renamable $r2, 0, 1, renamable $vpr :: (load 16 from %ir.src3, align 4)
125+
MVE_VSTRWU32 renamable $q0, killed renamable $r12, 0, 1, killed renamable $vpr :: (store 16 into %ir.dest3, align 4)
126+
$sp = t2LDMIA_RET $sp, 14, $noreg, def $r7, def $pc, implicit $q0
127+
128+
...
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
# NOTE: Assertions have been autogenerated by utils/update_mir_test_checks.py
2+
# RUN: llc -run-pass arm-mve-vpt %s -o - | FileCheck %s
3+
4+
--- |
5+
target datalayout = "e-m:e-p:32:32-Fi8-i64:64-v128:64:128-a:0:32-n32-S64"
6+
target triple = "thumbv8.1m.main-arm-none-eabi"
7+
8+
define hidden arm_aapcs_vfpcc <4 x float> @test_vminnmq_m_f32_v2(<4 x float> %inactive, <4 x float> %a, <4 x float> %b, i16 zeroext %p) local_unnamed_addr #0 {
9+
entry:
10+
%conv.i = zext i16 %p to i32
11+
%0 = tail call nnan ninf nsz <4 x float> @llvm.arm.mve.vminnm.m.v4f32.v4f32.v4f32.v4f32.i32(<4 x float> %inactive, <4 x float> %a, <4 x float> %b, i32 %conv.i) #2
12+
ret <4 x float> %0
13+
}
14+
15+
declare <4 x float> @llvm.arm.mve.vminnm.m.v4f32.v4f32.v4f32.v4f32.i32(<4 x float>, <4 x float>, <4 x float>, i32) #1
16+
17+
attributes #0 = { noinline optnone nounwind readnone "correctly-rounded-divide-sqrt-fp-math"="false" "denormal-fp-math"="preserve-sign" "disable-tail-calls"="false" "less-precise-fpmad"="false" "min-legal-vector-width"="128" "no-frame-pointer-elim"="false" "no-infs-fp-math"="true" "no-jump-tables"="false" "no-nans-fp-math"="true" "no-signed-zeros-fp-math"="true" "no-trapping-math"="true" "stack-protector-buffer-size"="8" "target-cpu"="generic" "target-features"="+armv8.1-m.main,+hwdiv,+mve.fp,+ras,+thumb-mode" "unsafe-fp-math"="false" "use-soft-float"="false" }
18+
attributes #1 = { nounwind readnone }
19+
attributes #2 = { nounwind }
20+
21+
22+
...
23+
---
24+
name: test_vminnmq_m_f32_v2
25+
alignment: 4
26+
exposesReturnsTwice: false
27+
legalized: false
28+
regBankSelected: false
29+
selected: false
30+
failedISel: false
31+
tracksRegLiveness: true
32+
hasWinCFI: false
33+
registers: []
34+
liveins:
35+
- { reg: '$q0', virtual-reg: '' }
36+
- { reg: '$q1', virtual-reg: '' }
37+
- { reg: '$q2', virtual-reg: '' }
38+
- { reg: '$r0', virtual-reg: '' }
39+
frameInfo:
40+
isFrameAddressTaken: false
41+
isReturnAddressTaken: false
42+
hasStackMap: false
43+
hasPatchPoint: false
44+
stackSize: 0
45+
offsetAdjustment: 0
46+
maxAlignment: 0
47+
adjustsStack: false
48+
hasCalls: false
49+
stackProtector: ''
50+
maxCallFrameSize: 0
51+
cvBytesOfCalleeSavedRegisters: 0
52+
hasOpaqueSPAdjustment: false
53+
hasVAStart: false
54+
hasMustTailInVarArgFunc: false
55+
localFrameSize: 0
56+
savePoint: ''
57+
restorePoint: ''
58+
fixedStack: []
59+
stack: []
60+
constants: []
61+
body: |
62+
bb.0.entry:
63+
liveins: $q0, $q1, $q2, $r0
64+
65+
; CHECK-LABEL: name: test_vminnmq_m_f32_v2
66+
; CHECK: liveins: $q0, $q1, $q2, $r0
67+
; CHECK: $vpr = VMSR_P0 killed $r0, 14, $noreg
68+
; CHECK: renamable $q0 = nnan ninf nsz MVE_VMINNMf32 killed renamable $q1, killed renamable $q2, 1, killed renamable $vpr, killed renamable $q0
69+
; CHECK: tBX_RET 14, $noreg, implicit $q0
70+
71+
$vpr = VMSR_P0 killed $r0, 14, $noreg
72+
renamable $q0 = nnan ninf nsz MVE_VMINNMf32 killed renamable $q1, killed renamable $q2, 1, killed renamable $vpr, killed renamable $q0
73+
tBX_RET 14, $noreg, implicit $q0
74+
75+
...

0 commit comments

Comments
 (0)