|           Line data    Source code 
       1             : //===- 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             : 
      10             : #include "llvm/MC/MCSymbolELF.h"
      11             : #include "llvm/BinaryFormat/ELF.h"
      12             : #include "llvm/MC/MCFixupKindInfo.h"
      13             : 
      14             : namespace llvm {
      15             : 
      16             : namespace {
      17             : enum {
      18             :   // Shift value for STT_* flags. 7 possible values. 3 bits.
      19             :   ELF_STT_Shift = 0,
      20             : 
      21             :   // Shift value for STB_* flags. 4 possible values, 2 bits.
      22             :   ELF_STB_Shift = 3,
      23             : 
      24             :   // Shift value for STV_* flags. 4 possible values, 2 bits.
      25             :   ELF_STV_Shift = 5,
      26             : 
      27             :   // Shift value for STO_* flags. 3 bits. All the values are between 0x20 and
      28             :   // 0xe0, so we shift right by 5 before storing.
      29             :   ELF_STO_Shift = 7,
      30             : 
      31             :   // One bit.
      32             :   ELF_IsSignature_Shift = 10,
      33             : 
      34             :   // One bit.
      35             :   ELF_WeakrefUsedInReloc_Shift = 11,
      36             : 
      37             :   // One bit.
      38             :   ELF_BindingSet_Shift = 12
      39             : };
      40             : }
      41             : 
      42     2993294 : void MCSymbolELF::setBinding(unsigned Binding) const {
      43     2993294 :   setIsBindingSet();
      44     2993295 :   if (getType() == ELF::STT_SECTION && Binding != ELF::STB_LOCAL)
      45           1 :     setType(ELF::STT_NOTYPE);
      46             :   unsigned Val;
      47     2993295 :   switch (Binding) {
      48           0 :   default:
      49           0 :     llvm_unreachable("Unsupported Binding");
      50             :   case ELF::STB_LOCAL:
      51             :     Val = 0;
      52             :     break;
      53       36727 :   case ELF::STB_GLOBAL:
      54             :     Val = 1;
      55       36727 :     break;
      56      214427 :   case ELF::STB_WEAK:
      57             :     Val = 2;
      58      214427 :     break;
      59           2 :   case ELF::STB_GNU_UNIQUE:
      60             :     Val = 3;
      61           2 :     break;
      62             :   }
      63     2993295 :   uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STB_Shift);
      64     2993295 :   setFlags(OtherFlags | (Val << ELF_STB_Shift));
      65     2993295 : }
      66             : 
      67    12395306 : unsigned MCSymbolELF::getBinding() const {
      68    12395306 :   if (isBindingSet()) {
      69     6471070 :     uint32_t Val = (getFlags() & (0x3 << ELF_STB_Shift)) >> ELF_STB_Shift;
      70     6471070 :     switch (Val) {
      71           0 :     default:
      72           0 :       llvm_unreachable("Invalid value");
      73             :     case 0:
      74             :       return ELF::STB_LOCAL;
      75      182355 :     case 1:
      76      182355 :       return ELF::STB_GLOBAL;
      77     3041613 :     case 2:
      78     3041613 :       return ELF::STB_WEAK;
      79           4 :     case 3:
      80           4 :       return ELF::STB_GNU_UNIQUE;
      81             :     }
      82             :   }
      83             : 
      84     5924236 :   if (isDefined())
      85             :     return ELF::STB_LOCAL;
      86      170409 :   if (isUsedInReloc())
      87             :     return ELF::STB_GLOBAL;
      88         459 :   if (isWeakrefUsedInReloc())
      89             :     return ELF::STB_WEAK;
      90         447 :   if (isSignature())
      91         414 :     return ELF::STB_LOCAL;
      92             :   return ELF::STB_GLOBAL;
      93             : }
      94             : 
      95     3220241 : void MCSymbolELF::setType(unsigned Type) const {
      96             :   unsigned Val;
      97     3220241 :   if (Type == ELF::STT_SECTION && getBinding() != ELF::STB_LOCAL)
      98             :     return;
      99     3220241 :   switch (Type) {
     100           0 :   default:
     101           0 :     llvm_unreachable("Unsupported Binding");
     102             :   case ELF::STT_NOTYPE:
     103             :     Val = 0;
     104             :     break;
     105      265162 :   case ELF::STT_OBJECT:
     106             :     Val = 1;
     107      265162 :     break;
     108      214320 :   case ELF::STT_FUNC:
     109             :     Val = 2;
     110      214320 :     break;
     111     2736489 :   case ELF::STT_SECTION:
     112             :     Val = 3;
     113     2736489 :     break;
     114           0 :   case ELF::STT_COMMON:
     115             :     Val = 4;
     116           0 :     break;
     117        1685 :   case ELF::STT_TLS:
     118             :     Val = 5;
     119        1685 :     break;
     120         335 :   case ELF::STT_GNU_IFUNC:
     121             :     Val = 6;
     122         335 :     break;
     123             :   }
     124     3220241 :   uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STT_Shift);
     125     3220241 :   setFlags(OtherFlags | (Val << ELF_STT_Shift));
     126             : }
     127             : 
     128    21379042 : unsigned MCSymbolELF::getType() const {
     129             :   uint32_t Val = (getFlags() & (0x7 << ELF_STT_Shift)) >> ELF_STT_Shift;
     130             :   switch (Val) {
     131           0 :   default:
     132           0 :     llvm_unreachable("Invalid value");
     133             :   case 0:
     134             :     return ELF::STT_NOTYPE;
     135             :   case 1:
     136             :     return ELF::STT_OBJECT;
     137             :   case 2:
     138             :     return ELF::STT_FUNC;
     139             :   case 3:
     140             :     return ELF::STT_SECTION;
     141             :   case 4:
     142             :     return ELF::STT_COMMON;
     143             :   case 5:
     144             :     return ELF::STT_TLS;
     145             :   case 6:
     146             :     return ELF::STT_GNU_IFUNC;
     147             :   }
     148             : }
     149             : 
     150        9735 : void MCSymbolELF::setVisibility(unsigned Visibility) {
     151             :   assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
     152             :          Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
     153             : 
     154        9735 :   uint32_t OtherFlags = getFlags() & ~(0x3 << ELF_STV_Shift);
     155        9735 :   setFlags(OtherFlags | (Visibility << ELF_STV_Shift));
     156        9735 : }
     157             : 
     158      695221 : unsigned MCSymbolELF::getVisibility() const {
     159      695221 :   unsigned Visibility = (getFlags() & (0x3 << ELF_STV_Shift)) >> ELF_STV_Shift;
     160             :   assert(Visibility == ELF::STV_DEFAULT || Visibility == ELF::STV_INTERNAL ||
     161             :          Visibility == ELF::STV_HIDDEN || Visibility == ELF::STV_PROTECTED);
     162      695221 :   return Visibility;
     163             : }
     164             : 
     165         258 : void MCSymbolELF::setOther(unsigned Other) {
     166             :   assert((Other & 0x1f) == 0);
     167         258 :   Other >>= 5;
     168             :   assert(Other <= 0x7);
     169         258 :   uint32_t OtherFlags = getFlags() & ~(0x7 << ELF_STO_Shift);
     170         258 :   setFlags(OtherFlags | (Other << ELF_STO_Shift));
     171         258 : }
     172             : 
     173      696356 : unsigned MCSymbolELF::getOther() const {
     174      696356 :   unsigned Other = (getFlags() & (0x7 << ELF_STO_Shift)) >> ELF_STO_Shift;
     175      696356 :   return Other << 5;
     176             : }
     177             : 
     178          12 : void MCSymbolELF::setIsWeakrefUsedInReloc() const {
     179          12 :   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_WeakrefUsedInReloc_Shift);
     180          12 :   setFlags(OtherFlags | (1 << ELF_WeakrefUsedInReloc_Shift));
     181          12 : }
     182             : 
     183     8546707 : bool MCSymbolELF::isWeakrefUsedInReloc() const {
     184     8546707 :   return getFlags() & (0x1 << ELF_WeakrefUsedInReloc_Shift);
     185             : }
     186             : 
     187      587773 : void MCSymbolELF::setIsSignature() const {
     188      587773 :   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_IsSignature_Shift);
     189      587773 :   setFlags(OtherFlags | (1 << ELF_IsSignature_Shift));
     190      587773 : }
     191             : 
     192     8546696 : bool MCSymbolELF::isSignature() const {
     193     8546696 :   return getFlags() & (0x1 << ELF_IsSignature_Shift);
     194             : }
     195             : 
     196     2993295 : void MCSymbolELF::setIsBindingSet() const {
     197     2993295 :   uint32_t OtherFlags = getFlags() & ~(0x1 << ELF_BindingSet_Shift);
     198     2993295 :   setFlags(OtherFlags | (1 << ELF_BindingSet_Shift));
     199     2993295 : }
     200             : 
     201    12399567 : bool MCSymbolELF::isBindingSet() const {
     202    12399567 :   return getFlags() & (0x1 << ELF_BindingSet_Shift);
     203             : }
     204             : }
 |