mirror of
https://github.com/corda/corda.git
synced 2025-01-22 12:28:11 +00:00
fix crash on exit due to order of operations bug in ~RawMonitorResource
The problem (which we've only been able to reproduce consistently with the openjdk-src process=interpret build on Linux virtual machines) was a race condition during VM shutdown. Thread "A" would exit, see there were other threads still running and thus enter ZombieState, which involves acquiring and releasing a lock using RawMonitorResource. Then the last thread (thread "B") would exit, wait for thread "A" to release the lock, then shut down the VM, freeing all memory. However, thread "A" writes to its Thread object one last time after releasing the lock (in ~Resource, the destructor of the superclass of RawMonitorResource, which sets Thread::resource). If thread "B" frees that Thread before ~Resource runs, we end up writing to freed memory. Thus, we need to update Thread::resource before releasing the lock. Apparently C++ destructors run in order from most derived to least derived, which is not what we want here. My solution to split Resource into two classes, one that has no destructor and another that extends it (called AutoResource) which does hafe a destructor. Now all the classes which used to extend Resource extend AutoResource, except for RawMonitorResource, which extends Resource directly so it can control the order of operations.
This commit is contained in:
parent
10056734e2
commit
4a83b671b3
@ -48,19 +48,19 @@ using namespace avian::util;
|
||||
#define ENTER(t, state) StateResource MAKE_NAME(stateResource_) (t, state)
|
||||
|
||||
#define THREAD_RESOURCE0(t, releaseBody) \
|
||||
class MAKE_NAME(Resource_): public Thread::Resource { \
|
||||
class MAKE_NAME(Resource_): public Thread::AutoResource { \
|
||||
public: \
|
||||
MAKE_NAME(Resource_)(Thread* t): Resource(t) { } \
|
||||
MAKE_NAME(Resource_)(Thread* t): AutoResource(t) { } \
|
||||
~MAKE_NAME(Resource_)() { releaseBody; } \
|
||||
virtual void release() \
|
||||
{ this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); } \
|
||||
} MAKE_NAME(resource_)(t);
|
||||
|
||||
#define OBJECT_RESOURCE(t, name, releaseBody) \
|
||||
class MAKE_NAME(Resource_): public Thread::Resource { \
|
||||
class MAKE_NAME(Resource_): public Thread::AutoResource { \
|
||||
public: \
|
||||
MAKE_NAME(Resource_)(Thread* t, object name): \
|
||||
Resource(t), name(name), protector(t, &(this->name)) { } \
|
||||
AutoResource(t), name(name), protector(t, &(this->name)) { } \
|
||||
~MAKE_NAME(Resource_)() { releaseBody; } \
|
||||
virtual void release() \
|
||||
{ this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); } \
|
||||
@ -71,10 +71,10 @@ using namespace avian::util;
|
||||
} MAKE_NAME(resource_)(t, name);
|
||||
|
||||
#define THREAD_RESOURCE(t, type, name, releaseBody) \
|
||||
class MAKE_NAME(Resource_): public Thread::Resource { \
|
||||
class MAKE_NAME(Resource_): public Thread::AutoResource { \
|
||||
public: \
|
||||
MAKE_NAME(Resource_)(Thread* t, type name): \
|
||||
Resource(t), name(name) { } \
|
||||
AutoResource(t), name(name) { } \
|
||||
~MAKE_NAME(Resource_)() { releaseBody; } \
|
||||
virtual void release() \
|
||||
{ this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); } \
|
||||
@ -84,10 +84,10 @@ using namespace avian::util;
|
||||
} MAKE_NAME(resource_)(t, name);
|
||||
|
||||
#define THREAD_RESOURCE2(t, type1, name1, type2, name2, releaseBody) \
|
||||
class MAKE_NAME(Resource_): public Thread::Resource { \
|
||||
class MAKE_NAME(Resource_): public Thread::AutoResource { \
|
||||
public: \
|
||||
MAKE_NAME(Resource_)(Thread* t, type1 name1, type2 name2): \
|
||||
Resource(t), name1(name1), name2(name2) { } \
|
||||
AutoResource(t), name1(name1), name2(name2) { } \
|
||||
~MAKE_NAME(Resource_)() { releaseBody; } \
|
||||
virtual void release() \
|
||||
{ this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); } \
|
||||
@ -1375,24 +1375,32 @@ class Thread {
|
||||
|
||||
class Resource {
|
||||
public:
|
||||
Resource(Thread* t): t(t), next(t->resource) {
|
||||
Resource(Thread* t, Resource* next): t(t), next(next) {
|
||||
t->resource = this;
|
||||
}
|
||||
|
||||
~Resource() {
|
||||
t->resource = next;
|
||||
}
|
||||
|
||||
virtual void release() = 0;
|
||||
|
||||
Thread* t;
|
||||
Resource* next;
|
||||
};
|
||||
|
||||
class ClassInitStack: public Resource {
|
||||
class AutoResource: public Resource {
|
||||
public:
|
||||
AutoResource(Thread* t): Resource(t, t->resource) { }
|
||||
|
||||
~AutoResource() {
|
||||
t->resource = next;
|
||||
}
|
||||
|
||||
virtual void release() = 0;
|
||||
|
||||
};
|
||||
|
||||
class ClassInitStack: public AutoResource {
|
||||
public:
|
||||
ClassInitStack(Thread* t, object class_):
|
||||
Resource(t),
|
||||
AutoResource(t),
|
||||
next(t->classInitStack),
|
||||
class_(class_),
|
||||
protector(t, &(this->class_))
|
||||
@ -1587,10 +1595,10 @@ class Classpath {
|
||||
#ifdef _MSC_VER
|
||||
|
||||
template <class T>
|
||||
class ThreadRuntimeArray: public Thread::Resource {
|
||||
class ThreadRuntimeArray: public Thread::AutoResource {
|
||||
public:
|
||||
ThreadRuntimeArray(Thread* t, unsigned size):
|
||||
Resource(t),
|
||||
AutoResource(t),
|
||||
body(static_cast<T*>(t->m->heap->allocate(size * sizeof(T)))),
|
||||
size(size)
|
||||
{ }
|
||||
@ -1644,10 +1652,10 @@ enterActiveState(Thread* t)
|
||||
enter(t, Thread::ActiveState);
|
||||
}
|
||||
|
||||
class StateResource: public Thread::Resource {
|
||||
class StateResource: public Thread::AutoResource {
|
||||
public:
|
||||
StateResource(Thread* t, Thread::State state):
|
||||
Resource(t), oldState(t->state)
|
||||
AutoResource(t), oldState(t->state)
|
||||
{
|
||||
enter(t, state);
|
||||
}
|
||||
@ -1733,10 +1741,10 @@ release(Thread* t, System::Monitor* m)
|
||||
m->release(t->systemThread);
|
||||
}
|
||||
|
||||
class MonitorResource: public Thread::Resource {
|
||||
class MonitorResource: public Thread::AutoResource {
|
||||
public:
|
||||
MonitorResource(Thread* t, System::Monitor* m):
|
||||
Resource(t), m(m)
|
||||
AutoResource(t), m(m)
|
||||
{
|
||||
acquire(t, m);
|
||||
}
|
||||
@ -1756,12 +1764,13 @@ class MonitorResource: public Thread::Resource {
|
||||
class RawMonitorResource: public Thread::Resource {
|
||||
public:
|
||||
RawMonitorResource(Thread* t, System::Monitor* m):
|
||||
Resource(t), m(m)
|
||||
Resource(t, t->resource), m(m)
|
||||
{
|
||||
m->acquire(t->systemThread);
|
||||
}
|
||||
|
||||
~RawMonitorResource() {
|
||||
t->resource = next;
|
||||
vm::release(t, m);
|
||||
}
|
||||
|
||||
|
@ -1009,9 +1009,9 @@ class BootContext {
|
||||
|
||||
class Context {
|
||||
public:
|
||||
class MyResource: public Thread::Resource {
|
||||
class MyResource: public Thread::AutoResource {
|
||||
public:
|
||||
MyResource(Context* c): Resource(c->thread), c(c) { }
|
||||
MyResource(Context* c): AutoResource(c->thread), c(c) { }
|
||||
|
||||
virtual void release() {
|
||||
c->dispose();
|
||||
@ -3857,9 +3857,9 @@ targetFieldOffset(Context* context, object field)
|
||||
|
||||
class Stack {
|
||||
public:
|
||||
class MyResource: public Thread::Resource {
|
||||
class MyResource: public Thread::AutoResource {
|
||||
public:
|
||||
MyResource(Stack* s): Resource(s->thread), s(s) { }
|
||||
MyResource(Stack* s): AutoResource(s->thread), s(s) { }
|
||||
|
||||
virtual void release() {
|
||||
s->zone.dispose();
|
||||
|
Loading…
Reference in New Issue
Block a user