| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 1 | //===-- TargetData.cpp - Data size & alignment routines --------------------==// | 
| Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 2 | // | 
| John Criswell | 482202a | 2003-10-20 19:43:21 +0000 | [diff] [blame] | 3 | //                     The LLVM Compiler Infrastructure | 
|  | 4 | // | 
|  | 5 | // This file was developed by the LLVM research group and is distributed under | 
|  | 6 | // the University of Illinois Open Source License. See LICENSE.TXT for details. | 
| Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 7 | // | 
| John Criswell | 482202a | 2003-10-20 19:43:21 +0000 | [diff] [blame] | 8 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 9 | // | 
|  | 10 | // This file defines target properties related to datatype size/offset/alignment | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 11 | // information. | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 12 | // | 
|  | 13 | // This structure should be created once, filled in if the defaults are not | 
|  | 14 | // correct and then passed around by const&.  None of the members functions | 
|  | 15 | // require modification to the object. | 
|  | 16 | // | 
|  | 17 | //===----------------------------------------------------------------------===// | 
|  | 18 |  | 
| Vikram S. Adve | 6cefc70 | 2001-09-18 12:58:33 +0000 | [diff] [blame] | 19 | #include "llvm/Target/TargetData.h" | 
| Chris Lattner | 42516ba | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 20 | #include "llvm/Module.h" | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 21 | #include "llvm/DerivedTypes.h" | 
| Chris Lattner | ca14237 | 2002-04-28 19:55:58 +0000 | [diff] [blame] | 22 | #include "llvm/Constants.h" | 
| Chris Lattner | 69193f9 | 2004-04-05 01:30:19 +0000 | [diff] [blame] | 23 | #include "llvm/Support/GetElementPtrTypeIterator.h" | 
| Reid Spencer | 7c16caa | 2004-09-01 22:55:40 +0000 | [diff] [blame] | 24 | #include "llvm/Support/MathExtras.h" | 
| Chris Lattner | 3434004 | 2005-03-13 19:04:41 +0000 | [diff] [blame] | 25 | #include <algorithm> | 
| Chris Lattner | bb661c1 | 2003-12-22 05:01:15 +0000 | [diff] [blame] | 26 | using namespace llvm; | 
| Brian Gaeke | 960707c | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 27 |  | 
| Misha Brukman | 7eb05a1 | 2003-08-18 14:43:39 +0000 | [diff] [blame] | 28 | // Handle the Pass registration stuff necessary to use TargetData's. | 
| Chris Lattner | 09cfdcb | 2002-09-25 23:46:55 +0000 | [diff] [blame] | 29 | namespace { | 
|  | 30 | // Register the default SparcV9 implementation... | 
|  | 31 | RegisterPass<TargetData> X("targetdata", "Target Data Layout"); | 
|  | 32 | } | 
|  | 33 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 34 | static inline void getTypeInfo(const Type *Ty, const TargetData *TD, | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 35 | uint64_t &Size, unsigned char &Alignment); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 36 |  | 
|  | 37 | //===----------------------------------------------------------------------===// | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 38 | // Support for StructLayout | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 39 | //===----------------------------------------------------------------------===// | 
|  | 40 |  | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 41 | StructLayout::StructLayout(const StructType *ST, const TargetData &TD) { | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 42 | StructAlignment = 0; | 
|  | 43 | StructSize = 0; | 
|  | 44 |  | 
|  | 45 | // Loop over each of the elements, placing them in memory... | 
| Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 46 | for (StructType::element_iterator TI = ST->element_begin(), | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 47 | TE = ST->element_end(); TI != TE; ++TI) { | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 48 | const Type *Ty = *TI; | 
|  | 49 | unsigned char A; | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 50 | unsigned TyAlign; | 
|  | 51 | uint64_t TySize; | 
|  | 52 | getTypeInfo(Ty, &TD, TySize, A); | 
|  | 53 | TyAlign = A; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 54 |  | 
| Misha Brukman | 7eb05a1 | 2003-08-18 14:43:39 +0000 | [diff] [blame] | 55 | // Add padding if necessary to make the data element aligned properly... | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 56 | if (StructSize % TyAlign != 0) | 
|  | 57 | StructSize = (StructSize/TyAlign + 1) * TyAlign;   // Add padding... | 
|  | 58 |  | 
|  | 59 | // Keep track of maximum alignment constraint | 
| Chris Lattner | 7f74a56 | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 60 | StructAlignment = std::max(TyAlign, StructAlignment); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 61 |  | 
|  | 62 | MemberOffsets.push_back(StructSize); | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 63 | StructSize += TySize;                 // Consume space for this data item | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 64 | } | 
|  | 65 |  | 
| Chris Lattner | 6532e42 | 2003-05-21 18:08:44 +0000 | [diff] [blame] | 66 | // Empty structures have alignment of 1 byte. | 
|  | 67 | if (StructAlignment == 0) StructAlignment = 1; | 
|  | 68 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 69 | // Add padding to the end of the struct so that it could be put in an array | 
|  | 70 | // and all array elements would be aligned correctly. | 
|  | 71 | if (StructSize % StructAlignment != 0) | 
|  | 72 | StructSize = (StructSize/StructAlignment + 1) * StructAlignment; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 73 | } | 
|  | 74 |  | 
| Chris Lattner | 3434004 | 2005-03-13 19:04:41 +0000 | [diff] [blame] | 75 |  | 
|  | 76 | /// getElementContainingOffset - Given a valid offset into the structure, | 
|  | 77 | /// return the structure index that contains it. | 
|  | 78 | unsigned StructLayout::getElementContainingOffset(uint64_t Offset) const { | 
|  | 79 | std::vector<uint64_t>::const_iterator SI = | 
|  | 80 | std::upper_bound(MemberOffsets.begin(), MemberOffsets.end(), | 
|  | 81 | Offset); | 
|  | 82 | assert(SI != MemberOffsets.begin() && "Offset not in structure type!"); | 
|  | 83 | --SI; | 
|  | 84 | assert(*SI <= Offset && "upper_bound didn't work"); | 
|  | 85 | assert((SI == MemberOffsets.begin() || *(SI-1) < Offset) && | 
|  | 86 | (SI+1 == MemberOffsets.end() || *(SI+1) > Offset) && | 
|  | 87 | "Upper bound didn't work!"); | 
|  | 88 | return SI-MemberOffsets.begin(); | 
|  | 89 | } | 
|  | 90 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 91 | //===----------------------------------------------------------------------===// | 
|  | 92 | //                       TargetData Class Implementation | 
|  | 93 | //===----------------------------------------------------------------------===// | 
|  | 94 |  | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 95 | TargetData::TargetData(const std::string &TargetName, | 
| Chris Lattner | 4f6cdbd | 2003-04-26 20:11:09 +0000 | [diff] [blame] | 96 | bool isLittleEndian, unsigned char PtrSize, | 
| Chris Lattner | 5156bba | 2002-10-14 22:41:13 +0000 | [diff] [blame] | 97 | unsigned char PtrAl, unsigned char DoubleAl, | 
| Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 98 | unsigned char FloatAl, unsigned char LongAl, | 
| Chris Lattner | 5156bba | 2002-10-14 22:41:13 +0000 | [diff] [blame] | 99 | unsigned char IntAl, unsigned char ShortAl, | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 100 | unsigned char ByteAl, unsigned char BoolAl) { | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 101 |  | 
| Chris Lattner | e483398 | 2003-04-25 02:50:45 +0000 | [diff] [blame] | 102 | // If this assert triggers, a pass "required" TargetData information, but the | 
| Brian Gaeke | 0174347 | 2004-04-14 21:21:56 +0000 | [diff] [blame] | 103 | // top level tool did not provide one for it.  We do not want to default | 
| Chris Lattner | e483398 | 2003-04-25 02:50:45 +0000 | [diff] [blame] | 104 | // construct, or else we might end up using a bad endianness or pointer size! | 
|  | 105 | // | 
|  | 106 | assert(!TargetName.empty() && | 
|  | 107 | "ERROR: Tool did not specify a target data to use!"); | 
|  | 108 |  | 
| Chris Lattner | 5156bba | 2002-10-14 22:41:13 +0000 | [diff] [blame] | 109 | LittleEndian     = isLittleEndian; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 110 | PointerSize      = PtrSize; | 
|  | 111 | PointerAlignment = PtrAl; | 
|  | 112 | DoubleAlignment  = DoubleAl; | 
|  | 113 | FloatAlignment   = FloatAl; | 
|  | 114 | LongAlignment    = LongAl; | 
|  | 115 | IntAlignment     = IntAl; | 
|  | 116 | ShortAlignment   = ShortAl; | 
|  | 117 | ByteAlignment    = ByteAl; | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 118 | BoolAlignment    = BoolAl; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 119 | } | 
|  | 120 |  | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 121 | TargetData::TargetData(const std::string &ToolName, const Module *M) { | 
| Chris Lattner | e34e200 | 2003-08-24 13:49:22 +0000 | [diff] [blame] | 122 | LittleEndian     = M->getEndianness() != Module::BigEndian; | 
|  | 123 | PointerSize      = M->getPointerSize() != Module::Pointer64 ? 4 : 8; | 
| Chris Lattner | 42516ba | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 124 | PointerAlignment = PointerSize; | 
| Chris Lattner | b05f851 | 2003-04-25 06:06:43 +0000 | [diff] [blame] | 125 | DoubleAlignment  = PointerSize; | 
| Chris Lattner | 42516ba | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 126 | FloatAlignment   = 4; | 
| Chris Lattner | cff3b41 | 2004-11-02 22:18:18 +0000 | [diff] [blame] | 127 | LongAlignment    = PointerSize; | 
| Chris Lattner | 42516ba | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 128 | IntAlignment     = 4; | 
|  | 129 | ShortAlignment   = 2; | 
|  | 130 | ByteAlignment    = 1; | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 131 | BoolAlignment    = 1; | 
| Chris Lattner | 42516ba | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 132 | } | 
|  | 133 |  | 
| Chris Lattner | a4de9ba | 2006-01-14 00:07:34 +0000 | [diff] [blame] | 134 | /// Layouts - The lazy cache of structure layout information maintained by | 
|  | 135 | /// TargetData. | 
|  | 136 | /// | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 137 | static std::map<std::pair<const TargetData*,const StructType*>, | 
|  | 138 | StructLayout> *Layouts = 0; | 
|  | 139 |  | 
|  | 140 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 141 | TargetData::~TargetData() { | 
| Chris Lattner | 7140e46 | 2004-02-26 08:02:17 +0000 | [diff] [blame] | 142 | if (Layouts) { | 
|  | 143 | // Remove any layouts for this TD. | 
|  | 144 | std::map<std::pair<const TargetData*, | 
|  | 145 | const StructType*>, StructLayout>::iterator | 
|  | 146 | I = Layouts->lower_bound(std::make_pair(this, (const StructType*)0)); | 
|  | 147 | while (I != Layouts->end() && I->first.first == this) | 
|  | 148 | Layouts->erase(I++); | 
|  | 149 | if (Layouts->empty()) { | 
|  | 150 | delete Layouts; | 
|  | 151 | Layouts = 0; | 
|  | 152 | } | 
|  | 153 | } | 
|  | 154 | } | 
|  | 155 |  | 
|  | 156 | const StructLayout *TargetData::getStructLayout(const StructType *Ty) const { | 
|  | 157 | if (Layouts == 0) | 
|  | 158 | Layouts = new std::map<std::pair<const TargetData*,const StructType*>, | 
|  | 159 | StructLayout>(); | 
|  | 160 | std::map<std::pair<const TargetData*,const StructType*>, | 
|  | 161 | StructLayout>::iterator | 
|  | 162 | I = Layouts->lower_bound(std::make_pair(this, Ty)); | 
|  | 163 | if (I != Layouts->end() && I->first.first == this && I->first.second == Ty) | 
|  | 164 | return &I->second; | 
|  | 165 | else { | 
|  | 166 | return &Layouts->insert(I, std::make_pair(std::make_pair(this, Ty), | 
|  | 167 | StructLayout(Ty, *this)))->second; | 
|  | 168 | } | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 169 | } | 
|  | 170 |  | 
| Chris Lattner | a4de9ba | 2006-01-14 00:07:34 +0000 | [diff] [blame] | 171 | /// InvalidateStructLayoutInfo - TargetData speculatively caches StructLayout | 
|  | 172 | /// objects.  If a TargetData object is alive when types are being refined and | 
|  | 173 | /// removed, this method must be called whenever a StructType is removed to | 
|  | 174 | /// avoid a dangling pointer in this cache. | 
|  | 175 | void TargetData::InvalidateStructLayoutInfo(const StructType *Ty) const { | 
|  | 176 | if (!Layouts) return;  // No cache. | 
|  | 177 |  | 
|  | 178 | std::map<std::pair<const TargetData*,const StructType*>, | 
|  | 179 | StructLayout>::iterator I = Layouts->find(std::make_pair(this, Ty)); | 
|  | 180 | if (I != Layouts->end()) | 
|  | 181 | Layouts->erase(I); | 
|  | 182 | } | 
|  | 183 |  | 
|  | 184 |  | 
|  | 185 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 186 | static inline void getTypeInfo(const Type *Ty, const TargetData *TD, | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 187 | uint64_t &Size, unsigned char &Alignment) { | 
| Chris Lattner | 6650d18 | 2001-12-13 00:46:11 +0000 | [diff] [blame] | 188 | assert(Ty->isSized() && "Cannot getTypeInfo() on a type that is unsized!"); | 
| Chris Lattner | 6b72759 | 2004-06-17 18:19:28 +0000 | [diff] [blame] | 189 | switch (Ty->getTypeID()) { | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 190 | case Type::BoolTyID:   Size = 1; Alignment = TD->getBoolAlignment(); return; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 191 | case Type::VoidTyID: | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 192 | case Type::UByteTyID: | 
|  | 193 | case Type::SByteTyID:  Size = 1; Alignment = TD->getByteAlignment(); return; | 
|  | 194 | case Type::UShortTyID: | 
|  | 195 | case Type::ShortTyID:  Size = 2; Alignment = TD->getShortAlignment(); return; | 
|  | 196 | case Type::UIntTyID: | 
|  | 197 | case Type::IntTyID:    Size = 4; Alignment = TD->getIntAlignment(); return; | 
|  | 198 | case Type::ULongTyID: | 
|  | 199 | case Type::LongTyID:   Size = 8; Alignment = TD->getLongAlignment(); return; | 
|  | 200 | case Type::FloatTyID:  Size = 4; Alignment = TD->getFloatAlignment(); return; | 
|  | 201 | case Type::DoubleTyID: Size = 8; Alignment = TD->getDoubleAlignment(); return; | 
|  | 202 | case Type::LabelTyID: | 
|  | 203 | case Type::PointerTyID: | 
|  | 204 | Size = TD->getPointerSize(); Alignment = TD->getPointerAlignment(); | 
|  | 205 | return; | 
|  | 206 | case Type::ArrayTyID: { | 
| Chris Lattner | 18a08e7 | 2004-07-01 17:32:59 +0000 | [diff] [blame] | 207 | const ArrayType *ATy = cast<ArrayType>(Ty); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 208 | getTypeInfo(ATy->getElementType(), TD, Size, Alignment); | 
| Brian Gaeke | f6d2471 | 2004-07-02 07:01:31 +0000 | [diff] [blame] | 209 | unsigned AlignedSize = (Size + Alignment - 1)/Alignment*Alignment; | 
| Chris Lattner | 18a08e7 | 2004-07-01 17:32:59 +0000 | [diff] [blame] | 210 | Size = AlignedSize*ATy->getNumElements(); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 211 | return; | 
|  | 212 | } | 
| Chris Lattner | e58bf09 | 2004-12-01 17:14:28 +0000 | [diff] [blame] | 213 | case Type::PackedTyID: { | 
|  | 214 | const PackedType *PTy = cast<PackedType>(Ty); | 
|  | 215 | getTypeInfo(PTy->getElementType(), TD, Size, Alignment); | 
|  | 216 | unsigned AlignedSize = (Size + Alignment - 1)/Alignment*Alignment; | 
|  | 217 | Size = AlignedSize*PTy->getNumElements(); | 
| Evan Cheng | a11d834 | 2006-03-31 22:33:42 +0000 | [diff] [blame] | 218 | // FIXME: The alignments of specific packed types are target dependent. | 
|  | 219 | // For now, just set it to be equal to Size. | 
| Chris Lattner | bf0016f | 2006-04-03 23:14:49 +0000 | [diff] [blame] | 220 | Alignment = Size; | 
| Chris Lattner | e58bf09 | 2004-12-01 17:14:28 +0000 | [diff] [blame] | 221 | return; | 
|  | 222 | } | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 223 | case Type::StructTyID: { | 
|  | 224 | // Get the layout annotation... which is lazily created on demand. | 
| Chris Lattner | 18a08e7 | 2004-07-01 17:32:59 +0000 | [diff] [blame] | 225 | const StructLayout *Layout = TD->getStructLayout(cast<StructType>(Ty)); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 226 | Size = Layout->StructSize; Alignment = Layout->StructAlignment; | 
|  | 227 | return; | 
|  | 228 | } | 
| Misha Brukman | 10468d8 | 2005-04-21 22:55:34 +0000 | [diff] [blame] | 229 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 230 | default: | 
|  | 231 | assert(0 && "Bad type for getTypeInfo!!!"); | 
|  | 232 | return; | 
|  | 233 | } | 
|  | 234 | } | 
|  | 235 |  | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 236 | uint64_t TargetData::getTypeSize(const Type *Ty) const { | 
|  | 237 | uint64_t Size; | 
|  | 238 | unsigned char Align; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 239 | getTypeInfo(Ty, this, Size, Align); | 
|  | 240 | return Size; | 
|  | 241 | } | 
|  | 242 |  | 
|  | 243 | unsigned char TargetData::getTypeAlignment(const Type *Ty) const { | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 244 | uint64_t Size; | 
|  | 245 | unsigned char Align; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 246 | getTypeInfo(Ty, this, Size, Align); | 
|  | 247 | return Align; | 
|  | 248 | } | 
|  | 249 |  | 
| Chris Lattner | 0c2ab8a | 2004-08-17 19:13:00 +0000 | [diff] [blame] | 250 | unsigned char TargetData::getTypeAlignmentShift(const Type *Ty) const { | 
|  | 251 | unsigned Align = getTypeAlignment(Ty); | 
|  | 252 | assert(!(Align & (Align-1)) && "Alignment is not a power of two!"); | 
| Chris Lattner | 6667bdb | 2005-08-02 19:26:06 +0000 | [diff] [blame] | 253 | return Log2_32(Align); | 
| Chris Lattner | 0c2ab8a | 2004-08-17 19:13:00 +0000 | [diff] [blame] | 254 | } | 
|  | 255 |  | 
| Chris Lattner | bb661c1 | 2003-12-22 05:01:15 +0000 | [diff] [blame] | 256 | /// getIntPtrType - Return an unsigned integer type that is the same size or | 
|  | 257 | /// greater to the host pointer size. | 
|  | 258 | const Type *TargetData::getIntPtrType() const { | 
|  | 259 | switch (getPointerSize()) { | 
|  | 260 | default: assert(0 && "Unknown pointer size!"); | 
|  | 261 | case 2: return Type::UShortTy; | 
|  | 262 | case 4: return Type::UIntTy; | 
|  | 263 | case 8: return Type::ULongTy; | 
|  | 264 | } | 
|  | 265 | } | 
|  | 266 |  | 
|  | 267 |  | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 268 | uint64_t TargetData::getIndexedOffset(const Type *ptrTy, | 
| Misha Brukman | 3faa865 | 2004-07-23 01:09:52 +0000 | [diff] [blame] | 269 | const std::vector<Value*> &Idx) const { | 
| Vikram S. Adve | 71895b2 | 2002-08-04 20:52:39 +0000 | [diff] [blame] | 270 | const Type *Ty = ptrTy; | 
|  | 271 | assert(isa<PointerType>(Ty) && "Illegal argument for getIndexedOffset()"); | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 272 | uint64_t Result = 0; | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 273 |  | 
| Chris Lattner | 69193f9 | 2004-04-05 01:30:19 +0000 | [diff] [blame] | 274 | generic_gep_type_iterator<std::vector<Value*>::const_iterator> | 
|  | 275 | TI = gep_type_begin(ptrTy, Idx.begin(), Idx.end()); | 
|  | 276 | for (unsigned CurIDX = 0; CurIDX != Idx.size(); ++CurIDX, ++TI) { | 
|  | 277 | if (const StructType *STy = dyn_cast<StructType>(*TI)) { | 
|  | 278 | assert(Idx[CurIDX]->getType() == Type::UIntTy && "Illegal struct idx"); | 
| Chris Lattner | 3462ae3 | 2001-12-03 22:26:30 +0000 | [diff] [blame] | 279 | unsigned FieldNo = cast<ConstantUInt>(Idx[CurIDX])->getValue(); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 280 |  | 
|  | 281 | // Get structure layout information... | 
|  | 282 | const StructLayout *Layout = getStructLayout(STy); | 
|  | 283 |  | 
|  | 284 | // Add in the offset, as calculated by the structure layout info... | 
| Vikram S. Adve | 8b83174 | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 285 | assert(FieldNo < Layout->MemberOffsets.size() &&"FieldNo out of range!"); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 286 | Result += Layout->MemberOffsets[FieldNo]; | 
| Vikram S. Adve | 71895b2 | 2002-08-04 20:52:39 +0000 | [diff] [blame] | 287 |  | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 288 | // Update Ty to refer to current element | 
| Chris Lattner | ac6db75 | 2004-02-09 04:37:31 +0000 | [diff] [blame] | 289 | Ty = STy->getElementType(FieldNo); | 
| Chris Lattner | 69193f9 | 2004-04-05 01:30:19 +0000 | [diff] [blame] | 290 | } else { | 
|  | 291 | // Update Ty to refer to current element | 
|  | 292 | Ty = cast<SequentialType>(Ty)->getElementType(); | 
|  | 293 |  | 
|  | 294 | // Get the array index and the size of each array element. | 
|  | 295 | int64_t arrayIdx = cast<ConstantInt>(Idx[CurIDX])->getRawValue(); | 
|  | 296 | Result += arrayIdx * (int64_t)getTypeSize(Ty); | 
| Chris Lattner | 53bbf07 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 297 | } | 
|  | 298 | } | 
|  | 299 |  | 
|  | 300 | return Result; | 
|  | 301 | } | 
| Brian Gaeke | 960707c | 2003-11-11 22:41:34 +0000 | [diff] [blame] | 302 |  |