xref: /openbsd-src/gnu/llvm/clang/lib/CodeGen/Address.h (revision 12c855180aad702bbcca06e0398d774beeafb155)
1e5dd7070Spatrick //===-- Address.h - An aligned address -------------------------*- C++ -*-===//
2e5dd7070Spatrick //
3e5dd7070Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e5dd7070Spatrick // See https://llvm.org/LICENSE.txt for license information.
5e5dd7070Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e5dd7070Spatrick //
7e5dd7070Spatrick //===----------------------------------------------------------------------===//
8e5dd7070Spatrick //
9e5dd7070Spatrick // This class provides a simple wrapper for a pair of a pointer and an
10e5dd7070Spatrick // alignment.
11e5dd7070Spatrick //
12e5dd7070Spatrick //===----------------------------------------------------------------------===//
13e5dd7070Spatrick 
14e5dd7070Spatrick #ifndef LLVM_CLANG_LIB_CODEGEN_ADDRESS_H
15e5dd7070Spatrick #define LLVM_CLANG_LIB_CODEGEN_ADDRESS_H
16e5dd7070Spatrick 
17e5dd7070Spatrick #include "clang/AST/CharUnits.h"
18*12c85518Srobert #include "llvm/ADT/PointerIntPair.h"
19*12c85518Srobert #include "llvm/IR/Constants.h"
20*12c85518Srobert #include "llvm/Support/MathExtras.h"
21e5dd7070Spatrick 
22e5dd7070Spatrick namespace clang {
23e5dd7070Spatrick namespace CodeGen {
24e5dd7070Spatrick 
25*12c85518Srobert // We try to save some space by using 6 bits over two PointerIntPairs to store
26*12c85518Srobert // the alignment. However, some arches don't support 3 bits in a PointerIntPair
27*12c85518Srobert // so we fallback to storing the alignment separately.
28*12c85518Srobert template <typename T, bool = alignof(llvm::Value *) >= 8> class AddressImpl {};
29*12c85518Srobert 
30*12c85518Srobert template <typename T> class AddressImpl<T, false> {
31*12c85518Srobert   llvm::Value *Pointer;
32*12c85518Srobert   llvm::Type *ElementType;
33*12c85518Srobert   CharUnits Alignment;
34*12c85518Srobert 
35*12c85518Srobert public:
AddressImpl(llvm::Value * Pointer,llvm::Type * ElementType,CharUnits Alignment)36*12c85518Srobert   AddressImpl(llvm::Value *Pointer, llvm::Type *ElementType,
37*12c85518Srobert               CharUnits Alignment)
38*12c85518Srobert       : Pointer(Pointer), ElementType(ElementType), Alignment(Alignment) {}
getPointer()39*12c85518Srobert   llvm::Value *getPointer() const { return Pointer; }
getElementType()40*12c85518Srobert   llvm::Type *getElementType() const { return ElementType; }
getAlignment()41*12c85518Srobert   CharUnits getAlignment() const { return Alignment; }
42*12c85518Srobert };
43*12c85518Srobert 
44*12c85518Srobert template <typename T> class AddressImpl<T, true> {
45*12c85518Srobert   // Int portion stores upper 3 bits of the log of the alignment.
46*12c85518Srobert   llvm::PointerIntPair<llvm::Value *, 3, unsigned> Pointer;
47*12c85518Srobert   // Int portion stores lower 3 bits of the log of the alignment.
48*12c85518Srobert   llvm::PointerIntPair<llvm::Type *, 3, unsigned> ElementType;
49*12c85518Srobert 
50*12c85518Srobert public:
AddressImpl(llvm::Value * Pointer,llvm::Type * ElementType,CharUnits Alignment)51*12c85518Srobert   AddressImpl(llvm::Value *Pointer, llvm::Type *ElementType,
52*12c85518Srobert               CharUnits Alignment)
53*12c85518Srobert       : Pointer(Pointer), ElementType(ElementType) {
54*12c85518Srobert     if (Alignment.isZero())
55*12c85518Srobert       return;
56*12c85518Srobert     // Currently the max supported alignment is much less than 1 << 63 and is
57*12c85518Srobert     // guaranteed to be a power of 2, so we can store the log of the alignment
58*12c85518Srobert     // into 6 bits.
59*12c85518Srobert     assert(Alignment.isPowerOfTwo() && "Alignment cannot be zero");
60*12c85518Srobert     auto AlignLog = llvm::Log2_64(Alignment.getQuantity());
61*12c85518Srobert     assert(AlignLog < (1 << 6) && "cannot fit alignment into 6 bits");
62*12c85518Srobert     this->Pointer.setInt(AlignLog >> 3);
63*12c85518Srobert     this->ElementType.setInt(AlignLog & 7);
64*12c85518Srobert   }
getPointer()65*12c85518Srobert   llvm::Value *getPointer() const { return Pointer.getPointer(); }
getElementType()66*12c85518Srobert   llvm::Type *getElementType() const { return ElementType.getPointer(); }
getAlignment()67*12c85518Srobert   CharUnits getAlignment() const {
68*12c85518Srobert     unsigned AlignLog = (Pointer.getInt() << 3) | ElementType.getInt();
69*12c85518Srobert     return CharUnits::fromQuantity(CharUnits::QuantityType(1) << AlignLog);
70*12c85518Srobert   }
71*12c85518Srobert };
72*12c85518Srobert 
73e5dd7070Spatrick /// An aligned address.
74e5dd7070Spatrick class Address {
75*12c85518Srobert   AddressImpl<void> A;
76*12c85518Srobert 
77*12c85518Srobert protected:
Address(std::nullptr_t)78*12c85518Srobert   Address(std::nullptr_t) : A(nullptr, nullptr, CharUnits::Zero()) {}
79*12c85518Srobert 
80e5dd7070Spatrick public:
Address(llvm::Value * Pointer,llvm::Type * ElementType,CharUnits Alignment)81*12c85518Srobert   Address(llvm::Value *Pointer, llvm::Type *ElementType, CharUnits Alignment)
82*12c85518Srobert       : A(Pointer, ElementType, Alignment) {
83*12c85518Srobert     assert(Pointer != nullptr && "Pointer cannot be null");
84*12c85518Srobert     assert(ElementType != nullptr && "Element type cannot be null");
85*12c85518Srobert     assert(llvm::cast<llvm::PointerType>(Pointer->getType())
86*12c85518Srobert                ->isOpaqueOrPointeeTypeMatches(ElementType) &&
87*12c85518Srobert            "Incorrect pointer element type");
88e5dd7070Spatrick   }
89e5dd7070Spatrick 
invalid()90*12c85518Srobert   static Address invalid() { return Address(nullptr); }
isValid()91*12c85518Srobert   bool isValid() const { return A.getPointer() != nullptr; }
92e5dd7070Spatrick 
getPointer()93e5dd7070Spatrick   llvm::Value *getPointer() const {
94e5dd7070Spatrick     assert(isValid());
95*12c85518Srobert     return A.getPointer();
96e5dd7070Spatrick   }
97e5dd7070Spatrick 
98e5dd7070Spatrick   /// Return the type of the pointer value.
getType()99e5dd7070Spatrick   llvm::PointerType *getType() const {
100e5dd7070Spatrick     return llvm::cast<llvm::PointerType>(getPointer()->getType());
101e5dd7070Spatrick   }
102e5dd7070Spatrick 
103e5dd7070Spatrick   /// Return the type of the values stored in this address.
getElementType()104e5dd7070Spatrick   llvm::Type *getElementType() const {
105*12c85518Srobert     assert(isValid());
106*12c85518Srobert     return A.getElementType();
107e5dd7070Spatrick   }
108e5dd7070Spatrick 
109e5dd7070Spatrick   /// Return the address space that this address resides in.
getAddressSpace()110e5dd7070Spatrick   unsigned getAddressSpace() const {
111e5dd7070Spatrick     return getType()->getAddressSpace();
112e5dd7070Spatrick   }
113e5dd7070Spatrick 
114e5dd7070Spatrick   /// Return the IR name of the pointer value.
getName()115e5dd7070Spatrick   llvm::StringRef getName() const {
116e5dd7070Spatrick     return getPointer()->getName();
117e5dd7070Spatrick   }
118e5dd7070Spatrick 
119e5dd7070Spatrick   /// Return the alignment of this pointer.
getAlignment()120e5dd7070Spatrick   CharUnits getAlignment() const {
121e5dd7070Spatrick     assert(isValid());
122*12c85518Srobert     return A.getAlignment();
123*12c85518Srobert   }
124*12c85518Srobert 
125*12c85518Srobert   /// Return address with different pointer, but same element type and
126*12c85518Srobert   /// alignment.
withPointer(llvm::Value * NewPointer)127*12c85518Srobert   Address withPointer(llvm::Value *NewPointer) const {
128*12c85518Srobert     return Address(NewPointer, getElementType(), getAlignment());
129*12c85518Srobert   }
130*12c85518Srobert 
131*12c85518Srobert   /// Return address with different alignment, but same pointer and element
132*12c85518Srobert   /// type.
withAlignment(CharUnits NewAlignment)133*12c85518Srobert   Address withAlignment(CharUnits NewAlignment) const {
134*12c85518Srobert     return Address(getPointer(), getElementType(), NewAlignment);
135e5dd7070Spatrick   }
136e5dd7070Spatrick };
137e5dd7070Spatrick 
138e5dd7070Spatrick /// A specialization of Address that requires the address to be an
139e5dd7070Spatrick /// LLVM Constant.
140e5dd7070Spatrick class ConstantAddress : public Address {
ConstantAddress(std::nullptr_t)141*12c85518Srobert   ConstantAddress(std::nullptr_t) : Address(nullptr) {}
142*12c85518Srobert 
143e5dd7070Spatrick public:
ConstantAddress(llvm::Constant * pointer,llvm::Type * elementType,CharUnits alignment)144*12c85518Srobert   ConstantAddress(llvm::Constant *pointer, llvm::Type *elementType,
145*12c85518Srobert                   CharUnits alignment)
146*12c85518Srobert       : Address(pointer, elementType, alignment) {}
147e5dd7070Spatrick 
invalid()148e5dd7070Spatrick   static ConstantAddress invalid() {
149*12c85518Srobert     return ConstantAddress(nullptr);
150e5dd7070Spatrick   }
151e5dd7070Spatrick 
getPointer()152e5dd7070Spatrick   llvm::Constant *getPointer() const {
153e5dd7070Spatrick     return llvm::cast<llvm::Constant>(Address::getPointer());
154e5dd7070Spatrick   }
155e5dd7070Spatrick 
getElementBitCast(llvm::Type * ElemTy)156*12c85518Srobert   ConstantAddress getElementBitCast(llvm::Type *ElemTy) const {
157*12c85518Srobert     llvm::Constant *BitCast = llvm::ConstantExpr::getBitCast(
158*12c85518Srobert         getPointer(), ElemTy->getPointerTo(getAddressSpace()));
159*12c85518Srobert     return ConstantAddress(BitCast, ElemTy, getAlignment());
160e5dd7070Spatrick   }
161e5dd7070Spatrick 
isaImpl(Address addr)162e5dd7070Spatrick   static bool isaImpl(Address addr) {
163e5dd7070Spatrick     return llvm::isa<llvm::Constant>(addr.getPointer());
164e5dd7070Spatrick   }
castImpl(Address addr)165e5dd7070Spatrick   static ConstantAddress castImpl(Address addr) {
166e5dd7070Spatrick     return ConstantAddress(llvm::cast<llvm::Constant>(addr.getPointer()),
167*12c85518Srobert                            addr.getElementType(), addr.getAlignment());
168e5dd7070Spatrick   }
169e5dd7070Spatrick };
170e5dd7070Spatrick 
171e5dd7070Spatrick }
172e5dd7070Spatrick 
173e5dd7070Spatrick // Present a minimal LLVM-like casting interface.
cast(CodeGen::Address addr)174e5dd7070Spatrick template <class U> inline U cast(CodeGen::Address addr) {
175e5dd7070Spatrick   return U::castImpl(addr);
176e5dd7070Spatrick }
isa(CodeGen::Address addr)177e5dd7070Spatrick template <class U> inline bool isa(CodeGen::Address addr) {
178e5dd7070Spatrick   return U::isaImpl(addr);
179e5dd7070Spatrick }
180e5dd7070Spatrick 
181e5dd7070Spatrick }
182e5dd7070Spatrick 
183e5dd7070Spatrick #endif
184