| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 1 | //===-- TargetData.cpp - Data size & alignment routines --------------------==// | 
|  | 2 | // | 
|  | 3 | // This file defines target properties related to datatype size/offset/alignment | 
|  | 4 | // information.  It uses lazy annotations to cache information about how | 
|  | 5 | // structure types are laid out and used. | 
|  | 6 | // | 
|  | 7 | // This structure should be created once, filled in if the defaults are not | 
|  | 8 | // correct and then passed around by const&.  None of the members functions | 
|  | 9 | // require modification to the object. | 
|  | 10 | // | 
|  | 11 | //===----------------------------------------------------------------------===// | 
|  | 12 |  | 
| Vikram S. Adve | 0799fc4 | 2001-09-18 12:58:33 +0000 | [diff] [blame] | 13 | #include "llvm/Target/TargetData.h" | 
| Chris Lattner | 53a0c38 | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 14 | #include "llvm/Module.h" | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 15 | #include "llvm/DerivedTypes.h" | 
| Chris Lattner | 31bcdb8 | 2002-04-28 19:55:58 +0000 | [diff] [blame] | 16 | #include "llvm/Constants.h" | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 17 |  | 
| Chris Lattner | aa31ad0 | 2002-09-25 23:46:55 +0000 | [diff] [blame] | 18 | // Handle the Pass registration stuff neccesary to use TargetData's. | 
|  | 19 | namespace { | 
|  | 20 | // Register the default SparcV9 implementation... | 
|  | 21 | RegisterPass<TargetData> X("targetdata", "Target Data Layout"); | 
|  | 22 | } | 
|  | 23 |  | 
|  | 24 |  | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 25 | static inline void getTypeInfo(const Type *Ty, const TargetData *TD, | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 26 | uint64_t &Size, unsigned char &Alignment); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 27 |  | 
|  | 28 | //===----------------------------------------------------------------------===// | 
|  | 29 | // Support for StructLayout Annotation | 
|  | 30 | //===----------------------------------------------------------------------===// | 
|  | 31 |  | 
|  | 32 | StructLayout::StructLayout(const StructType *ST, const TargetData &TD) | 
|  | 33 | : Annotation(TD.getStructLayoutAID()) { | 
|  | 34 | StructAlignment = 0; | 
|  | 35 | StructSize = 0; | 
|  | 36 |  | 
|  | 37 | // Loop over each of the elements, placing them in memory... | 
|  | 38 | for (StructType::ElementTypes::const_iterator | 
|  | 39 | TI = ST->getElementTypes().begin(), | 
|  | 40 | TE = ST->getElementTypes().end(); TI != TE; ++TI) { | 
|  | 41 | const Type *Ty = *TI; | 
|  | 42 | unsigned char A; | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 43 | unsigned TyAlign; | 
|  | 44 | uint64_t TySize; | 
|  | 45 | getTypeInfo(Ty, &TD, TySize, A); | 
|  | 46 | TyAlign = A; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 47 |  | 
|  | 48 | // Add padding if neccesary to make the data element aligned properly... | 
|  | 49 | if (StructSize % TyAlign != 0) | 
|  | 50 | StructSize = (StructSize/TyAlign + 1) * TyAlign;   // Add padding... | 
|  | 51 |  | 
|  | 52 | // Keep track of maximum alignment constraint | 
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 53 | StructAlignment = std::max(TyAlign, StructAlignment); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 54 |  | 
|  | 55 | MemberOffsets.push_back(StructSize); | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 56 | StructSize += TySize;                 // Consume space for this data item | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 57 | } | 
|  | 58 |  | 
|  | 59 | // Add padding to the end of the struct so that it could be put in an array | 
|  | 60 | // and all array elements would be aligned correctly. | 
|  | 61 | if (StructSize % StructAlignment != 0) | 
|  | 62 | StructSize = (StructSize/StructAlignment + 1) * StructAlignment; | 
|  | 63 |  | 
|  | 64 | if (StructSize == 0) { | 
|  | 65 | StructSize = 1;           // Empty struct is 1 byte | 
|  | 66 | StructAlignment = 1; | 
|  | 67 | } | 
|  | 68 | } | 
|  | 69 |  | 
| Chris Lattner | 97b7311 | 2001-09-07 16:40:04 +0000 | [diff] [blame] | 70 | Annotation *TargetData::TypeAnFactory(AnnotationID AID, const Annotable *T, | 
|  | 71 | void *D) { | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 72 | const TargetData &TD = *(const TargetData*)D; | 
|  | 73 | assert(AID == TD.AID && "Target data annotation ID mismatch!"); | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 74 | const Type *Ty = cast<const Type>((const Value *)T); | 
| Chris Lattner | 9b62503 | 2002-05-06 16:15:30 +0000 | [diff] [blame] | 75 | assert(isa<StructType>(Ty) && | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 76 | "Can only create StructLayout annotation on structs!"); | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 77 | return new StructLayout((const StructType *)Ty, TD); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 78 | } | 
|  | 79 |  | 
|  | 80 | //===----------------------------------------------------------------------===// | 
|  | 81 | //                       TargetData Class Implementation | 
|  | 82 | //===----------------------------------------------------------------------===// | 
|  | 83 |  | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 84 | TargetData::TargetData(const std::string &TargetName, | 
| Chris Lattner | c56406c | 2002-10-29 21:48:17 +0000 | [diff] [blame] | 85 | bool isLittleEndian, unsigned char SubWordSize, | 
| Chris Lattner | 85131c8 | 2002-10-14 22:41:13 +0000 | [diff] [blame] | 86 | unsigned char IntRegSize, unsigned char PtrSize, | 
|  | 87 | unsigned char PtrAl, unsigned char DoubleAl, | 
|  | 88 | unsigned char FloatAl, unsigned char LongAl, | 
|  | 89 | unsigned char IntAl, unsigned char ShortAl, | 
|  | 90 | unsigned char ByteAl) | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 91 | : AID(AnnotationManager::getID("TargetData::" + TargetName)) { | 
|  | 92 | AnnotationManager::registerAnnotationFactory(AID, TypeAnFactory, this); | 
|  | 93 |  | 
| Chris Lattner | 46326d9 | 2003-04-25 02:50:45 +0000 | [diff] [blame] | 94 | // If this assert triggers, a pass "required" TargetData information, but the | 
|  | 95 | // top level tool did not provide once for it.  We do not want to default | 
|  | 96 | // construct, or else we might end up using a bad endianness or pointer size! | 
|  | 97 | // | 
|  | 98 | assert(!TargetName.empty() && | 
|  | 99 | "ERROR: Tool did not specify a target data to use!"); | 
|  | 100 |  | 
| Chris Lattner | 85131c8 | 2002-10-14 22:41:13 +0000 | [diff] [blame] | 101 | LittleEndian     = isLittleEndian; | 
| Chris Lattner | c56406c | 2002-10-29 21:48:17 +0000 | [diff] [blame] | 102 | SubWordDataSize  = SubWordSize; | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 103 | IntegerRegSize   = IntRegSize; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 104 | PointerSize      = PtrSize; | 
|  | 105 | PointerAlignment = PtrAl; | 
|  | 106 | DoubleAlignment  = DoubleAl; | 
|  | 107 | FloatAlignment   = FloatAl; | 
|  | 108 | LongAlignment    = LongAl; | 
|  | 109 | IntAlignment     = IntAl; | 
|  | 110 | ShortAlignment   = ShortAl; | 
|  | 111 | ByteAlignment    = ByteAl; | 
|  | 112 | } | 
|  | 113 |  | 
| Chris Lattner | 53a0c38 | 2003-04-24 19:09:05 +0000 | [diff] [blame] | 114 | TargetData::TargetData(const std::string &ToolName, const Module *M) | 
|  | 115 | : AID(AnnotationManager::getID("TargetData::" + ToolName)) { | 
|  | 116 | AnnotationManager::registerAnnotationFactory(AID, TypeAnFactory, this); | 
|  | 117 |  | 
|  | 118 | LittleEndian     = M->isLittleEndian(); | 
|  | 119 | SubWordDataSize  = 1; | 
|  | 120 | IntegerRegSize   = 8; | 
|  | 121 | PointerSize      = M->has32BitPointers() ? 32 : 64; | 
|  | 122 | PointerAlignment = PointerSize; | 
|  | 123 | DoubleAlignment  = 8; | 
|  | 124 | FloatAlignment   = 4; | 
|  | 125 | LongAlignment    = 8; | 
|  | 126 | IntAlignment     = 4; | 
|  | 127 | ShortAlignment   = 2; | 
|  | 128 | ByteAlignment    = 1; | 
|  | 129 | } | 
|  | 130 |  | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 131 | TargetData::~TargetData() { | 
|  | 132 | AnnotationManager::registerAnnotationFactory(AID, 0);   // Deregister factory | 
|  | 133 | } | 
|  | 134 |  | 
|  | 135 | static inline void getTypeInfo(const Type *Ty, const TargetData *TD, | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 136 | uint64_t &Size, unsigned char &Alignment) { | 
| Chris Lattner | f59ce92 | 2001-12-13 00:46:11 +0000 | [diff] [blame] | 137 | assert(Ty->isSized() && "Cannot getTypeInfo() on a type that is unsized!"); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 138 | switch (Ty->getPrimitiveID()) { | 
|  | 139 | case Type::VoidTyID: | 
|  | 140 | case Type::BoolTyID: | 
|  | 141 | case Type::UByteTyID: | 
|  | 142 | case Type::SByteTyID:  Size = 1; Alignment = TD->getByteAlignment(); return; | 
|  | 143 | case Type::UShortTyID: | 
|  | 144 | case Type::ShortTyID:  Size = 2; Alignment = TD->getShortAlignment(); return; | 
|  | 145 | case Type::UIntTyID: | 
|  | 146 | case Type::IntTyID:    Size = 4; Alignment = TD->getIntAlignment(); return; | 
|  | 147 | case Type::ULongTyID: | 
|  | 148 | case Type::LongTyID:   Size = 8; Alignment = TD->getLongAlignment(); return; | 
|  | 149 | case Type::FloatTyID:  Size = 4; Alignment = TD->getFloatAlignment(); return; | 
|  | 150 | case Type::DoubleTyID: Size = 8; Alignment = TD->getDoubleAlignment(); return; | 
|  | 151 | case Type::LabelTyID: | 
|  | 152 | case Type::PointerTyID: | 
|  | 153 | Size = TD->getPointerSize(); Alignment = TD->getPointerAlignment(); | 
|  | 154 | return; | 
|  | 155 | case Type::ArrayTyID: { | 
|  | 156 | const ArrayType *ATy = (const ArrayType *)Ty; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 157 | getTypeInfo(ATy->getElementType(), TD, Size, Alignment); | 
|  | 158 | Size *= ATy->getNumElements(); | 
|  | 159 | return; | 
|  | 160 | } | 
|  | 161 | case Type::StructTyID: { | 
|  | 162 | // Get the layout annotation... which is lazily created on demand. | 
|  | 163 | const StructLayout *Layout = TD->getStructLayout((const StructType*)Ty); | 
|  | 164 | Size = Layout->StructSize; Alignment = Layout->StructAlignment; | 
|  | 165 | return; | 
|  | 166 | } | 
|  | 167 |  | 
|  | 168 | case Type::TypeTyID: | 
|  | 169 | default: | 
|  | 170 | assert(0 && "Bad type for getTypeInfo!!!"); | 
|  | 171 | return; | 
|  | 172 | } | 
|  | 173 | } | 
|  | 174 |  | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 175 | uint64_t TargetData::getTypeSize(const Type *Ty) const { | 
|  | 176 | uint64_t Size; | 
|  | 177 | unsigned char Align; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 178 | getTypeInfo(Ty, this, Size, Align); | 
|  | 179 | return Size; | 
|  | 180 | } | 
|  | 181 |  | 
|  | 182 | unsigned char TargetData::getTypeAlignment(const Type *Ty) const { | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 183 | uint64_t Size; | 
|  | 184 | unsigned char Align; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 185 | getTypeInfo(Ty, this, Size, Align); | 
|  | 186 | return Align; | 
|  | 187 | } | 
|  | 188 |  | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 189 | uint64_t TargetData::getIndexedOffset(const Type *ptrTy, | 
| Chris Lattner | 697954c | 2002-01-20 22:54:45 +0000 | [diff] [blame] | 190 | const std::vector<Value*> &Idx) const { | 
| Vikram S. Adve | ed0030e | 2002-08-04 20:52:39 +0000 | [diff] [blame] | 191 | const Type *Ty = ptrTy; | 
|  | 192 | assert(isa<PointerType>(Ty) && "Illegal argument for getIndexedOffset()"); | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 193 | uint64_t Result = 0; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 194 |  | 
| Chris Lattner | 3cac88a | 2002-09-11 01:21:33 +0000 | [diff] [blame] | 195 | for (unsigned CurIDX = 0; CurIDX != Idx.size(); ++CurIDX) { | 
|  | 196 | if (Idx[CurIDX]->getType() == Type::LongTy) { | 
| Vikram S. Adve | ca710e9 | 2002-08-13 18:17:56 +0000 | [diff] [blame] | 197 | // Update Ty to refer to current element | 
|  | 198 | Ty = cast<SequentialType>(Ty)->getElementType(); | 
|  | 199 |  | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 200 | // Get the array index and the size of each array element. | 
| Vikram S. Adve | 8e22ead | 2002-10-13 21:47:44 +0000 | [diff] [blame] | 201 | // Both must be known constants, or the index shd be 0; else this fails. | 
| Chris Lattner | 3cac88a | 2002-09-11 01:21:33 +0000 | [diff] [blame] | 202 | int64_t arrayIdx = cast<ConstantSInt>(Idx[CurIDX])->getValue(); | 
| Vikram S. Adve | 8e22ead | 2002-10-13 21:47:44 +0000 | [diff] [blame] | 203 | Result += arrayIdx == 0? 0 | 
|  | 204 | : (uint64_t) (arrayIdx * (int64_t) getTypeSize(Ty)); | 
| Vikram S. Adve | ed0030e | 2002-08-04 20:52:39 +0000 | [diff] [blame] | 205 |  | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 206 | } else if (const StructType *STy = dyn_cast<const StructType>(Ty)) { | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 207 | assert(Idx[CurIDX]->getType() == Type::UByteTy && "Illegal struct idx"); | 
| Chris Lattner | e9bb2df | 2001-12-03 22:26:30 +0000 | [diff] [blame] | 208 | unsigned FieldNo = cast<ConstantUInt>(Idx[CurIDX])->getValue(); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 209 |  | 
|  | 210 | // Get structure layout information... | 
|  | 211 | const StructLayout *Layout = getStructLayout(STy); | 
|  | 212 |  | 
|  | 213 | // Add in the offset, as calculated by the structure layout info... | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 214 | assert(FieldNo < Layout->MemberOffsets.size() &&"FieldNo out of range!"); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 215 | Result += Layout->MemberOffsets[FieldNo]; | 
| Vikram S. Adve | ed0030e | 2002-08-04 20:52:39 +0000 | [diff] [blame] | 216 |  | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 217 | // Update Ty to refer to current element | 
|  | 218 | Ty = STy->getElementTypes()[FieldNo]; | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 219 | } else { | 
| Vikram S. Adve | f66723f | 2002-05-19 15:28:02 +0000 | [diff] [blame] | 220 | assert(0 && "Indexing type that is not struct or array?"); | 
| Chris Lattner | e7fb360 | 2001-08-27 16:00:15 +0000 | [diff] [blame] | 221 | return 0;                         // Load directly through ptr | 
|  | 222 | } | 
|  | 223 | } | 
|  | 224 |  | 
|  | 225 | return Result; | 
|  | 226 | } |