Move prepared-file work into app runtime
This commit is contained in:
37
src/app.cpp
37
src/app.cpp
@@ -643,6 +643,29 @@ void App::rec_export(std::string path)
|
||||
|
||||
namespace
|
||||
{
|
||||
template <typename CanvasDocument>
|
||||
struct RecordingWorkerIterationContext
|
||||
{
|
||||
Canvas* legacy_canvas = nullptr;
|
||||
CanvasDocument* canvas_document = nullptr;
|
||||
CanvasEncoder* encoder = nullptr;
|
||||
pp::app::RecordingWorkerIterationPlan plan{};
|
||||
};
|
||||
|
||||
template <typename CanvasDocument>
|
||||
RecordingWorkerIterationContext<CanvasDocument> make_recording_worker_iteration_context(App& app)
|
||||
{
|
||||
RecordingWorkerIterationContext<CanvasDocument> context;
|
||||
context.legacy_canvas = Canvas::I;
|
||||
context.canvas_document = app.canvas ? app.canvas->m_canvas.get() : nullptr;
|
||||
context.encoder = context.legacy_canvas ? context.legacy_canvas->m_encoder.get() : nullptr;
|
||||
context.plan = pp::app::plan_recording_worker_iteration(
|
||||
app.rec_running,
|
||||
context.encoder != nullptr,
|
||||
context.legacy_canvas != nullptr && context.canvas_document != nullptr);
|
||||
return context;
|
||||
}
|
||||
|
||||
template <typename CanvasDocument>
|
||||
void encode_recording_frame(
|
||||
App& app,
|
||||
@@ -676,18 +699,12 @@ void App::rec_loop()
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(rec_mutex);
|
||||
rec_cv.wait(lock/*, [this] { return !(rec_frames.empty() && rec_running); }*/);
|
||||
auto* legacy_canvas = Canvas::I;
|
||||
auto* canvas_document = canvas ? canvas->m_canvas.get() : nullptr;
|
||||
auto* encoder = legacy_canvas ? legacy_canvas->m_encoder.get() : nullptr;
|
||||
const auto plan = pp::app::plan_recording_worker_iteration(
|
||||
rec_running,
|
||||
encoder != nullptr,
|
||||
legacy_canvas != nullptr && canvas_document != nullptr);
|
||||
if (!plan.continue_running)
|
||||
const auto iteration = make_recording_worker_iteration_context<CanvasDocument>(*this);
|
||||
if (!iteration.plan.continue_running)
|
||||
break;
|
||||
|
||||
if (plan.encode_frame && legacy_canvas && canvas_document && encoder)
|
||||
encode_recording_frame(*this, plan, legacy_canvas, canvas_document, encoder);
|
||||
if (iteration.plan.encode_frame && iteration.legacy_canvas && iteration.canvas_document && iteration.encoder)
|
||||
encode_recording_frame(*this, iteration.plan, iteration.legacy_canvas, iteration.canvas_document, iteration.encoder);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -6,12 +6,7 @@
|
||||
#include "app_core/document_sharing.h"
|
||||
#include "platform_api/platform_services.h"
|
||||
|
||||
#include <condition_variable>
|
||||
#include <deque>
|
||||
#include <functional>
|
||||
#include <mutex>
|
||||
#include <stop_token>
|
||||
#include <thread>
|
||||
|
||||
#ifdef __LINUX__
|
||||
#include <GLFW/glfw3.h>
|
||||
@@ -22,88 +17,6 @@
|
||||
|
||||
namespace {
|
||||
|
||||
class LegacyPreparedFileWorker final {
|
||||
public:
|
||||
LegacyPreparedFileWorker()
|
||||
: worker_([this](std::stop_token stop_token) {
|
||||
run(stop_token);
|
||||
})
|
||||
{
|
||||
}
|
||||
|
||||
~LegacyPreparedFileWorker()
|
||||
{
|
||||
shutdown();
|
||||
}
|
||||
|
||||
void post(std::function<void()> task)
|
||||
{
|
||||
{
|
||||
std::lock_guard<std::mutex> lock(mutex_);
|
||||
if (stopping_)
|
||||
return;
|
||||
tasks_.push_back(std::move(task));
|
||||
}
|
||||
cv_.notify_one();
|
||||
}
|
||||
|
||||
private:
|
||||
void shutdown()
|
||||
{
|
||||
{
|
||||
std::lock_guard<std::mutex> lock(mutex_);
|
||||
stopping_ = true;
|
||||
}
|
||||
cv_.notify_all();
|
||||
}
|
||||
|
||||
void run(std::stop_token stop_token)
|
||||
{
|
||||
for (;;) {
|
||||
std::function<void()> task;
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(mutex_);
|
||||
cv_.wait(lock, [&] {
|
||||
return stopping_ || stop_token.stop_requested() || !tasks_.empty();
|
||||
});
|
||||
if ((stopping_ || stop_token.stop_requested()) && tasks_.empty())
|
||||
break;
|
||||
task = std::move(tasks_.front());
|
||||
tasks_.pop_front();
|
||||
}
|
||||
|
||||
if (task) {
|
||||
try {
|
||||
task();
|
||||
} catch (...) {
|
||||
LOG("prepared file worker task failed");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
std::mutex mutex_;
|
||||
std::condition_variable cv_;
|
||||
std::deque<std::function<void()>> tasks_;
|
||||
bool stopping_ = false;
|
||||
std::jthread worker_;
|
||||
};
|
||||
|
||||
struct RetainedPreparedFileWorkerState final {
|
||||
void post(std::function<void()> task)
|
||||
{
|
||||
worker.post(std::move(task));
|
||||
}
|
||||
|
||||
LegacyPreparedFileWorker worker;
|
||||
};
|
||||
|
||||
RetainedPreparedFileWorkerState& retained_prepared_file_worker_state()
|
||||
{
|
||||
static RetainedPreparedFileWorkerState state;
|
||||
return state;
|
||||
}
|
||||
|
||||
[[nodiscard]] GLint rgba8_internal_format() noexcept
|
||||
{
|
||||
return static_cast<GLint>(pp::renderer::gl::rgba8_internal_format());
|
||||
@@ -293,7 +206,7 @@ void App::pick_file_save(const std::string& type, const std::string& default_nam
|
||||
LOG("App::pick_file_save %s", target.path.c_str());
|
||||
if (target.write_on_background_thread) {
|
||||
auto* app = this;
|
||||
retained_prepared_file_worker_state().post([
|
||||
runtime_.prepared_file_task([
|
||||
app,
|
||||
writer = std::move(writer),
|
||||
callback = std::move(callback),
|
||||
|
||||
@@ -3,6 +3,19 @@
|
||||
|
||||
#include "app.h"
|
||||
|
||||
AppRuntime::AppRuntime()
|
||||
: prepared_file_worker_([this](std::stop_token stop_token)
|
||||
{
|
||||
prepared_file_worker_main(stop_token);
|
||||
})
|
||||
{
|
||||
}
|
||||
|
||||
AppRuntime::~AppRuntime()
|
||||
{
|
||||
prepared_file_worker_stop();
|
||||
}
|
||||
|
||||
bool AppRuntime::is_render_thread() const noexcept
|
||||
{
|
||||
return std::this_thread::get_id() == render_thread_id_;
|
||||
@@ -23,6 +36,62 @@ void AppRuntime::notify_ui_worker() noexcept
|
||||
ui_cv_.notify_all();
|
||||
}
|
||||
|
||||
void AppRuntime::prepared_file_task(std::function<void()> task)
|
||||
{
|
||||
{
|
||||
std::lock_guard<std::mutex> lock(prepared_file_task_mutex_);
|
||||
if (!prepared_file_running_)
|
||||
return;
|
||||
prepared_file_tasklist_.push_back(std::move(task));
|
||||
}
|
||||
prepared_file_cv_.notify_one();
|
||||
}
|
||||
|
||||
void AppRuntime::prepared_file_worker_main(std::stop_token stop_token)
|
||||
{
|
||||
for (;;)
|
||||
{
|
||||
std::function<void()> task;
|
||||
{
|
||||
std::unique_lock<std::mutex> lock(prepared_file_task_mutex_);
|
||||
prepared_file_cv_.wait(lock, [this, &stop_token]
|
||||
{
|
||||
return stop_token.stop_requested() || !prepared_file_running_ || !prepared_file_tasklist_.empty();
|
||||
});
|
||||
if ((stop_token.stop_requested() || !prepared_file_running_) && prepared_file_tasklist_.empty())
|
||||
break;
|
||||
task = std::move(prepared_file_tasklist_.front());
|
||||
prepared_file_tasklist_.pop_front();
|
||||
}
|
||||
|
||||
if (task)
|
||||
{
|
||||
try
|
||||
{
|
||||
task();
|
||||
}
|
||||
catch (...)
|
||||
{
|
||||
LOG("prepared file worker task failed");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void AppRuntime::prepared_file_worker_stop()
|
||||
{
|
||||
{
|
||||
std::lock_guard<std::mutex> lock(prepared_file_task_mutex_);
|
||||
prepared_file_running_ = false;
|
||||
}
|
||||
prepared_file_cv_.notify_all();
|
||||
if (prepared_file_worker_.joinable())
|
||||
{
|
||||
prepared_file_worker_.request_stop();
|
||||
prepared_file_worker_.join();
|
||||
}
|
||||
}
|
||||
|
||||
void AppRuntime::render_thread_tick(App& app)
|
||||
{
|
||||
static uint32_t count = 0;
|
||||
|
||||
@@ -34,6 +34,9 @@ struct AppTask : public std::packaged_task<void()>
|
||||
class AppRuntime
|
||||
{
|
||||
public:
|
||||
AppRuntime();
|
||||
~AppRuntime();
|
||||
|
||||
[[nodiscard]] bool is_render_thread() const noexcept;
|
||||
[[nodiscard]] bool is_ui_thread() const noexcept;
|
||||
[[nodiscard]] bool request_redraw() const noexcept { return request_redraw_; }
|
||||
@@ -41,6 +44,7 @@ public:
|
||||
|
||||
void notify_render_worker() noexcept;
|
||||
void notify_ui_worker() noexcept;
|
||||
void prepared_file_task(std::function<void()> task);
|
||||
|
||||
void render_thread_tick(App& app);
|
||||
void render_thread_main(App& app, std::stop_token stop_token);
|
||||
@@ -199,6 +203,15 @@ public:
|
||||
}
|
||||
|
||||
private:
|
||||
void prepared_file_worker_main(std::stop_token stop_token);
|
||||
void prepared_file_worker_stop();
|
||||
|
||||
std::deque<std::function<void()>> prepared_file_tasklist_;
|
||||
std::mutex prepared_file_task_mutex_;
|
||||
std::condition_variable prepared_file_cv_;
|
||||
std::jthread prepared_file_worker_;
|
||||
bool prepared_file_running_ = true;
|
||||
|
||||
std::deque<AppTask> render_tasklist_;
|
||||
std::mutex render_task_mutex_;
|
||||
std::condition_variable render_cv_;
|
||||
|
||||
@@ -250,6 +250,19 @@ template <typename GridT>
|
||||
};
|
||||
}
|
||||
|
||||
template <typename ModesT>
|
||||
[[nodiscard]] inline auto make_legacy_canvas_draw_merge_current_modes_draw(
|
||||
ModesT* modes,
|
||||
const glm::mat4& ortho_proj,
|
||||
const glm::mat4& proj,
|
||||
const glm::mat4& camera)
|
||||
{
|
||||
return [modes, ortho_proj, proj, camera] {
|
||||
for (auto& mode : *modes)
|
||||
mode->on_Draw(ortho_proj, proj, camera);
|
||||
};
|
||||
}
|
||||
|
||||
struct LegacyCanvasDrawMergeSmaskFacesExecution {
|
||||
std::function<void()> set_active_texture_unit;
|
||||
std::function<void()> enable_blend;
|
||||
|
||||
@@ -770,10 +770,11 @@ void NodeCanvas::draw()
|
||||
mode->on_Draw(ortho_proj, proj, camera);
|
||||
},
|
||||
.draw_heightmap = pp::panopainter::make_legacy_canvas_draw_merge_heightmap_draw(App::I->grid, proj, camera),
|
||||
.draw_current_modes = [&] {
|
||||
for (auto& mode : *m_canvas->m_mode)
|
||||
mode->on_Draw(ortho_proj, proj, camera);
|
||||
},
|
||||
.draw_current_modes = pp::panopainter::make_legacy_canvas_draw_merge_current_modes_draw(
|
||||
m_canvas->m_mode,
|
||||
ortho_proj,
|
||||
proj,
|
||||
camera),
|
||||
});
|
||||
|
||||
if (m_density != 1.f)
|
||||
|
||||
Reference in New Issue
Block a user