205 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===- lib/MC/MCSymbolELF.cpp ---------------------------------------------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/MC/MCSymbolELF.h"
 | |
| #include "llvm/BinaryFormat/ELF.h"
 | |
| #include "llvm/MC/MCFixupKindInfo.h"
 | |
| 
 | |
| namespace llvm {
 | |
| 
 | |
| namespace {
 | |
| enum {
 | |
|   // Shift value for STT_* flags. 7 possible values. 3 bits.
 | |
|   ELF_STT_Shift = 0,
 | |
| 
 | |
|   // Shift value for STB_* flags. 4 possible values, 2 bits.
 | |
|   ELF_STB_Shift = 3,
 | |
| 
 | |
|   // Shift value for STV_* flags. 4 possible values, 2 bits.
 | |
|   ELF_STV_Shift = 5,
 | |
| 
 | |
|   // Shift value for STO_* flags. 3 bits. All the values are between 0x20 and
 | |
|   // 0xe0, so we shift right by 5 before storing.
 | |
|   ELF_STO_Shift = 7,
 | |
| 
 | |
|   // One bit.
 | |
|   ELF_IsSignature_Shift = 10,
 | |
| 
 | |
|   // One bit.
 | |
|   ELF_WeakrefUsedInReloc_Shift = 11,
 | |
| 
 | |
|   // One bit.
 | |
|   ELF_BindingSet_Shift = 12
 | |
| };
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setBinding(unsigned Binding) const {
 | |
|   setIsBindingSet();
 | |
|   if (getType() == ELF::STT_SECTION && Binding != ELF::STB_LOCAL)
 | |
|     setType(ELF::STT_NOTYPE);
 | |
|   unsigned Val;
 | |
|   switch (Binding) {
 | |
|   default:
 | |
|     llvm_unreachable("Unsupported Binding");
 | |
|   case ELF::STB_LOCAL:
 | |
|     Val = 0;
 | |
|     break;
 | |
|   case ELF::STB_GLOBAL:
 | |
|     Val = 1;
 | |
|     break;
 | |
|   case ELF::STB_WEAK:
 | |
|     Val = 2;
 | |
|     break;
 | |
|   case ELF::STB_GNU_UNIQUE:
 | |
|     Val = 3;
 | |
|     break;
 | |
|   }
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STB_Shift);
 | |
|   setFlags(OtherFlags | (Val << ELF_STB_Shift));
 | |
| }
 | |
| 
 | |
| unsigned MCSymbolELF::getBinding() const {
 | |
|   if (isBindingSet()) {
 | |
|     uint32_t Val = (getFlags() & (0x3 << ELF_STB_Shift)) >> ELF_STB_Shift;
 | |
|     switch (Val) {
 | |
|     default:
 | |
|       llvm_unreachable("Invalid value");
 | |
|     case 0:
 | |
|       return ELF::STB_LOCAL;
 | |
|     case 1:
 | |
|       return ELF::STB_GLOBAL;
 | |
|     case 2:
 | |
|       return ELF::STB_WEAK;
 | |
|     case 3:
 | |
|       return ELF::STB_GNU_UNIQUE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (isDefined())
 | |
|     return ELF::STB_LOCAL;
 | |
|   if (isUsedInReloc())
 | |
|     return ELF::STB_GLOBAL;
 | |
|   if (isWeakrefUsedInReloc())
 | |
|     return ELF::STB_WEAK;
 | |
|   if (isSignature())
 | |
|     return ELF::STB_LOCAL;
 | |
|   return ELF::STB_GLOBAL;
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setType(unsigned Type) const {
 | |
|   unsigned Val;
 | |
|   if (Type == ELF::STT_SECTION && getBinding() != ELF::STB_LOCAL)
 | |
|     return;
 | |
|   switch (Type) {
 | |
|   default:
 | |
|     llvm_unreachable("Unsupported Binding");
 | |
|   case ELF::STT_NOTYPE:
 | |
|     Val = 0;
 | |
|     break;
 | |
|   case ELF::STT_OBJECT:
 | |
|     Val = 1;
 | |
|     break;
 | |
|   case ELF::STT_FUNC:
 | |
|     Val = 2;
 | |
|     break;
 | |
|   case ELF::STT_SECTION:
 | |
|     Val = 3;
 | |
|     break;
 | |
|   case ELF::STT_COMMON:
 | |
|     Val = 4;
 | |
|     break;
 | |
|   case ELF::STT_TLS:
 | |
|     Val = 5;
 | |
|     break;
 | |
|   case ELF::STT_GNU_IFUNC:
 | |
|     Val = 6;
 | |
|     break;
 | |
|   }
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STT_Shift);
 | |
|   setFlags(OtherFlags | (Val << ELF_STT_Shift));
 | |
| }
 | |
| 
 | |
| unsigned MCSymbolELF::getType() const {
 | |
|   uint32_t Val = (getFlags() & (0x7 << ELF_STT_Shift)) >> ELF_STT_Shift;
 | |
|   switch (Val) {
 | |
|   default:
 | |
|     llvm_unreachable("Invalid value");
 | |
|   case 0:
 | |
|     return ELF::STT_NOTYPE;
 | |
|   case 1:
 | |
|     return ELF::STT_OBJECT;
 | |
|   case 2:
 | |
|     return ELF::STT_FUNC;
 | |
|   case 3:
 | |
|     return ELF::STT_SECTION;
 | |
|   case 4:
 | |
|     return ELF::STT_COMMON;
 | |
|   case 5:
 | |
|     return ELF::STT_TLS;
 | |
|   case 6:
 | |
|     return ELF::STT_GNU_IFUNC;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setVisibility(unsigned Visibility) {
 | |
|   assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
 | |
|          Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
 | |
| 
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STV_Shift);
 | |
|   setFlags(OtherFlags | (Visibility << ELF_STV_Shift));
 | |
| }
 | |
| 
 | |
| unsigned MCSymbolELF::getVisibility() const {
 | |
|   unsigned Visibility = (getFlags() & (0x3 << ELF_STV_Shift)) >> ELF_STV_Shift;
 | |
|   assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
 | |
|          Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
 | |
|   return Visibility;
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setOther(unsigned Other) {
 | |
|   assert((Other & 0x1f) == 0);
 | |
|   Other >>= 5;
 | |
|   assert(Other <= 0x7);
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift);
 | |
|   setFlags(OtherFlags | (Other << ELF_STO_Shift));
 | |
| }
 | |
| 
 | |
| unsigned MCSymbolELF::getOther() const {
 | |
|   unsigned Other = (getFlags() & (0x7 << ELF_STO_Shift)) >> ELF_STO_Shift;
 | |
|   return Other << 5;
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setIsWeakrefUsedInReloc() const {
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift);
 | |
|   setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift));
 | |
| }
 | |
| 
 | |
| bool MCSymbolELF::isWeakrefUsedInReloc() const {
 | |
|   return getFlags() & (0x1 << ELF_WeakrefUsedInReloc_Shift);
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setIsSignature() const {
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_IsSignature_Shift);
 | |
|   setFlags(OtherFlags | (1 << ELF_IsSignature_Shift));
 | |
| }
 | |
| 
 | |
| bool MCSymbolELF::isSignature() const {
 | |
|   return getFlags() & (0x1 << ELF_IsSignature_Shift);
 | |
| }
 | |
| 
 | |
| void MCSymbolELF::setIsBindingSet() const {
 | |
|   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_BindingSet_Shift);
 | |
|   setFlags(OtherFlags | (1 << ELF_BindingSet_Shift));
 | |
| }
 | |
| 
 | |
| bool MCSymbolELF::isBindingSet() const {
 | |
|   return getFlags() & (0x1 << ELF_BindingSet_Shift);
 | |
| }
 | |
| }
 |