mirror of
https://github.com/kmc-jp/n64-emu.git
synced 2025-04-02 10:21:43 -04:00
135 lines
3.6 KiB
C++
135 lines
3.6 KiB
C++
/* Copyright (c) 2020 Themaister
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#include <chrono>
|
|
#include "command_ring.hpp"
|
|
#include "rdp_device.hpp"
|
|
#include "thread_id.hpp"
|
|
#include <assert.h>
|
|
|
|
namespace RDP
|
|
{
|
|
void CommandRing::init(
|
|
#ifdef PARALLEL_RDP_SHADER_DIR
|
|
Granite::Global::GlobalManagersHandle global_handles_,
|
|
#endif
|
|
CommandProcessor *processor_, unsigned count)
|
|
{
|
|
assert((count & (count - 1)) == 0);
|
|
teardown_thread();
|
|
processor = processor_;
|
|
ring.resize(count);
|
|
write_count = 0;
|
|
read_count = 0;
|
|
#ifdef PARALLEL_RDP_SHADER_DIR
|
|
global_handles = std::move(global_handles_);
|
|
#endif
|
|
thr = std::thread(&CommandRing::thread_loop, this);
|
|
}
|
|
|
|
void CommandRing::teardown_thread()
|
|
{
|
|
if (thr.joinable())
|
|
{
|
|
enqueue_command(0, nullptr);
|
|
thr.join();
|
|
}
|
|
}
|
|
|
|
CommandRing::~CommandRing()
|
|
{
|
|
teardown_thread();
|
|
}
|
|
|
|
void CommandRing::drain()
|
|
{
|
|
std::unique_lock<std::mutex> holder{lock};
|
|
cond.wait(holder, [this]() {
|
|
return write_count == completed_count;
|
|
});
|
|
}
|
|
|
|
void CommandRing::enqueue_command(unsigned num_words, const uint32_t *words)
|
|
{
|
|
std::unique_lock<std::mutex> holder{lock};
|
|
cond.wait(holder, [this, num_words]() {
|
|
return write_count + num_words + 1 <= read_count + ring.size();
|
|
});
|
|
|
|
size_t mask = ring.size() - 1;
|
|
ring[write_count++ & mask] = num_words;
|
|
for (unsigned i = 0; i < num_words; i++)
|
|
ring[write_count++ & mask] = words[i];
|
|
|
|
cond.notify_one();
|
|
}
|
|
|
|
void CommandRing::thread_loop()
|
|
{
|
|
Util::register_thread_index(0);
|
|
|
|
#ifdef PARALLEL_RDP_SHADER_DIR
|
|
// Here to let the RDP play nice with full Granite.
|
|
// When we move to standalone Granite, we won't need to interact with global subsystems like this.
|
|
Granite::Global::set_thread_context(*global_handles);
|
|
global_handles.reset();
|
|
#endif
|
|
|
|
std::vector<uint32_t> tmp_buffer;
|
|
tmp_buffer.reserve(64);
|
|
size_t mask = ring.size() - 1;
|
|
|
|
for (;;)
|
|
{
|
|
bool is_idle = false;
|
|
{
|
|
std::unique_lock<std::mutex> holder{lock};
|
|
if (cond.wait_for(holder, std::chrono::microseconds(500), [this]() { return write_count > read_count; }))
|
|
{
|
|
uint32_t num_words = ring[read_count++ & mask];
|
|
tmp_buffer.resize(num_words);
|
|
for (uint32_t i = 0; i < num_words; i++)
|
|
tmp_buffer[i] = ring[read_count++ & mask];
|
|
}
|
|
else
|
|
{
|
|
// If we don't receive commands at a steady pace,
|
|
// notify rendering thread that we should probably kick some work.
|
|
tmp_buffer.resize(1);
|
|
tmp_buffer[0] = uint32_t(Op::MetaIdle) << 24;
|
|
is_idle = true;
|
|
}
|
|
}
|
|
|
|
if (tmp_buffer.empty())
|
|
break;
|
|
|
|
processor->enqueue_command_direct(tmp_buffer.size(), tmp_buffer.data());
|
|
if (!is_idle)
|
|
{
|
|
std::lock_guard<std::mutex> holder{lock};
|
|
completed_count = read_count;
|
|
cond.notify_one();
|
|
}
|
|
}
|
|
}
|
|
}
|