mirror of
https://github.com/RPCS3/llvm.git
synced 2025-05-17 19:06:09 +00:00

to reflect the new license. We understand that people may be surprised that we're moving the header entirely to discuss the new license. We checked this carefully with the Foundation's lawyer and we believe this is the correct approach. Essentially, all code in the project is now made available by the LLVM project under our new license, so you will see that the license headers include that license only. Some of our contributors have contributed code under our old license, and accordingly, we have retained a copy of our old license notice in the top-level files in each project and repository. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@351636 91177308-0d34-0410-b5e6-96231b3b80d8
239 lines
7.5 KiB
C++
239 lines
7.5 KiB
C++
//===----- GDBRegistrationListener.cpp - Registers objects with GDB -------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm-c/ExecutionEngine.h"
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ExecutionEngine/JITEventListener.h"
|
|
#include "llvm/Object/ObjectFile.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
#include "llvm/Support/ManagedStatic.h"
|
|
#include "llvm/Support/Mutex.h"
|
|
#include "llvm/Support/MutexGuard.h"
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::object;
|
|
|
|
// 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.
|
|
struct jit_descriptor __jit_debug_descriptor = { 1, 0, nullptr, nullptr };
|
|
|
|
// Debuggers puts a breakpoint in this function.
|
|
LLVM_ATTRIBUTE_NOINLINE void __jit_debug_register_code() {
|
|
// The noinline and the asm prevent calls to this function from being
|
|
// optimized out.
|
|
#if !defined(_MSC_VER)
|
|
asm volatile("":::"memory");
|
|
#endif
|
|
}
|
|
|
|
}
|
|
|
|
namespace {
|
|
|
|
struct RegisteredObjectInfo {
|
|
RegisteredObjectInfo() {}
|
|
|
|
RegisteredObjectInfo(std::size_t Size, jit_code_entry *Entry,
|
|
OwningBinary<ObjectFile> Obj)
|
|
: Size(Size), Entry(Entry), Obj(std::move(Obj)) {}
|
|
|
|
std::size_t Size;
|
|
jit_code_entry *Entry;
|
|
OwningBinary<ObjectFile> Obj;
|
|
};
|
|
|
|
// Buffer for an in-memory object file in executable memory
|
|
typedef llvm::DenseMap<JITEventListener::ObjectKey, RegisteredObjectInfo>
|
|
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 GDBJITRegistrationListener : public JITEventListener {
|
|
/// A map of in-memory object files that have been registered with the
|
|
/// JIT interface.
|
|
RegisteredObjectBufferMap ObjectBufferMap;
|
|
|
|
public:
|
|
/// Instantiates the JIT service.
|
|
GDBJITRegistrationListener() : ObjectBufferMap() {}
|
|
|
|
/// Unregisters each object that was previously registered and releases all
|
|
/// internal resources.
|
|
~GDBJITRegistrationListener() override;
|
|
|
|
/// 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 notifyObjectLoaded(ObjectKey K, const ObjectFile &Obj,
|
|
const RuntimeDyld::LoadedObjectInfo &L) override;
|
|
|
|
/// Removes the internal registration of @p Object, and
|
|
/// frees associated resources.
|
|
/// Returns true if @p Object was found in ObjectBufferMap.
|
|
void notifyFreeingObject(ObjectKey K) override;
|
|
|
|
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.
|
|
ManagedStatic<sys::Mutex> JITDebugLock;
|
|
|
|
/// Do the registration.
|
|
void NotifyDebugger(jit_code_entry* JITCodeEntry) {
|
|
__jit_debug_descriptor.action_flag = JIT_REGISTER_FN;
|
|
|
|
// Insert this entry at the head of the list.
|
|
JITCodeEntry->prev_entry = nullptr;
|
|
jit_code_entry* NextEntry = __jit_debug_descriptor.first_entry;
|
|
JITCodeEntry->next_entry = NextEntry;
|
|
if (NextEntry) {
|
|
NextEntry->prev_entry = JITCodeEntry;
|
|
}
|
|
__jit_debug_descriptor.first_entry = JITCodeEntry;
|
|
__jit_debug_descriptor.relevant_entry = JITCodeEntry;
|
|
__jit_debug_register_code();
|
|
}
|
|
|
|
GDBJITRegistrationListener::~GDBJITRegistrationListener() {
|
|
// Free all registered object files.
|
|
llvm::MutexGuard locked(*JITDebugLock);
|
|
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 GDBJITRegistrationListener::notifyObjectLoaded(
|
|
ObjectKey K, const ObjectFile &Obj,
|
|
const RuntimeDyld::LoadedObjectInfo &L) {
|
|
|
|
OwningBinary<ObjectFile> DebugObj = L.getObjectForDebug(Obj);
|
|
|
|
// Bail out if debug objects aren't supported.
|
|
if (!DebugObj.getBinary())
|
|
return;
|
|
|
|
const char *Buffer = DebugObj.getBinary()->getMemoryBufferRef().getBufferStart();
|
|
size_t Size = DebugObj.getBinary()->getMemoryBufferRef().getBufferSize();
|
|
|
|
llvm::MutexGuard locked(*JITDebugLock);
|
|
assert(ObjectBufferMap.find(K) == ObjectBufferMap.end() &&
|
|
"Second attempt to perform debug registration.");
|
|
jit_code_entry* JITCodeEntry = new jit_code_entry();
|
|
|
|
if (!JITCodeEntry) {
|
|
llvm::report_fatal_error(
|
|
"Allocation failed when registering a JIT entry!\n");
|
|
} else {
|
|
JITCodeEntry->symfile_addr = Buffer;
|
|
JITCodeEntry->symfile_size = Size;
|
|
|
|
ObjectBufferMap[K] =
|
|
RegisteredObjectInfo(Size, JITCodeEntry, std::move(DebugObj));
|
|
NotifyDebugger(JITCodeEntry);
|
|
}
|
|
}
|
|
|
|
void GDBJITRegistrationListener::notifyFreeingObject(ObjectKey K) {
|
|
llvm::MutexGuard locked(*JITDebugLock);
|
|
RegisteredObjectBufferMap::iterator I = ObjectBufferMap.find(K);
|
|
|
|
if (I != ObjectBufferMap.end()) {
|
|
deregisterObjectInternal(I);
|
|
ObjectBufferMap.erase(I);
|
|
}
|
|
}
|
|
|
|
void GDBJITRegistrationListener::deregisterObjectInternal(
|
|
RegisteredObjectBufferMap::iterator I) {
|
|
|
|
jit_code_entry*& JITCodeEntry = I->second.Entry;
|
|
|
|
// Do the unregistration.
|
|
{
|
|
__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 = nullptr;
|
|
}
|
|
|
|
llvm::ManagedStatic<GDBJITRegistrationListener> GDBRegListener;
|
|
|
|
} // end namespace
|
|
|
|
namespace llvm {
|
|
|
|
JITEventListener* JITEventListener::createGDBRegistrationListener() {
|
|
return &*GDBRegListener;
|
|
}
|
|
|
|
} // namespace llvm
|
|
|
|
LLVMJITEventListenerRef LLVMCreateGDBRegistrationListener(void)
|
|
{
|
|
return wrap(JITEventListener::createGDBRegistrationListener());
|
|
}
|