1 //===-- RelocVisitor.h - Visitor for object file relocations -*- 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 // This file provides a wrapper around all the different types of relocations 11 // in different file formats, such that a client can handle them in a unified 12 // manner by only implementing a minimal number of functions. 13 // 14 //===----------------------------------------------------------------------===// 15 16 #ifndef LLVM_OBJECT_RELOCVISITOR_H 17 #define LLVM_OBJECT_RELOCVISITOR_H 18 19 #include "llvm/ADT/StringRef.h" 20 #include "llvm/Object/ObjectFile.h" 21 #include "llvm/Object/ELF.h" 22 #include "llvm/Support/Debug.h" 23 #include "llvm/Support/ELF.h" 24 #include "llvm/Support/raw_ostream.h" 25 26 namespace llvm { 27 namespace object { 28 29 struct RelocToApply { 30 // The computed value after applying the relevant relocations. 31 int64_t Value; 32 33 // The width of the value; how many bytes to touch when applying the 34 // relocation. 35 char Width; RelocToApplyRelocToApply36 RelocToApply(const RelocToApply &In) : Value(In.Value), Width(In.Width) {} RelocToApplyRelocToApply37 RelocToApply(int64_t Value, char Width) : Value(Value), Width(Width) {} RelocToApplyRelocToApply38 RelocToApply() : Value(0), Width(0) {} 39 }; 40 41 /// @brief Base class for object file relocation visitors. 42 class RelocVisitor { 43 public: RelocVisitor(StringRef FileFormat)44 explicit RelocVisitor(StringRef FileFormat) 45 : FileFormat(FileFormat), HasError(false) {} 46 47 // TODO: Should handle multiple applied relocations via either passing in the 48 // previously computed value or just count paired relocations as a single 49 // visit. 50 RelocToApply visit(uint32_t RelocType, RelocationRef R, uint64_t SecAddr = 0, 51 uint64_t Value = 0) { 52 if (FileFormat == "ELF64-x86-64") { 53 switch (RelocType) { 54 case llvm::ELF::R_X86_64_NONE: 55 return visitELF_X86_64_NONE(R); 56 case llvm::ELF::R_X86_64_64: 57 return visitELF_X86_64_64(R, Value); 58 case llvm::ELF::R_X86_64_PC32: 59 return visitELF_X86_64_PC32(R, Value, SecAddr); 60 case llvm::ELF::R_X86_64_32: 61 return visitELF_X86_64_32(R, Value); 62 case llvm::ELF::R_X86_64_32S: 63 return visitELF_X86_64_32S(R, Value); 64 default: 65 HasError = true; 66 return RelocToApply(); 67 } 68 } else if (FileFormat == "ELF32-i386") { 69 switch (RelocType) { 70 case llvm::ELF::R_386_NONE: 71 return visitELF_386_NONE(R); 72 case llvm::ELF::R_386_32: 73 return visitELF_386_32(R, Value); 74 case llvm::ELF::R_386_PC32: 75 return visitELF_386_PC32(R, Value, SecAddr); 76 default: 77 HasError = true; 78 return RelocToApply(); 79 } 80 } else if (FileFormat == "ELF64-ppc64") { 81 switch (RelocType) { 82 case llvm::ELF::R_PPC64_ADDR32: 83 return visitELF_PPC64_ADDR32(R, Value); 84 case llvm::ELF::R_PPC64_ADDR64: 85 return visitELF_PPC64_ADDR64(R, Value); 86 default: 87 HasError = true; 88 return RelocToApply(); 89 } 90 } else if (FileFormat == "ELF32-ppc") { 91 switch (RelocType) { 92 case llvm::ELF::R_PPC_ADDR32: 93 return visitELF_PPC_ADDR32(R, Value); 94 default: 95 HasError = true; 96 return RelocToApply(); 97 } 98 } else if (FileFormat == "ELF32-mips") { 99 switch (RelocType) { 100 case llvm::ELF::R_MIPS_32: 101 return visitELF_MIPS_32(R, Value); 102 default: 103 HasError = true; 104 return RelocToApply(); 105 } 106 } else if (FileFormat == "ELF64-aarch64") { 107 switch (RelocType) { 108 case llvm::ELF::R_AARCH64_ABS32: 109 return visitELF_AARCH64_ABS32(R, Value); 110 case llvm::ELF::R_AARCH64_ABS64: 111 return visitELF_AARCH64_ABS64(R, Value); 112 default: 113 HasError = true; 114 return RelocToApply(); 115 } 116 } else if (FileFormat == "ELF64-s390") { 117 switch (RelocType) { 118 case llvm::ELF::R_390_32: 119 return visitELF_390_32(R, Value); 120 case llvm::ELF::R_390_64: 121 return visitELF_390_64(R, Value); 122 default: 123 HasError = true; 124 return RelocToApply(); 125 } 126 } 127 HasError = true; 128 return RelocToApply(); 129 } 130 error()131 bool error() { return HasError; } 132 133 private: 134 StringRef FileFormat; 135 bool HasError; 136 getAddend32LE(RelocationRef R)137 int64_t getAddend32LE(RelocationRef R) { 138 const ELF32LEObjectFile *Obj = cast<ELF32LEObjectFile>(R.getObjectFile()); 139 DataRefImpl DRI = R.getRawDataRefImpl(); 140 int64_t Addend; 141 Obj->getRelocationAddend(DRI, Addend); 142 return Addend; 143 } 144 getAddend64LE(RelocationRef R)145 int64_t getAddend64LE(RelocationRef R) { 146 const ELF64LEObjectFile *Obj = cast<ELF64LEObjectFile>(R.getObjectFile()); 147 DataRefImpl DRI = R.getRawDataRefImpl(); 148 int64_t Addend; 149 Obj->getRelocationAddend(DRI, Addend); 150 return Addend; 151 } 152 getAddend32BE(RelocationRef R)153 int64_t getAddend32BE(RelocationRef R) { 154 const ELF32BEObjectFile *Obj = cast<ELF32BEObjectFile>(R.getObjectFile()); 155 DataRefImpl DRI = R.getRawDataRefImpl(); 156 int64_t Addend; 157 Obj->getRelocationAddend(DRI, Addend); 158 return Addend; 159 } 160 getAddend64BE(RelocationRef R)161 int64_t getAddend64BE(RelocationRef R) { 162 const ELF64BEObjectFile *Obj = cast<ELF64BEObjectFile>(R.getObjectFile()); 163 DataRefImpl DRI = R.getRawDataRefImpl(); 164 int64_t Addend; 165 Obj->getRelocationAddend(DRI, Addend); 166 return Addend; 167 } 168 /// Operations 169 170 /// 386-ELF visitELF_386_NONE(RelocationRef R)171 RelocToApply visitELF_386_NONE(RelocationRef R) { 172 return RelocToApply(0, 0); 173 } 174 175 // Ideally the Addend here will be the addend in the data for 176 // the relocation. It's not actually the case for Rel relocations. visitELF_386_32(RelocationRef R,uint64_t Value)177 RelocToApply visitELF_386_32(RelocationRef R, uint64_t Value) { 178 int64_t Addend = getAddend32LE(R); 179 return RelocToApply(Value + Addend, 4); 180 } 181 visitELF_386_PC32(RelocationRef R,uint64_t Value,uint64_t SecAddr)182 RelocToApply visitELF_386_PC32(RelocationRef R, uint64_t Value, 183 uint64_t SecAddr) { 184 int64_t Addend = getAddend32LE(R); 185 uint64_t Address; 186 R.getOffset(Address); 187 return RelocToApply(Value + Addend - Address, 4); 188 } 189 190 /// X86-64 ELF visitELF_X86_64_NONE(RelocationRef R)191 RelocToApply visitELF_X86_64_NONE(RelocationRef R) { 192 return RelocToApply(0, 0); 193 } visitELF_X86_64_64(RelocationRef R,uint64_t Value)194 RelocToApply visitELF_X86_64_64(RelocationRef R, uint64_t Value) { 195 int64_t Addend = getAddend64LE(R); 196 return RelocToApply(Value + Addend, 8); 197 } visitELF_X86_64_PC32(RelocationRef R,uint64_t Value,uint64_t SecAddr)198 RelocToApply visitELF_X86_64_PC32(RelocationRef R, uint64_t Value, 199 uint64_t SecAddr) { 200 int64_t Addend = getAddend64LE(R); 201 uint64_t Address; 202 R.getOffset(Address); 203 return RelocToApply(Value + Addend - Address, 4); 204 } visitELF_X86_64_32(RelocationRef R,uint64_t Value)205 RelocToApply visitELF_X86_64_32(RelocationRef R, uint64_t Value) { 206 int64_t Addend = getAddend64LE(R); 207 uint32_t Res = (Value + Addend) & 0xFFFFFFFF; 208 return RelocToApply(Res, 4); 209 } visitELF_X86_64_32S(RelocationRef R,uint64_t Value)210 RelocToApply visitELF_X86_64_32S(RelocationRef R, uint64_t Value) { 211 int64_t Addend = getAddend64LE(R); 212 int32_t Res = (Value + Addend) & 0xFFFFFFFF; 213 return RelocToApply(Res, 4); 214 } 215 216 /// PPC64 ELF visitELF_PPC64_ADDR32(RelocationRef R,uint64_t Value)217 RelocToApply visitELF_PPC64_ADDR32(RelocationRef R, uint64_t Value) { 218 int64_t Addend = getAddend64BE(R); 219 uint32_t Res = (Value + Addend) & 0xFFFFFFFF; 220 return RelocToApply(Res, 4); 221 } visitELF_PPC64_ADDR64(RelocationRef R,uint64_t Value)222 RelocToApply visitELF_PPC64_ADDR64(RelocationRef R, uint64_t Value) { 223 int64_t Addend = getAddend64BE(R); 224 return RelocToApply(Value + Addend, 8); 225 } 226 227 /// PPC32 ELF visitELF_PPC_ADDR32(RelocationRef R,uint64_t Value)228 RelocToApply visitELF_PPC_ADDR32(RelocationRef R, uint64_t Value) { 229 int64_t Addend = getAddend32BE(R); 230 uint32_t Res = (Value + Addend) & 0xFFFFFFFF; 231 return RelocToApply(Res, 4); 232 } 233 234 /// MIPS ELF visitELF_MIPS_32(RelocationRef R,uint64_t Value)235 RelocToApply visitELF_MIPS_32(RelocationRef R, uint64_t Value) { 236 int64_t Addend; 237 getELFRelocationAddend(R, Addend); 238 uint32_t Res = (Value + Addend) & 0xFFFFFFFF; 239 return RelocToApply(Res, 4); 240 } 241 242 // AArch64 ELF visitELF_AARCH64_ABS32(RelocationRef R,uint64_t Value)243 RelocToApply visitELF_AARCH64_ABS32(RelocationRef R, uint64_t Value) { 244 int64_t Addend = getAddend64LE(R); 245 int64_t Res = Value + Addend; 246 247 // Overflow check allows for both signed and unsigned interpretation. 248 if (Res < INT32_MIN || Res > UINT32_MAX) 249 HasError = true; 250 251 return RelocToApply(static_cast<uint32_t>(Res), 4); 252 } 253 visitELF_AARCH64_ABS64(RelocationRef R,uint64_t Value)254 RelocToApply visitELF_AARCH64_ABS64(RelocationRef R, uint64_t Value) { 255 int64_t Addend = getAddend64LE(R); 256 return RelocToApply(Value + Addend, 8); 257 } 258 259 // SystemZ ELF visitELF_390_32(RelocationRef R,uint64_t Value)260 RelocToApply visitELF_390_32(RelocationRef R, uint64_t Value) { 261 int64_t Addend = getAddend64BE(R); 262 int64_t Res = Value + Addend; 263 264 // Overflow check allows for both signed and unsigned interpretation. 265 if (Res < INT32_MIN || Res > UINT32_MAX) 266 HasError = true; 267 268 return RelocToApply(static_cast<uint32_t>(Res), 4); 269 } 270 visitELF_390_64(RelocationRef R,uint64_t Value)271 RelocToApply visitELF_390_64(RelocationRef R, uint64_t Value) { 272 int64_t Addend = getAddend64BE(R); 273 return RelocToApply(Value + Addend, 8); 274 } 275 }; 276 277 } 278 } 279 #endif 280