mirror of
https://github.com/ppy/osu
synced 2025-01-05 05:39:49 +00:00
Add flag to guard against deadlocks during blocking operations
This commit is contained in:
parent
11f0f3c17d
commit
791ea0308f
@ -89,10 +89,15 @@ namespace osu.Game.Database
|
||||
|
||||
private Realm? updateRealm;
|
||||
|
||||
private bool isSendingNotificationResetEvents;
|
||||
|
||||
public Realm Realm => ensureUpdateRealm();
|
||||
|
||||
private Realm ensureUpdateRealm()
|
||||
{
|
||||
if (isSendingNotificationResetEvents)
|
||||
throw new InvalidOperationException("Cannot retrieve a realm context from a notification callback during a blocking operation.");
|
||||
|
||||
if (!ThreadSafety.IsUpdateThread)
|
||||
throw new InvalidOperationException(@$"Use {nameof(getRealmInstance)} when performing realm operations from a non-update thread");
|
||||
|
||||
@ -613,9 +618,21 @@ namespace osu.Game.Database
|
||||
// This is because realm may fire event callbacks between the `unregisterAllSubscriptions` and `updateRealm.Dispose`
|
||||
// calls above.
|
||||
syncContext?.Send(_ =>
|
||||
{
|
||||
// Flag ensures that we don't get in a deadlocked scenario due to a callback attempting to access `RealmAccess.Realm` or `RealmAccess.Run`
|
||||
// and hitting `realmRetrievalLock` a second time. Generally such usages should not exist, and as such we throw when an attempt is made
|
||||
// to use in this fashion.
|
||||
isSendingNotificationResetEvents = true;
|
||||
|
||||
try
|
||||
{
|
||||
foreach (var action in notificationsResetMap.Values)
|
||||
action();
|
||||
}
|
||||
finally
|
||||
{
|
||||
isSendingNotificationResetEvents = false;
|
||||
}
|
||||
}, null);
|
||||
|
||||
const int sleep_length = 200;
|
||||
|
Loading…
Reference in New Issue
Block a user