1 //===- HexagonStoreWidening.cpp -------------------------------------------===//
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
7 //===----------------------------------------------------------------------===//
8 // Replace sequences of "narrow" stores to adjacent memory locations with
9 // a fewer "wide" stores that have the same effect.
10 // For example, replace:
11 // S4_storeirb_io %100, 0, 0 ; store-immediate-byte
12 // S4_storeirb_io %100, 1, 0 ; store-immediate-byte
14 // S4_storeirh_io %100, 0, 0 ; store-immediate-halfword
15 // The above is the general idea. The actual cases handled by the code
16 // may be a bit more complex.
17 // The purpose of this pass is to reduce the number of outstanding stores,
18 // or as one could say, "reduce store queue pressure". Also, wide stores
19 // mean fewer stores, and since there are only two memory instructions allowed
20 // per packet, it also means fewer packets, and ultimately fewer cycles.
21 //===---------------------------------------------------------------------===//
23 #define DEBUG_TYPE "hexagon-widen-stores"
25 #include "HexagonInstrInfo.h"
26 #include "HexagonRegisterInfo.h"
27 #include "HexagonSubtarget.h"
28 #include "llvm/ADT/SmallPtrSet.h"
29 #include "llvm/Analysis/AliasAnalysis.h"
30 #include "llvm/Analysis/MemoryLocation.h"
31 #include "llvm/CodeGen/MachineBasicBlock.h"
32 #include "llvm/CodeGen/MachineFunction.h"
33 #include "llvm/CodeGen/MachineFunctionPass.h"
34 #include "llvm/CodeGen/MachineInstr.h"
35 #include "llvm/CodeGen/MachineInstrBuilder.h"
36 #include "llvm/CodeGen/MachineMemOperand.h"
37 #include "llvm/CodeGen/MachineOperand.h"
38 #include "llvm/CodeGen/MachineRegisterInfo.h"
39 #include "llvm/IR/DebugLoc.h"
40 #include "llvm/MC/MCInstrDesc.h"
41 #include "llvm/Pass.h"
42 #include "llvm/Support/Debug.h"
43 #include "llvm/Support/ErrorHandling.h"
44 #include "llvm/Support/MathExtras.h"
45 #include "llvm/Support/raw_ostream.h"
56 FunctionPass
*createHexagonStoreWidening();
57 void initializeHexagonStoreWideningPass(PassRegistry
&);
59 } // end namespace llvm
63 struct HexagonStoreWidening
: public MachineFunctionPass
{
64 const HexagonInstrInfo
*TII
;
65 const HexagonRegisterInfo
*TRI
;
66 const MachineRegisterInfo
*MRI
;
73 HexagonStoreWidening() : MachineFunctionPass(ID
) {
74 initializeHexagonStoreWideningPass(*PassRegistry::getPassRegistry());
77 bool runOnMachineFunction(MachineFunction
&MF
) override
;
79 StringRef
getPassName() const override
{ return "Hexagon Store Widening"; }
81 void getAnalysisUsage(AnalysisUsage
&AU
) const override
{
82 AU
.addRequired
<AAResultsWrapperPass
>();
83 AU
.addPreserved
<AAResultsWrapperPass
>();
84 MachineFunctionPass::getAnalysisUsage(AU
);
87 static bool handledStoreType(const MachineInstr
*MI
);
90 static const int MaxWideSize
= 4;
92 using InstrGroup
= std::vector
<MachineInstr
*>;
93 using InstrGroupList
= std::vector
<InstrGroup
>;
95 bool instrAliased(InstrGroup
&Stores
, const MachineMemOperand
&MMO
);
96 bool instrAliased(InstrGroup
&Stores
, const MachineInstr
*MI
);
97 void createStoreGroup(MachineInstr
*BaseStore
, InstrGroup::iterator Begin
,
98 InstrGroup::iterator End
, InstrGroup
&Group
);
99 void createStoreGroups(MachineBasicBlock
&MBB
,
100 InstrGroupList
&StoreGroups
);
101 bool processBasicBlock(MachineBasicBlock
&MBB
);
102 bool processStoreGroup(InstrGroup
&Group
);
103 bool selectStores(InstrGroup::iterator Begin
, InstrGroup::iterator End
,
104 InstrGroup
&OG
, unsigned &TotalSize
, unsigned MaxSize
);
105 bool createWideStores(InstrGroup
&OG
, InstrGroup
&NG
, unsigned TotalSize
);
106 bool replaceStores(InstrGroup
&OG
, InstrGroup
&NG
);
107 bool storesAreAdjacent(const MachineInstr
*S1
, const MachineInstr
*S2
);
110 } // end anonymous namespace
112 char HexagonStoreWidening::ID
= 0;
114 INITIALIZE_PASS_BEGIN(HexagonStoreWidening
, "hexagon-widen-stores",
115 "Hexason Store Widening", false, false)
116 INITIALIZE_PASS_DEPENDENCY(AAResultsWrapperPass
)
117 INITIALIZE_PASS_END(HexagonStoreWidening
, "hexagon-widen-stores",
118 "Hexagon Store Widening", false, false)
120 // Some local helper functions...
121 static unsigned getBaseAddressRegister(const MachineInstr
*MI
) {
122 const MachineOperand
&MO
= MI
->getOperand(0);
123 assert(MO
.isReg() && "Expecting register operand");
127 static int64_t getStoreOffset(const MachineInstr
*MI
) {
128 unsigned OpC
= MI
->getOpcode();
129 assert(HexagonStoreWidening::handledStoreType(MI
) && "Unhandled opcode");
132 case Hexagon::S4_storeirb_io
:
133 case Hexagon::S4_storeirh_io
:
134 case Hexagon::S4_storeiri_io
: {
135 const MachineOperand
&MO
= MI
->getOperand(1);
136 assert(MO
.isImm() && "Expecting immediate offset");
141 llvm_unreachable("Store offset calculation missing for a handled opcode");
145 static const MachineMemOperand
&getStoreTarget(const MachineInstr
*MI
) {
146 assert(!MI
->memoperands_empty() && "Expecting memory operands");
147 return **MI
->memoperands_begin();
150 // Filtering function: any stores whose opcodes are not "approved" of by
151 // this function will not be subjected to widening.
152 inline bool HexagonStoreWidening::handledStoreType(const MachineInstr
*MI
) {
153 // For now, only handle stores of immediate values.
154 // Also, reject stores to stack slots.
155 unsigned Opc
= MI
->getOpcode();
157 case Hexagon::S4_storeirb_io
:
158 case Hexagon::S4_storeirh_io
:
159 case Hexagon::S4_storeiri_io
:
160 // Base address must be a register. (Implement FI later.)
161 return MI
->getOperand(0).isReg();
167 // Check if the machine memory operand MMO is aliased with any of the
168 // stores in the store group Stores.
169 bool HexagonStoreWidening::instrAliased(InstrGroup
&Stores
,
170 const MachineMemOperand
&MMO
) {
174 MemoryLocation
L(MMO
.getValue(), MMO
.getSize(), MMO
.getAAInfo());
176 for (auto SI
: Stores
) {
177 const MachineMemOperand
&SMO
= getStoreTarget(SI
);
181 MemoryLocation
SL(SMO
.getValue(), SMO
.getSize(), SMO
.getAAInfo());
182 if (AA
->alias(L
, SL
))
189 // Check if the machine instruction MI accesses any storage aliased with
190 // any store in the group Stores.
191 bool HexagonStoreWidening::instrAliased(InstrGroup
&Stores
,
192 const MachineInstr
*MI
) {
193 for (auto &I
: MI
->memoperands())
194 if (instrAliased(Stores
, *I
))
199 // Inspect a machine basic block, and generate store groups out of stores
200 // encountered in the block.
202 // A store group is a group of stores that use the same base register,
203 // and which can be reordered within that group without altering the
204 // semantics of the program. A single store group could be widened as
205 // a whole, if there existed a single store instruction with the same
206 // semantics as the entire group. In many cases, a single store group
207 // may need more than one wide store.
208 void HexagonStoreWidening::createStoreGroups(MachineBasicBlock
&MBB
,
209 InstrGroupList
&StoreGroups
) {
212 // Copy all instruction pointers from the basic block to a temporary
213 // list. This will allow operating on the list, and modifying its
214 // elements without affecting the basic block.
216 AllInsns
.push_back(&I
);
218 // Traverse all instructions in the AllInsns list, and if we encounter
219 // a store, then try to create a store group starting at that instruction
220 // i.e. a sequence of independent stores that can be widened.
221 for (auto I
= AllInsns
.begin(), E
= AllInsns
.end(); I
!= E
; ++I
) {
222 MachineInstr
*MI
= *I
;
223 // Skip null pointers (processed instructions).
224 if (!MI
|| !handledStoreType(MI
))
227 // Found a store. Try to create a store group.
229 createStoreGroup(MI
, I
+1, E
, G
);
231 StoreGroups
.push_back(G
);
235 // Create a single store group. The stores need to be independent between
236 // themselves, and also there cannot be other instructions between them
237 // that could read or modify storage being stored into.
238 void HexagonStoreWidening::createStoreGroup(MachineInstr
*BaseStore
,
239 InstrGroup::iterator Begin
, InstrGroup::iterator End
, InstrGroup
&Group
) {
240 assert(handledStoreType(BaseStore
) && "Unexpected instruction");
241 unsigned BaseReg
= getBaseAddressRegister(BaseStore
);
244 Group
.push_back(BaseStore
);
246 for (auto I
= Begin
; I
!= End
; ++I
) {
247 MachineInstr
*MI
= *I
;
251 if (handledStoreType(MI
)) {
252 // If this store instruction is aliased with anything already in the
253 // group, terminate the group now.
254 if (instrAliased(Group
, getStoreTarget(MI
)))
256 // If this store is aliased to any of the memory instructions we have
257 // seen so far (that are not a part of this group), terminate the group.
258 if (instrAliased(Other
, getStoreTarget(MI
)))
261 unsigned BR
= getBaseAddressRegister(MI
);
269 // Assume calls are aliased to everything.
270 if (MI
->isCall() || MI
->hasUnmodeledSideEffects())
273 if (MI
->mayLoad() || MI
->mayStore()) {
274 if (MI
->hasOrderedMemoryRef() || instrAliased(Group
, MI
))
281 // Check if store instructions S1 and S2 are adjacent. More precisely,
282 // S2 has to access memory immediately following that accessed by S1.
283 bool HexagonStoreWidening::storesAreAdjacent(const MachineInstr
*S1
,
284 const MachineInstr
*S2
) {
285 if (!handledStoreType(S1
) || !handledStoreType(S2
))
288 const MachineMemOperand
&S1MO
= getStoreTarget(S1
);
290 // Currently only handling immediate stores.
291 int Off1
= S1
->getOperand(1).getImm();
292 int Off2
= S2
->getOperand(1).getImm();
294 return (Off1
>= 0) ? Off1
+S1MO
.getSize() == unsigned(Off2
)
295 : int(Off1
+S1MO
.getSize()) == Off2
;
298 /// Given a sequence of adjacent stores, and a maximum size of a single wide
299 /// store, pick a group of stores that can be replaced by a single store
300 /// of size not exceeding MaxSize. The selected sequence will be recorded
301 /// in OG ("old group" of instructions).
302 /// OG should be empty on entry, and should be left empty if the function
304 bool HexagonStoreWidening::selectStores(InstrGroup::iterator Begin
,
305 InstrGroup::iterator End
, InstrGroup
&OG
, unsigned &TotalSize
,
307 assert(Begin
!= End
&& "No instructions to analyze");
308 assert(OG
.empty() && "Old group not empty on entry");
310 if (std::distance(Begin
, End
) <= 1)
313 MachineInstr
*FirstMI
= *Begin
;
314 assert(!FirstMI
->memoperands_empty() && "Expecting some memory operands");
315 const MachineMemOperand
&FirstMMO
= getStoreTarget(FirstMI
);
316 unsigned Alignment
= FirstMMO
.getAlignment();
317 unsigned SizeAccum
= FirstMMO
.getSize();
318 unsigned FirstOffset
= getStoreOffset(FirstMI
);
320 // The initial value of SizeAccum should always be a power of 2.
321 assert(isPowerOf2_32(SizeAccum
) && "First store size not a power of 2");
323 // If the size of the first store equals to or exceeds the limit, do nothing.
324 if (SizeAccum
>= MaxSize
)
327 // If the size of the first store is greater than or equal to the address
328 // stored to, then the store cannot be made any wider.
329 if (SizeAccum
>= Alignment
)
332 // The offset of a store will put restrictions on how wide the store can be.
333 // Offsets in stores of size 2^n bytes need to have the n lowest bits be 0.
334 // If the first store already exhausts the offset limits, quit. Test this
335 // by checking if the next wider size would exceed the limit.
336 if ((2*SizeAccum
-1) & FirstOffset
)
339 OG
.push_back(FirstMI
);
340 MachineInstr
*S1
= FirstMI
, *S2
= *(Begin
+1);
341 InstrGroup::iterator I
= Begin
+1;
343 // Pow2Num will be the largest number of elements in OG such that the sum
344 // of sizes of stores 0...Pow2Num-1 will be a power of 2.
345 unsigned Pow2Num
= 1;
346 unsigned Pow2Size
= SizeAccum
;
348 // Be greedy: keep accumulating stores as long as they are to adjacent
349 // memory locations, and as long as the total number of bytes stored
350 // does not exceed the limit (MaxSize).
351 // Keep track of when the total size covered is a power of 2, since
352 // this is a size a single store can cover.
355 // Stores are sorted, so if S1 and S2 are not adjacent, there won't be
356 // any other store to fill the "hole".
357 if (!storesAreAdjacent(S1
, S2
))
360 unsigned S2Size
= getStoreTarget(S2
).getSize();
361 if (SizeAccum
+ S2Size
> std::min(MaxSize
, Alignment
))
366 if (isPowerOf2_32(SizeAccum
)) {
368 Pow2Size
= SizeAccum
;
370 if ((2*Pow2Size
-1) & FirstOffset
)
377 // The stores don't add up to anything that can be widened. Clean up.
383 // Only leave the stored being widened.
385 TotalSize
= Pow2Size
;
389 /// Given an "old group" OG of stores, create a "new group" NG of instructions
390 /// to replace them. Ideally, NG would only have a single instruction in it,
391 /// but that may only be possible for store-immediate.
392 bool HexagonStoreWidening::createWideStores(InstrGroup
&OG
, InstrGroup
&NG
,
393 unsigned TotalSize
) {
394 // XXX Current limitations:
395 // - only expect stores of immediate values in OG,
396 // - only handle a TotalSize of up to 4.
401 unsigned Acc
= 0; // Value accumulator.
404 for (InstrGroup::iterator I
= OG
.begin(), E
= OG
.end(); I
!= E
; ++I
) {
405 MachineInstr
*MI
= *I
;
406 const MachineMemOperand
&MMO
= getStoreTarget(MI
);
407 MachineOperand
&SO
= MI
->getOperand(2); // Source.
408 assert(SO
.isImm() && "Expecting an immediate operand");
410 unsigned NBits
= MMO
.getSize()*8;
411 unsigned Mask
= (0xFFFFFFFFU
>> (32-NBits
));
412 unsigned Val
= (SO
.getImm() & Mask
) << Shift
;
417 MachineInstr
*FirstSt
= OG
.front();
418 DebugLoc DL
= OG
.back()->getDebugLoc();
419 const MachineMemOperand
&OldM
= getStoreTarget(FirstSt
);
420 MachineMemOperand
*NewM
=
421 MF
->getMachineMemOperand(OldM
.getPointerInfo(), OldM
.getFlags(),
422 TotalSize
, OldM
.getAlignment(),
426 // Create mem[hw] = #Acc
427 unsigned WOpc
= (TotalSize
== 2) ? Hexagon::S4_storeirh_io
:
428 (TotalSize
== 4) ? Hexagon::S4_storeiri_io
: 0;
429 assert(WOpc
&& "Unexpected size");
431 int Val
= (TotalSize
== 2) ? int16_t(Acc
) : int(Acc
);
432 const MCInstrDesc
&StD
= TII
->get(WOpc
);
433 MachineOperand
&MR
= FirstSt
->getOperand(0);
434 int64_t Off
= FirstSt
->getOperand(1).getImm();
436 BuildMI(*MF
, DL
, StD
)
437 .addReg(MR
.getReg(), getKillRegState(MR
.isKill()), MR
.getSubReg())
440 StI
->addMemOperand(*MF
, NewM
);
443 // Create vreg = A2_tfrsi #Acc; mem[hw] = vreg
444 const MCInstrDesc
&TfrD
= TII
->get(Hexagon::A2_tfrsi
);
445 const TargetRegisterClass
*RC
= TII
->getRegClass(TfrD
, 0, TRI
, *MF
);
446 unsigned VReg
= MF
->getRegInfo().createVirtualRegister(RC
);
447 MachineInstr
*TfrI
= BuildMI(*MF
, DL
, TfrD
, VReg
)
451 unsigned WOpc
= (TotalSize
== 2) ? Hexagon::S2_storerh_io
:
452 (TotalSize
== 4) ? Hexagon::S2_storeri_io
: 0;
453 assert(WOpc
&& "Unexpected size");
455 const MCInstrDesc
&StD
= TII
->get(WOpc
);
456 MachineOperand
&MR
= FirstSt
->getOperand(0);
457 int64_t Off
= FirstSt
->getOperand(1).getImm();
459 BuildMI(*MF
, DL
, StD
)
460 .addReg(MR
.getReg(), getKillRegState(MR
.isKill()), MR
.getSubReg())
462 .addReg(VReg
, RegState::Kill
);
463 StI
->addMemOperand(*MF
, NewM
);
470 // Replace instructions from the old group OG with instructions from the
471 // new group NG. Conceptually, remove all instructions in OG, and then
472 // insert all instructions in NG, starting at where the first instruction
473 // from OG was (in the order in which they appeared in the basic block).
474 // (The ordering in OG does not have to match the order in the basic block.)
475 bool HexagonStoreWidening::replaceStores(InstrGroup
&OG
, InstrGroup
&NG
) {
477 dbgs() << "Replacing:\n";
485 MachineBasicBlock
*MBB
= OG
.back()->getParent();
486 MachineBasicBlock::iterator InsertAt
= MBB
->end();
488 // Need to establish the insertion point. The best one is right before
489 // the first store in the OG, but in the order in which the stores occur
490 // in the program list. Since the ordering in OG does not correspond
491 // to the order in the program list, we need to do some work to find
492 // the insertion point.
494 // Create a set of all instructions in OG (for quick lookup).
495 SmallPtrSet
<MachineInstr
*, 4> InstrSet
;
499 // Traverse the block, until we hit an instruction from OG.
500 for (auto &I
: *MBB
) {
501 if (InstrSet
.count(&I
)) {
507 assert((InsertAt
!= MBB
->end()) && "Cannot locate any store from the group");
509 bool AtBBStart
= false;
511 // InsertAt points at the first instruction that will be removed. We need
512 // to move it out of the way, so it remains valid after removing all the
513 // old stores, and so we are able to recover it back to the proper insertion
515 if (InsertAt
!= MBB
->begin())
521 I
->eraseFromParent();
526 InsertAt
= MBB
->begin();
529 MBB
->insert(InsertAt
, I
);
534 // Break up the group into smaller groups, each of which can be replaced by
535 // a single wide store. Widen each such smaller group and replace the old
536 // instructions with the widened ones.
537 bool HexagonStoreWidening::processStoreGroup(InstrGroup
&Group
) {
538 bool Changed
= false;
539 InstrGroup::iterator I
= Group
.begin(), E
= Group
.end();
540 InstrGroup OG
, NG
; // Old and new groups.
541 unsigned CollectedSize
;
547 bool Succ
= selectStores(I
++, E
, OG
, CollectedSize
, MaxWideSize
) &&
548 createWideStores(OG
, NG
, CollectedSize
) &&
549 replaceStores(OG
, NG
);
553 assert(OG
.size() > 1 && "Created invalid group");
554 assert(distance(I
, E
)+1 >= int(OG
.size()) && "Too many elements");
563 // Process a single basic block: create the store groups, and replace them
564 // with the widened stores, if possible. Processing of each basic block
565 // is independent from processing of any other basic block. This transfor-
566 // mation could be stopped after having processed any basic block without
567 // any ill effects (other than not having performed widening in the unpro-
568 // cessed blocks). Also, the basic blocks can be processed in any order.
569 bool HexagonStoreWidening::processBasicBlock(MachineBasicBlock
&MBB
) {
571 bool Changed
= false;
573 createStoreGroups(MBB
, SGs
);
575 auto Less
= [] (const MachineInstr
*A
, const MachineInstr
*B
) -> bool {
576 return getStoreOffset(A
) < getStoreOffset(B
);
578 for (auto &G
: SGs
) {
579 assert(G
.size() > 1 && "Store group with fewer than 2 elements");
582 Changed
|= processStoreGroup(G
);
588 bool HexagonStoreWidening::runOnMachineFunction(MachineFunction
&MFn
) {
589 if (skipFunction(MFn
.getFunction()))
593 auto &ST
= MFn
.getSubtarget
<HexagonSubtarget
>();
594 TII
= ST
.getInstrInfo();
595 TRI
= ST
.getRegisterInfo();
596 MRI
= &MFn
.getRegInfo();
597 AA
= &getAnalysis
<AAResultsWrapperPass
>().getAAResults();
599 bool Changed
= false;
602 Changed
|= processBasicBlock(B
);
607 FunctionPass
*llvm::createHexagonStoreWidening() {
608 return new HexagonStoreWidening();