/* * Copyright 2011, The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef ELF_OBJECT_H #define ELF_OBJECT_H #include "ELFTypes.h" #include "MemChunk.h" #include "utils/rsl_assert.h" #include <llvm/ADT/OwningPtr.h> #include <string> #include <vector> template <unsigned Bitwidth> class ELFObject { public: ELF_TYPE_INTRO_TO_TEMPLATE_SCOPE(Bitwidth); private: llvm::OwningPtr<ELFHeaderTy> header; llvm::OwningPtr<ELFSectionHeaderTableTy> shtab; std::vector<ELFSectionTy *> stab; MemChunk SHNCommonData; unsigned char *SHNCommonDataPtr; size_t SHNCommonDataFreeSize; // TODO: Need refactor! bool initSHNCommonDataSize(size_t SHNCommonDataSize) { rsl_assert(!SHNCommonDataPtr && "Can't init twice."); if (!SHNCommonData.allocate(SHNCommonDataSize)) { return false; } SHNCommonDataPtr = SHNCommonData.getBuffer(); SHNCommonDataFreeSize = SHNCommonDataSize; return true; } private: ELFObject() : SHNCommonDataPtr(NULL) { } public: template <typename Archiver> static ELFObject *read(Archiver &AR); ELFHeaderTy const *getHeader() const { return header.get(); } ELFSectionHeaderTableTy const *getSectionHeaderTable() const { return shtab.get(); } char const *getSectionName(size_t i) const; ELFSectionTy const *getSectionByIndex(size_t i) const; ELFSectionTy *getSectionByIndex(size_t i); ELFSectionTy const *getSectionByName(std::string const &str) const; ELFSectionTy *getSectionByName(std::string const &str); void *allocateSHNCommonData(size_t size, size_t align = 1) { rsl_assert(size > 0 && align != 0); rsl_assert(SHNCommonDataPtr && "Must init common data size before use!"); // Ensure alignment size_t rem = ((uintptr_t)SHNCommonDataPtr) % align; if (rem != 0) { SHNCommonDataPtr += align - rem; SHNCommonDataFreeSize -= align - rem; } // Ensure the free size is sufficient if (SHNCommonDataFreeSize < size) { return NULL; } // Allcoate void *result = SHNCommonDataPtr; SHNCommonDataPtr += size; SHNCommonDataFreeSize -= size; return result; } void relocate(void *(*find_sym)(void *context, char const *name), void *context); void print() const; ~ELFObject() { for (size_t i = 0; i < stab.size(); ++i) { // Delete will check the pointer is nullptr or not by himself. delete stab[i]; } } private: void relocateARM(void *(*find_sym)(void *context, char const *name), void *context, ELFSectionRelTableTy *reltab, ELFSectionProgBitsTy *text); void relocateX86_32(void *(*find_sym)(void *context, char const *name), void *context, ELFSectionRelTableTy *reltab, ELFSectionProgBitsTy *text); void relocateX86_64(void *(*find_sym)(void *context, char const *name), void *context, ELFSectionRelTableTy *reltab, ELFSectionProgBitsTy *text); }; #include "impl/ELFObject.hxx" #endif // ELF_OBJECT_H