forked from OSchip/llvm-project
Add files which were not included by commit 154868.
llvm-svn: 154872
This commit is contained in:
parent
f594b178c6
commit
e185ecb28b
|
@ -0,0 +1,214 @@
|
|||
//===-- GDBRegistrar.cpp - Registers objects with GDB ---------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "JITRegistrar.h"
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/Support/MutexGuard.h"
|
||||
#include "llvm/Support/Mutex.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
#include "llvm/Support/Compiler.h"
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
// This must be kept in sync with gdb/gdb/jit.h .
|
||||
extern "C" {
|
||||
|
||||
typedef enum {
|
||||
JIT_NOACTION = 0,
|
||||
JIT_REGISTER_FN,
|
||||
JIT_UNREGISTER_FN
|
||||
} jit_actions_t;
|
||||
|
||||
struct jit_code_entry {
|
||||
struct jit_code_entry *next_entry;
|
||||
struct jit_code_entry *prev_entry;
|
||||
const char *symfile_addr;
|
||||
uint64_t symfile_size;
|
||||
};
|
||||
|
||||
struct jit_descriptor {
|
||||
uint32_t version;
|
||||
// This should be jit_actions_t, but we want to be specific about the
|
||||
// bit-width.
|
||||
uint32_t action_flag;
|
||||
struct jit_code_entry *relevant_entry;
|
||||
struct jit_code_entry *first_entry;
|
||||
};
|
||||
|
||||
// We put information about the JITed function in this global, which the
|
||||
// debugger reads. Make sure to specify the version statically, because the
|
||||
// debugger checks the version before we can set it during runtime.
|
||||
static struct jit_descriptor __jit_debug_descriptor = { 1, 0, 0, 0 };
|
||||
|
||||
// Debuggers puts a breakpoint in this function.
|
||||
LLVM_ATTRIBUTE_NOINLINE void __jit_debug_register_code() { }
|
||||
|
||||
}
|
||||
|
||||
namespace {
|
||||
|
||||
// Buffer for an in-memory object file in executable memory
|
||||
typedef llvm::DenseMap< const char*,
|
||||
std::pair<std::size_t, jit_code_entry*> >
|
||||
RegisteredObjectBufferMap;
|
||||
|
||||
/// Global access point for the JIT debugging interface designed for use with a
|
||||
/// singleton toolbox. Handles thread-safe registration and deregistration of
|
||||
/// object files that are in executable memory managed by the client of this
|
||||
/// class.
|
||||
class GDBJITRegistrar : public JITRegistrar {
|
||||
/// A map of in-memory object files that have been registered with the
|
||||
/// JIT interface.
|
||||
RegisteredObjectBufferMap ObjectBufferMap;
|
||||
|
||||
public:
|
||||
/// Instantiates the JIT service.
|
||||
GDBJITRegistrar() : ObjectBufferMap() {}
|
||||
|
||||
/// Unregisters each object that was previously registered and releases all
|
||||
/// internal resources.
|
||||
virtual ~GDBJITRegistrar();
|
||||
|
||||
/// Creates an entry in the JIT registry for the buffer @p Object,
|
||||
/// which must contain an object file in executable memory with any
|
||||
/// debug information for the debugger.
|
||||
void registerObject(const MemoryBuffer &Object);
|
||||
|
||||
/// Removes the internal registration of @p Object, and
|
||||
/// frees associated resources.
|
||||
/// Returns true if @p Object was found in ObjectBufferMap.
|
||||
bool deregisterObject(const MemoryBuffer &Object);
|
||||
|
||||
private:
|
||||
/// Deregister the debug info for the given object file from the debugger
|
||||
/// and delete any temporary copies. This private method does not remove
|
||||
/// the function from Map so that it can be called while iterating over Map.
|
||||
void deregisterObjectInternal(RegisteredObjectBufferMap::iterator I);
|
||||
};
|
||||
|
||||
/// Lock used to serialize all jit registration events, since they
|
||||
/// modify global variables.
|
||||
llvm::sys::Mutex JITDebugLock;
|
||||
|
||||
/// Acquire the lock and do the registration.
|
||||
void NotifyDebugger(jit_code_entry* JITCodeEntry) {
|
||||
llvm::MutexGuard locked(JITDebugLock);
|
||||
__jit_debug_descriptor.action_flag = JIT_REGISTER_FN;
|
||||
|
||||
// Insert this entry at the head of the list.
|
||||
JITCodeEntry->prev_entry = NULL;
|
||||
jit_code_entry* NextEntry = __jit_debug_descriptor.first_entry;
|
||||
JITCodeEntry->next_entry = NextEntry;
|
||||
if (NextEntry != NULL) {
|
||||
NextEntry->prev_entry = JITCodeEntry;
|
||||
}
|
||||
__jit_debug_descriptor.first_entry = JITCodeEntry;
|
||||
__jit_debug_descriptor.relevant_entry = JITCodeEntry;
|
||||
__jit_debug_register_code();
|
||||
}
|
||||
|
||||
GDBJITRegistrar::~GDBJITRegistrar() {
|
||||
// Free all registered object files.
|
||||
for (RegisteredObjectBufferMap::iterator I = ObjectBufferMap.begin(), E = ObjectBufferMap.end();
|
||||
I != E; ++I) {
|
||||
// Call the private method that doesn't update the map so our iterator
|
||||
// doesn't break.
|
||||
deregisterObjectInternal(I);
|
||||
}
|
||||
ObjectBufferMap.clear();
|
||||
}
|
||||
|
||||
void GDBJITRegistrar::registerObject(const MemoryBuffer &Object) {
|
||||
|
||||
const char *Buffer = Object.getBufferStart();
|
||||
size_t Size = Object.getBufferSize();
|
||||
|
||||
assert(Buffer && "Attempt to register a null object with a debugger.");
|
||||
assert(ObjectBufferMap.find(Buffer) == ObjectBufferMap.end() &&
|
||||
"Second attempt to perform debug registration.");
|
||||
jit_code_entry* JITCodeEntry = new jit_code_entry();
|
||||
|
||||
if (JITCodeEntry == 0) {
|
||||
llvm::report_fatal_error(
|
||||
"Allocation failed when registering a JIT entry!\n");
|
||||
}
|
||||
else {
|
||||
JITCodeEntry->symfile_addr = Buffer;
|
||||
JITCodeEntry->symfile_size = Size;
|
||||
|
||||
ObjectBufferMap[Buffer] = std::make_pair(Size, JITCodeEntry);
|
||||
NotifyDebugger(JITCodeEntry);
|
||||
}
|
||||
}
|
||||
|
||||
bool GDBJITRegistrar::deregisterObject(const MemoryBuffer& Object) {
|
||||
const char *Buffer = Object.getBufferStart();
|
||||
RegisteredObjectBufferMap::iterator I = ObjectBufferMap.find(Buffer);
|
||||
|
||||
if (I != ObjectBufferMap.end()) {
|
||||
deregisterObjectInternal(I);
|
||||
ObjectBufferMap.erase(I);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void GDBJITRegistrar::deregisterObjectInternal(
|
||||
RegisteredObjectBufferMap::iterator I) {
|
||||
|
||||
jit_code_entry*& JITCodeEntry = I->second.second;
|
||||
|
||||
// Acquire the lock and do the unregistration.
|
||||
{
|
||||
llvm::MutexGuard locked(JITDebugLock);
|
||||
__jit_debug_descriptor.action_flag = JIT_UNREGISTER_FN;
|
||||
|
||||
// Remove the jit_code_entry from the linked list.
|
||||
jit_code_entry* PrevEntry = JITCodeEntry->prev_entry;
|
||||
jit_code_entry* NextEntry = JITCodeEntry->next_entry;
|
||||
|
||||
if (NextEntry) {
|
||||
NextEntry->prev_entry = PrevEntry;
|
||||
}
|
||||
if (PrevEntry) {
|
||||
PrevEntry->next_entry = NextEntry;
|
||||
}
|
||||
else {
|
||||
assert(__jit_debug_descriptor.first_entry == JITCodeEntry);
|
||||
__jit_debug_descriptor.first_entry = NextEntry;
|
||||
}
|
||||
|
||||
// Tell the debugger which entry we removed, and unregister the code.
|
||||
__jit_debug_descriptor.relevant_entry = JITCodeEntry;
|
||||
__jit_debug_register_code();
|
||||
}
|
||||
|
||||
delete JITCodeEntry;
|
||||
JITCodeEntry = NULL;
|
||||
}
|
||||
|
||||
} // end namespace
|
||||
|
||||
namespace llvm {
|
||||
|
||||
JITRegistrar& JITRegistrar::getGDBRegistrar() {
|
||||
static GDBJITRegistrar* sRegistrar = NULL;
|
||||
if (sRegistrar == NULL) {
|
||||
// The mutex is here so that it won't slow down access once the registrar
|
||||
// is instantiated
|
||||
llvm::MutexGuard locked(JITDebugLock);
|
||||
// Check again to be sure another thread didn't create this while we waited
|
||||
if (sRegistrar == NULL) {
|
||||
sRegistrar = new GDBJITRegistrar;
|
||||
}
|
||||
}
|
||||
return *sRegistrar;
|
||||
}
|
||||
|
||||
} // namespace llvm
|
|
@ -0,0 +1,43 @@
|
|||
//===-- JITRegistrar.h - Registers objects with a debugger ----------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_EXECUTION_ENGINE_JIT_REGISTRAR_H
|
||||
#define LLVM_EXECUTION_ENGINE_JIT_REGISTRAR_H
|
||||
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
|
||||
namespace llvm {
|
||||
|
||||
/// Global access point for the JIT debugging interface.
|
||||
class JITRegistrar {
|
||||
public:
|
||||
/// Instantiates the JIT service.
|
||||
JITRegistrar() {}
|
||||
|
||||
/// Unregisters each object that was previously registered and releases all
|
||||
/// internal resources.
|
||||
virtual ~JITRegistrar() {}
|
||||
|
||||
/// Creates an entry in the JIT registry for the buffer @p Object,
|
||||
/// which must contain an object file in executable memory with any
|
||||
/// debug information for the debugger.
|
||||
virtual void registerObject(const MemoryBuffer &Object) = 0;
|
||||
|
||||
/// Removes the internal registration of @p Object, and
|
||||
/// frees associated resources.
|
||||
/// Returns true if @p Object was previously registered.
|
||||
virtual bool deregisterObject(const MemoryBuffer &Object) = 0;
|
||||
|
||||
/// Returns a reference to a GDB JIT registrar singleton
|
||||
static JITRegistrar& getGDBRegistrar();
|
||||
};
|
||||
|
||||
} // end namespace llvm
|
||||
|
||||
#endif // LLVM_EXECUTION_ENGINE_JIT_REGISTRAR_H
|
|
@ -0,0 +1,59 @@
|
|||
//===---- ObjectImage.h - Format independent executuable object image -----===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file declares a file format independent ObjectImage class.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_RUNTIMEDYLD_OBJECT_IMAGE_H
|
||||
#define LLVM_RUNTIMEDYLD_OBJECT_IMAGE_H
|
||||
|
||||
#include "llvm/Object/ObjectFile.h"
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class ObjectImage {
|
||||
ObjectImage(); // = delete
|
||||
ObjectImage(const ObjectImage &other); // = delete
|
||||
protected:
|
||||
object::ObjectFile *ObjFile;
|
||||
|
||||
public:
|
||||
ObjectImage(object::ObjectFile *Obj) { ObjFile = Obj; }
|
||||
virtual ~ObjectImage() {}
|
||||
|
||||
virtual object::symbol_iterator begin_symbols() const
|
||||
{ return ObjFile->begin_symbols(); }
|
||||
virtual object::symbol_iterator end_symbols() const
|
||||
{ return ObjFile->end_symbols(); }
|
||||
|
||||
virtual object::section_iterator begin_sections() const
|
||||
{ return ObjFile->begin_sections(); }
|
||||
virtual object::section_iterator end_sections() const
|
||||
{ return ObjFile->end_sections(); }
|
||||
|
||||
virtual /* Triple::ArchType */ unsigned getArch() const
|
||||
{ return ObjFile->getArch(); }
|
||||
|
||||
// Subclasses can override these methods to update the image with loaded
|
||||
// addresses for sections and common symbols
|
||||
virtual void updateSectionAddress(const object::SectionRef &Sec,
|
||||
uint64_t Addr) {}
|
||||
virtual void updateSymbolAddress(const object::SymbolRef &Sym, uint64_t Addr)
|
||||
{}
|
||||
|
||||
// Subclasses can override this method to provide JIT debugging support
|
||||
virtual void registerWithDebugger() {}
|
||||
virtual void deregisterWithDebugger() {}
|
||||
};
|
||||
|
||||
} // end namespace llvm
|
||||
|
||||
#endif // LLVM_RUNTIMEDYLD_OBJECT_IMAGE_H
|
||||
|
Loading…
Reference in New Issue