2012-01-20 12:48:44 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
#ifndef CustomElf_h
|
|
|
|
#define CustomElf_h
|
|
|
|
|
|
|
|
#include "ElfLoader.h"
|
2014-10-07 02:42:18 +04:00
|
|
|
#include "BaseElf.h"
|
2012-01-20 12:48:44 +04:00
|
|
|
#include "Logging.h"
|
2012-04-25 11:05:02 +04:00
|
|
|
#include "Elfxx.h"
|
2012-01-20 12:48:44 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Library Handle class for ELF libraries we don't let the system linker
|
|
|
|
* handle.
|
|
|
|
*/
|
2014-10-07 02:42:18 +04:00
|
|
|
class CustomElf: public BaseElf, private ElfLoader::link_map
|
2012-01-20 12:48:44 +04:00
|
|
|
{
|
2012-01-20 12:49:03 +04:00
|
|
|
friend class ElfLoader;
|
2012-02-22 11:12:15 +04:00
|
|
|
friend class SEGVHandler;
|
2012-01-20 12:48:44 +04:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Returns a new CustomElf using the given file descriptor to map ELF
|
|
|
|
* content. The file descriptor ownership is stolen, and it will be closed
|
|
|
|
* in CustomElf's destructor if an instance is created, or by the Load
|
|
|
|
* method otherwise. The path corresponds to the file descriptor, and flags
|
|
|
|
* are the same kind of flags that would be given to dlopen(), though
|
|
|
|
* currently, none are supported and the behaviour is more or less that of
|
|
|
|
* RTLD_GLOBAL | RTLD_BIND_NOW.
|
|
|
|
*/
|
2015-06-17 17:00:52 +03:00
|
|
|
static already_AddRefed<LibHandle> Load(Mappable *mappable,
|
2012-01-20 12:48:44 +04:00
|
|
|
const char *path, int flags);
|
|
|
|
|
|
|
|
/**
|
2014-10-07 02:42:18 +04:00
|
|
|
* Inherited from LibHandle/BaseElf
|
2012-01-20 12:48:44 +04:00
|
|
|
*/
|
|
|
|
virtual ~CustomElf();
|
2013-10-31 19:40:32 +04:00
|
|
|
|
2013-04-12 12:23:12 +04:00
|
|
|
protected:
|
|
|
|
virtual Mappable *GetMappable() const;
|
|
|
|
|
|
|
|
public:
|
2014-10-16 04:19:45 +04:00
|
|
|
/**
|
|
|
|
* Returns the instance, casted as BaseElf. (short of a better way to do
|
|
|
|
* this without RTTI)
|
|
|
|
*/
|
|
|
|
virtual BaseElf *AsBaseElf() { return this; }
|
|
|
|
|
2012-01-20 12:48:44 +04:00
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* Scan dependent libraries to find the address corresponding to the
|
|
|
|
* given symbol name. This is used to find symbols that are undefined
|
|
|
|
* in the Elf object.
|
|
|
|
*/
|
|
|
|
void *GetSymbolPtrInDeps(const char *symbol) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Private constructor
|
|
|
|
*/
|
2012-01-20 12:48:44 +04:00
|
|
|
CustomElf(Mappable *mappable, const char *path)
|
2014-10-16 04:19:45 +04:00
|
|
|
: BaseElf(path, mappable)
|
2014-10-15 09:46:09 +04:00
|
|
|
, link_map()
|
2014-01-16 04:13:42 +04:00
|
|
|
, init(0)
|
|
|
|
, fini(0)
|
|
|
|
, initialized(false)
|
|
|
|
, has_text_relocs(false)
|
2012-01-20 12:48:44 +04:00
|
|
|
{ }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads an Elf segment defined by the given PT_LOAD header.
|
|
|
|
* Returns whether this succeeded or failed.
|
|
|
|
*/
|
|
|
|
bool LoadSegment(const Elf::Phdr *pt_load) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes the library according to information found in the given
|
|
|
|
* PT_DYNAMIC header.
|
|
|
|
* Returns whether this succeeded or failed.
|
|
|
|
*/
|
|
|
|
bool InitDyn(const Elf::Phdr *pt_dyn);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply .rel.dyn/.rela.dyn relocations.
|
|
|
|
* Returns whether this succeeded or failed.
|
|
|
|
*/
|
|
|
|
bool Relocate();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Apply .rel.plt/.rela.plt relocations.
|
|
|
|
* Returns whether this succeeded or failed.
|
|
|
|
*/
|
|
|
|
bool RelocateJumps();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call initialization functions (.init/.init_array)
|
|
|
|
* Returns true;
|
|
|
|
*/
|
|
|
|
bool CallInit();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call destructor functions (.fini_array/.fini)
|
|
|
|
* Returns whether this succeeded or failed.
|
|
|
|
*/
|
|
|
|
void CallFini();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call a function given a pointer to its location.
|
|
|
|
*/
|
|
|
|
void CallFunction(void *ptr) const
|
|
|
|
{
|
|
|
|
/* C++ doesn't allow direct conversion between pointer-to-object
|
|
|
|
* and pointer-to-function. */
|
|
|
|
union {
|
|
|
|
void *ptr;
|
|
|
|
void (*func)(void);
|
|
|
|
} f;
|
|
|
|
f.ptr = ptr;
|
2013-06-27 04:35:49 +04:00
|
|
|
DEBUG_LOG("%s: Calling function @%p", GetPath(), ptr);
|
2012-01-20 12:48:44 +04:00
|
|
|
f.func();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Call a function given a an address relative to the library base
|
|
|
|
*/
|
|
|
|
void CallFunction(Elf::Addr addr) const
|
|
|
|
{
|
|
|
|
return CallFunction(GetPtr(addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* List of dependent libraries */
|
2015-10-18 08:24:48 +03:00
|
|
|
std::vector<RefPtr<LibHandle> > dependencies;
|
2012-01-20 12:48:44 +04:00
|
|
|
|
|
|
|
/* List of .rel.dyn/.rela.dyn relocations */
|
|
|
|
Array<Elf::Reloc> relocations;
|
|
|
|
|
|
|
|
/* List of .rel.plt/.rela.plt relocation */
|
|
|
|
Array<Elf::Reloc> jumprels;
|
|
|
|
|
|
|
|
/* Relative address of the initialization and destruction functions
|
|
|
|
* (.init/.fini) */
|
|
|
|
Elf::Addr init, fini;
|
|
|
|
|
|
|
|
/* List of initialization and destruction functions
|
|
|
|
* (.init_array/.fini_array) */
|
|
|
|
Array<void *> init_array, fini_array;
|
|
|
|
|
|
|
|
bool initialized;
|
2013-10-31 19:40:32 +04:00
|
|
|
|
2014-01-16 04:13:42 +04:00
|
|
|
bool has_text_relocs;
|
2012-01-20 12:48:44 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* CustomElf_h */
|