Hi,
I’m the developer of Rewired. I have a user who is reporting an issue in his project that I cannot reproduce. He is using Unity 2018.1.1f1 (also tested on 2018.4.3) and building to IL2CPP on an Xbox One home console in development mode. He is reporting a concurrency::invalid_operation exception being thrown by a call to Task::get in Rewired_UWP.dll, a native UWP C++/CX library when pressing the Xbox/Guide button on his controller and then resuming the application. It only happens when he presses the Xbox/Guide button, the Xbox overlay comes up, and then he closes the overlay screen.
According to the documentation here, this exception will be thrown if Task::get is called from an STA thread:
I can confirm this exception should only be thrown on an STA thread by looking at ppltasks.h:
#if (_PPL_TASK_CONTEXT_CONTROL_ENABLED)
if (_IsNonBlockingThread())
{
// In order to prevent Windows Runtime STA threads from blocking the UI, calling task.wait() task.get() is illegal
// if task has not been completed.
if (!_IsCompleted() && !_IsCanceled())
{
#ifdef BUILD_WINDOWS
_THROW(invalid_operation, "Illegal to wait on a task in a Windows Runtime STA");
#else
_THROW(invalid_operation("Illegal to wait on a task in a Windows Runtime STA"));
#endif
}
else
{
// Task Continuations are 'scheduled' *inside* the chore that is executing on the ancestor's task group. If a continuation
// needs to be marshalled to a different apartment instead of scheduling, we make a synchronous cross-apartment COM
// call to execute the continuation. If it then happens to do something which waits on the ancestor (say it calls .get(),
// which task based continuations are wont to do), waiting on the task group results in waiting on the chore that is making
// this synchronous callback, which causes a deadlock. To avoid this, we test the state of the ancestor's event,
// and we will NOT wait on it if it has finished execution (which means now we are in the inline synchronous callback).
_DoWait = false;
}
}
#endif
According to Unity documentation, Unity always runs on a separate thread:
Let’s take a closer look at AppCallbacks class. When you create it, Unity creates a new thread called “AppThread”. This is done because there’s a restriction from Microsoft - if your application does not become responsive after 5 seconds you’ll fail to pass WACK (Windows Application Certification). (You can read more here - http://msdn.microsoft.com/en-us/library/windowsphone/develop/hh184840(v=vs.105).aspx) Imagine if your first level is pretty big and takes a significant amount of time to load. Because your application is running on UI thread, the UI will be unresponsive until your level is fully loaded. __*That’s why Unity always runs your game on different thread.*__
I can confirm through my own testing that Update and FixedUpdate run on an MTA thread, not STA, using the following code:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
public class UWPCheck : MonoBehaviour {
private string msg;
private string error;
private void FixedUpdate() {
Go();
}
private void Update() {
Go();
}
private void Go() {
#if ENABLE_WINMD_SUPPORT
msg = System.Threading.Thread.CurrentThread.GetApartmentState().ToString();
if(msg != "MTA") error = "Error: " + msg;
#else
msg = "UWP check";
#endif
}
private void OnGUI() {
GUILayout.Label(msg);
GUILayout.Label(error);
}
}
There is never a time I get any result but MTA in my testing.
Making this issue even stranger, this user is reporting that after loading a saved game in his game, the error stops happening when opening and closing the overlay. Upon restarting the game, the problem returns.
It seems very far-fetched, but the only thing I can theorize at this point that is somehow, Update (or Awake) is being called on an STA thread at least once upon resuming the application causing Task::get to throw the exception. I fail to see any other way calling Task::get could throw an concurrency::invalid_operation exception.
Does Unity ever run Update on the UI thread, in this case, when resuming the application? Does it ever run on a thread pool and might this thread pool include an STA thread?
Rewired never calls this function that is throwing the exception except on Awake, Update, or Fixed Update (depending on user’s settings).
Another remote possibility: Does IL2CPP ever make P/Invoke calls a thread other than the one making the function call, with the possibility of it being called on an STA thread?
I cannot reproduce this issue, and neither can the user when testing in a new scene. It only happens in his game which likely includes may calls to various parts of the Unity API. My theory is that something his game is doing is triggering Unity to do something to make the Update callback run on the UI thread. Is something like this even possible?