Skip to content

Commit

Permalink
Thread-safe signalling inside Mutex class.
Browse files Browse the repository at this point in the history
  • Loading branch information
RadiantUwU committed Nov 26, 2024
1 parent d09d82d commit d8f5f57
Show file tree
Hide file tree
Showing 5 changed files with 90 additions and 13 deletions.
55 changes: 55 additions & 0 deletions core/core_bind.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1301,6 +1301,61 @@ void Mutex::unlock() {
mutex.unlock();
}

bool Mutex::_mt_disconnect(const StringName &p_signal, const Callable &p_callable, bool p_force) {
MutexLock lock(mutex);
return Object::_disconnect(p_signal, p_callable, p_force);
}

Error Mutex::emit_signalp(const StringName &p_name, const Variant **p_args, int p_argcount) {
MutexLock lock(mutex);
return Object::emit_signalp(p_name, p_args, p_argcount);
}

bool Mutex::has_signal(const StringName &p_name) const {
MutexLock lock(mutex);
return Object::has_signal(p_name);
}

void Mutex::get_signal_list(List<MethodInfo> *p_signals) const {
MutexLock lock(mutex);
Object::get_signal_list(p_signals);
}

void Mutex::get_signal_connection_list(const StringName &p_signal, List<Connection> *p_connections) const {
MutexLock lock(mutex);
Object::get_signal_connection_list(p_signal, p_connections);
}

void Mutex::get_all_signal_connections(List<Connection> *p_connections) const {
MutexLock lock(mutex);
Object::get_all_signal_connections(p_connections);
}

int Mutex::get_persistent_signal_connection_count() const {
MutexLock lock(mutex);
return Object::get_persistent_signal_connection_count();
}

void Mutex::get_signals_connected_to_this(List<Connection> *p_connections) const {
MutexLock lock(mutex);
Object::get_signals_connected_to_this(p_connections);
}

Error Mutex::connect(const StringName &p_signal, const Callable &p_callable, uint32_t p_flags) {
MutexLock lock(mutex);
return Object::connect(p_signal, p_callable, p_flags);
}

void Mutex::disconnect(const StringName &p_signal, const Callable &p_callable) {
MutexLock lock(mutex);
Object::disconnect(p_signal, p_callable);
}

bool Mutex::is_connected(const StringName &p_signal, const Callable &p_callable) const {
MutexLock lock(mutex);
return Object::is_connected(p_signal, p_callable);
}

void Mutex::_bind_methods() {
ClassDB::bind_method(D_METHOD("lock"), &Mutex::lock);
ClassDB::bind_method(D_METHOD("try_lock"), &Mutex::try_lock);
Expand Down
15 changes: 15 additions & 0 deletions core/core_bind.h
Original file line number Diff line number Diff line change
Expand Up @@ -407,10 +407,25 @@ class Mutex : public RefCounted {

static void _bind_methods();

protected:
virtual bool _mt_disconnect(const StringName &p_signal, const Callable &p_callable, bool p_force = false) override;

public:
void lock();
bool try_lock();
void unlock();

virtual Error emit_signalp(const StringName &p_name, const Variant **p_args, int p_argcount) override;
virtual bool has_signal(const StringName &p_name) const override;
virtual void get_signal_list(List<MethodInfo> *p_signals) const override;
virtual void get_signal_connection_list(const StringName &p_signal, List<Connection> *p_connections) const override;
virtual void get_all_signal_connections(List<Connection> *p_connections) const override;
virtual int get_persistent_signal_connection_count() const override;
virtual void get_signals_connected_to_this(List<Connection> *p_connections) const override;

virtual Error connect(const StringName &p_signal, const Callable &p_callable, uint32_t p_flags = 0) override;
virtual void disconnect(const StringName &p_signal, const Callable &p_callable) override;
virtual bool is_connected(const StringName &p_signal, const Callable &p_callable) const override;
};

class Semaphore : public RefCounted {
Expand Down
2 changes: 1 addition & 1 deletion core/object/object.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2153,7 +2153,7 @@ Object::~Object() {
Object *obj = c.callable.get_object();
bool disconnected = false;
if (likely(obj)) {
disconnected = c.signal.get_object()->_disconnect(c.signal.get_name(), c.callable, true);
disconnected = c.signal.get_object()->_mt_disconnect(c.signal.get_name(), c.callable, true);
}
if (unlikely(!disconnected)) {
// If the disconnect has failed, abandon the connection to avoid getting trapped in an infinite loop here.
Expand Down
29 changes: 17 additions & 12 deletions core/object/object.h
Original file line number Diff line number Diff line change
Expand Up @@ -756,6 +756,11 @@ class Object {

bool _disconnect(const StringName &p_signal, const Callable &p_callable, bool p_force = false);

// When a multi-threaded guard is required.
virtual bool _mt_disconnect(const StringName &p_signal, const Callable &p_callable, bool p_force = false) {
return _disconnect(p_signal, p_callable, p_force);
}

#ifdef TOOLS_ENABLED
struct VirtualMethodTracker {
void **method;
Expand Down Expand Up @@ -921,18 +926,18 @@ class Object {
return emit_signalp(p_name, sizeof...(p_args) == 0 ? nullptr : (const Variant **)argptrs, sizeof...(p_args));
}

MTVIRTUAL Error emit_signalp(const StringName &p_name, const Variant **p_args, int p_argcount);
MTVIRTUAL bool has_signal(const StringName &p_name) const;
MTVIRTUAL void get_signal_list(List<MethodInfo> *p_signals) const;
MTVIRTUAL void get_signal_connection_list(const StringName &p_signal, List<Connection> *p_connections) const;
MTVIRTUAL void get_all_signal_connections(List<Connection> *p_connections) const;
MTVIRTUAL int get_persistent_signal_connection_count() const;
MTVIRTUAL void get_signals_connected_to_this(List<Connection> *p_connections) const;

MTVIRTUAL Error connect(const StringName &p_signal, const Callable &p_callable, uint32_t p_flags = 0);
MTVIRTUAL void disconnect(const StringName &p_signal, const Callable &p_callable);
MTVIRTUAL bool is_connected(const StringName &p_signal, const Callable &p_callable) const;
MTVIRTUAL bool has_connections(const StringName &p_signal) const;
virtual Error emit_signalp(const StringName &p_name, const Variant **p_args, int p_argcount);
virtual bool has_signal(const StringName &p_name) const;
virtual void get_signal_list(List<MethodInfo> *p_signals) const;
virtual void get_signal_connection_list(const StringName &p_signal, List<Connection> *p_connections) const;
virtual void get_all_signal_connections(List<Connection> *p_connections) const;
virtual int get_persistent_signal_connection_count() const;
virtual void get_signals_connected_to_this(List<Connection> *p_connections) const;

virtual Error connect(const StringName &p_signal, const Callable &p_callable, uint32_t p_flags = 0);
virtual void disconnect(const StringName &p_signal, const Callable &p_callable);
virtual bool is_connected(const StringName &p_signal, const Callable &p_callable) const;
virtual bool has_connections(const StringName &p_signal) const;

template <typename... VarArgs>
void call_deferred(const StringName &p_name, VarArgs... p_args) {
Expand Down
2 changes: 2 additions & 0 deletions doc/classes/Mutex.xml
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@
[b]Warning:[/b] To ensure proper cleanup without crashes or deadlocks, the following conditions must be met:
- When a [Mutex]'s reference count reaches zero and it is therefore destroyed, no threads (including the one on which the destruction will happen) must have it locked.
- When a [Thread]'s reference count reaches zero and it is therefore destroyed, it must not have any mutex locked.

[b]Note:[/b] You can now use `add_user_signal` on mutexes to create thread-safe signals on them.
</description>
<tutorials>
<link title="Using multiple threads">$DOCS_URL/tutorials/performance/using_multiple_threads.html</link>
Expand Down

0 comments on commit d8f5f57

Please sign in to comment.