1 //===- ELFObjectFile.cpp - ELF object file implementation -------*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 // 10 // Part of the ELFObjectFile class implementation. 11 // 12 //===----------------------------------------------------------------------===// 13 14 #include "llvm/Object/ELFObjectFile.h" 15 #include "llvm/Support/MathExtras.h" 16 17 namespace llvm { 18 using namespace object; 19 20 ELFObjectFileBase::ELFObjectFileBase(unsigned int Type, MemoryBufferRef Source) 21 : ObjectFile(Type, Source) {} 22 23 ErrorOr<std::unique_ptr<ObjectFile>> 24 ObjectFile::createELFObjectFile(MemoryBufferRef Obj) { 25 std::pair<unsigned char, unsigned char> Ident = 26 getElfArchType(Obj.getBuffer()); 27 std::size_t MaxAlignment = 28 1ULL << countTrailingZeros(uintptr_t(Obj.getBufferStart())); 29 30 if (MaxAlignment < 2) 31 return object_error::parse_failed; 32 33 std::error_code EC; 34 std::unique_ptr<ObjectFile> R; 35 if (Ident.first == ELF::ELFCLASS32) { 36 if (Ident.second == ELF::ELFDATA2LSB) 37 R.reset(new ELFObjectFile<ELFType<support::little, false>>(Obj, EC)); 38 else if (Ident.second == ELF::ELFDATA2MSB) 39 R.reset(new ELFObjectFile<ELFType<support::big, false>>(Obj, EC)); 40 else 41 return object_error::parse_failed; 42 } else if (Ident.first == ELF::ELFCLASS64) { 43 if (Ident.second == ELF::ELFDATA2LSB) 44 R.reset(new ELFObjectFile<ELFType<support::little, true>>(Obj, EC)); 45 else if (Ident.second == ELF::ELFDATA2MSB) 46 R.reset(new ELFObjectFile<ELFType<support::big, true>>(Obj, EC)); 47 else 48 return object_error::parse_failed; 49 } else { 50 return object_error::parse_failed; 51 } 52 53 if (EC) 54 return EC; 55 return std::move(R); 56 } 57 58 SubtargetFeatures ELFObjectFileBase::getFeatures() const { 59 switch (getEMachine()) { 60 case ELF::EM_MIPS: { 61 SubtargetFeatures Features; 62 unsigned PlatformFlags; 63 getPlatformFlags(PlatformFlags); 64 65 switch (PlatformFlags & ELF::EF_MIPS_ARCH) { 66 case ELF::EF_MIPS_ARCH_1: 67 break; 68 case ELF::EF_MIPS_ARCH_2: 69 Features.AddFeature("mips2"); 70 break; 71 case ELF::EF_MIPS_ARCH_3: 72 Features.AddFeature("mips3"); 73 break; 74 case ELF::EF_MIPS_ARCH_4: 75 Features.AddFeature("mips4"); 76 break; 77 case ELF::EF_MIPS_ARCH_5: 78 Features.AddFeature("mips5"); 79 break; 80 case ELF::EF_MIPS_ARCH_32: 81 Features.AddFeature("mips32"); 82 break; 83 case ELF::EF_MIPS_ARCH_64: 84 Features.AddFeature("mips64"); 85 break; 86 case ELF::EF_MIPS_ARCH_32R2: 87 Features.AddFeature("mips32r2"); 88 break; 89 case ELF::EF_MIPS_ARCH_64R2: 90 Features.AddFeature("mips64r2"); 91 break; 92 case ELF::EF_MIPS_ARCH_32R6: 93 Features.AddFeature("mips32r6"); 94 break; 95 case ELF::EF_MIPS_ARCH_64R6: 96 Features.AddFeature("mips64r6"); 97 break; 98 default: 99 llvm_unreachable("Unknown EF_MIPS_ARCH value"); 100 } 101 102 switch (PlatformFlags & ELF::EF_MIPS_MACH) { 103 case ELF::EF_MIPS_MACH_NONE: 104 // No feature associated with this value. 105 break; 106 case ELF::EF_MIPS_MACH_OCTEON: 107 Features.AddFeature("cnmips"); 108 break; 109 default: 110 llvm_unreachable("Unknown EF_MIPS_ARCH value"); 111 } 112 113 if (PlatformFlags & ELF::EF_MIPS_ARCH_ASE_M16) 114 Features.AddFeature("mips16"); 115 if (PlatformFlags & ELF::EF_MIPS_MICROMIPS) 116 Features.AddFeature("micromips"); 117 118 return Features; 119 } 120 default: 121 return SubtargetFeatures(); 122 } 123 } 124 125 // FIXME Encode from a tablegen description or target parser. 126 void ELFObjectFileBase::setARMSubArch(Triple &TheTriple) const { 127 if (TheTriple.getSubArch() != Triple::NoSubArch) 128 return; 129 130 ARMAttributeParser Attributes; 131 std::error_code EC = getBuildAttributes(Attributes); 132 if (EC) 133 return; 134 135 std::string Triple; 136 // Default to ARM, but use the triple if it's been set. 137 if (TheTriple.getArch() == Triple::thumb || 138 TheTriple.getArch() == Triple::thumbeb) 139 Triple = "thumb"; 140 else 141 Triple = "arm"; 142 143 if (Attributes.hasAttribute(ARMBuildAttrs::CPU_arch)) { 144 switch(Attributes.getAttributeValue(ARMBuildAttrs::CPU_arch)) { 145 case ARMBuildAttrs::v4: 146 Triple += "v4"; 147 break; 148 case ARMBuildAttrs::v4T: 149 Triple += "v4t"; 150 break; 151 case ARMBuildAttrs::v5T: 152 Triple += "v5t"; 153 break; 154 case ARMBuildAttrs::v5TE: 155 Triple += "v5te"; 156 break; 157 case ARMBuildAttrs::v5TEJ: 158 Triple += "v5tej"; 159 break; 160 case ARMBuildAttrs::v6: 161 Triple += "v6"; 162 break; 163 case ARMBuildAttrs::v6KZ: 164 Triple += "v6kz"; 165 break; 166 case ARMBuildAttrs::v6T2: 167 Triple += "v6t2"; 168 break; 169 case ARMBuildAttrs::v6K: 170 Triple += "v6k"; 171 break; 172 case ARMBuildAttrs::v7: 173 Triple += "v7"; 174 break; 175 case ARMBuildAttrs::v6_M: 176 Triple += "v6m"; 177 break; 178 case ARMBuildAttrs::v6S_M: 179 Triple += "v6sm"; 180 break; 181 case ARMBuildAttrs::v7E_M: 182 Triple += "v7em"; 183 break; 184 } 185 } 186 if (!isLittleEndian()) 187 Triple += "eb"; 188 189 TheTriple.setArchName(Triple); 190 } 191 192 } // end namespace llvm 193