|
24 | 24 | #include "llvm/CodeGen/MachineInstrBuilder.h"
|
25 | 25 | #include "llvm/CodeGen/MachineRegisterInfo.h"
|
26 | 26 | #include "llvm/CodeGen/SelectionDAGISel.h"
|
| 27 | +#include "llvm/IR/ConstantRange.h" |
27 | 28 | #include "llvm/IR/Function.h"
|
28 | 29 | #include "llvm/IR/Instructions.h"
|
29 | 30 | #include "llvm/IR/Intrinsics.h"
|
@@ -1571,7 +1572,15 @@ bool X86DAGToDAGISel::selectMOV64Imm32(SDValue N, SDValue &Imm) {
|
1571 | 1572 | return false;
|
1572 | 1573 |
|
1573 | 1574 | Imm = N;
|
1574 |
| - return TM.getCodeModel() == CodeModel::Small; |
| 1575 | + if (N->getOpcode() != ISD::TargetGlobalAddress) |
| 1576 | + return TM.getCodeModel() == CodeModel::Small; |
| 1577 | + |
| 1578 | + Optional<ConstantRange> CR = |
| 1579 | + cast<GlobalAddressSDNode>(N)->getGlobal()->getAbsoluteSymbolRange(); |
| 1580 | + if (!CR) |
| 1581 | + return TM.getCodeModel() == CodeModel::Small; |
| 1582 | + |
| 1583 | + return CR->getUnsignedMax().ult(1ull << 32); |
1575 | 1584 | }
|
1576 | 1585 |
|
1577 | 1586 | bool X86DAGToDAGISel::selectLEA64_32Addr(SDValue N, SDValue &Base,
|
@@ -1710,10 +1719,39 @@ bool X86DAGToDAGISel::selectRelocImm(SDValue N, SDValue &Op) {
|
1710 | 1719 | return true;
|
1711 | 1720 | }
|
1712 | 1721 |
|
| 1722 | + // Keep track of the original value type and whether this value was |
| 1723 | + // truncated. If we see a truncation from pointer type to VT that truncates |
| 1724 | + // bits that are known to be zero, we can use a narrow reference. |
| 1725 | + EVT VT = N.getValueType(); |
| 1726 | + bool WasTruncated = false; |
| 1727 | + if (N.getOpcode() == ISD::TRUNCATE) { |
| 1728 | + WasTruncated = true; |
| 1729 | + N = N.getOperand(0); |
| 1730 | + } |
| 1731 | + |
1713 | 1732 | if (N.getOpcode() != X86ISD::Wrapper)
|
1714 | 1733 | return false;
|
1715 | 1734 |
|
1716 |
| - Op = N.getOperand(0); |
| 1735 | + // We can only use non-GlobalValues as immediates if they were not truncated, |
| 1736 | + // as we do not have any range information. If we have a GlobalValue and the |
| 1737 | + // address was not truncated, we can select it as an operand directly. |
| 1738 | + unsigned Opc = N.getOperand(0)->getOpcode(); |
| 1739 | + if (Opc != ISD::TargetGlobalAddress || !WasTruncated) { |
| 1740 | + Op = N.getOperand(0); |
| 1741 | + // We can only select the operand directly if we didn't have to look past a |
| 1742 | + // truncate. |
| 1743 | + return !WasTruncated; |
| 1744 | + } |
| 1745 | + |
| 1746 | + // Check that the global's range fits into VT. |
| 1747 | + auto *GA = cast<GlobalAddressSDNode>(N.getOperand(0)); |
| 1748 | + Optional<ConstantRange> CR = GA->getGlobal()->getAbsoluteSymbolRange(); |
| 1749 | + if (!CR || CR->getUnsignedMax().uge(1ull << VT.getSizeInBits())) |
| 1750 | + return false; |
| 1751 | + |
| 1752 | + // Okay, we can use a narrow reference. |
| 1753 | + Op = CurDAG->getTargetGlobalAddress(GA->getGlobal(), SDLoc(N), VT, |
| 1754 | + GA->getOffset(), GA->getTargetFlags()); |
1717 | 1755 | return true;
|
1718 | 1756 | }
|
1719 | 1757 |
|
|
0 commit comments