]> CyberLeo.Net >> Repos - FreeBSD/stable/9.git/blob - contrib/llvm/include/llvm/Object/RelocVisitor.h
MFC r244628:
[FreeBSD/stable/9.git] / contrib / llvm / include / llvm / Object / RelocVisitor.h
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
17 #define _LLVM_OBJECT_RELOCVISITOR
18
19 #include "llvm/Support/Debug.h"
20 #include "llvm/Support/raw_ostream.h"
21 #include "llvm/Object/ObjectFile.h"
22 #include "llvm/Object/ELF.h"
23 #include "llvm/ADT/StringRef.h"
24
25 namespace llvm {
26 namespace object {
27
28 struct RelocToApply {
29   // The computed value after applying the relevant relocations.
30   int64_t Value;
31
32   // The width of the value; how many bytes to touch when applying the
33   // relocation.
34   char Width;
35   RelocToApply(const RelocToApply &In) : Value(In.Value), Width(In.Width) {}
36   RelocToApply(int64_t Value, char Width) : Value(Value), Width(Width) {}
37   RelocToApply() : Value(0), Width(0) {}
38 };
39
40 /// @brief Base class for object file relocation visitors.
41 class RelocVisitor {
42 public:
43   explicit RelocVisitor(llvm::StringRef FileFormat)
44     : FileFormat(FileFormat), HasError(false) {}
45
46   // TODO: Should handle multiple applied relocations via either passing in the
47   // previously computed value or just count paired relocations as a single
48   // visit.
49   RelocToApply visit(uint32_t RelocType, RelocationRef R, uint64_t SecAddr = 0,
50                      uint64_t Value = 0) {
51     if (FileFormat == "ELF64-x86-64") {
52       switch (RelocType) {
53         case llvm::ELF::R_X86_64_NONE:
54           return visitELF_X86_64_NONE(R);
55         case llvm::ELF::R_X86_64_64:
56           return visitELF_X86_64_64(R, Value);
57         case llvm::ELF::R_X86_64_PC32:
58           return visitELF_X86_64_PC32(R, Value, SecAddr);
59         case llvm::ELF::R_X86_64_32:
60           return visitELF_X86_64_32(R, Value);
61         case llvm::ELF::R_X86_64_32S:
62           return visitELF_X86_64_32S(R, Value);
63         default:
64           HasError = true;
65           return RelocToApply();
66       }
67     }
68     return RelocToApply();
69   }
70
71   bool error() { return HasError; }
72
73 private:
74   llvm::StringRef FileFormat;
75   bool HasError;
76
77   /// Operations
78
79   // Width is the width in bytes of the extend.
80   RelocToApply zeroExtend(RelocToApply r, char Width) {
81     if (Width == r.Width)
82       return r;
83     r.Value &= (1 << ((Width * 8))) - 1;
84     return r;
85   }
86   RelocToApply signExtend(RelocToApply r, char Width) {
87     if (Width == r.Width)
88       return r;
89     bool SignBit = r.Value & (1 << ((Width * 8) - 1));
90     if (SignBit) {
91       r.Value |= ~((1 << (Width * 8)) - 1);
92     } else {
93       r.Value &= (1 << (Width * 8)) - 1;
94     }
95     return r;
96   }
97
98   /// X86-64 ELF
99   RelocToApply visitELF_X86_64_NONE(RelocationRef R) {
100     return RelocToApply(0, 0);
101   }
102   RelocToApply visitELF_X86_64_64(RelocationRef R, uint64_t Value) {
103     int64_t Addend;
104     R.getAdditionalInfo(Addend);
105     return RelocToApply(Value + Addend, 8);
106   }
107   RelocToApply visitELF_X86_64_PC32(RelocationRef R, uint64_t Value,
108                                     uint64_t SecAddr) {
109     int64_t Addend;
110     R.getAdditionalInfo(Addend);
111     uint64_t Address;
112     R.getAddress(Address);
113     return RelocToApply(Value + Addend - Address, 4);
114   }
115   RelocToApply visitELF_X86_64_32(RelocationRef R, uint64_t Value) {
116     int64_t Addend;
117     R.getAdditionalInfo(Addend);
118     uint32_t Res = (Value + Addend) & 0xFFFFFFFF;
119     return RelocToApply(Res, 4);
120   }
121   RelocToApply visitELF_X86_64_32S(RelocationRef R, uint64_t Value) {
122     int64_t Addend;
123     R.getAdditionalInfo(Addend);
124     int32_t Res = (Value + Addend) & 0xFFFFFFFF;
125     return RelocToApply(Res, 4);
126   }
127 };
128
129 }
130 }
131 #endif