Home | History | Annotate | Download | only in Hexagon
      1 //===- HexagonPLT.cpp -----------------------------------------------------===//
      2 //
      3 //                     The MCLinker Project
      4 //
      5 // This file is distributed under the University of Illinois Open Source
      6 // License. See LICENSE.TXT for details.
      7 //
      8 //===----------------------------------------------------------------------===//
      9 #include "HexagonPLT.h"
     10 #include "HexagonRelocationFunctions.h"
     11 
     12 #include "mcld/LD/LDSection.h"
     13 #include "mcld/LinkerConfig.h"
     14 #include "mcld/Support/MsgHandling.h"
     15 
     16 #include <llvm/Support/ELF.h>
     17 #include <llvm/Support/Casting.h>
     18 
     19 namespace mcld {
     20 
     21 //===----------------------------------------------------------------------===//
     22 // PLT entry data
     23 //===----------------------------------------------------------------------===//
     24 HexagonPLT0::HexagonPLT0(SectionData& pParent)
     25     : PLT::Entry<sizeof(hexagon_plt0)>(pParent) {
     26 }
     27 
     28 HexagonPLT1::HexagonPLT1(SectionData& pParent)
     29     : PLT::Entry<sizeof(hexagon_plt1)>(pParent) {
     30 }
     31 
     32 //===----------------------------------------------------------------------===//
     33 // HexagonPLT
     34 //===----------------------------------------------------------------------===//
     35 HexagonPLT::HexagonPLT(LDSection& pSection,
     36                        HexagonGOTPLT& pGOTPLT,
     37                        const LinkerConfig& pConfig)
     38     : PLT(pSection), m_GOTPLT(pGOTPLT) {
     39   assert(LinkerConfig::DynObj == pConfig.codeGenType() ||
     40          LinkerConfig::Exec == pConfig.codeGenType() ||
     41          LinkerConfig::Binary == pConfig.codeGenType());
     42 
     43   m_PLT0 = hexagon_plt0;
     44   m_PLT0Size = sizeof(hexagon_plt0);
     45   // create PLT0
     46   new HexagonPLT0(*m_pSectionData);
     47   pSection.setAlign(16);
     48 }
     49 
     50 HexagonPLT::~HexagonPLT() {
     51 }
     52 
     53 PLTEntryBase* HexagonPLT::getPLT0() const {
     54   iterator first = m_pSectionData->getFragmentList().begin();
     55 
     56   assert(first != m_pSectionData->getFragmentList().end() &&
     57          "FragmentList is empty, getPLT0 failed!");
     58 
     59   PLTEntryBase* plt0 = &(llvm::cast<PLTEntryBase>(*first));
     60 
     61   return plt0;
     62 }
     63 
     64 void HexagonPLT::finalizeSectionSize() {
     65   uint64_t size = 0;
     66   // plt0 size
     67   size = getPLT0()->size();
     68 
     69   // get first plt1 entry
     70   HexagonPLT::iterator it = begin();
     71   ++it;
     72   if (end() != it) {
     73     // plt1 size
     74     PLTEntryBase* plt1 = &(llvm::cast<PLTEntryBase>(*it));
     75     size += (m_pSectionData->size() - 1) * plt1->size();
     76   }
     77   m_Section.setSize(size);
     78 
     79   uint32_t offset = 0;
     80   SectionData::iterator frag, fragEnd = m_pSectionData->end();
     81   for (frag = m_pSectionData->begin(); frag != fragEnd; ++frag) {
     82     frag->setOffset(offset);
     83     offset += frag->size();
     84   }
     85 }
     86 
     87 bool HexagonPLT::hasPLT1() const {
     88   return (m_pSectionData->size() > 1);
     89 }
     90 
     91 HexagonPLT1* HexagonPLT::create() {
     92   return new HexagonPLT1(*m_pSectionData);
     93 }
     94 
     95 void HexagonPLT::applyPLT0() {
     96   PLTEntryBase* plt0 = getPLT0();
     97   uint64_t pltBase = m_Section.addr();
     98 
     99   unsigned char* data = 0;
    100   data = static_cast<unsigned char*>(malloc(plt0->size()));
    101 
    102   if (!data)
    103     fatal(diag::fail_allocate_memory_plt);
    104 
    105   memcpy(data, m_PLT0, plt0->size());
    106   uint32_t gotpltAddr = m_GOTPLT.addr();
    107 
    108   int32_t* dest = reinterpret_cast<int32_t*>(data);
    109   int32_t result = ((gotpltAddr - pltBase) >> 6);
    110   *dest |= ApplyMask<int32_t>(0xfff3fff, result);
    111   dest = dest + 1;
    112   // Already calculated using pltBase
    113   result = (gotpltAddr - pltBase);
    114   *(dest) |= ApplyMask<int32_t>(0x1f80, result);
    115 
    116   plt0->setValue(data);
    117 }
    118 
    119 void HexagonPLT::applyPLT1() {
    120   uint64_t plt_base = m_Section.addr();
    121   assert(plt_base && ".plt base address is NULL!");
    122 
    123   uint64_t got_base = m_GOTPLT.addr();
    124   assert(got_base && ".got base address is NULL!");
    125 
    126   HexagonPLT::iterator it = m_pSectionData->begin();
    127   HexagonPLT::iterator ie = m_pSectionData->end();
    128   assert(it != ie && "FragmentList is empty, applyPLT1 failed!");
    129 
    130   uint32_t GOTEntrySize = HexagonGOTEntry::EntrySize;
    131   uint32_t GOTEntryAddress = got_base + GOTEntrySize * 4;
    132 
    133   uint64_t PLTEntryAddress =
    134       plt_base + HexagonPLT0::EntrySize;  // Offset of PLT0
    135 
    136   ++it;  // skip PLT0
    137   uint64_t PLT1EntrySize = HexagonPLT1::EntrySize;
    138   HexagonPLT1* plt1 = NULL;
    139 
    140   uint32_t* Out = NULL;
    141   while (it != ie) {
    142     plt1 = &(llvm::cast<HexagonPLT1>(*it));
    143     Out = static_cast<uint32_t*>(malloc(HexagonPLT1::EntrySize));
    144 
    145     if (!Out)
    146       fatal(diag::fail_allocate_memory_plt);
    147 
    148     memcpy(Out, hexagon_plt1, plt1->size());
    149 
    150     int32_t* dest = reinterpret_cast<int32_t*>(Out);
    151     int32_t result = ((GOTEntryAddress - PLTEntryAddress) >> 6);
    152     *dest |= ApplyMask<int32_t>(0xfff3fff, result);
    153     dest = dest + 1;
    154     result = (GOTEntryAddress - PLTEntryAddress);
    155     *(dest) |= ApplyMask<int32_t>(0x1f80, result);
    156 
    157     // Address in the PLT entries point to the corresponding GOT entries
    158     // TODO: Fixup plt to point to the corresponding GOTEntryAddress
    159     // We need to borrow the same relocation code to fix the relocation
    160     plt1->setValue(reinterpret_cast<unsigned char*>(Out));
    161     ++it;
    162 
    163     GOTEntryAddress += GOTEntrySize;
    164     PLTEntryAddress += PLT1EntrySize;
    165   }
    166 }
    167 
    168 uint64_t HexagonPLT::emit(MemoryRegion& pRegion) {
    169   uint64_t result = 0x0;
    170   iterator it = begin();
    171 
    172   unsigned char* buffer = pRegion.begin();
    173   memcpy(buffer,
    174          llvm::cast<HexagonPLT0>((*it)).getValue(),
    175          HexagonPLT0::EntrySize);
    176   result += HexagonPLT0::EntrySize;
    177   ++it;
    178 
    179   HexagonPLT1* plt1 = 0;
    180   HexagonPLT::iterator ie = end();
    181   while (it != ie) {
    182     plt1 = &(llvm::cast<HexagonPLT1>(*it));
    183     memcpy(buffer + result, plt1->getValue(), HexagonPLT1::EntrySize);
    184     result += HexagonPLT1::EntrySize;
    185     ++it;
    186   }
    187   return result;
    188 }
    189 
    190 }  // namespace mcld
    191