| /* @LICENSE(UNSW_OZPLB) */ |
| |
| /* |
| * Australian Public Licence B (OZPLB) |
| * |
| * Version 1-0 |
| * |
| * Copyright (c) 1999-2004 University of New South Wales |
| * |
| * All rights reserved. |
| * |
| * Developed by: Operating Systems and Distributed Systems Group (DiSy) |
| * University of New South Wales |
| * http://www.disy.cse.unsw.edu.au |
| * |
| * Permission is granted by University of New South Wales, free of charge, to |
| * any person obtaining a copy of this software and any associated |
| * documentation files (the "Software") to deal with the Software without |
| * restriction, including (without limitation) the rights to use, copy, |
| * modify, adapt, merge, publish, distribute, communicate to the public, |
| * sublicense, and/or sell, lend or rent out copies of the Software, and |
| * to permit persons to whom the Software is furnished to do so, subject |
| * to the following conditions: |
| * |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimers. |
| * |
| * * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following |
| * disclaimers in the documentation and/or other materials provided |
| * with the distribution. |
| * |
| * * Neither the name of University of New South Wales, nor the names of its |
| * contributors, may be used to endorse or promote products derived |
| * from this Software without specific prior written permission. |
| * |
| * EXCEPT AS EXPRESSLY STATED IN THIS LICENCE AND TO THE FULL EXTENT |
| * PERMITTED BY APPLICABLE LAW, THE SOFTWARE IS PROVIDED "AS-IS", AND |
| * NATIONAL ICT AUSTRALIA AND ITS CONTRIBUTORS MAKE NO REPRESENTATIONS, |
| * WARRANTIES OR CONDITIONS OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING |
| * BUT NOT LIMITED TO ANY REPRESENTATIONS, WARRANTIES OR CONDITIONS |
| * REGARDING THE CONTENTS OR ACCURACY OF THE SOFTWARE, OR OF TITLE, |
| * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, NONINFRINGEMENT, |
| * THE ABSENCE OF LATENT OR OTHER DEFECTS, OR THE PRESENCE OR ABSENCE OF |
| * ERRORS, WHETHER OR NOT DISCOVERABLE. |
| * |
| * TO THE FULL EXTENT PERMITTED BY APPLICABLE LAW, IN NO EVENT SHALL |
| * NATIONAL ICT AUSTRALIA OR ITS CONTRIBUTORS BE LIABLE ON ANY LEGAL |
| * THEORY (INCLUDING, WITHOUT LIMITATION, IN AN ACTION OF CONTRACT, |
| * NEGLIGENCE OR OTHERWISE) FOR ANY CLAIM, LOSS, DAMAGES OR OTHER |
| * LIABILITY, INCLUDING (WITHOUT LIMITATION) LOSS OF PRODUCTION OR |
| * OPERATION TIME, LOSS, DAMAGE OR CORRUPTION OF DATA OR RECORDS; OR LOSS |
| * OF ANTICIPATED SAVINGS, OPPORTUNITY, REVENUE, PROFIT OR GOODWILL, OR |
| * OTHER ECONOMIC LOSS; OR ANY SPECIAL, INCIDENTAL, INDIRECT, |
| * CONSEQUENTIAL, PUNITIVE OR EXEMPLARY DAMAGES, ARISING OUT OF OR IN |
| * CONNECTION WITH THIS LICENCE, THE SOFTWARE OR THE USE OF OR OTHER |
| * DEALINGS WITH THE SOFTWARE, EVEN IF NATIONAL ICT AUSTRALIA OR ITS |
| * CONTRIBUTORS HAVE BEEN ADVISED OF THE POSSIBILITY OF SUCH CLAIM, LOSS, |
| * DAMAGES OR OTHER LIABILITY. |
| * |
| * If applicable legislation implies representations, warranties, or |
| * conditions, or imposes obligations or liability on University of New South |
| * Wales or one of its contributors in respect of the Software that |
| * cannot be wholly or partly excluded, restricted or modified, the |
| * liability of University of New South Wales or the contributor is limited, to |
| * the full extent permitted by the applicable legislation, at its |
| * option, to: |
| * a. in the case of goods, any one or more of the following: |
| * i. the replacement of the goods or the supply of equivalent goods; |
| * ii. the repair of the goods; |
| * iii. the payment of the cost of replacing the goods or of acquiring |
| * equivalent goods; |
| * iv. the payment of the cost of having the goods repaired; or |
| * b. in the case of services: |
| * i. the supplying of the services again; or |
| * ii. the payment of the cost of having the services supplied again. |
| * |
| * The construction, validity and performance of this licence is governed |
| * by the laws in force in New South Wales, Australia. |
| */ |
| |
| /* |
| Authors: Luke Deller, Ben Leslie |
| Created: 24/Sep/1999 |
| */ |
| |
| /** |
| \file |
| |
| \brief Generic ELF library |
| |
| The ELF library is designed to make the task of parsing and getting information |
| out of an ELF file easier. |
| |
| It provides function to obtain the various different fields in the ELF header, and |
| the program and segment information. |
| |
| Also importantly, it provides a function elf_loadFile which will load a given |
| ELF file into memory. |
| |
| */ |
| |
| #pragma once |
| |
| #include <stdbool.h> |
| #include <stddef.h> |
| #include <stdint.h> |
| #include <elf.h> |
| |
| struct elf { |
| void *elfFile; |
| size_t elfSize; |
| unsigned char elfClass; /* 32-bit or 64-bit */ |
| }; |
| typedef struct elf elf_t; |
| |
| enum elf_addr_type { |
| VIRTUAL, |
| PHYSICAL |
| }; |
| typedef enum elf_addr_type elf_addr_type_t; |
| |
| /* ELF header functions */ |
| /** |
| * Initialises an elf_t structure and checks that the ELF file is valid. |
| * This function must be called to validate the ELF before any other function. |
| * Otherwise, attempting to call other functions with an invalid ELF file may |
| * result in undefined behaviour. |
| * |
| * @param file ELF file to use |
| * @param size Size of the ELF file |
| * @param res elf_t to initialise |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_newFile(void *file, size_t size, elf_t *res); |
| |
| /** |
| * Initialises and elf_t structure and checks that the ELF file is valid. |
| * The validity of a potential ELF file can be determined by the arguments |
| * check_pht and check_st. |
| * If both check_pht and check_st are true, this function is equivalent to |
| * elf_newFile. |
| * Calling other functions with an invalid ELF file may result in undefined |
| * behaviour. |
| * |
| * @param file ELF file to use |
| * @param size Size of the ELF file |
| * @param check_pht Whether to check the ELF program header table is valid |
| * @param check_st Whether to check the ELF section table is valid |
| * @param res elf_t to initialise |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_newFile_maybe_unsafe(void *file, size_t size, bool check_pht, bool check_st, elf_t *res); |
| |
| /** |
| * Checks that file starts with the ELF magic number. |
| * File must be at least 4 bytes (SELFMAG). |
| * |
| * @param file to check |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_check_magic(char *file); |
| |
| /** |
| * Checks that elfFile points to an ELF file with a valid ELF header. |
| * |
| * @param elfFile Potential ELF file to check |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_checkFile(elf_t *elfFile); |
| |
| /** |
| * Checks that elfFile points to an ELF file with a valid program header table. |
| * |
| * @param elfFile Potential ELF file to check |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_checkProgramHeaderTable(elf_t *elfFile); |
| |
| /** |
| * Checks that elfFile points to an ELF file with a valid section table. |
| * |
| * @param elfFile Potential ELF file to check |
| * |
| * \return 0 on success, otherwise < 0 |
| */ |
| int elf_checkSectionTable(elf_t *elfFile); |
| |
| /** |
| * Find the entry point of an ELF file. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * |
| * \return The entry point address. |
| */ |
| uintptr_t elf_getEntryPoint(elf_t *elfFile); |
| |
| /** |
| * Determine number of program headers in an ELF file. |
| * |
| * @param elfFile Pointer to a valid ELF structure. |
| * |
| * \return Number of program headers in the ELF file. |
| */ |
| size_t elf_getNumProgramHeaders(elf_t *elfFile); |
| |
| /** |
| * Determine number of sections in an ELF file. |
| * |
| * @param elfFile Pointer to a valid ELF structure. |
| * |
| * \return Number of sections in the ELF file. |
| */ |
| size_t elf_getNumSections(elf_t *elfFile); |
| |
| /** |
| * Get the index of the section header string table of an ELF file. |
| * |
| * @param elf Pointer to a valid ELF structure. |
| * |
| * \return The index of the section header string table. |
| */ |
| size_t elf_getSectionStringTableIndex(elf_t *elf); |
| |
| /** |
| * Get a string table section of an ELF file. |
| * |
| * @param elfFile Pointer to a valid ELF structure. |
| * @param string_section The section number of the string table. |
| * |
| * \return The string table, or NULL if the section is not a string table. |
| */ |
| const char *elf_getStringTable(elf_t *elfFile, size_t string_segment); |
| |
| /** |
| * Get the string table for section header names. |
| * |
| * @param elfFile Pointer to a valid ELF structure. |
| * |
| * \return The string table, or NULL if there is no table. |
| */ |
| const char *elf_getSectionStringTable(elf_t *elfFile); |
| |
| |
| /* Section header functions */ |
| /** |
| * Get a section of an ELF file. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i The section number |
| * |
| * \return The section, or NULL if there is no section. |
| */ |
| void *elf_getSection(elf_t *elfFile, size_t i); |
| |
| /** |
| * Get the section of an ELF file with a given name. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param str Name of the section |
| * @param i Pointer to store the section number |
| * |
| * \return The section, or NULL if there is no section. |
| */ |
| void *elf_getSectionNamed(elf_t *elfFile, const char *str, size_t *i); |
| |
| /** |
| * Return the name of a given section. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The name of a given section. |
| */ |
| const char *elf_getSectionName(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the offset to the name of a given section in the section header |
| * string table. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The offset to the name of a given section in the section header |
| * string table. |
| */ |
| size_t elf_getSectionNameOffset(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the type of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The type of a given section. |
| */ |
| uint32_t elf_getSectionType(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the flags of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The flags of a given section. |
| */ |
| size_t elf_getSectionFlags(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the address of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The address of a given section. |
| */ |
| uintptr_t elf_getSectionAddr(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the offset of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The offset of a given section. |
| */ |
| size_t elf_getSectionOffset(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the size of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The size of a given section. |
| */ |
| size_t elf_getSectionSize(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the related section index of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The related section index of a given section. |
| */ |
| uint32_t elf_getSectionLink(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return extra information of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return Extra information of a given section. |
| */ |
| uint32_t elf_getSectionInfo(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the alignment of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The alignment of a given section. |
| */ |
| size_t elf_getSectionAddrAlign(elf_t *elfFile, size_t i); |
| |
| /** |
| * Return the entry size of a given section |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param i Index of the section |
| * |
| * \return The entry size of a given section. |
| */ |
| size_t elf_getSectionEntrySize(elf_t *elfFile, size_t i); |
| |
| |
| /* Program header functions */ |
| |
| /** |
| * Return the segment data for a given program header. |
| * |
| * @param elf Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return Pointer to the segment data |
| */ |
| void *elf_getProgramSegment(elf_t *elf, size_t ph); |
| |
| /** |
| * Return the type for a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The type of a given program header. |
| */ |
| uint32_t elf_getProgramHeaderType(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the segment offset for a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The offset of this program header from the start of the file. |
| */ |
| size_t elf_getProgramHeaderOffset(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the base virtual address of given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The memory size of the specified program header. |
| */ |
| uintptr_t elf_getProgramHeaderVaddr(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the base physical address of given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The memory size of the specified program header. |
| */ |
| uintptr_t elf_getProgramHeaderPaddr(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the file size of a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The file size of the specified program header. |
| */ |
| size_t elf_getProgramHeaderFileSize(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the memory size of a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The memory size of the specified program header. |
| */ |
| size_t elf_getProgramHeaderMemorySize(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the flags for a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The flags of a given program header. |
| */ |
| uint32_t elf_getProgramHeaderFlags(elf_t *elfFile, size_t ph); |
| |
| /** |
| * Return the alignment for a given program header. |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param ph Index of the program header |
| * |
| * \return The alignment of the given program header. |
| */ |
| size_t elf_getProgramHeaderAlign(elf_t *elfFile, size_t ph); |
| |
| |
| /* Utility functions */ |
| |
| /** |
| * Determine the memory bounds of an ELF file |
| * |
| * @param elfFile Pointer to a valid ELF structure |
| * @param addr_type If PHYSICAL return bounds of physical memory, otherwise |
| * return bounds of virtual memory |
| * @param min Pointer to return value of the minimum |
| * @param max Pointer to return value of the maximum |
| * |
| * \return true on success. false on failure, if for example, it is an invalid ELF file |
| */ |
| int elf_getMemoryBounds(elf_t *elfFile, elf_addr_type_t addr_type, uintptr_t *min, uintptr_t *max); |
| |
| /** |
| * |
| * \return true if the address in in this program header |
| */ |
| int elf_vaddrInProgramHeader(elf_t *elfFile, size_t ph, uintptr_t vaddr); |
| |
| /** |
| * Return the physical translation of a physical address, with respect |
| * to a given program header |
| * |
| */ |
| uintptr_t elf_vtopProgramHeader(elf_t *elfFile, size_t ph, uintptr_t vaddr); |
| |
| /** |
| * Load an ELF file into memory |
| * |
| * @param elfFile Pointer to a valid ELF file |
| * @param addr_type If PHYSICAL load using the physical address, otherwise using the |
| * virtual addresses |
| * |
| * \return true on success, false on failure. |
| * |
| * The function assumes that the ELF file is loaded in memory at some |
| * address different to the target address at which it will be loaded. |
| * It also assumes direct access to the source and destination address, i.e: |
| * Memory must be able to be loaded with a simple memcpy. |
| * |
| * Obviously this also means that if we are loading a 64bit ELF on a 32bit |
| * platform, we assume that any memory addresses are within the first 4GB. |
| * |
| */ |
| int elf_loadFile(elf_t *elfFile, elf_addr_type_t addr_type); |