Squashed 'external/parallel-rdp/parallel-rdp-standalone/' content from commit 3f59f61f2c
git-subtree-dir: external/parallel-rdp/parallel-rdp-standalone git-subtree-split: 3f59f61f2c1c56424356003041df5e4a10612049
This commit is contained in:
135
parallel-rdp/command_ring.cpp
Normal file
135
parallel-rdp/command_ring.cpp
Normal file
@@ -0,0 +1,135 @@
|
||||
/* 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();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user