|  | // elfcpp_file.h -- file access for elfcpp   -*- C++ -*- | 
|  |  | 
|  | // Copyright (C) 2006-2023 Free Software Foundation, Inc. | 
|  | // Written by Ian Lance Taylor <iant@google.com>. | 
|  |  | 
|  | // This file is part of elfcpp. | 
|  |  | 
|  | // This program is free software; you can redistribute it and/or | 
|  | // modify it under the terms of the GNU Library General Public License | 
|  | // as published by the Free Software Foundation; either version 2, or | 
|  | // (at your option) any later version. | 
|  |  | 
|  | // In addition to the permissions in the GNU Library General Public | 
|  | // License, the Free Software Foundation gives you unlimited | 
|  | // permission to link the compiled version of this file into | 
|  | // combinations with other programs, and to distribute those | 
|  | // combinations without any restriction coming from the use of this | 
|  | // file.  (The Library Public License restrictions do apply in other | 
|  | // respects; for example, they cover modification of the file, and | 
|  | /// distribution when not linked into a combined executable.) | 
|  |  | 
|  | // This program is distributed in the hope that it will be useful, but | 
|  | // WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | // Library General Public License for more details. | 
|  |  | 
|  | // You should have received a copy of the GNU Library General Public | 
|  | // License along with this program; if not, write to the Free Software | 
|  | // Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA | 
|  | // 02110-1301, USA. | 
|  |  | 
|  | // This header file defines the class Elf_file which can be used to | 
|  | // read useful data from an ELF file.  The functions here are all | 
|  | // templates which take a file interface object as a parameter.  This | 
|  | // type must have a subtype View.  This type must support two methods: | 
|  | //     View view(off_t file_offset, off_t data_size) | 
|  | // returns a View for the specified part of the file. | 
|  | //     void error(const char* printf_format, ...) | 
|  | // prints an error message and does not return.  The subtype View must | 
|  | // support a method | 
|  | //     const unsigned char* data() | 
|  | // which returns a pointer to a buffer containing the requested data. | 
|  | // This general interface is used to read data from the file.  Objects | 
|  | // of type View will never survive longer than the elfcpp function. | 
|  |  | 
|  | // Some of these functions must return a reference to part of the | 
|  | // file.  To use these, the file interface must support a subtype | 
|  | // Location: | 
|  | //    Location(off_t file_offset, off_t data_size) | 
|  | // To use this in conjunction with the accessors types Shdr, etc., the | 
|  | // file interface should support an overload of view: | 
|  | //    View view(Location) | 
|  | // This permits writing | 
|  | //    elfcpp::Shdr shdr(file, ef.section_header(n)); | 
|  |  | 
|  | #ifndef ELFCPP_FILE_H | 
|  | #define ELFCPP_FILE_H | 
|  |  | 
|  | #include <string> | 
|  | #include <cstdio> | 
|  | #include <cstring> | 
|  |  | 
|  | #include "elfcpp.h" | 
|  |  | 
|  | namespace elfcpp | 
|  | { | 
|  |  | 
|  | // A simple helper class to recognize if a file has an ELF header. | 
|  |  | 
|  | class Elf_recognizer | 
|  | { | 
|  | public: | 
|  | // Maximum header size.  The user should try to read this much of | 
|  | // the file when using this class. | 
|  |  | 
|  | static const int max_header_size = Elf_sizes<64>::ehdr_size; | 
|  |  | 
|  | // Checks if the file contains the ELF magic.  Other header fields | 
|  | // are not checked. | 
|  |  | 
|  | static bool | 
|  | is_elf_file(const unsigned char* ehdr_buf, int size); | 
|  |  | 
|  | // Check if EHDR_BUF/BUFSIZE is a valid header of a 32-bit or | 
|  | // 64-bit, little-endian or big-endian ELF file.  Assumes | 
|  | // is_elf_file() has been checked to be true.  If the header is not | 
|  | // valid, *ERROR contains a human-readable error message.  If is is, | 
|  | // *SIZE is set to either 32 or 64, *BIG_ENDIAN is set to indicate | 
|  | // whether the file is big-endian. | 
|  |  | 
|  | static bool | 
|  | is_valid_header(const unsigned char* ehdr_buf, off_t bufsize, | 
|  | int* size, bool* big_endian, | 
|  | std::string* error); | 
|  | }; | 
|  |  | 
|  | // This object is used to read an ELF file. | 
|  | //   SIZE: The size of file, 32 or 64. | 
|  | //   BIG_ENDIAN: Whether the file is in big-endian format. | 
|  | //   FILE: A file reading type as described above. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | class Elf_file | 
|  | { | 
|  | private: | 
|  | typedef Elf_file<size, big_endian, File> This; | 
|  |  | 
|  | public: | 
|  | static const int ehdr_size = Elf_sizes<size>::ehdr_size; | 
|  | static const int phdr_size = Elf_sizes<size>::phdr_size; | 
|  | static const int shdr_size = Elf_sizes<size>::shdr_size; | 
|  | static const int sym_size = Elf_sizes<size>::sym_size; | 
|  | static const int rel_size = Elf_sizes<size>::rel_size; | 
|  | static const int rela_size = Elf_sizes<size>::rela_size; | 
|  |  | 
|  | typedef Ehdr<size, big_endian> Ef_ehdr; | 
|  | typedef Phdr<size, big_endian> Ef_phdr; | 
|  | typedef Shdr<size, big_endian> Ef_shdr; | 
|  | typedef Sym<size, big_endian> Ef_sym; | 
|  |  | 
|  | // Construct an Elf_file given an ELF file header. | 
|  | Elf_file(File* file, const Ef_ehdr& ehdr) | 
|  | { this->construct(file, ehdr); } | 
|  |  | 
|  | // Construct an ELF file. | 
|  | inline | 
|  | Elf_file(File* file); | 
|  |  | 
|  | // Return the file offset to the section headers. | 
|  | off_t | 
|  | shoff() const | 
|  | { return this->shoff_; } | 
|  |  | 
|  | // Find the first section with an sh_type field equal to TYPE and | 
|  | // return its index.  Returns SHN_UNDEF if there is no such section. | 
|  | unsigned int | 
|  | find_section_by_type(unsigned int type); | 
|  |  | 
|  | // Return the number of sections. | 
|  | unsigned int | 
|  | shnum() | 
|  | { | 
|  | this->initialize_shnum(); | 
|  | return this->shnum_; | 
|  | } | 
|  |  | 
|  | unsigned int | 
|  | shnum() const | 
|  | { | 
|  | if (this->shnum_ == 0 && this->shoff_ != 0) | 
|  | this->file_->error(_("ELF file has not been initialized yet" | 
|  | " (internal error)")); | 
|  | return this->shnum_; | 
|  | } | 
|  |  | 
|  | // Return the section index of the section name string table. | 
|  | unsigned int | 
|  | shstrndx() | 
|  | { | 
|  | this->initialize_shnum(); | 
|  | return this->shstrndx_; | 
|  | } | 
|  |  | 
|  | unsigned int | 
|  | shstrndx() const | 
|  | { | 
|  | if (this->shstrndx_ == SHN_XINDEX && this->shoff_ != 0) | 
|  | { | 
|  | this->file_->error(_("ELF file has not been initialized yet" | 
|  | " (internal error)")); | 
|  | return 0; | 
|  | } | 
|  | return this->shstrndx_; | 
|  | } | 
|  |  | 
|  | // Return the value to subtract from section indexes >= | 
|  | // SHN_LORESERVE.  See the comment in initialize_shnum. | 
|  | int | 
|  | large_shndx_offset() | 
|  | { | 
|  | this->initialize_shnum(); | 
|  | return this->large_shndx_offset_; | 
|  | } | 
|  |  | 
|  | int | 
|  | large_shndx_offset() const | 
|  | { | 
|  | if (this->shstrndx_ == SHN_XINDEX && this->shoff_ != 0) | 
|  | this->file_->error(_("ELF file has not been initialized yet" | 
|  | " (internal error)")); | 
|  | return this->large_shndx_offset_; | 
|  | } | 
|  |  | 
|  | // Return the location of the header of section SHNDX. | 
|  | typename File::Location | 
|  | section_header(unsigned int shndx) | 
|  | { | 
|  | return typename File::Location(this->section_header_offset(shndx), | 
|  | shdr_size); | 
|  | } | 
|  |  | 
|  | // Return the name of section SHNDX. | 
|  | std::string | 
|  | section_name(unsigned int shndx) const; | 
|  |  | 
|  | // Return the location of the contents of section SHNDX. | 
|  | typename File::Location | 
|  | section_contents(unsigned int shndx); | 
|  |  | 
|  | // Return the size of section SHNDX. | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | section_size(unsigned int shndx); | 
|  |  | 
|  | // Return the flags of section SHNDX. | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | section_flags(unsigned int shndx); | 
|  |  | 
|  | // Return the address of section SHNDX. | 
|  | typename Elf_types<size>::Elf_Addr | 
|  | section_addr(unsigned int shndx); | 
|  |  | 
|  | // Return the type of section SHNDX. | 
|  | Elf_Word | 
|  | section_type(unsigned int shndx); | 
|  |  | 
|  | // Return the link field of section SHNDX. | 
|  | Elf_Word | 
|  | section_link(unsigned int shndx); | 
|  |  | 
|  | // Return the info field of section SHNDX. | 
|  | Elf_Word | 
|  | section_info(unsigned int shndx); | 
|  |  | 
|  | // Return the addralign field of section SHNDX. | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | section_addralign(unsigned int shndx); | 
|  |  | 
|  | private: | 
|  | // Shared constructor code. | 
|  | void | 
|  | construct(File* file, const Ef_ehdr& ehdr); | 
|  |  | 
|  | // Initialize shnum_ and shstrndx_. | 
|  | void | 
|  | initialize_shnum(); | 
|  |  | 
|  | // Return the file offset of the header of section SHNDX. | 
|  | off_t | 
|  | section_header_offset(unsigned int shndx) const; | 
|  |  | 
|  | // The file we are reading. | 
|  | File* file_; | 
|  | // The file offset to the section headers. | 
|  | off_t shoff_; | 
|  | // The number of sections. | 
|  | unsigned int shnum_; | 
|  | // The section index of the section name string table. | 
|  | unsigned int shstrndx_; | 
|  | // Offset to add to sections larger than SHN_LORESERVE. | 
|  | int large_shndx_offset_; | 
|  | }; | 
|  |  | 
|  | // A small wrapper around SHT_STRTAB data mapped to memory. It checks that the | 
|  | // index is not out of bounds and the string is NULL-terminated. | 
|  |  | 
|  | class Elf_strtab | 
|  | { | 
|  | public: | 
|  | // Construct an Elf_strtab for a section with contents *P and size SIZE. | 
|  | Elf_strtab(const unsigned char* p, size_t size); | 
|  |  | 
|  | // Return the file offset to the section headers. | 
|  | bool | 
|  | get_c_string(size_t offset, const char** cstring) const | 
|  | { | 
|  | if (offset >= this->usable_size_) | 
|  | return false; | 
|  | *cstring = this->base_ + offset; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Contents of the section mapped to memory. | 
|  | const char* base_; | 
|  | // One larger that the position of the last NULL character in the section. | 
|  | // For valid SHT_STRTAB sections, this is the size of the section. | 
|  | size_t usable_size_; | 
|  | }; | 
|  |  | 
|  | // Inline function definitions. | 
|  |  | 
|  | // Check for presence of the ELF magic number. | 
|  |  | 
|  | inline bool | 
|  | Elf_recognizer::is_elf_file(const unsigned char* ehdr_buf, int size) | 
|  | { | 
|  | if (size < 4) | 
|  | return false; | 
|  |  | 
|  | static unsigned char elfmagic[4] = | 
|  | { | 
|  | elfcpp::ELFMAG0, elfcpp::ELFMAG1, | 
|  | elfcpp::ELFMAG2, elfcpp::ELFMAG3 | 
|  | }; | 
|  | return memcmp(ehdr_buf, elfmagic, 4) == 0; | 
|  | } | 
|  |  | 
|  | namespace | 
|  | { | 
|  |  | 
|  | // Print a number to a string. | 
|  |  | 
|  | inline std::string | 
|  | internal_printf_int(const char* format, int arg) | 
|  | { | 
|  | char buf[256]; | 
|  | snprintf(buf, sizeof(buf), format, arg); | 
|  | return std::string(buf); | 
|  | } | 
|  |  | 
|  | }  // End anonymous namespace. | 
|  |  | 
|  | // Check the validity of the ELF header. | 
|  |  | 
|  | inline bool | 
|  | Elf_recognizer::is_valid_header( | 
|  | const unsigned char* ehdr_buf, | 
|  | off_t bufsize, | 
|  | int* size, | 
|  | bool* big_endian, | 
|  | std::string* error) | 
|  | { | 
|  | if (bufsize < elfcpp::EI_NIDENT) | 
|  | { | 
|  | *error = _("ELF file too short"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | int v = ehdr_buf[elfcpp::EI_VERSION]; | 
|  | if (v != elfcpp::EV_CURRENT) | 
|  | { | 
|  | if (v == elfcpp::EV_NONE) | 
|  | *error = _("invalid ELF version 0"); | 
|  | else | 
|  | *error = internal_printf_int(_("unsupported ELF version %d"), v); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | int c = ehdr_buf[elfcpp::EI_CLASS]; | 
|  | if (c == elfcpp::ELFCLASSNONE) | 
|  | { | 
|  | *error = _("invalid ELF class 0"); | 
|  | return false; | 
|  | } | 
|  | else if (c != elfcpp::ELFCLASS32 | 
|  | && c != elfcpp::ELFCLASS64) | 
|  | { | 
|  | *error = internal_printf_int(_("unsupported ELF class %d"), c); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | int d = ehdr_buf[elfcpp::EI_DATA]; | 
|  | if (d == elfcpp::ELFDATANONE) | 
|  | { | 
|  | *error = _("invalid ELF data encoding"); | 
|  | return false; | 
|  | } | 
|  | else if (d != elfcpp::ELFDATA2LSB | 
|  | && d != elfcpp::ELFDATA2MSB) | 
|  | { | 
|  | *error = internal_printf_int(_("unsupported ELF data encoding %d"), d); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | *big_endian = (d == elfcpp::ELFDATA2MSB); | 
|  |  | 
|  | if (c == elfcpp::ELFCLASS32) | 
|  | { | 
|  | if (bufsize < elfcpp::Elf_sizes<32>::ehdr_size) | 
|  | { | 
|  | *error = _("ELF file too short"); | 
|  | return false; | 
|  | } | 
|  | *size = 32; | 
|  | } | 
|  | else | 
|  | { | 
|  | if (bufsize < elfcpp::Elf_sizes<64>::ehdr_size) | 
|  | { | 
|  | *error = _("ELF file too short"); | 
|  | return false; | 
|  | } | 
|  | *size = 64; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Template function definitions. | 
|  |  | 
|  | // Construct an Elf_file given an ELF file header. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | void | 
|  | Elf_file<size, big_endian, File>::construct(File* file, const Ef_ehdr& ehdr) | 
|  | { | 
|  | this->file_ = file; | 
|  | this->shoff_ = ehdr.get_e_shoff(); | 
|  | this->shnum_ = ehdr.get_e_shnum(); | 
|  | this->shstrndx_ = ehdr.get_e_shstrndx(); | 
|  | this->large_shndx_offset_ = 0; | 
|  | if (ehdr.get_e_ehsize() != This::ehdr_size) | 
|  | file->error(_("bad e_ehsize (%d != %d)"), | 
|  | ehdr.get_e_ehsize(), This::ehdr_size); | 
|  | if (ehdr.get_e_shentsize() != This::shdr_size) | 
|  | file->error(_("bad e_shentsize (%d != %d)"), | 
|  | ehdr.get_e_shentsize(), This::shdr_size); | 
|  | } | 
|  |  | 
|  | // Construct an ELF file. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | inline | 
|  | Elf_file<size, big_endian, File>::Elf_file(File* file) | 
|  | { | 
|  | typename File::View v(file->view(file_header_offset, This::ehdr_size)); | 
|  | this->construct(file, Ef_ehdr(v.data())); | 
|  | } | 
|  |  | 
|  | // Initialize the shnum_ and shstrndx_ fields, handling overflow. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | void | 
|  | Elf_file<size, big_endian, File>::initialize_shnum() | 
|  | { | 
|  | if ((this->shnum_ == 0 || this->shstrndx_ == SHN_XINDEX) | 
|  | && this->shoff_ != 0) | 
|  | { | 
|  | typename File::View v(this->file_->view(this->shoff_, This::shdr_size)); | 
|  | Ef_shdr shdr(v.data()); | 
|  |  | 
|  | if (this->shnum_ == 0) | 
|  | this->shnum_ = shdr.get_sh_size(); | 
|  |  | 
|  | if (this->shstrndx_ == SHN_XINDEX) | 
|  | { | 
|  | this->shstrndx_ = shdr.get_sh_link(); | 
|  |  | 
|  | // Versions of the GNU binutils between 2.12 and 2.18 did | 
|  | // not handle objects with more than SHN_LORESERVE sections | 
|  | // correctly.  All large section indexes were offset by | 
|  | // 0x100.  Some information can be found here: | 
|  | // http://sourceware.org/bugzilla/show_bug.cgi?id=5900 . | 
|  | // Fortunately these object files are easy to detect, as the | 
|  | // GNU binutils always put the section header string table | 
|  | // near the end of the list of sections.  Thus if the | 
|  | // section header string table index is larger than the | 
|  | // number of sections, then we know we have to subtract | 
|  | // 0x100 to get the real section index. | 
|  | if (this->shstrndx_ >= this->shnum_) | 
|  | { | 
|  | if (this->shstrndx_ >= elfcpp::SHN_LORESERVE + 0x100) | 
|  | { | 
|  | this->large_shndx_offset_ = - 0x100; | 
|  | this->shstrndx_ -= 0x100; | 
|  | } | 
|  | if (this->shstrndx_ >= this->shnum_) | 
|  | this->file_->error(_("bad shstrndx: %u >= %u"), | 
|  | this->shstrndx_, this->shnum_); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Find section with sh_type equal to TYPE and return its index. | 
|  | // Returns SHN_UNDEF if not found. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | unsigned int | 
|  | Elf_file<size, big_endian, File>::find_section_by_type(unsigned int type) | 
|  | { | 
|  | unsigned int shnum = this->shnum(); | 
|  | typename File::View v(this->file_->view(this->shoff_, | 
|  | This::shdr_size * shnum)); | 
|  | for (unsigned int i = 0; i < shnum; i++) | 
|  | { | 
|  | Ef_shdr shdr(v.data() + This::shdr_size * i); | 
|  | if (shdr.get_sh_type() == type) | 
|  | return i; | 
|  | } | 
|  | return SHN_UNDEF; | 
|  | } | 
|  |  | 
|  | // Return the file offset of the section header of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | off_t | 
|  | Elf_file<size, big_endian, File>::section_header_offset(unsigned int shndx) const | 
|  | { | 
|  | if (shndx >= this->shnum()) | 
|  | this->file_->error(_("section_header_offset: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  | return this->shoff_ + This::shdr_size * shndx; | 
|  | } | 
|  |  | 
|  | // Return the name of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | std::string | 
|  | Elf_file<size, big_endian, File>::section_name(unsigned int shndx) const | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | // Get the section name offset. | 
|  | unsigned int sh_name; | 
|  | { | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  | Ef_shdr shdr(v.data()); | 
|  | sh_name = shdr.get_sh_name(); | 
|  | } | 
|  |  | 
|  | // Get the file offset for the section name string table data. | 
|  | off_t shstr_off; | 
|  | typename Elf_types<size>::Elf_WXword shstr_size; | 
|  | { | 
|  | const unsigned int shstrndx = this->shstrndx_; | 
|  | typename File::View v(file->view(this->section_header_offset(shstrndx), | 
|  | This::shdr_size)); | 
|  | Ef_shdr shstr_shdr(v.data()); | 
|  | shstr_off = shstr_shdr.get_sh_offset(); | 
|  | shstr_size = shstr_shdr.get_sh_size(); | 
|  | } | 
|  |  | 
|  | if (sh_name >= shstr_size) | 
|  | file->error(_("bad section name offset for section %u: %u"), | 
|  | shndx, sh_name); | 
|  |  | 
|  | typename File::View v(file->view(shstr_off, shstr_size)); | 
|  |  | 
|  | const unsigned char* datau = v.data(); | 
|  | const char* data = reinterpret_cast<const char*>(datau); | 
|  | const void* p = ::memchr(data + sh_name, '\0', shstr_size - sh_name); | 
|  | if (p == NULL) | 
|  | file->error(_("missing null terminator for name of section %u"), | 
|  | shndx); | 
|  |  | 
|  | size_t len = static_cast<const char*>(p) - (data + sh_name); | 
|  |  | 
|  | return std::string(data + sh_name, len); | 
|  | } | 
|  |  | 
|  | // Return the contents of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | typename File::Location | 
|  | Elf_file<size, big_endian, File>::section_contents(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_contents: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  | Ef_shdr shdr(v.data()); | 
|  | return typename File::Location(shdr.get_sh_offset(), shdr.get_sh_size()); | 
|  | } | 
|  |  | 
|  | // Get the size of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | Elf_file<size, big_endian, File>::section_size(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_size: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_size(); | 
|  | } | 
|  |  | 
|  | // Return the section flags of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | Elf_file<size, big_endian, File>::section_flags(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_flags: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_flags(); | 
|  | } | 
|  |  | 
|  | // Return the address of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | typename Elf_types<size>::Elf_Addr | 
|  | Elf_file<size, big_endian, File>::section_addr(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_flags: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_addr(); | 
|  | } | 
|  |  | 
|  | // Return the type of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | Elf_Word | 
|  | Elf_file<size, big_endian, File>::section_type(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_type: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_type(); | 
|  | } | 
|  |  | 
|  | // Return the sh_link field of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | Elf_Word | 
|  | Elf_file<size, big_endian, File>::section_link(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_link: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_link(); | 
|  | } | 
|  |  | 
|  | // Return the sh_info field of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | Elf_Word | 
|  | Elf_file<size, big_endian, File>::section_info(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_info: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_info(); | 
|  | } | 
|  |  | 
|  | // Return the sh_addralign field of section SHNDX. | 
|  |  | 
|  | template<int size, bool big_endian, typename File> | 
|  | typename Elf_types<size>::Elf_WXword | 
|  | Elf_file<size, big_endian, File>::section_addralign(unsigned int shndx) | 
|  | { | 
|  | File* const file = this->file_; | 
|  |  | 
|  | if (shndx >= this->shnum()) | 
|  | file->error(_("section_addralign: bad shndx %u >= %u"), | 
|  | shndx, this->shnum()); | 
|  |  | 
|  | typename File::View v(file->view(this->section_header_offset(shndx), | 
|  | This::shdr_size)); | 
|  |  | 
|  | Ef_shdr shdr(v.data()); | 
|  | return shdr.get_sh_addralign(); | 
|  | } | 
|  |  | 
|  | inline | 
|  | Elf_strtab::Elf_strtab(const unsigned char* p, size_t size) | 
|  | { | 
|  | // Check if the section is NUL-terminated. If it isn't, we ignore | 
|  | // the last part to make sure we don't return non-NUL-terminated | 
|  | // strings. | 
|  | while (size > 0 && p[size - 1] != 0) | 
|  | size--; | 
|  | this->base_ = reinterpret_cast<const char*>(p); | 
|  | this->usable_size_ = size; | 
|  | } | 
|  |  | 
|  | } // End namespace elfcpp. | 
|  |  | 
|  | #endif // !defined(ELFCPP_FILE_H) |