| // elfcpp_file.h -- file access for elfcpp -*- C++ -*- |
| |
| // Copyright (C) 2006-2021 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) |