Android: Don't rely on onPause for pausing before destroying surface

Fixes a crash which was uncovered (or just made more likely?)
by the previous commit.
This commit is contained in:
JosJuice 2021-08-08 16:22:52 +02:00
parent 446e2d9119
commit 3eb07e9772
2 changed files with 20 additions and 1 deletions

View File

@ -186,6 +186,7 @@ public final class EmulationFragment extends Fragment implements SurfaceHolder.C
{ {
Log.debug("[EmulationFragment] Surface destroyed."); Log.debug("[EmulationFragment] Surface destroyed.");
NativeLibrary.SurfaceDestroyed(); NativeLibrary.SurfaceDestroyed();
mRunWhenSurfaceIsValid = true;
} }
public void stopEmulation() public void stopEmulation()

View File

@ -443,7 +443,25 @@ JNIEXPORT void JNICALL Java_org_dolphinemu_dolphinemu_NativeLibrary_SurfaceChang
JNIEXPORT void JNICALL Java_org_dolphinemu_dolphinemu_NativeLibrary_SurfaceDestroyed(JNIEnv*, JNIEXPORT void JNICALL Java_org_dolphinemu_dolphinemu_NativeLibrary_SurfaceDestroyed(JNIEnv*,
jclass) jclass)
{ {
std::lock_guard<std::mutex> guard(s_surface_lock); {
// If emulation continues running without a valid surface, we will probably crash,
// so pause emulation until we get a valid surface again. EmulationFragment handles resuming.
std::unique_lock host_identity_guard(s_host_identity_lock);
while (s_is_booting.IsSet())
{
// Need to wait for boot to finish before we can pause
host_identity_guard.unlock();
std::this_thread::sleep_for(std::chrono::milliseconds(1));
host_identity_guard.lock();
}
if (Core::GetState() == Core::State::Running)
Core::SetState(Core::State::Paused);
}
std::lock_guard surface_guard(s_surface_lock);
if (g_renderer) if (g_renderer)
g_renderer->ChangeSurface(nullptr); g_renderer->ChangeSurface(nullptr);