Skip to content

Commit d342895

Browse files
committed
Fix for WaitForBackground
Unit tests were sometimes failing because the WaitForBackground awaiter was sometimes resuming on the main thread.
1 parent 3ba458a commit d342895

File tree

3 files changed

+62
-12
lines changed

3 files changed

+62
-12
lines changed

Runtime/Awaiters.cs

+44-2
Original file line numberDiff line numberDiff line change
@@ -28,9 +28,51 @@ public static class Awaiters
2828

2929
public class WaitForBackground
3030
{
31-
public ConfiguredTaskAwaitable.ConfiguredTaskAwaiter GetAwaiter()
31+
private class BackgroundThreadJoinAwaiter : IAwaitable
3232
{
33-
return Task.Run(() => {}).ConfigureAwait(false).GetAwaiter();
33+
private Action _continuation = null;
34+
private bool isCompleted = false;
35+
public bool IsCompleted => isCompleted;
36+
37+
public void Complete()
38+
{
39+
isCompleted = true;
40+
_continuation?.Invoke();
41+
}
42+
43+
public void GetResult()
44+
{
45+
do
46+
{
47+
} while (!isCompleted);
48+
}
49+
50+
void INotifyCompletion.OnCompleted(Action continuation)
51+
{
52+
if (isCompleted)
53+
{
54+
throw new InvalidOperationException("Continuation is invalid. This awaiter is already completed.");
55+
}
56+
_continuation = continuation;
57+
}
58+
}
59+
60+
public IAwaitable GetAwaiter()
61+
{
62+
//Doing Task.Run(()=>{}).ConfigureAwait(false) will apparently sometimes still resume on the main thread
63+
//Updated to the below pattern to ensure we actually will be running in the background when we resume
64+
var awaiter = new BackgroundThreadJoinAwaiter();
65+
Task.Run(async () =>
66+
{
67+
//Doing complete immediately without a yield appears to cause the awaiter to never resume
68+
//I'm not entirely sure as to why.
69+
//I suspected maybe Complete() was getting called before the the method doing the awaiting could add its continuation
70+
//However when I added a check and exception for this I did not see it get thrown.
71+
//Adding the await Task.Yield however appeared to get Unit tests to consistently pass.
72+
await Task.Yield();
73+
awaiter.Complete();
74+
});
75+
return awaiter;
3476
}
3577
}
3678

Runtime/UnityTaskUtil.cs

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,8 @@ private static void Install()
1919
UnityTaskFactory = new TaskFactory<UnityEngine.Object>(UnityTaskScheduler);
2020
}
2121

22-
public static bool CurrentThreadIsUnityThread => SynchronizationContext.Current == UnitySynchronizationContext;
23-
22+
public static bool CurrentThreadIsUnityThread => UnityThreadId == Thread.CurrentThread.ManagedThreadId;
23+
2424
public static TaskScheduler UnityTaskScheduler { get; private set; }
2525

2626
public static TaskFactory<UnityEngine.Object> UnityTaskFactory { get; private set; }

Tests/Runtime/AwaitersTests.cs

+16-8
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
using System.Collections;
2+
using System.Threading;
23
using System.Threading.Tasks;
34
using NUnit.Framework;
45
using UnityEngine;
@@ -8,25 +9,32 @@ namespace Gameframe.Async.Tests
89
{
910
public class AwaitersTests
1011
{
11-
[UnityTest]
12+
[UnityTest, Timeout(1000)]
1213
public IEnumerator TestBackgroundAndMainThreadAwaiters()
1314
{
15+
yield return null;
1416
var task = DoTest();
1517
yield return task.AsIEnumerator();
1618
}
1719

1820
private async Task DoTest()
1921
{
22+
Assert.IsTrue(UnityTaskUtil.UnitySynchronizationContext != null);
23+
2024
//Start on unity thread
2125
Assert.IsTrue(UnityTaskUtil.CurrentThreadIsUnityThread);
26+
27+
//Test Migration Multiple times just to be sure
28+
for (int i = 0; i < 10; i++)
29+
{
30+
//Migrate to background thread
31+
await Awaiters.BackgroundThread;
32+
Assert.IsFalse(UnityTaskUtil.CurrentThreadIsUnityThread,$"Expected to be on background thread. CurrentThread:{Thread.CurrentThread.ManagedThreadId} UnityThreadId:{UnityTaskUtil.UnityThreadId}");
2233

23-
//Migrate to background thread
24-
await Awaiters.BackgroundThread;
25-
Assert.IsFalse(UnityTaskUtil.CurrentThreadIsUnityThread);
26-
27-
//Migrate back to main thread
28-
await Awaiters.MainUnityThread;
29-
Assert.IsTrue(UnityTaskUtil.CurrentThreadIsUnityThread);
34+
//Migrate back to main thread
35+
await Awaiters.MainUnityThread;
36+
Assert.IsTrue(UnityTaskUtil.CurrentThreadIsUnityThread,$"Expected to be on main thread. CurrentThread:{Thread.CurrentThread.ManagedThreadId} UnityThreadId:{UnityTaskUtil.UnityThreadId}");
37+
}
3038

3139
//Await the main thread when already on the main thread should do nothing
3240
await Awaiters.MainUnityThread;

0 commit comments

Comments
 (0)