1 | //===- AMDGPUGlobalISelUtils.cpp ---------------------------------*- C++ -*-==// |
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 | //===----------------------------------------------------------------------===// |
8 | |
9 | #include "AMDGPUGlobalISelUtils.h" |
10 | #include "GCNSubtarget.h" |
11 | #include "llvm/CodeGen/GlobalISel/GISelKnownBits.h" |
12 | #include "llvm/CodeGen/GlobalISel/MIPatternMatch.h" |
13 | #include "llvm/CodeGenTypes/LowLevelType.h" |
14 | #include "llvm/IR/Constants.h" |
15 | |
16 | using namespace llvm; |
17 | using namespace MIPatternMatch; |
18 | |
19 | std::pair<Register, unsigned> |
20 | AMDGPU::getBaseWithConstantOffset(MachineRegisterInfo &MRI, Register Reg, |
21 | GISelKnownBits *KnownBits, bool CheckNUW) { |
22 | MachineInstr *Def = getDefIgnoringCopies(Reg, MRI); |
23 | if (Def->getOpcode() == TargetOpcode::G_CONSTANT) { |
24 | unsigned Offset; |
25 | const MachineOperand &Op = Def->getOperand(i: 1); |
26 | if (Op.isImm()) |
27 | Offset = Op.getImm(); |
28 | else |
29 | Offset = Op.getCImm()->getZExtValue(); |
30 | |
31 | return std::pair(Register(), Offset); |
32 | } |
33 | |
34 | int64_t Offset; |
35 | if (Def->getOpcode() == TargetOpcode::G_ADD) { |
36 | // A 32-bit (address + offset) should not cause unsigned 32-bit integer |
37 | // wraparound, because s_load instructions perform the addition in 64 bits. |
38 | if (CheckNUW && !Def->getFlag(Flag: MachineInstr::NoUWrap)) { |
39 | assert(MRI.getType(Reg).getScalarSizeInBits() == 32); |
40 | return std::pair(Reg, 0); |
41 | } |
42 | // TODO: Handle G_OR used for add case |
43 | if (mi_match(R: Def->getOperand(i: 2).getReg(), MRI, P: m_ICst(Cst&: Offset))) |
44 | return std::pair(Def->getOperand(i: 1).getReg(), Offset); |
45 | |
46 | // FIXME: matcher should ignore copies |
47 | if (mi_match(R: Def->getOperand(i: 2).getReg(), MRI, P: m_Copy(Src: m_ICst(Cst&: Offset)))) |
48 | return std::pair(Def->getOperand(i: 1).getReg(), Offset); |
49 | } |
50 | |
51 | Register Base; |
52 | if (KnownBits && mi_match(R: Reg, MRI, P: m_GOr(L: m_Reg(R&: Base), R: m_ICst(Cst&: Offset))) && |
53 | KnownBits->maskedValueIsZero(Val: Base, Mask: APInt(32, Offset))) |
54 | return std::pair(Base, Offset); |
55 | |
56 | // Handle G_PTRTOINT (G_PTR_ADD base, const) case |
57 | if (Def->getOpcode() == TargetOpcode::G_PTRTOINT) { |
58 | MachineInstr *Base; |
59 | if (mi_match(R: Def->getOperand(i: 1).getReg(), MRI, |
60 | P: m_GPtrAdd(L: m_MInstr(MI&: Base), R: m_ICst(Cst&: Offset)))) { |
61 | // If Base was int converted to pointer, simply return int and offset. |
62 | if (Base->getOpcode() == TargetOpcode::G_INTTOPTR) |
63 | return std::pair(Base->getOperand(i: 1).getReg(), Offset); |
64 | |
65 | // Register returned here will be of pointer type. |
66 | return std::pair(Base->getOperand(i: 0).getReg(), Offset); |
67 | } |
68 | } |
69 | |
70 | return std::pair(Reg, 0); |
71 | } |
72 | |