Files
video-shader-toys/apps/RenderCadenceCompositor/video/SyntheticInputProducer.cpp
Aiden 0a8b335048
All checks were successful
CI / React UI Build (push) Successful in 10s
CI / Native Windows Build And Tests (push) Successful in 2m59s
CI / Windows Release Package (push) Has been skipped
INput
2026-05-12 18:39:08 +10:00

112 lines
3.7 KiB
C++

#include "SyntheticInputProducer.h"
#include "VideoIOFormat.h"
#include <algorithm>
#include <cmath>
namespace RenderCadenceCompositor
{
SyntheticInputProducer::SyntheticInputProducer(InputFrameMailbox& mailbox, SyntheticInputProducerConfig config) :
mMailbox(mailbox),
mConfig(config)
{
}
SyntheticInputProducer::~SyntheticInputProducer()
{
Stop();
}
bool SyntheticInputProducer::Start()
{
if (mThread.joinable())
return true;
if (mConfig.width == 0 || mConfig.height == 0)
return false;
mStopping.store(false, std::memory_order_release);
mThread = std::thread([this]() { ThreadMain(); });
return true;
}
void SyntheticInputProducer::Stop()
{
mStopping.store(true, std::memory_order_release);
if (mThread.joinable())
mThread.join();
}
SyntheticInputProducerMetrics SyntheticInputProducer::Metrics() const
{
SyntheticInputProducerMetrics metrics;
metrics.generatedFrames = mGeneratedFrames.load(std::memory_order_relaxed);
metrics.submitMisses = mSubmitMisses.load(std::memory_order_relaxed);
return metrics;
}
void SyntheticInputProducer::ThreadMain()
{
const unsigned rowBytes = VideoIORowBytes(VideoIOPixelFormat::Bgra8, mConfig.width);
std::vector<unsigned char> buffer(static_cast<std::size_t>(rowBytes) * static_cast<std::size_t>(mConfig.height));
const auto frameDuration = std::chrono::duration_cast<std::chrono::steady_clock::duration>(
std::chrono::duration<double, std::milli>(mConfig.frameDurationMilliseconds));
uint64_t frameIndex = 0;
auto nextFrameTime = std::chrono::steady_clock::now();
while (!mStopping.load(std::memory_order_acquire))
{
const auto now = std::chrono::steady_clock::now();
if (now < nextFrameTime)
{
std::this_thread::sleep_for((std::min)(std::chrono::milliseconds(1), std::chrono::duration_cast<std::chrono::milliseconds>(nextFrameTime - now)));
continue;
}
GenerateFrame(frameIndex, buffer);
if (!mMailbox.SubmitFrame(buffer.data(), rowBytes, frameIndex))
mSubmitMisses.fetch_add(1, std::memory_order_relaxed);
mGeneratedFrames.fetch_add(1, std::memory_order_relaxed);
++frameIndex;
nextFrameTime += frameDuration;
if (std::chrono::steady_clock::now() - nextFrameTime > frameDuration * 4)
nextFrameTime = std::chrono::steady_clock::now() + frameDuration;
}
}
void SyntheticInputProducer::GenerateFrame(uint64_t frameIndex, std::vector<unsigned char>& buffer) const
{
const float t = static_cast<float>(frameIndex) / 60.0f;
const unsigned boxWidth = (std::max)(1u, mConfig.width / 5u);
const unsigned boxHeight = (std::max)(1u, mConfig.height / 6u);
const unsigned maxX = mConfig.width > boxWidth ? mConfig.width - boxWidth : 0u;
const unsigned maxY = mConfig.height > boxHeight ? mConfig.height - boxHeight : 0u;
const unsigned boxX = static_cast<unsigned>((0.5f + 0.5f * std::sin(t * 1.4f)) * static_cast<float>(maxX));
const unsigned boxY = static_cast<unsigned>((0.5f + 0.5f * std::sin(t * 0.9f + 1.2f)) * static_cast<float>(maxY));
const unsigned rowBytes = VideoIORowBytes(VideoIOPixelFormat::Bgra8, mConfig.width);
for (unsigned y = 0; y < mConfig.height; ++y)
{
for (unsigned x = 0; x < mConfig.width; ++x)
{
const std::size_t offset = static_cast<std::size_t>(y) * rowBytes + static_cast<std::size_t>(x) * 4;
const unsigned checker = ((x / 80u) + (y / 80u) + static_cast<unsigned>(frameIndex / 15u)) & 1u;
unsigned char red = checker ? 42 : 16;
unsigned char green = checker ? 70 : 28;
unsigned char blue = checker ? 110 : 55;
if (x >= boxX && x < boxX + boxWidth && y >= boxY && y < boxY + boxHeight)
{
red = 245;
green = static_cast<unsigned char>(160 + 60 * (0.5f + 0.5f * std::sin(t * 2.1f)));
blue = 35;
}
buffer[offset + 0] = blue;
buffer[offset + 1] = green;
buffer[offset + 2] = red;
buffer[offset + 3] = 255;
}
}
}
}