1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-14 03:25:11 +08:00

Fix notification reset events potentially arriving out of order if a block operation times out

This commit is contained in:
Dean Herbert 2022-01-25 16:26:06 +09:00
parent 0d575006fb
commit 1bb1366c9f

View File

@ -334,25 +334,6 @@ namespace osu.Game.Database
} }
} }
/// <summary>
/// Unregister all subscriptions when the realm instance is to be recycled.
/// Subscriptions will still remain and will be re-subscribed when the realm instance returns.
/// </summary>
private void unregisterAllSubscriptions()
{
lock (realmLock)
{
foreach (var action in notificationsResetMap.Values)
action();
foreach (var action in customSubscriptionsResetMap)
{
action.Value?.Dispose();
customSubscriptionsResetMap[action.Key] = null;
}
}
}
private Realm getRealmInstance() private Realm getRealmInstance()
{ {
if (isDisposed) if (isDisposed)
@ -605,9 +586,16 @@ namespace osu.Game.Database
throw new InvalidOperationException(@$"{nameof(BlockAllOperations)} must be called from the update thread."); throw new InvalidOperationException(@$"{nameof(BlockAllOperations)} must be called from the update thread.");
syncContext = SynchronizationContext.Current; syncContext = SynchronizationContext.Current;
}
unregisterAllSubscriptions(); // Before disposing the update context, clean up all subscriptions.
// Note that in the case of realm notification subscriptions, this is not really required (they will be cleaned up by disposal).
// In the case of custom subscriptions, we want them to fire before the update realm is disposed in case they do any follow-up work.
foreach (var action in customSubscriptionsResetMap)
{
action.Value?.Dispose();
customSubscriptionsResetMap[action.Key] = null;
}
}
Logger.Log(@"Blocking realm operations.", LoggingTarget.Database); Logger.Log(@"Blocking realm operations.", LoggingTarget.Database);
@ -615,6 +603,16 @@ namespace osu.Game.Database
updateRealm = null; updateRealm = null;
} }
// In order to ensure events arrive in the correct order, these *must* be fired post disposal of the update realm,
// and must be posted to the synchronization context.
// This is because realm may fire event callbacks between the `unregisterAllSubscriptions` and `updateRealm.Dispose`
// calls above.
syncContext?.Post(_ =>
{
foreach (var action in notificationsResetMap.Values)
action();
}, null);
const int sleep_length = 200; const int sleep_length = 200;
int timeout = 5000; int timeout = 5000;