#include #include #include #include #include #include "Shared.hpp" using namespace Aquamarine; using namespace Hyprutils::Memory; using namespace Hyprutils::Math; #define SP CSharedPointer #define TIMESPEC_NSEC_PER_SEC 1000000000L static void timespecAddNs(timespec* pTimespec, int64_t delta) { int delta_ns_low = delta % TIMESPEC_NSEC_PER_SEC; int delta_s_high = delta / TIMESPEC_NSEC_PER_SEC; pTimespec->tv_sec += delta_s_high; pTimespec->tv_nsec += (long)delta_ns_low; if (pTimespec->tv_nsec >= TIMESPEC_NSEC_PER_SEC) { pTimespec->tv_nsec -= TIMESPEC_NSEC_PER_SEC; ++pTimespec->tv_sec; } } Aquamarine::CHeadlessOutput::CHeadlessOutput(const std::string& name_, Hyprutils::Memory::CWeakPointer backend_) : backend(backend_) { name = name_; framecb = makeShared>([this]() { frameScheduled = false; events.frame.emit(); }); } Aquamarine::CHeadlessOutput::~CHeadlessOutput() { backend->backend->removeIdleEvent(framecb); events.destroy.emit(); } bool Aquamarine::CHeadlessOutput::commit() { events.commit.emit(); state->onCommit(); needsFrame = false; return true; } bool Aquamarine::CHeadlessOutput::test() { return true; } std::vector Aquamarine::CHeadlessOutput::getRenderFormats() { // not sure if this is right but prob doest matter return backend->getRenderFormats(); } Hyprutils::Memory::CSharedPointer Aquamarine::CHeadlessOutput::getBackend() { return backend.lock(); } void Aquamarine::CHeadlessOutput::scheduleFrame(const scheduleFrameReason reason) { TRACE(backend->backend->log(AQ_LOG_TRACE, std::format("CHeadlessOutput::scheduleFrame: reason {}, needsFrame {}, frameScheduled {}", (uint32_t)reason, needsFrame, frameScheduled))); // FIXME: limit fps to the committed framerate. needsFrame = true; if (frameScheduled) return; frameScheduled = true; backend->backend->addIdleEvent(framecb); } bool Aquamarine::CHeadlessOutput::destroy() { events.destroy.emit(); std::erase(backend->outputs, self.lock()); return true; } Aquamarine::CHeadlessBackend::~CHeadlessBackend() { ; } Aquamarine::CHeadlessBackend::CHeadlessBackend(SP backend_) : backend(backend_) { timers.timerfd = timerfd_create(CLOCK_MONOTONIC, TFD_CLOEXEC); } eBackendType Aquamarine::CHeadlessBackend::type() { return eBackendType::AQ_BACKEND_HEADLESS; } bool Aquamarine::CHeadlessBackend::start() { return true; } std::vector> Aquamarine::CHeadlessBackend::pollFDs() { return {makeShared(timers.timerfd, [this]() { dispatchTimers(); })}; } int Aquamarine::CHeadlessBackend::drmFD() { return -1; } bool Aquamarine::CHeadlessBackend::dispatchEvents() { return true; } uint32_t Aquamarine::CHeadlessBackend::capabilities() { return 0; } bool Aquamarine::CHeadlessBackend::setCursor(SP buffer, const Hyprutils::Math::Vector2D& hotspot) { return false; } void Aquamarine::CHeadlessBackend::onReady() { ; } std::vector Aquamarine::CHeadlessBackend::getRenderFormats() { // FIXME: allow any modifier. Maybe set INVALID to mean that? or a special value? return {SDRMFormat{.drmFormat = DRM_FORMAT_RGBA8888, .modifiers = {DRM_FORMAT_MOD_LINEAR}}, SDRMFormat{.drmFormat = DRM_FORMAT_RGBA1010102, .modifiers = {DRM_FORMAT_MOD_LINEAR}}}; } std::vector Aquamarine::CHeadlessBackend::getCursorFormats() { return {}; // No cursor support } bool Aquamarine::CHeadlessBackend::createOutput(const std::string& name) { auto output = SP(new CHeadlessOutput(name.empty() ? std::format("HEADLESS-{}", ++outputIDCounter) : name, self.lock())); outputs.emplace_back(output); output->swapchain = CSwapchain::create(backend->primaryAllocator, self.lock()); output->self = output; backend->events.newOutput.emit(SP(output)); return true; } void Aquamarine::CHeadlessBackend::dispatchTimers() { std::vector toFire; for (size_t i = 0; i < timers.timers.size(); ++i) { if (timers.timers.at(i).expired()) { toFire.emplace_back(timers.timers.at(i)); timers.timers.erase(timers.timers.begin() + i); i--; continue; } } for (auto& copy : toFire) { if (copy.what) copy.what(); } updateTimerFD(); } void Aquamarine::CHeadlessBackend::updateTimerFD() { long long lowestNs = TIMESPEC_NSEC_PER_SEC * 240 /* 240s, 4 mins */; const auto clocknow = std::chrono::steady_clock::now(); bool any = false; for (auto& t : timers.timers) { auto delta = std::chrono::duration_cast(t.when - clocknow).count() * 1000 /* µs -> ns */; if (delta < lowestNs) lowestNs = delta; } if (lowestNs < 0) lowestNs = 0; timespec now; clock_gettime(CLOCK_MONOTONIC, &now); timespecAddNs(&now, lowestNs); itimerspec ts = {.it_value = now}; if (timerfd_settime(timers.timerfd, TFD_TIMER_ABSTIME, &ts, nullptr)) backend->log(AQ_LOG_ERROR, std::format("headless: failed to arm timerfd: {}", strerror(errno))); } SP Aquamarine::CHeadlessBackend::preferredAllocator() { return backend->primaryAllocator; } bool Aquamarine::CHeadlessBackend::CTimer::expired() { return std::chrono::steady_clock::now() > when; }