| //===- SectionMap.cpp -----------------------------------------------------===// |
| // |
| // The MCLinker Project |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| #include <cassert> |
| #include <cstring> |
| #include <mcld/LD/SectionMap.h> |
| |
| using namespace mcld; |
| |
| //===----------------------------------------------------------------------===// |
| // SectionMap |
| SectionMap::SectionMap() |
| { |
| } |
| |
| SectionMap::~SectionMap() |
| { |
| } |
| |
| const std::string& SectionMap::getOutputSectName(const std::string& pInput) |
| { |
| iterator it; |
| for (it = begin(); it != end(); ++it) { |
| if (0 == strncmp(pInput.c_str(), |
| (*it).inputSubStr.c_str(), |
| (*it).inputSubStr.length())) |
| break; |
| // wildcard to a user-defined output section. |
| else if (0 == strcmp("*", (*it).inputSubStr.c_str())) |
| break; |
| } |
| // if still no matching, just let a output seciton has the same input name |
| if (it == end()) |
| return pInput; |
| return (*it).outputStr; |
| } |
| |
| bool SectionMap::push_back(const std::string& pInput, |
| const std::string& pOutput) |
| { |
| // Now only check if the mapping exists in the map already |
| // TODO: handle the cases such as overriding the exist mapping and drawing |
| // exception from the given SECTIONS command |
| iterator it; |
| for (it = m_SectMap.begin(); it != m_SectMap.end(); ++it) { |
| if (pInput == (*it).inputSubStr) |
| return false; |
| } |
| struct Mapping mapping = { |
| pInput, |
| pOutput, |
| }; |
| m_SectMap.push_back(mapping); |
| return true; |
| } |
| |
| SectionMap::iterator SectionMap::find(const std::string& pInput) |
| { |
| iterator it; |
| for (it = begin(); it != end(); ++it) { |
| if(pInput == (*it).inputSubStr) |
| break; |
| } |
| return it; |
| } |
| |
| SectionMap::Mapping* SectionMap::at(const std::string& pInput) |
| { |
| iterator it; |
| for (it = begin(); it != end(); ++it) { |
| if(pInput == (*it).inputSubStr) |
| break; |
| } |
| if (end() == it) |
| return NULL; |
| return &(*it); |
| } |
| |
| // Common mappings of ELF and other formants. Now only ELF specific mappings are added |
| const SectionMap::SectionNameMapping SectionMap::m_StdSectionMap[] = |
| { |
| {".text", ".text"}, |
| {".rodata", ".rodata"}, |
| {".data.rel.ro.local", ".data.rel.ro.local"}, |
| {".data.rel.ro", ".data.rel.ro"}, |
| {".data", ".data"}, |
| {".bss", ".bss"}, |
| {".tdata", ".tdata"}, |
| {".tbss", ".tbss"}, |
| {".init_array", ".init_array"}, |
| {".fini_array", ".fini_array"}, |
| // TODO: Support DT_INIT_ARRAY for all constructors? |
| {".ctors", ".ctors"}, |
| {".dtors", ".dtors"}, |
| {".sdata", ".sdata"}, |
| {".sbss", ".sbss"}, |
| // FIXME: in GNU ld, if we are creating a shared object .sdata2 and .sbss2 |
| // sections would be handled differently. |
| {".sdata2", ".sdata"}, |
| {".sbss2", ".sbss"}, |
| {".lrodata", ".lrodata"}, |
| {".ldata", ".ldata"}, |
| {".lbss", ".lbss"}, |
| {".gcc_except_table", ".gcc_except_table"}, |
| {".gnu.linkonce.d.rel.ro.local", ".data.rel.ro.local"}, |
| {".gnu.linkonce.d.rel.ro", ".data.rel.ro"}, |
| {".gnu.linkonce.t", ".text"}, |
| {".gnu.linkonce.r", ".rodata"}, |
| {".gnu.linkonce.d", ".data"}, |
| {".gnu.linkonce.b", ".bss"}, |
| {".gnu.linkonce.s", ".sdata"}, |
| {".gnu.linkonce.sb", ".sbss"}, |
| {".gnu.linkonce.s2", ".sdata"}, |
| {".gnu.linkonce.sb2", ".sbss"}, |
| {".gnu.linkonce.wi", ".debug_info"}, |
| {".gnu.linkonce.td", ".tdata"}, |
| {".gnu.linkonce.tb", ".tbss"}, |
| {".gnu.linkonce.lr", ".lrodata"}, |
| {".gnu.linkonce.l", ".ldata"}, |
| {".gnu.linkonce.lb", ".lbss"}, |
| }; |
| |
| const int SectionMap::m_StdSectionMapSize = |
| (sizeof(SectionMap::m_StdSectionMap) / sizeof(SectionMap::m_StdSectionMap[0])); |
| |
| bool SectionMap::initStdSectionMap() |
| { |
| for (int i = 0; i < m_StdSectionMapSize; ++i) { |
| struct Mapping mapping = { m_StdSectionMap[i].from, m_StdSectionMap[i].to}; |
| m_SectMap.push_back(mapping); |
| } |
| return true; |
| } |