xref: /freebsd/contrib/llvm-project/llvm/lib/Target/AArch64/AArch64CompressJumpTables.cpp (revision 9c77fb6aaa366cbabc80ee1b834bcfe4df135491)
1 //==-- AArch64CompressJumpTables.cpp - Compress jump tables for AArch64 --====//
2 //
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
6 //
7 // This pass looks at the basic blocks each jump-table refers to and works out
8 // whether they can be emitted in a compressed form (with 8 or 16-bit
9 // entries). If so, it changes the opcode and flags them in the associated
10 // AArch64FunctionInfo.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "AArch64.h"
15 #include "AArch64MachineFunctionInfo.h"
16 #include "AArch64Subtarget.h"
17 #include "llvm/ADT/Statistic.h"
18 #include "llvm/CodeGen/MachineFunctionPass.h"
19 #include "llvm/CodeGen/MachineJumpTableInfo.h"
20 #include "llvm/CodeGen/TargetInstrInfo.h"
21 #include "llvm/CodeGen/TargetSubtargetInfo.h"
22 #include "llvm/Support/Alignment.h"
23 
24 using namespace llvm;
25 
26 #define DEBUG_TYPE "aarch64-jump-tables"
27 
28 STATISTIC(NumJT8, "Number of jump-tables with 1-byte entries");
29 STATISTIC(NumJT16, "Number of jump-tables with 2-byte entries");
30 STATISTIC(NumJT32, "Number of jump-tables with 4-byte entries");
31 
32 namespace {
33 class AArch64CompressJumpTables : public MachineFunctionPass {
34   const TargetInstrInfo *TII;
35   MachineFunction *MF;
36   SmallVector<int, 8> BlockInfo;
37 
38   /// Returns the size of instructions in the block \p MBB, or std::nullopt if
39   /// we couldn't get a safe upper bound.
40   std::optional<int> computeBlockSize(MachineBasicBlock &MBB);
41 
42   /// Gather information about the function, returns false if we can't perform
43   /// this optimization for some reason.
44   bool scanFunction();
45 
46   bool compressJumpTable(MachineInstr &MI, int Offset);
47 
48 public:
49   static char ID;
50   AArch64CompressJumpTables() : MachineFunctionPass(ID) {}
51 
52   bool runOnMachineFunction(MachineFunction &MF) override;
53 
54   MachineFunctionProperties getRequiredProperties() const override {
55     return MachineFunctionProperties().setNoVRegs();
56   }
57   StringRef getPassName() const override {
58     return "AArch64 Compress Jump Tables";
59   }
60 };
61 char AArch64CompressJumpTables::ID = 0;
62 } // namespace
63 
64 INITIALIZE_PASS(AArch64CompressJumpTables, DEBUG_TYPE,
65                 "AArch64 compress jump tables pass", false, false)
66 
67 std::optional<int>
68 AArch64CompressJumpTables::computeBlockSize(MachineBasicBlock &MBB) {
69   int Size = 0;
70   for (const MachineInstr &MI : MBB) {
71     // Inline asm may contain some directives like .bytes which we don't
72     // currently have the ability to parse accurately. To be safe, just avoid
73     // computing a size and bail out.
74     if (MI.getOpcode() == AArch64::INLINEASM ||
75         MI.getOpcode() == AArch64::INLINEASM_BR)
76       return std::nullopt;
77     Size += TII->getInstSizeInBytes(MI);
78   }
79   return Size;
80 }
81 
82 bool AArch64CompressJumpTables::scanFunction() {
83   BlockInfo.clear();
84   BlockInfo.resize(MF->getNumBlockIDs());
85 
86   // NOTE: BlockSize, Offset, OffsetAfterAlignment are all upper bounds.
87 
88   unsigned Offset = 0;
89   for (MachineBasicBlock &MBB : *MF) {
90     const Align Alignment = MBB.getAlignment();
91     unsigned OffsetAfterAlignment = Offset;
92     // We don't know the exact size of MBB so assume worse case padding.
93     if (Alignment > Align(4))
94       OffsetAfterAlignment += Alignment.value() - 4;
95     BlockInfo[MBB.getNumber()] = OffsetAfterAlignment;
96     auto BlockSize = computeBlockSize(MBB);
97     if (!BlockSize)
98       return false;
99     Offset = OffsetAfterAlignment + *BlockSize;
100   }
101   return true;
102 }
103 
104 bool AArch64CompressJumpTables::compressJumpTable(MachineInstr &MI,
105                                                   int Offset) {
106   if (MI.getOpcode() != AArch64::JumpTableDest32)
107     return false;
108 
109   int JTIdx = MI.getOperand(4).getIndex();
110   auto &JTInfo = *MF->getJumpTableInfo();
111   const MachineJumpTableEntry &JT = JTInfo.getJumpTables()[JTIdx];
112 
113   // The jump-table might have been optimized away.
114   if (JT.MBBs.empty())
115     return false;
116 
117   int MaxOffset = std::numeric_limits<int>::min(),
118       MinOffset = std::numeric_limits<int>::max();
119   MachineBasicBlock *MinBlock = nullptr;
120   for (auto *Block : JT.MBBs) {
121     int BlockOffset = BlockInfo[Block->getNumber()];
122     assert(BlockOffset % 4 == 0 && "misaligned basic block");
123 
124     MaxOffset = std::max(MaxOffset, BlockOffset);
125     if (BlockOffset <= MinOffset) {
126       MinOffset = BlockOffset;
127       MinBlock = Block;
128     }
129   }
130   assert(MinBlock && "Failed to find minimum offset block");
131 
132   // The ADR instruction needed to calculate the address of the first reachable
133   // basic block can address +/-1MB.
134   if (!isInt<21>(MinOffset - Offset)) {
135     ++NumJT32;
136     return false;
137   }
138 
139   int Span = MaxOffset - MinOffset;
140   auto *AFI = MF->getInfo<AArch64FunctionInfo>();
141   if (isUInt<8>(Span / 4)) {
142     AFI->setJumpTableEntryInfo(JTIdx, 1, MinBlock->getSymbol());
143     MI.setDesc(TII->get(AArch64::JumpTableDest8));
144     ++NumJT8;
145     return true;
146   }
147   if (isUInt<16>(Span / 4)) {
148     AFI->setJumpTableEntryInfo(JTIdx, 2, MinBlock->getSymbol());
149     MI.setDesc(TII->get(AArch64::JumpTableDest16));
150     ++NumJT16;
151     return true;
152   }
153 
154   ++NumJT32;
155   return false;
156 }
157 
158 bool AArch64CompressJumpTables::runOnMachineFunction(MachineFunction &MFIn) {
159   bool Changed = false;
160   MF = &MFIn;
161 
162   const auto &ST = MF->getSubtarget<AArch64Subtarget>();
163   TII = ST.getInstrInfo();
164 
165   if (ST.force32BitJumpTables() && !MF->getFunction().hasMinSize())
166     return false;
167 
168   if (!scanFunction())
169     return false;
170 
171   for (MachineBasicBlock &MBB : *MF) {
172     int Offset = BlockInfo[MBB.getNumber()];
173     for (MachineInstr &MI : MBB) {
174       Changed |= compressJumpTable(MI, Offset);
175       Offset += TII->getInstSizeInBytes(MI);
176     }
177   }
178 
179   return Changed;
180 }
181 
182 FunctionPass *llvm::createAArch64CompressJumpTablesPass() {
183   return new AArch64CompressJumpTables();
184 }
185