blob: a07c56c64f8416dd5308513797b1f0472446baa6 [file] [log] [blame]
Rafael Espindola95fb9b92015-06-02 20:38:46 +00001//===- lib/MC/MCSymbolELF.cpp ---------------------------------------------===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Rafael Espindola95fb9b92015-06-02 20:38:46 +00006//
7//===----------------------------------------------------------------------===//
8
Rafael Espindola95fb9b92015-06-02 20:38:46 +00009#include "llvm/MC/MCSymbolELF.h"
Zachary Turner264b5d92017-06-07 03:48:56 +000010#include "llvm/BinaryFormat/ELF.h"
Rafael Espindola95fb9b92015-06-02 20:38:46 +000011#include "llvm/MC/MCFixupKindInfo.h"
Rafael Espindola95fb9b92015-06-02 20:38:46 +000012
13namespace llvm {
14
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000015namespace {
16enum {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000017 // Shift value for STT_* flags. 7 possible values. 3 bits.
18 ELF_STT_Shift = 0,
19
20 // Shift value for STB_* flags. 4 possible values, 2 bits.
21 ELF_STB_Shift = 3,
22
23 // Shift value for STV_* flags. 4 possible values, 2 bits.
24 ELF_STV_Shift = 5,
25
26 // Shift value for STO_* flags. 3 bits. All the values are between 0x20 and
27 // 0xe0, so we shift right by 5 before storing.
28 ELF_STO_Shift = 7,
29
30 // One bit.
31 ELF_IsSignature_Shift = 10,
32
33 // One bit.
34 ELF_WeakrefUsedInReloc_Shift = 11,
35
36 // One bit.
Rafael Espindoladfe2d352015-06-17 20:08:20 +000037 ELF_BindingSet_Shift = 12
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000038};
Alexander Kornienkof00654e2015-06-23 09:49:53 +000039}
Rafael Espindolaf8794ff2015-06-04 00:47:43 +000040
Rafael Espindola95fb9b92015-06-02 20:38:46 +000041void MCSymbolELF::setBinding(unsigned Binding) const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000042 setIsBindingSet();
Rafael Espindola13a79bb2017-02-02 21:26:06 +000043 if (getType() == ELF::STT_SECTION && Binding != ELF::STB_LOCAL)
44 setType(ELF::STT_NOTYPE);
Rafael Espindola8c006ee2015-06-04 05:59:23 +000045 unsigned Val;
46 switch (Binding) {
47 default:
48 llvm_unreachable("Unsupported Binding");
49 case ELF::STB_LOCAL:
50 Val = 0;
51 break;
52 case ELF::STB_GLOBAL:
53 Val = 1;
54 break;
55 case ELF::STB_WEAK:
56 Val = 2;
57 break;
58 case ELF::STB_GNU_UNIQUE:
59 Val = 3;
60 break;
61 }
62 uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STB_Shift);
63 setFlags(OtherFlags | (Val << ELF_STB_Shift));
Rafael Espindola95fb9b92015-06-02 20:38:46 +000064}
65
66unsigned MCSymbolELF::getBinding() const {
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000067 if (isBindingSet()) {
Fangrui Songbbd798f2019-04-10 10:30:22 +000068 uint32_t Val = (Flags >> ELF_STB_Shift) & 3;
Rafael Espindola8c006ee2015-06-04 05:59:23 +000069 switch (Val) {
70 default:
71 llvm_unreachable("Invalid value");
72 case 0:
73 return ELF::STB_LOCAL;
74 case 1:
75 return ELF::STB_GLOBAL;
76 case 2:
77 return ELF::STB_WEAK;
78 case 3:
79 return ELF::STB_GNU_UNIQUE;
80 }
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000081 }
82
83 if (isDefined())
84 return ELF::STB_LOCAL;
85 if (isUsedInReloc())
86 return ELF::STB_GLOBAL;
Rafael Espindola212fdde2015-06-03 21:52:06 +000087 if (isWeakrefUsedInReloc())
88 return ELF::STB_WEAK;
Rafael Espindola8c52a9b2015-06-03 21:41:59 +000089 if (isSignature())
90 return ELF::STB_LOCAL;
91 return ELF::STB_GLOBAL;
Rafael Espindola95fb9b92015-06-02 20:38:46 +000092}
93
94void MCSymbolELF::setType(unsigned Type) const {
Rafael Espindola8c006ee2015-06-04 05:59:23 +000095 unsigned Val;
Rafael Espindola13a79bb2017-02-02 21:26:06 +000096 if (Type == ELF::STT_SECTION && getBinding() != ELF::STB_LOCAL)
97 return;
Rafael Espindola8c006ee2015-06-04 05:59:23 +000098 switch (Type) {
99 default:
100 llvm_unreachable("Unsupported Binding");
101 case ELF::STT_NOTYPE:
102 Val = 0;
103 break;
104 case ELF::STT_OBJECT:
105 Val = 1;
106 break;
107 case ELF::STT_FUNC:
108 Val = 2;
109 break;
110 case ELF::STT_SECTION:
111 Val = 3;
112 break;
113 case ELF::STT_COMMON:
114 Val = 4;
115 break;
116 case ELF::STT_TLS:
117 Val = 5;
118 break;
119 case ELF::STT_GNU_IFUNC:
120 Val = 6;
121 break;
122 }
123 uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STT_Shift);
124 setFlags(OtherFlags | (Val << ELF_STT_Shift));
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000125}
126
127unsigned MCSymbolELF::getType() const {
Fangrui Songbbd798f2019-04-10 10:30:22 +0000128 uint32_t Val = (Flags >> ELF_STT_Shift) & 7;
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000129 switch (Val) {
130 default:
131 llvm_unreachable("Invalid value");
132 case 0:
133 return ELF::STT_NOTYPE;
134 case 1:
135 return ELF::STT_OBJECT;
136 case 2:
137 return ELF::STT_FUNC;
138 case 3:
139 return ELF::STT_SECTION;
140 case 4:
141 return ELF::STT_COMMON;
142 case 5:
143 return ELF::STT_TLS;
144 case 6:
145 return ELF::STT_GNU_IFUNC;
146 }
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000147}
148
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000149void MCSymbolELF::setVisibility(unsigned Visibility) {
150 assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
151 Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
152
153 uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STV_Shift);
154 setFlags(OtherFlags | (Visibility << ELF_STV_Shift));
155}
156
157unsigned MCSymbolELF::getVisibility() const {
Fangrui Songbbd798f2019-04-10 10:30:22 +0000158 unsigned Visibility = (Flags >> ELF_STV_Shift) & 3;
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000159 return Visibility;
160}
161
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000162void MCSymbolELF::setOther(unsigned Other) {
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000163 assert((Other & 0x1f) == 0);
164 Other >>= 5;
165 assert(Other <= 0x7);
166 uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift);
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000167 setFlags(OtherFlags | (Other << ELF_STO_Shift));
168}
169
170unsigned MCSymbolELF::getOther() const {
Fangrui Songbbd798f2019-04-10 10:30:22 +0000171 unsigned Other = (Flags >> ELF_STO_Shift) & 7;
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000172 return Other << 5;
Rafael Espindola95fb9b92015-06-02 20:38:46 +0000173}
Rafael Espindolaada43f62015-06-03 21:30:10 +0000174
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000175void MCSymbolELF::setIsWeakrefUsedInReloc() const {
176 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift);
177 setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift));
178}
Rafael Espindola212fdde2015-06-03 21:52:06 +0000179
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000180bool MCSymbolELF::isWeakrefUsedInReloc() const {
181 return getFlags() & (0x1 << ELF_WeakrefUsedInReloc_Shift);
182}
Rafael Espindola212fdde2015-06-03 21:52:06 +0000183
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000184void MCSymbolELF::setIsSignature() const {
185 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_IsSignature_Shift);
186 setFlags(OtherFlags | (1 << ELF_IsSignature_Shift));
187}
Rafael Espindola8c52a9b2015-06-03 21:41:59 +0000188
Rafael Espindola8c006ee2015-06-04 05:59:23 +0000189bool MCSymbolELF::isSignature() const {
190 return getFlags() & (0x1 << ELF_IsSignature_Shift);
191}
192
193void MCSymbolELF::setIsBindingSet() const {
194 uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_BindingSet_Shift);
195 setFlags(OtherFlags | (1 << ELF_BindingSet_Shift));
196}
197
198bool MCSymbolELF::isBindingSet() const {
199 return getFlags() & (0x1 << ELF_BindingSet_Shift);
200}
Alexander Kornienkof00654e2015-06-23 09:49:53 +0000201}