forked from OSchip/llvm-project
ManagedStatic: remove from GDBRegistrationListener
An earlier version of this change originally landed as part of
e6f1f06245
(D129120), which caused a
Fuchsia buildbot regression in ExecutionEngine tests.
Careful review suggests that the issue was that in the earlier version,
the destructor of the JITDebugLock was run before the destructor of
GDBJITRegistrationListener. The new version of the change moves the lock
to a member variable of the (singleton!) GDBJITRegistartionListener so
that destructors are run in the right order.
This commit is contained in:
parent
d55985789b
commit
c4ccf608c2
|
@ -12,7 +12,6 @@
|
||||||
#include "llvm/Object/ObjectFile.h"
|
#include "llvm/Object/ObjectFile.h"
|
||||||
#include "llvm/Support/Compiler.h"
|
#include "llvm/Support/Compiler.h"
|
||||||
#include "llvm/Support/ErrorHandling.h"
|
#include "llvm/Support/ErrorHandling.h"
|
||||||
#include "llvm/Support/ManagedStatic.h"
|
|
||||||
#include "llvm/Support/MemoryBuffer.h"
|
#include "llvm/Support/MemoryBuffer.h"
|
||||||
#include "llvm/Support/Mutex.h"
|
#include "llvm/Support/Mutex.h"
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
|
@ -91,11 +90,18 @@ typedef llvm::DenseMap<JITEventListener::ObjectKey, RegisteredObjectInfo>
|
||||||
/// object files that are in executable memory managed by the client of this
|
/// object files that are in executable memory managed by the client of this
|
||||||
/// class.
|
/// class.
|
||||||
class GDBJITRegistrationListener : public JITEventListener {
|
class GDBJITRegistrationListener : public JITEventListener {
|
||||||
|
/// Lock used to serialize all jit registration events, since they
|
||||||
|
/// modify global variables.
|
||||||
|
///
|
||||||
|
/// Only a single instance of GDBJITRegistrationListener is ever created,
|
||||||
|
/// and so the lock can be a member variable of that instance. This ensures
|
||||||
|
/// destructors are run in the correct order.
|
||||||
|
sys::Mutex JITDebugLock;
|
||||||
|
|
||||||
/// A map of in-memory object files that have been registered with the
|
/// A map of in-memory object files that have been registered with the
|
||||||
/// JIT interface.
|
/// JIT interface.
|
||||||
RegisteredObjectBufferMap ObjectBufferMap;
|
RegisteredObjectBufferMap ObjectBufferMap;
|
||||||
|
|
||||||
public:
|
|
||||||
/// Instantiates the JIT service.
|
/// Instantiates the JIT service.
|
||||||
GDBJITRegistrationListener() = default;
|
GDBJITRegistrationListener() = default;
|
||||||
|
|
||||||
|
@ -103,6 +109,12 @@ public:
|
||||||
/// internal resources.
|
/// internal resources.
|
||||||
~GDBJITRegistrationListener() override;
|
~GDBJITRegistrationListener() override;
|
||||||
|
|
||||||
|
public:
|
||||||
|
static GDBJITRegistrationListener &instance() {
|
||||||
|
static GDBJITRegistrationListener Instance;
|
||||||
|
return Instance;
|
||||||
|
}
|
||||||
|
|
||||||
/// Creates an entry in the JIT registry for the buffer @p Object,
|
/// Creates an entry in the JIT registry for the buffer @p Object,
|
||||||
/// which must contain an object file in executable memory with any
|
/// which must contain an object file in executable memory with any
|
||||||
/// debug information for the debugger.
|
/// debug information for the debugger.
|
||||||
|
@ -121,10 +133,6 @@ private:
|
||||||
void deregisterObjectInternal(RegisteredObjectBufferMap::iterator I);
|
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.
|
/// Do the registration.
|
||||||
void NotifyDebugger(jit_code_entry* JITCodeEntry) {
|
void NotifyDebugger(jit_code_entry* JITCodeEntry) {
|
||||||
__jit_debug_descriptor.action_flag = JIT_REGISTER_FN;
|
__jit_debug_descriptor.action_flag = JIT_REGISTER_FN;
|
||||||
|
@ -143,7 +151,7 @@ void NotifyDebugger(jit_code_entry* JITCodeEntry) {
|
||||||
|
|
||||||
GDBJITRegistrationListener::~GDBJITRegistrationListener() {
|
GDBJITRegistrationListener::~GDBJITRegistrationListener() {
|
||||||
// Free all registered object files.
|
// Free all registered object files.
|
||||||
std::lock_guard<llvm::sys::Mutex> locked(*JITDebugLock);
|
std::lock_guard<llvm::sys::Mutex> locked(JITDebugLock);
|
||||||
for (RegisteredObjectBufferMap::iterator I = ObjectBufferMap.begin(),
|
for (RegisteredObjectBufferMap::iterator I = ObjectBufferMap.begin(),
|
||||||
E = ObjectBufferMap.end();
|
E = ObjectBufferMap.end();
|
||||||
I != E; ++I) {
|
I != E; ++I) {
|
||||||
|
@ -167,7 +175,7 @@ void GDBJITRegistrationListener::notifyObjectLoaded(
|
||||||
const char *Buffer = DebugObj.getBinary()->getMemoryBufferRef().getBufferStart();
|
const char *Buffer = DebugObj.getBinary()->getMemoryBufferRef().getBufferStart();
|
||||||
size_t Size = DebugObj.getBinary()->getMemoryBufferRef().getBufferSize();
|
size_t Size = DebugObj.getBinary()->getMemoryBufferRef().getBufferSize();
|
||||||
|
|
||||||
std::lock_guard<llvm::sys::Mutex> locked(*JITDebugLock);
|
std::lock_guard<llvm::sys::Mutex> locked(JITDebugLock);
|
||||||
assert(ObjectBufferMap.find(K) == ObjectBufferMap.end() &&
|
assert(ObjectBufferMap.find(K) == ObjectBufferMap.end() &&
|
||||||
"Second attempt to perform debug registration.");
|
"Second attempt to perform debug registration.");
|
||||||
jit_code_entry* JITCodeEntry = new jit_code_entry();
|
jit_code_entry* JITCodeEntry = new jit_code_entry();
|
||||||
|
@ -186,7 +194,7 @@ void GDBJITRegistrationListener::notifyObjectLoaded(
|
||||||
}
|
}
|
||||||
|
|
||||||
void GDBJITRegistrationListener::notifyFreeingObject(ObjectKey K) {
|
void GDBJITRegistrationListener::notifyFreeingObject(ObjectKey K) {
|
||||||
std::lock_guard<llvm::sys::Mutex> locked(*JITDebugLock);
|
std::lock_guard<llvm::sys::Mutex> locked(JITDebugLock);
|
||||||
RegisteredObjectBufferMap::iterator I = ObjectBufferMap.find(K);
|
RegisteredObjectBufferMap::iterator I = ObjectBufferMap.find(K);
|
||||||
|
|
||||||
if (I != ObjectBufferMap.end()) {
|
if (I != ObjectBufferMap.end()) {
|
||||||
|
@ -228,14 +236,12 @@ void GDBJITRegistrationListener::deregisterObjectInternal(
|
||||||
JITCodeEntry = nullptr;
|
JITCodeEntry = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
llvm::ManagedStatic<GDBJITRegistrationListener> GDBRegListener;
|
|
||||||
|
|
||||||
} // end namespace
|
} // end namespace
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
JITEventListener* JITEventListener::createGDBRegistrationListener() {
|
JITEventListener* JITEventListener::createGDBRegistrationListener() {
|
||||||
return &*GDBRegListener;
|
return &GDBJITRegistrationListener::instance();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace llvm
|
} // namespace llvm
|
||||||
|
|
Loading…
Reference in New Issue