blob: 67449eb6dcf938247ed5324015cb19401b7bf992 [file] [log] [blame]
Rafael Espindola95fb9b92015-06-02 20:38:46 +00001//===- lib/MC/MCSymbolELF.cpp ---------------------------------------------===//
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
Rafael Espindola95fb9b92015-06-02 20:38:46 +000010#include "llvm/MC/MCSymbolELF.h"
Zachary Turner264b5d92017-06-07 03:48:56 +000011#include "llvm/BinaryFormat/ELF.h"
Chandler Carruth6bda14b2017-06-06 11:49:48 +000012#include "llvm/MC/MCAssembler.h"
Rafael Espindola95fb9b92015-06-02 20:38:46 +000013#include "llvm/MC/MCFixupKindInfo.h"
Rafael Espindola95fb9b92015-06-02 20:38:46 +000014
15namespace llvm {
16
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000017namespace {
18enum {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000019 // Shift value for STT_* flags. 7 possible values. 3 bits.
20 ELF_STT_Shift = 0,
21
22 // Shift value for STB_* flags. 4 possible values, 2 bits.
23 ELF_STB_Shift = 3,
24
25 // Shift value for STV_* flags. 4 possible values, 2 bits.
26 ELF_STV_Shift = 5,
27
28 // Shift value for STO_* flags. 3 bits. All the values are between 0x20 and
29 // 0xe0, so we shift right by 5 before storing.
30 ELF_STO_Shift = 7,
31
32 // One bit.
33 ELF_IsSignature_Shift = 10,
34
35 // One bit.
36 ELF_WeakrefUsedInReloc_Shift = 11,
37
38 // One bit.
Rafael Espindoladfe2d352015-06-17 20:08:20 +000039 ELF_BindingSet_Shift = 12
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000040};
Alexander Kornienkof00654e2015-06-23 09:49:53 +000041}
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000042
Rafael Espindola95fb9b92015-06-02 20:38:46 +000043void MCSymbolELF::setBinding(unsigned Binding) const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000044 setIsBindingSet();
Rafael Espindola13a79bb2017-02-02 21:26:06 +000045 if (getType() == ELF::STT_SECTION && Binding != ELF::STB_LOCAL)
46 setType(ELF::STT_NOTYPE);
Rafael Espindola8c006ee2015-06-04 05:59:23 +000047 unsigned Val;
48 switch (Binding) {
49 default:
50 llvm_unreachable("Unsupported Binding");
51 case ELF::STB_LOCAL:
52 Val = 0;
53 break;
54 case ELF::STB_GLOBAL:
55 Val = 1;
56 break;
57 case ELF::STB_WEAK:
58 Val = 2;
59 break;
60 case ELF::STB_GNU_UNIQUE:
61 Val = 3;
62 break;
63 }
64 uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STB_Shift);
65 setFlags(OtherFlags | (Val << ELF_STB_Shift));
Rafael Espindola95fb9b92015-06-02 20:38:46 +000066}
67
68unsigned MCSymbolELF::getBinding() const {
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000069 if (isBindingSet()) {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000070 uint32_t Val = (getFlags() & (0x3 << ELF_STB_Shift)) >> ELF_STB_Shift;
71 switch (Val) {
72 default:
73 llvm_unreachable("Invalid value");
74 case 0:
75 return ELF::STB_LOCAL;
76 case 1:
77 return ELF::STB_GLOBAL;
78 case 2:
79 return ELF::STB_WEAK;
80 case 3:
81 return ELF::STB_GNU_UNIQUE;
82 }
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000083 }
84
85 if (isDefined())
86 return ELF::STB_LOCAL;
87 if (isUsedInReloc())
88 return ELF::STB_GLOBAL;
Rafael Espindola212fdde2015-06-03 21:52:06 +000089 if (isWeakrefUsedInReloc())
90 return ELF::STB_WEAK;
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000091 if (isSignature())
92 return ELF::STB_LOCAL;
93 return ELF::STB_GLOBAL;
Rafael Espindola95fb9b92015-06-02 20:38:46 +000094}
95
96void MCSymbolELF::setType(unsigned Type) const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000097 unsigned Val;
Rafael Espindola13a79bb2017-02-02 21:26:06 +000098 if (Type == ELF::STT_SECTION && getBinding() != ELF::STB_LOCAL)
99 return;
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000100 switch (Type) {
101 default:
102 llvm_unreachable("Unsupported Binding");
103 case ELF::STT_NOTYPE:
104 Val = 0;
105 break;
106 case ELF::STT_OBJECT:
107 Val = 1;
108 break;
109 case ELF::STT_FUNC:
110 Val = 2;
111 break;
112 case ELF::STT_SECTION:
113 Val = 3;
114 break;
115 case ELF::STT_COMMON:
116 Val = 4;
117 break;
118 case ELF::STT_TLS:
119 Val = 5;
120 break;
121 case ELF::STT_GNU_IFUNC:
122 Val = 6;
123 break;
124 }
125 uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STT_Shift);
126 setFlags(OtherFlags | (Val << ELF_STT_Shift));
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000127}
128
129unsigned MCSymbolELF::getType() const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000130 uint32_t Val = (getFlags() & (0x7 << ELF_STT_Shift)) >> ELF_STT_Shift;
131 switch (Val) {
132 default:
133 llvm_unreachable("Invalid value");
134 case 0:
135 return ELF::STT_NOTYPE;
136 case 1:
137 return ELF::STT_OBJECT;
138 case 2:
139 return ELF::STT_FUNC;
140 case 3:
141 return ELF::STT_SECTION;
142 case 4:
143 return ELF::STT_COMMON;
144 case 5:
145 return ELF::STT_TLS;
146 case 6:
147 return ELF::STT_GNU_IFUNC;
148 }
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000149}
150
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000151void MCSymbolELF::setVisibility(unsigned Visibility) {
152 assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
153 Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
154
155 uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STV_Shift);
156 setFlags(OtherFlags | (Visibility << ELF_STV_Shift));
157}
158
159unsigned MCSymbolELF::getVisibility() const {
160 unsigned Visibility = (getFlags() & (0x3 << ELF_STV_Shift)) >> ELF_STV_Shift;
161 assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
162 Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
163 return Visibility;
164}
165
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000166void MCSymbolELF::setOther(unsigned Other) {
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000167 assert((Other & 0x1f) == 0);
168 Other >>= 5;
169 assert(Other <= 0x7);
170 uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift);
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000171 setFlags(OtherFlags | (Other << ELF_STO_Shift));
172}
173
174unsigned MCSymbolELF::getOther() const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000175 unsigned Other = (getFlags() & (0x7 << ELF_STO_Shift)) >> ELF_STO_Shift;
176 return Other << 5;
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000177}
Rafael Espindolaada43f62015-06-03 21:30:10 +0000178
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000179void MCSymbolELF::setIsWeakrefUsedInReloc() const {
180 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift);
181 setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift));
182}
Rafael Espindola212fdde2015-06-03 21:52:06 +0000183
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000184bool MCSymbolELF::isWeakrefUsedInReloc() const {
185 return getFlags() & (0x1 << ELF_WeakrefUsedInReloc_Shift);
186}
Rafael Espindola212fdde2015-06-03 21:52:06 +0000187
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000188void MCSymbolELF::setIsSignature() const {
189 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_IsSignature_Shift);
190 setFlags(OtherFlags | (1 << ELF_IsSignature_Shift));
191}
Rafael Espindola8c52a9b2015-06-03 21:41:59 +0000192
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000193bool MCSymbolELF::isSignature() const {
194 return getFlags() & (0x1 << ELF_IsSignature_Shift);
195}
196
197void MCSymbolELF::setIsBindingSet() const {
198 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_BindingSet_Shift);
199 setFlags(OtherFlags | (1 << ELF_BindingSet_Shift));
200}
201
202bool MCSymbolELF::isBindingSet() const {
203 return getFlags() & (0x1 << ELF_BindingSet_Shift);
204}
Alexander Kornienkof00654e2015-06-23 09:49:53 +0000205}