2015-05-24 06:32:32 +02:00
|
|
|
// Copyright 2015 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2015-05-24 06:32:32 +02:00
|
|
|
|
2021-12-09 18:22:16 -08:00
|
|
|
#include "VideoCommon/AsyncRequests.h"
|
|
|
|
|
2016-01-17 16:54:31 -05:00
|
|
|
#include <mutex>
|
|
|
|
|
2022-12-09 20:01:25 +01:00
|
|
|
#include "Core/System.h"
|
2023-01-30 03:40:15 +13:00
|
|
|
|
|
|
|
#include "VideoCommon/BoundingBox.h"
|
2015-03-05 17:12:24 +01:00
|
|
|
#include "VideoCommon/Fifo.h"
|
2023-01-30 23:59:54 +13:00
|
|
|
#include "VideoCommon/Present.h"
|
2015-01-31 11:38:23 +01:00
|
|
|
#include "VideoCommon/RenderBase.h"
|
2019-03-04 01:25:33 +00:00
|
|
|
#include "VideoCommon/Statistics.h"
|
2018-03-26 21:33:16 +02:00
|
|
|
#include "VideoCommon/VertexManagerBase.h"
|
2016-01-17 16:54:31 -05:00
|
|
|
#include "VideoCommon/VideoBackendBase.h"
|
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2023-01-30 23:59:54 +13:00
|
|
|
#include "VideoCommon/VideoEvents.h"
|
2023-01-31 17:29:16 +13:00
|
|
|
#include "VideoCommon/VideoState.h"
|
2015-01-31 11:38:23 +01:00
|
|
|
|
|
|
|
AsyncRequests AsyncRequests::s_singleton;
|
|
|
|
|
2018-04-01 19:01:55 -04:00
|
|
|
AsyncRequests::AsyncRequests() = default;
|
2015-01-31 11:38:23 +01:00
|
|
|
|
|
|
|
void AsyncRequests::PullEventsInternal()
|
|
|
|
{
|
2018-03-26 21:33:16 +02:00
|
|
|
// This is only called if the queue isn't empty.
|
|
|
|
// So just flush the pipeline to get accurate results.
|
|
|
|
g_vertex_manager->Flush();
|
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
2016-08-05 16:04:39 +02:00
|
|
|
m_empty.Set();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
while (!m_queue.empty())
|
|
|
|
{
|
2015-05-01 18:58:11 +02:00
|
|
|
Event e = m_queue.front();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-05-01 18:58:11 +02:00
|
|
|
// try to merge as many efb pokes as possible
|
|
|
|
// it's a bit hacky, but some games render a complete frame in this way
|
|
|
|
if ((e.type == Event::EFB_POKE_COLOR || e.type == Event::EFB_POKE_Z))
|
|
|
|
{
|
|
|
|
m_merged_efb_pokes.clear();
|
|
|
|
Event first_event = m_queue.front();
|
2017-01-23 02:51:46 -05:00
|
|
|
const auto t = first_event.type == Event::EFB_POKE_COLOR ? EFBAccessType::PokeColor :
|
|
|
|
EFBAccessType::PokeZ;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-05-01 18:58:11 +02:00
|
|
|
do
|
|
|
|
{
|
|
|
|
e = m_queue.front();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-05-01 18:58:11 +02:00
|
|
|
EfbPokeData d;
|
|
|
|
d.data = e.efb_poke.data;
|
|
|
|
d.x = e.efb_poke.x;
|
|
|
|
d.y = e.efb_poke.y;
|
|
|
|
m_merged_efb_pokes.push_back(d);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-05-01 18:58:11 +02:00
|
|
|
m_queue.pop();
|
|
|
|
} while (!m_queue.empty() && m_queue.front().type == first_event.type);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-05-01 18:58:11 +02:00
|
|
|
lock.unlock();
|
2015-12-20 00:34:56 +10:00
|
|
|
g_renderer->PokeEFB(t, m_merged_efb_pokes.data(), m_merged_efb_pokes.size());
|
2015-05-01 18:58:11 +02:00
|
|
|
lock.lock();
|
|
|
|
continue;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
lock.unlock();
|
|
|
|
HandleEvent(e);
|
|
|
|
lock.lock();
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
m_queue.pop();
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
if (m_wake_me_up_again)
|
|
|
|
{
|
|
|
|
m_wake_me_up_again = false;
|
|
|
|
m_cond.notify_all();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsyncRequests::PushEvent(const AsyncRequests::Event& event, bool blocking)
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
2015-01-31 12:01:01 +01:00
|
|
|
|
|
|
|
if (m_passthrough)
|
|
|
|
{
|
|
|
|
HandleEvent(event);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-05 16:04:39 +02:00
|
|
|
m_empty.Clear();
|
2015-01-31 11:38:23 +01:00
|
|
|
m_wake_me_up_again |= blocking;
|
|
|
|
|
|
|
|
if (!m_enable)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_queue.push(event);
|
|
|
|
|
2022-12-09 20:01:25 +01:00
|
|
|
auto& system = Core::System::GetInstance();
|
2023-12-18 21:31:32 -05:00
|
|
|
system.GetFifo().RunGpu();
|
2015-01-31 11:38:23 +01:00
|
|
|
if (blocking)
|
|
|
|
{
|
|
|
|
m_cond.wait(lock, [this] { return m_queue.empty(); });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-07 19:37:32 +02:00
|
|
|
void AsyncRequests::WaitForEmptyQueue()
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
m_cond.wait(lock, [this] { return m_queue.empty(); });
|
|
|
|
}
|
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
void AsyncRequests::SetEnable(bool enable)
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
m_enable = enable;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
if (!enable)
|
|
|
|
{
|
|
|
|
// flush the queue on disabling
|
|
|
|
while (!m_queue.empty())
|
|
|
|
m_queue.pop();
|
|
|
|
if (m_wake_me_up_again)
|
|
|
|
m_cond.notify_all();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsyncRequests::HandleEvent(const AsyncRequests::Event& e)
|
|
|
|
{
|
|
|
|
switch (e.type)
|
|
|
|
{
|
|
|
|
case Event::EFB_POKE_COLOR:
|
2015-12-20 00:34:56 +10:00
|
|
|
{
|
2019-07-10 23:34:50 -04:00
|
|
|
INCSTAT(g_stats.this_frame.num_efb_pokes);
|
2015-12-20 00:34:56 +10:00
|
|
|
EfbPokeData poke = {e.efb_poke.x, e.efb_poke.y, e.efb_poke.data};
|
2017-01-23 02:51:46 -05:00
|
|
|
g_renderer->PokeEFB(EFBAccessType::PokeColor, &poke, 1);
|
2015-12-20 00:34:56 +10:00
|
|
|
}
|
2015-01-31 11:38:23 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
case Event::EFB_POKE_Z:
|
2015-12-20 00:34:56 +10:00
|
|
|
{
|
2019-07-10 23:34:50 -04:00
|
|
|
INCSTAT(g_stats.this_frame.num_efb_pokes);
|
2015-12-20 00:34:56 +10:00
|
|
|
EfbPokeData poke = {e.efb_poke.x, e.efb_poke.y, e.efb_poke.data};
|
2017-01-23 02:51:46 -05:00
|
|
|
g_renderer->PokeEFB(EFBAccessType::PokeZ, &poke, 1);
|
2015-12-20 00:34:56 +10:00
|
|
|
}
|
2015-01-31 11:38:23 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
case Event::EFB_PEEK_COLOR:
|
2019-07-10 23:34:50 -04:00
|
|
|
INCSTAT(g_stats.this_frame.num_efb_peeks);
|
2017-01-23 02:51:46 -05:00
|
|
|
*e.efb_peek.data =
|
|
|
|
g_renderer->AccessEFB(EFBAccessType::PeekColor, e.efb_peek.x, e.efb_peek.y, 0);
|
2015-01-31 11:38:23 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 11:38:23 +01:00
|
|
|
case Event::EFB_PEEK_Z:
|
2019-07-10 23:34:50 -04:00
|
|
|
INCSTAT(g_stats.this_frame.num_efb_peeks);
|
2017-01-23 02:51:46 -05:00
|
|
|
*e.efb_peek.data = g_renderer->AccessEFB(EFBAccessType::PeekZ, e.efb_peek.x, e.efb_peek.y, 0);
|
2015-01-31 11:38:23 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 12:01:01 +01:00
|
|
|
case Event::SWAP_EVENT:
|
2023-01-30 23:59:54 +13:00
|
|
|
g_presenter->ViSwap(e.swap_event.xfbAddr, e.swap_event.fbWidth, e.swap_event.fbStride,
|
|
|
|
e.swap_event.fbHeight, e.time);
|
2015-01-31 12:01:01 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-31 12:43:58 +01:00
|
|
|
case Event::BBOX_READ:
|
2023-01-30 03:40:15 +13:00
|
|
|
*e.bbox.data = g_bounding_box->Get(e.bbox.index);
|
2015-01-31 13:09:25 +01:00
|
|
|
break;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2022-08-18 13:38:37 -07:00
|
|
|
case Event::FIFO_RESET:
|
2022-12-09 20:01:25 +01:00
|
|
|
Core::System::GetInstance().GetFifo().ResetVideoBuffer();
|
2022-08-18 13:38:37 -07:00
|
|
|
break;
|
|
|
|
|
2015-01-31 13:09:25 +01:00
|
|
|
case Event::PERF_QUERY:
|
|
|
|
g_perf_query->FlushResults();
|
|
|
|
break;
|
2019-06-29 18:35:12 +10:00
|
|
|
|
|
|
|
case Event::DO_SAVE_STATE:
|
2019-06-29 19:27:53 +10:00
|
|
|
VideoCommon_DoState(*e.do_save_state.p);
|
2019-06-29 18:35:12 +10:00
|
|
|
break;
|
2015-01-31 11:38:23 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-31 12:01:01 +01:00
|
|
|
void AsyncRequests::SetPassthrough(bool enable)
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
m_passthrough = enable;
|
|
|
|
}
|