Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixes for shutting down during async operations #1141

Open
wants to merge 25 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
43dce92
WIP: Fixes for shutting down during async operations
bghgary Sep 29, 2022
925d243
Change to explicitly call Rundown
bghgary Sep 29, 2022
c999153
Cannot throw in destructors
bghgary Sep 30, 2022
204efb9
Merge remote-tracking branch 'origin/master' into async-fixes
bghgary Oct 26, 2022
9d1e098
Fix merge issues
bghgary Oct 26, 2022
124def4
Merge with new timeout code with work queue fixes
bghgary Oct 27, 2022
309c4d9
Fix Canvas
bghgary Oct 28, 2022
ad88d30
Fix Android build
bghgary Oct 28, 2022
11ca510
Merge remote-tracking branch 'origin/master' into async-fixes
bghgary Mar 29, 2023
31248b6
Merge branch 'async-fixes' of https://github.com/bghgary/BabylonNativ…
bghgary Mar 29, 2023
9322360
Update arcana.cpp
bghgary Apr 4, 2023
b4a3c73
Temp fixes for MediaStream
bghgary Apr 5, 2023
9305ca8
Add missing std::forward calls for perfect forwarding
bghgary Apr 19, 2023
df246e2
Work queue shutdown fixes
bghgary Apr 28, 2023
7350715
Miscellaneous Windows AppRuntime fixes
bghgary Apr 28, 2023
77dc8f3
Fix typo in AppRuntime for Win32
bghgary Apr 28, 2023
def047c
Better fix for work queue shutdown issue
bghgary May 2, 2023
bdaf1b9
Merge remote-tracking branch 'origin/master' into async-fixes
bghgary May 4, 2023
0ae6cf4
Fix build issues from merge
bghgary May 4, 2023
71efaf4
Update arcana.cpp to include continuation fix
bghgary May 9, 2023
a0db419
Minor style fixes
bghgary May 9, 2023
6909f55
Update comment
bghgary May 9, 2023
e9d229d
Update comment 2
bghgary May 9, 2023
423ed5e
More style fixes
bghgary May 9, 2023
36a4661
Merge AppRuntime and WorkQueue to fix race condition
bghgary May 24, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions Core/AppRuntime/Source/AppRuntime.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,12 @@ namespace Babylon

AppRuntime::~AppRuntime()
{
// Notify the JsRuntime on the JavaScript thread that the JavaScript
// runtime shutdown sequence has begun. The JsRuntimeScheduler will
// use this signal to gracefully cancel asynchronous operations.
Dispatch([](Napi::Env env) {
JsRuntime::NotifyDisposing(JsRuntime::GetFromJavaScript(env));
});
}

void AppRuntime::Run(Napi::Env env)
Expand Down
20 changes: 6 additions & 14 deletions Core/AppRuntime/Source/AppRuntime_Chakra.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -22,31 +22,23 @@ namespace Babylon

void AppRuntime::RunEnvironmentTier(const char*)
{
using DispatchFunction = std::function<void(std::function<void()>)>;
DispatchFunction dispatchFunction =
[this](std::function<void()> action) {
Dispatch([action = std::move(action)](Napi::Env) {
action();
});
};

JsRuntimeHandle jsRuntime;
ThrowIfFailed(JsCreateRuntime(JsRuntimeAttributeNone, nullptr, &jsRuntime));
JsContextRef context;
ThrowIfFailed(JsCreateContext(jsRuntime, &context));
ThrowIfFailed(JsSetCurrentContext(context));
ThrowIfFailed(JsSetPromiseContinuationCallback(
[](JsValueRef task, void* callbackState) {
auto* pThis = reinterpret_cast<AppRuntime*>(callbackState);
bghgary marked this conversation as resolved.
Show resolved Hide resolved
ThrowIfFailed(JsAddRef(task, nullptr));
auto* dispatch = reinterpret_cast<DispatchFunction*>(callbackState);
dispatch->operator()([task]() {
JsValueRef undefined;
ThrowIfFailed(JsGetUndefinedValue(&undefined));
ThrowIfFailed(JsCallFunction(task, &undefined, 1, nullptr));
pThis->Dispatch([task](auto) {
JsValueRef global;
ThrowIfFailed(JsGetGlobalObject(&global));
ThrowIfFailed(JsCallFunction(task, &global, 1, nullptr));
ThrowIfFailed(JsRelease(task, nullptr));
});
},
&dispatchFunction));
this));
ThrowIfFailed(JsProjectWinRTNamespace(L"Windows"));

#if defined(_DEBUG)
Expand Down
23 changes: 7 additions & 16 deletions Core/AppRuntime/Source/AppRuntime_Win32.cpp
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#include "AppRuntime.h"

#include <Objbase.h>
#include <Windows.h>
#include <winrt/base.h>
#include <roapi.h>

#include <gsl/gsl>
#include <cassert>
Expand All @@ -10,23 +10,14 @@

namespace Babylon
{
namespace
{
constexpr size_t FILENAME_BUFFER_SIZE = 1024;
}

void AppRuntime::RunPlatformTier()
{
HRESULT hr = CoInitializeEx(nullptr, COINIT_APARTMENTTHREADED);
assert(SUCCEEDED(hr));
_CRT_UNUSED(hr);
auto coInitScopeGuard = gsl::finally([] { CoUninitialize(); });
winrt::check_hresult(Windows::Foundation::Initialize(RO_INIT_MULTITHREADED));

char executablePath[1024];
winrt::check_bool(GetModuleFileNameA(nullptr, executablePath, ARRAYSIZE(executablePath)) != 0);

char filename[FILENAME_BUFFER_SIZE];
auto result = GetModuleFileNameA(nullptr, filename, static_cast<DWORD>(std::size(filename)));
assert(result != 0);
(void)result;
RunEnvironmentTier(filename);
RunEnvironmentTier(executablePath);
}

void AppRuntime::DefaultUnhandledExceptionHandler(const std::exception& error)
Expand Down
5 changes: 4 additions & 1 deletion Core/AppRuntime/Source/AppRuntime_WinRT.cpp
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#include "AppRuntime.h"

#include <Windows.h>
#include <winrt/base.h>
#include <roapi.h>

#include <exception>
#include <sstream>
Expand All @@ -9,6 +10,8 @@ namespace Babylon
{
void AppRuntime::RunPlatformTier()
{
winrt::check_hresult(Windows::Foundation::Initialize(RO_INIT_MULTITHREADED));

RunEnvironmentTier();
}

Expand Down
18 changes: 14 additions & 4 deletions Core/AppRuntime/Source/WorkQueue.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,12 @@ namespace Babylon
Resume();
}

m_cancelSource.cancel();
m_dispatcher.cancelled();
// Dispatch a cancel to signal the Run function to gracefully end.
// It must be dispatched and not canceled directly to ensure that
// existing work is executed and executed in the correct order.
m_dispatcher([this]() {
m_cancellationSource.cancel();
});

m_thread.join();
}
Expand All @@ -37,13 +41,19 @@ namespace Babylon
void WorkQueue::Run(Napi::Env env)
{
m_env = std::make_optional(env);

m_dispatcher.set_affinity(std::this_thread::get_id());

while (!m_cancelSource.cancelled())
while (!m_cancellationSource.cancelled())
{
m_dispatcher.blocking_tick(m_cancelSource);
m_dispatcher.blocking_tick(m_cancellationSource);
}

// The dispatcher can be non-empty if something is dispatched after cancellation.
// For example, Chakra's JsSetPromiseContinuationCallback may potentially dispatch
// a continuation after cancellation.
m_dispatcher.clear();

m_env.reset();
}
}
21 changes: 8 additions & 13 deletions Core/AppRuntime/Source/WorkQueue.h
Original file line number Diff line number Diff line change
Expand Up @@ -23,11 +23,15 @@ namespace Babylon
// copyable callable if necessary.
if constexpr (std::is_copy_constructible<CallableT>::value)
{
m_dispatcher.queue([this, callable = std::move(callable)]() { Invoke(callable); });
m_dispatcher.queue([this, callable = std::move(callable)]() {
callable(m_env.value());
});
}
else
{
m_dispatcher.queue([this, callablePtr = std::make_shared<CallableT>(std::move(callable))]() { Invoke(*callablePtr); });
m_dispatcher.queue([this, callablePtr = std::make_shared<CallableT>(std::move(callable))]() {
(*callablePtr)(m_env.value());
});
}
}

Expand All @@ -36,19 +40,10 @@ namespace Babylon
void Run(Napi::Env);

private:
template<typename CallableT>
void Invoke(CallableT& callable)
{
callable(m_env.value());
}

std::optional<Napi::Env> m_env{};

std::optional<std::scoped_lock<std::mutex>> m_suspensionLock{};

arcana::cancellation_source m_cancelSource{};
arcana::cancellation_source m_cancellationSource{};
arcana::manual_dispatcher<128> m_dispatcher{};

std::thread m_thread;
std::thread m_thread{};
};
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@

#include <napi/env.h>

#include <bx/allocator.h>
#include <bgfx/bgfx.h>
#include <bgfx/platform.h>

Expand Down Expand Up @@ -94,7 +95,7 @@ namespace Babylon::Graphics

Update GetUpdate(const char* updateName);

void RequestScreenShot(std::function<void(std::vector<uint8_t>)> callback);
arcana::task<std::vector<uint8_t>, std::exception_ptr> RequestScreenShotAsync();

arcana::task<void, std::exception_ptr> ReadTextureAsync(bgfx::TextureHandle handle, gsl::span<uint8_t> data, uint8_t mipLevel = 0);

Expand All @@ -115,12 +116,16 @@ namespace Babylon::Graphics
void RemoveTexture(bgfx::TextureHandle handle);
TextureInfo GetTextureInfo(bgfx::TextureHandle handle);

bx::AllocatorI& Allocator() { return m_allocator; }

private:
friend UpdateToken;

DeviceImpl& m_graphicsImpl;

std::unordered_map<uint16_t, TextureInfo> m_textureHandleToInfo{};
std::mutex m_textureHandleToInfoMutex{};

bx::DefaultAllocator m_allocator{};
};
}
4 changes: 2 additions & 2 deletions Core/Graphics/Source/DeviceContext.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -56,9 +56,9 @@ namespace Babylon::Graphics
return {m_graphicsImpl.GetSafeTimespanGuarantor(updateName), *this};
}

void DeviceContext::RequestScreenShot(std::function<void(std::vector<uint8_t>)> callback)
arcana::task<std::vector<uint8_t>, std::exception_ptr> DeviceContext::RequestScreenShotAsync()
{
return m_graphicsImpl.RequestScreenShot(std::move(callback));
return m_graphicsImpl.RequestScreenShotAsync();
}

arcana::task<void, std::exception_ptr> DeviceContext::ReadTextureAsync(bgfx::TextureHandle handle, gsl::span<uint8_t> data, uint8_t mipLevel)
Expand Down
10 changes: 8 additions & 2 deletions Core/Graphics/Source/DeviceImpl.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -300,9 +300,15 @@ namespace Babylon::Graphics
return m_afterRenderDispatcher.scheduler();
}

void DeviceImpl::RequestScreenShot(std::function<void(std::vector<uint8_t>)> callback)
arcana::task<std::vector<uint8_t>, std::exception_ptr> DeviceImpl::RequestScreenShotAsync()
{
m_screenShotCallbacks.push(std::move(callback));
arcana::task_completion_source<std::vector<uint8_t>, std::exception_ptr> taskCompletionSource{};

m_screenShotCallbacks.push([taskCompletionSource](std::vector<uint8_t> bytes) mutable {
taskCompletionSource.complete(std::move(bytes));
});

return taskCompletionSource.as_task();
}

arcana::task<void, std::exception_ptr> DeviceImpl::ReadTextureAsync(bgfx::TextureHandle handle, gsl::span<uint8_t> data, uint8_t mipLevel)
Expand Down
2 changes: 1 addition & 1 deletion Core/Graphics/Source/DeviceImpl.h
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ namespace Babylon::Graphics
continuation_scheduler<>& BeforeRenderScheduler();
continuation_scheduler<>& AfterRenderScheduler();

void RequestScreenShot(std::function<void(std::vector<uint8_t>)> callback);
arcana::task<std::vector<uint8_t>, std::exception_ptr> RequestScreenShotAsync();

arcana::task<void, std::exception_ptr> ReadTextureAsync(bgfx::TextureHandle handle, gsl::span<uint8_t> data, uint8_t mipLevel);

Expand Down
31 changes: 25 additions & 6 deletions Core/JsRuntime/Include/Babylon/JsRuntime.h
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

#include <functional>
#include <mutex>
#include <vector>

namespace Babylon
{
Expand All @@ -22,30 +23,48 @@ namespace Babylon
}
};

struct InternalState;
friend struct InternalState;
JsRuntime(const JsRuntime&) = delete;
JsRuntime& operator=(const JsRuntime&) = delete;

// Any JavaScript errors that occur will bubble up as a Napi::Error C++ exception.
// JsRuntime expects the provided dispatch function to handle this exception,
// such as with a try/catch and logging the exception message.
using DispatchFunctionT = std::function<void(std::function<void(Napi::Env)>)>;

// Creates the JsRuntime object owned by the JavaScript environment.
// Note: It is the contract of JsRuntime that its dispatch function must be usable
// at the moment of construction. JsRuntime cannot be built with dispatch function
// that captures a reference to a not-yet-completed object that will be completed
// later -- an instance of an inheriting type, for example. The dispatch function
// must be safely callable as soon as it is passed to the JsRuntime constructor.
static JsRuntime& CreateForJavaScript(Napi::Env, DispatchFunctionT);

// Gets the JsRuntime from the given N-API environment.
static JsRuntime& GetFromJavaScript(Napi::Env);
void Dispatch(std::function<void(Napi::Env)>);

protected:
JsRuntime(const JsRuntime&) = delete;
JsRuntime& operator=(const JsRuntime&) = delete;
struct IDisposingCallback
{
virtual void Disposing() = 0;
};
bghgary marked this conversation as resolved.
Show resolved Hide resolved

// Notifies the JsRuntime that the JavaScript environment will begin shutting down.
// Calling this function will signal callbacks registered with RegisterDisposing.
static void NotifyDisposing(JsRuntime&);

// Registers a callback for when the JavaScript environment will begin shutting down.
static void RegisterDisposing(JsRuntime&, IDisposingCallback*);

// Unregisters a callback for when the JavaScript environment will begin shutting down.
static void UnregisterDisposing(JsRuntime&, IDisposingCallback*);

// Dispatches work onto the JavaScript thread and provides access to the N-API
// environment.
void Dispatch(std::function<void(Napi::Env)>);

private:
JsRuntime(Napi::Env, DispatchFunctionT);

DispatchFunctionT m_dispatchFunction{};
std::vector<IDisposingCallback*> m_disposingCallbacks{};
};
}
Loading