2024-06-18 18:45:05 +02:00
|
|
|
#include <aquamarine/backend/Wayland.hpp>
|
|
|
|
#include <wayland.hpp>
|
|
|
|
#include <xdg-shell.hpp>
|
2024-06-19 22:40:23 +02:00
|
|
|
#include "Shared.hpp"
|
|
|
|
#include <string.h>
|
|
|
|
#include <xf86drm.h>
|
|
|
|
#include <gbm.h>
|
|
|
|
#include <fcntl.h>
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
using namespace Aquamarine;
|
|
|
|
using namespace Hyprutils::Memory;
|
|
|
|
using namespace Hyprutils::Math;
|
|
|
|
#define SP CSharedPointer
|
|
|
|
|
|
|
|
Aquamarine::CWaylandBackend::~CWaylandBackend() {
|
2024-06-19 22:40:23 +02:00
|
|
|
if (drmState.fd >= 0)
|
|
|
|
close(drmState.fd);
|
2024-06-18 18:45:05 +02:00
|
|
|
}
|
2024-06-19 22:40:23 +02:00
|
|
|
|
2024-06-18 18:45:05 +02:00
|
|
|
eBackendType Aquamarine::CWaylandBackend::type() {
|
|
|
|
return AQ_BACKEND_WAYLAND;
|
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
Aquamarine::CWaylandBackend::CWaylandBackend(SP<CBackend> backend_) : backend(backend_) {
|
2024-06-18 18:45:05 +02:00
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Aquamarine::CWaylandBackend::start() {
|
|
|
|
backend->log(AQ_LOG_DEBUG, "Starting the Wayland backend!");
|
|
|
|
|
|
|
|
waylandState.display = wl_display_connect(nullptr);
|
|
|
|
|
|
|
|
if (!waylandState.display) {
|
|
|
|
backend->log(AQ_LOG_ERROR, "Wayland backend cannot start: wl_display_connect failed (is a wayland compositor running?)");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.registry = makeShared<CCWlRegistry>((wl_proxy*)wl_display_get_registry(waylandState.display));
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
backend->log(AQ_LOG_DEBUG, std::format("Got registry at 0x{:x}", (uintptr_t)waylandState.registry->resource()));
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.registry->setGlobal([this](CCWlRegistry* r, uint32_t id, const char* name, uint32_t version) {
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->log(AQ_LOG_TRACE, std::format(" | received global: {} (version {}) with id {}", name, version, id));
|
|
|
|
|
|
|
|
const std::string NAME = name;
|
|
|
|
|
|
|
|
if (NAME == "wl_seat") {
|
|
|
|
backend->log(AQ_LOG_TRACE, std::format(" > binding to global: {} (version {}) with id {}", name, 9, id));
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.seat = makeShared<CCWlSeat>((wl_proxy*)wl_registry_bind((wl_registry*)waylandState.registry->resource(), id, &wl_seat_interface, 9));
|
2024-06-18 18:45:05 +02:00
|
|
|
initSeat();
|
|
|
|
} else if (NAME == "xdg_wm_base") {
|
|
|
|
backend->log(AQ_LOG_TRACE, std::format(" > binding to global: {} (version {}) with id {}", name, 6, id));
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdg = makeShared<CCXdgWmBase>((wl_proxy*)wl_registry_bind((wl_registry*)waylandState.registry->resource(), id, &xdg_wm_base_interface, 6));
|
2024-06-18 18:45:05 +02:00
|
|
|
initShell();
|
|
|
|
} else if (NAME == "wl_compositor") {
|
|
|
|
backend->log(AQ_LOG_TRACE, std::format(" > binding to global: {} (version {}) with id {}", name, 6, id));
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.compositor = makeShared<CCWlCompositor>((wl_proxy*)wl_registry_bind((wl_registry*)waylandState.registry->resource(), id, &wl_compositor_interface, 6));
|
2024-06-19 22:40:23 +02:00
|
|
|
} else if (NAME == "zwp_linux_dmabuf_v1") {
|
|
|
|
backend->log(AQ_LOG_TRACE, std::format(" > binding to global: {} (version {}) with id {}", name, 5, id));
|
|
|
|
waylandState.dmabuf =
|
2024-06-20 19:24:43 +02:00
|
|
|
makeShared<CCZwpLinuxDmabufV1>((wl_proxy*)wl_registry_bind((wl_registry*)waylandState.registry->resource(), id, &zwp_linux_dmabuf_v1_interface, 5));
|
2024-06-19 22:40:23 +02:00
|
|
|
if (!initDmabuf()) {
|
|
|
|
backend->log(AQ_LOG_ERROR, "Wayland backend cannot start: zwp_linux_dmabuf_v1 init failed");
|
|
|
|
waylandState.dmabufFailed = true;
|
|
|
|
}
|
2024-06-18 18:45:05 +02:00
|
|
|
}
|
|
|
|
});
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.registry->setGlobalRemove([this](CCWlRegistry* r, uint32_t id) { backend->log(AQ_LOG_DEBUG, std::format("Global {} removed", id)); });
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
wl_display_roundtrip(waylandState.display);
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
if (!waylandState.xdg || !waylandState.compositor || !waylandState.seat || !waylandState.dmabuf || waylandState.dmabufFailed) {
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->log(AQ_LOG_ERROR, "Wayland backend cannot start: Missing protocols");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatchEvents();
|
|
|
|
|
|
|
|
createOutput("WAYLAND1");
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
int Aquamarine::CWaylandBackend::drmFD() {
|
|
|
|
return drmState.fd;
|
|
|
|
}
|
|
|
|
|
2024-06-18 18:45:05 +02:00
|
|
|
void Aquamarine::CWaylandBackend::createOutput(const std::string& szName) {
|
2024-06-20 19:24:43 +02:00
|
|
|
auto o = outputs.emplace_back(SP<CWaylandOutput>(new CWaylandOutput(szName, self)));
|
|
|
|
o->self = o;
|
2024-06-21 15:49:28 +02:00
|
|
|
idleCallbacks.emplace_back([this, o]() { backend->events.newOutput.emit(SP<IOutput>(o)); });
|
2024-06-18 18:45:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int Aquamarine::CWaylandBackend::pollFD() {
|
|
|
|
if (!waylandState.display)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return wl_display_get_fd(waylandState.display);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Aquamarine::CWaylandBackend::dispatchEvents() {
|
|
|
|
wl_display_flush(waylandState.display);
|
|
|
|
|
|
|
|
if (wl_display_prepare_read(waylandState.display) == 0) {
|
|
|
|
wl_display_read_events(waylandState.display);
|
|
|
|
wl_display_dispatch_pending(waylandState.display);
|
|
|
|
} else
|
|
|
|
wl_display_dispatch(waylandState.display);
|
|
|
|
|
|
|
|
int ret = 0;
|
|
|
|
do {
|
|
|
|
ret = wl_display_dispatch_pending(waylandState.display);
|
|
|
|
wl_display_flush(waylandState.display);
|
|
|
|
} while (ret > 0);
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
// dispatch frames
|
2024-06-21 15:49:28 +02:00
|
|
|
if (backend->ready) {
|
|
|
|
for (auto& f : idleCallbacks) {
|
|
|
|
f();
|
|
|
|
}
|
|
|
|
idleCallbacks.clear();
|
2024-06-20 19:24:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t Aquamarine::CWaylandBackend::capabilities() {
|
|
|
|
return AQ_BACKEND_CAPABILITY_POINTER;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Aquamarine::CWaylandBackend::setCursor(Hyprutils::Memory::CSharedPointer<IBuffer> buffer, const Hyprutils::Math::Vector2D& hotspot) {
|
|
|
|
// TODO:
|
2024-06-18 18:45:05 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
void Aquamarine::CWaylandBackend::onReady() {
|
|
|
|
for (auto& o : outputs) {
|
|
|
|
o->swapchain = makeShared<CSwapchain>(backend->allocator);
|
|
|
|
if (!o->swapchain) {
|
|
|
|
backend->log(AQ_LOG_ERROR, std::format("Output {} failed: swapchain creation failed", o->name));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Aquamarine::CWaylandKeyboard::CWaylandKeyboard(SP<CCWlKeyboard> keyboard_, Hyprutils::Memory::CWeakPointer<CWaylandBackend> backend_) : keyboard(keyboard_), backend(backend_) {
|
2024-06-18 18:45:05 +02:00
|
|
|
if (!keyboard->resource())
|
|
|
|
return;
|
|
|
|
|
|
|
|
backend->backend->log(AQ_LOG_DEBUG, "New wayland keyboard wl_keyboard");
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
keyboard->setKey([this](CCWlKeyboard* r, uint32_t serial, uint32_t timeMs, uint32_t key, wl_keyboard_key_state state) {
|
2024-06-18 18:45:05 +02:00
|
|
|
events.key.emit(SKeyEvent{
|
|
|
|
.timeMs = timeMs,
|
|
|
|
.key = key,
|
|
|
|
.pressed = state == WL_KEYBOARD_KEY_STATE_PRESSED,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
keyboard->setModifiers([this](CCWlKeyboard* r, uint32_t serial, uint32_t depressed, uint32_t latched, uint32_t locked, uint32_t group) {
|
2024-06-19 22:40:23 +02:00
|
|
|
events.modifiers.emit(SModifiersEvent{
|
2024-06-18 18:45:05 +02:00
|
|
|
.depressed = depressed,
|
|
|
|
.latched = latched,
|
|
|
|
.locked = locked,
|
|
|
|
.group = group,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Aquamarine::CWaylandKeyboard::~CWaylandKeyboard() {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& Aquamarine::CWaylandKeyboard::getName() {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
Aquamarine::CWaylandPointer::CWaylandPointer(SP<CCWlPointer> pointer_, Hyprutils::Memory::CWeakPointer<CWaylandBackend> backend_) : pointer(pointer_), backend(backend_) {
|
2024-06-18 18:45:05 +02:00
|
|
|
if (!pointer->resource())
|
|
|
|
return;
|
|
|
|
|
|
|
|
backend->backend->log(AQ_LOG_DEBUG, "New wayland pointer wl_pointer");
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
pointer->setMotion([this](CCWlPointer* r, uint32_t serial, wl_fixed_t x, wl_fixed_t y) {
|
|
|
|
if (!backend->focusedOutput || (!backend->focusedOutput->state->mode && !backend->focusedOutput->state->customMode))
|
2024-06-18 18:45:05 +02:00
|
|
|
return;
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
const Vector2D size = backend->focusedOutput->state->customMode ? backend->focusedOutput->state->customMode->pixelSize : backend->focusedOutput->state->mode->pixelSize;
|
2024-06-19 22:40:23 +02:00
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
Vector2D local = {wl_fixed_to_double(x), wl_fixed_to_double(y)};
|
|
|
|
local = local / size;
|
2024-06-18 18:45:05 +02:00
|
|
|
|
2024-06-21 15:49:28 +02:00
|
|
|
timespec now;
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
|
|
|
2024-06-18 18:45:05 +02:00
|
|
|
events.warp.emit(SWarpEvent{
|
2024-06-21 15:49:28 +02:00
|
|
|
.timeMs = (uint32_t)(now.tv_sec * 1000 + now.tv_nsec / 1000000),
|
2024-06-18 18:45:05 +02:00
|
|
|
.absolute = local,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
pointer->setEnter([this](CCWlPointer* r, uint32_t serial, wl_proxy* surface, wl_fixed_t x, wl_fixed_t y) {
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->lastEnterSerial = serial;
|
|
|
|
|
|
|
|
for (auto& o : backend->outputs) {
|
|
|
|
if (o->waylandState.surface->resource() != surface)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
backend->focusedOutput = o;
|
|
|
|
backend->backend->log(AQ_LOG_DEBUG, std::format("[wayland] focus changed: {}", o->name));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
pointer->setButton([this](CCWlPointer* r, uint32_t serial, uint32_t timeMs, uint32_t button, wl_pointer_button_state state) {
|
2024-06-18 18:45:05 +02:00
|
|
|
events.button.emit(SButtonEvent{
|
|
|
|
.timeMs = timeMs,
|
|
|
|
.button = button,
|
|
|
|
.pressed = state == WL_POINTER_BUTTON_STATE_PRESSED,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
pointer->setAxis([this](CCWlPointer* r, uint32_t timeMs, wl_pointer_axis axis, wl_fixed_t value) {
|
2024-06-18 18:45:05 +02:00
|
|
|
events.axis.emit(SAxisEvent{
|
|
|
|
.timeMs = timeMs,
|
|
|
|
.axis = axis == WL_POINTER_AXIS_HORIZONTAL_SCROLL ? AQ_POINTER_AXIS_HORIZONTAL : AQ_POINTER_AXIS_VERTICAL,
|
2024-06-21 15:49:28 +02:00
|
|
|
.delta = wl_fixed_to_double(value),
|
2024-06-18 18:45:05 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
pointer->setFrame([this](CCWlPointer* r) { events.frame.emit(); });
|
2024-06-18 18:45:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Aquamarine::CWaylandPointer::~CWaylandPointer() {
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& Aquamarine::CWaylandPointer::getName() {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandBackend::initSeat() {
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.seat->setCapabilities([this](CCWlSeat* r, wl_seat_capability cap) {
|
2024-06-18 18:45:05 +02:00
|
|
|
const bool HAS_KEYBOARD = ((uint32_t)cap) & WL_SEAT_CAPABILITY_KEYBOARD;
|
|
|
|
const bool HAS_POINTER = ((uint32_t)cap) & WL_SEAT_CAPABILITY_POINTER;
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
if (HAS_KEYBOARD && keyboards.empty()) {
|
2024-06-20 19:24:43 +02:00
|
|
|
auto k = keyboards.emplace_back(makeShared<CWaylandKeyboard>(makeShared<CCWlKeyboard>(waylandState.seat->sendGetKeyboard()), self));
|
2024-06-21 15:49:28 +02:00
|
|
|
idleCallbacks.emplace_back([this, k]() { backend->events.newKeyboard.emit(SP<IKeyboard>(k)); });
|
2024-06-19 22:40:23 +02:00
|
|
|
} else if (!HAS_KEYBOARD && !keyboards.empty())
|
2024-06-18 18:45:05 +02:00
|
|
|
keyboards.clear();
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
if (HAS_POINTER && pointers.empty()) {
|
2024-06-20 19:24:43 +02:00
|
|
|
auto p = pointers.emplace_back(makeShared<CWaylandPointer>(makeShared<CCWlPointer>(waylandState.seat->sendGetPointer()), self));
|
2024-06-21 15:49:28 +02:00
|
|
|
idleCallbacks.emplace_back([this, p]() { backend->events.newPointer.emit(SP<IPointer>(p)); });
|
2024-06-19 22:40:23 +02:00
|
|
|
} else if (!HAS_POINTER && !pointers.empty())
|
|
|
|
pointers.clear();
|
2024-06-18 18:45:05 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandBackend::initShell() {
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdg->setPing([](CCXdgWmBase* r, uint32_t serial) { r->sendPong(serial); });
|
2024-06-18 18:45:05 +02:00
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
bool Aquamarine::CWaylandBackend::initDmabuf() {
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.dmabufFeedback = makeShared<CCZwpLinuxDmabufFeedbackV1>(waylandState.dmabuf->sendGetDefaultFeedback());
|
2024-06-19 22:40:23 +02:00
|
|
|
if (!waylandState.dmabufFeedback) {
|
|
|
|
backend->log(AQ_LOG_ERROR, "initDmabuf: failed to get default feedback");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.dmabufFeedback->setDone([this](CCZwpLinuxDmabufFeedbackV1* r) {
|
2024-06-19 22:40:23 +02:00
|
|
|
// no-op
|
|
|
|
backend->log(AQ_LOG_DEBUG, "zwp_linux_dmabuf_v1: Got done");
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.dmabufFeedback->setMainDevice([this](CCZwpLinuxDmabufFeedbackV1* r, wl_array* deviceArr) {
|
2024-06-19 22:40:23 +02:00
|
|
|
backend->log(AQ_LOG_DEBUG, "zwp_linux_dmabuf_v1: Got main device");
|
|
|
|
|
|
|
|
dev_t device;
|
|
|
|
ASSERT(deviceArr->size == sizeof(device));
|
|
|
|
memcpy(&device, deviceArr->data, sizeof(device));
|
|
|
|
|
|
|
|
drmDevice* drmDev;
|
|
|
|
if (drmGetDeviceFromDevId(device, /* flags */ 0, &drmDev) != 0) {
|
|
|
|
backend->log(AQ_LOG_ERROR, "zwp_linux_dmabuf_v1: drmGetDeviceFromDevId failed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* name = nullptr;
|
|
|
|
if (drmDev->available_nodes & (1 << DRM_NODE_RENDER))
|
|
|
|
name = drmDev->nodes[DRM_NODE_RENDER];
|
|
|
|
else {
|
|
|
|
// Likely a split display/render setup. Pick the primary node and hope
|
|
|
|
// Mesa will open the right render node under-the-hood.
|
|
|
|
ASSERT(drmDev->available_nodes & (1 << DRM_NODE_PRIMARY));
|
|
|
|
name = drmDev->nodes[DRM_NODE_PRIMARY];
|
|
|
|
backend->log(AQ_LOG_WARNING, "zwp_linux_dmabuf_v1: DRM device has no render node, using primary.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!name) {
|
|
|
|
backend->log(AQ_LOG_ERROR, "zwp_linux_dmabuf_v1: no node name");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
drmState.nodeName = name;
|
|
|
|
|
|
|
|
drmFreeDevice(&drmDev);
|
|
|
|
|
|
|
|
backend->log(AQ_LOG_DEBUG, std::format("zwp_linux_dmabuf_v1: Got node {}", drmState.nodeName));
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO: format table and tranche
|
|
|
|
|
|
|
|
wl_display_roundtrip(waylandState.display);
|
|
|
|
|
|
|
|
if (!drmState.nodeName.empty()) {
|
|
|
|
drmState.fd = open(drmState.nodeName.c_str(), O_RDWR | O_NONBLOCK | O_CLOEXEC);
|
|
|
|
if (drmState.fd < 0) {
|
|
|
|
backend->log(AQ_LOG_ERROR, std::format("zwp_linux_dmabuf_v1: Failed to open node {}", drmState.nodeName));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
backend->log(AQ_LOG_DEBUG, std::format("zwp_linux_dmabuf_v1: opened node {} with fd {}", drmState.nodeName, drmState.fd));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
Aquamarine::CWaylandOutput::CWaylandOutput(const std::string& name_, Hyprutils::Memory::CWeakPointer<CWaylandBackend> backend_) : backend(backend_) {
|
|
|
|
name = name_;
|
2024-06-18 18:45:05 +02:00
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.surface = makeShared<CCWlSurface>(backend->waylandState.compositor->sendCreateSurface());
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
if (!waylandState.surface->resource()) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {} failed: no surface given. Errno: {}", name, errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdgSurface = makeShared<CCXdgSurface>(backend->waylandState.xdg->sendGetXdgSurface(waylandState.surface->resource()));
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
if (!waylandState.xdgSurface->resource()) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {} failed: no xdgSurface given. Errno: {}", name, errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdgSurface->setConfigure([this](CCXdgSurface* r, uint32_t serial) {
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->backend->log(AQ_LOG_DEBUG, std::format("Output {}: configure surface with {}", name, serial));
|
|
|
|
r->sendAckConfigure(serial);
|
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdgToplevel = makeShared<CCXdgToplevel>(waylandState.xdgSurface->sendGetToplevel());
|
2024-06-18 18:45:05 +02:00
|
|
|
|
|
|
|
if (!waylandState.xdgToplevel->resource()) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {} failed: no xdgToplevel given. Errno: {}", name, errno));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
waylandState.xdgToplevel->setWmCapabilities(
|
2024-06-20 19:24:43 +02:00
|
|
|
[this](CCXdgToplevel* r, wl_array* arr) { backend->backend->log(AQ_LOG_DEBUG, std::format("Output {}: wm_capabilities received", name)); });
|
2024-06-19 22:40:23 +02:00
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.xdgToplevel->setConfigure([this](CCXdgToplevel* r, int32_t w, int32_t h, wl_array* arr) {
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->backend->log(AQ_LOG_DEBUG, std::format("Output {}: configure toplevel with {}x{}", name, w, h));
|
|
|
|
events.state.emit(SStateEvent{.size = {w, h}});
|
2024-06-19 22:40:23 +02:00
|
|
|
sendFrameAndSetCallback();
|
2024-06-18 18:45:05 +02:00
|
|
|
});
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
auto inputRegion = makeShared<CCWlRegion>(backend->waylandState.compositor->sendCreateRegion());
|
2024-06-19 22:40:23 +02:00
|
|
|
inputRegion->sendAdd(0, 0, INT32_MAX, INT32_MAX);
|
|
|
|
|
|
|
|
waylandState.surface->sendSetInputRegion(inputRegion.get());
|
2024-06-18 18:45:05 +02:00
|
|
|
waylandState.surface->sendAttach(nullptr, 0, 0);
|
|
|
|
waylandState.surface->sendCommit();
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
inputRegion->sendDestroy();
|
|
|
|
|
2024-06-18 18:45:05 +02:00
|
|
|
backend->backend->log(AQ_LOG_DEBUG, std::format("Output {}: initialized", name));
|
|
|
|
}
|
|
|
|
|
|
|
|
Aquamarine::CWaylandOutput::~CWaylandOutput() {
|
|
|
|
if (waylandState.xdgToplevel)
|
|
|
|
waylandState.xdgToplevel->sendDestroy();
|
|
|
|
if (waylandState.xdgSurface)
|
|
|
|
waylandState.xdgSurface->sendDestroy();
|
|
|
|
if (waylandState.surface)
|
|
|
|
waylandState.surface->sendDestroy();
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
bool Aquamarine::CWaylandOutput::test() {
|
|
|
|
return true; // TODO:
|
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
bool Aquamarine::CWaylandOutput::commit() {
|
|
|
|
Vector2D pixelSize = {};
|
|
|
|
uint32_t refreshRate = 0;
|
|
|
|
|
|
|
|
if (state->customMode)
|
|
|
|
pixelSize = state->customMode->pixelSize;
|
|
|
|
else if (state->mode)
|
|
|
|
pixelSize = state->mode->pixelSize;
|
|
|
|
else {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: pending state rejected: invalid mode", name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t format = state->drmFormat;
|
|
|
|
|
|
|
|
if (format == DRM_FORMAT_INVALID) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: pending state rejected: invalid format", name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
if (!swapchain) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: no swapchain, lying because it will soon be here", name));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
if (!swapchain->reconfigure(SSwapchainOptions{.length = 2, .size = pixelSize, .format = format})) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: pending state rejected: swapchain failed reconfiguring", name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!state->buffer) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: pending state rejected: no buffer", name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto wlBuffer = wlBufferFromBuffer(state->buffer);
|
|
|
|
|
|
|
|
if (!wlBuffer) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, std::format("Output {}: pending state rejected: no wlBuffer??", name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wlBuffer->pendingRelease)
|
|
|
|
backend->backend->log(AQ_LOG_WARNING, std::format("Output {}: pending state has a non-released buffer??", name));
|
|
|
|
|
|
|
|
wlBuffer->pendingRelease = true;
|
|
|
|
|
|
|
|
waylandState.surface->sendAttach(wlBuffer->waylandState.buffer.get(), 0, 0);
|
|
|
|
waylandState.surface->sendDamageBuffer(0, 0, INT32_MAX, INT32_MAX);
|
|
|
|
waylandState.surface->sendCommit();
|
|
|
|
|
2024-06-21 15:49:28 +02:00
|
|
|
readyForFrameCallback = true;
|
|
|
|
|
2024-06-21 18:37:09 +02:00
|
|
|
events.commit.emit();
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
SP<IBackendImplementation> Aquamarine::CWaylandOutput::getBackend() {
|
|
|
|
return SP<IBackendImplementation>(backend.lock());
|
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
SP<CWaylandBuffer> Aquamarine::CWaylandOutput::wlBufferFromBuffer(SP<IBuffer> buffer) {
|
|
|
|
std::erase_if(backendState.buffers, [this](const auto& el) { return el.first.expired() || !swapchain->contains(el.first.lock()); });
|
|
|
|
|
|
|
|
for (auto& [k, v] : backendState.buffers) {
|
|
|
|
if (k != buffer)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
// create a new one
|
|
|
|
auto wlBuffer = makeShared<CWaylandBuffer>(buffer, backend);
|
|
|
|
|
|
|
|
if (!wlBuffer->good())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
backendState.buffers.emplace_back(std::make_pair<>(buffer, wlBuffer));
|
|
|
|
|
|
|
|
return wlBuffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandOutput::sendFrameAndSetCallback() {
|
|
|
|
events.frame.emit();
|
2024-06-21 15:49:28 +02:00
|
|
|
frameScheduled = false;
|
|
|
|
if (waylandState.frameCallback || !readyForFrameCallback)
|
2024-06-19 22:40:23 +02:00
|
|
|
return;
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.frameCallback = makeShared<CCWlCallback>(waylandState.surface->sendFrame());
|
2024-06-21 15:49:28 +02:00
|
|
|
waylandState.frameCallback->setDone([this](CCWlCallback* r, uint32_t ms) { onFrameDone(); });
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandOutput::onFrameDone() {
|
|
|
|
waylandState.frameCallback.reset();
|
|
|
|
readyForFrameCallback = false;
|
|
|
|
|
|
|
|
if (frameScheduledWhileWaiting)
|
|
|
|
sendFrameAndSetCallback();
|
|
|
|
else
|
2024-06-19 22:40:23 +02:00
|
|
|
events.frame.emit();
|
2024-06-21 15:49:28 +02:00
|
|
|
|
|
|
|
frameScheduledWhileWaiting = false;
|
2024-06-19 22:40:23 +02:00
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
bool Aquamarine::CWaylandOutput::setCursor(Hyprutils::Memory::CSharedPointer<IBuffer> buffer, const Hyprutils::Math::Vector2D& hotspot) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandOutput::moveCursor(const Hyprutils::Math::Vector2D& coord) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Aquamarine::CWaylandOutput::scheduleFrame() {
|
2024-06-21 15:49:28 +02:00
|
|
|
if (frameScheduled)
|
2024-06-20 19:24:43 +02:00
|
|
|
return;
|
|
|
|
|
2024-06-21 15:49:28 +02:00
|
|
|
frameScheduled = true;
|
|
|
|
|
|
|
|
if (waylandState.frameCallback)
|
|
|
|
frameScheduledWhileWaiting = true;
|
|
|
|
else
|
|
|
|
backend->idleCallbacks.emplace_back([this]() { sendFrameAndSetCallback(); });
|
2024-06-20 19:24:43 +02:00
|
|
|
}
|
|
|
|
|
2024-06-19 22:40:23 +02:00
|
|
|
Aquamarine::CWaylandBuffer::CWaylandBuffer(SP<IBuffer> buffer_, Hyprutils::Memory::CWeakPointer<CWaylandBackend> backend_) : buffer(buffer_), backend(backend_) {
|
2024-06-20 19:24:43 +02:00
|
|
|
auto params = makeShared<CCZwpLinuxBufferParamsV1>(backend->waylandState.dmabuf->sendCreateParams());
|
2024-06-19 22:40:23 +02:00
|
|
|
|
|
|
|
if (!params) {
|
|
|
|
backend->backend->log(AQ_LOG_ERROR, "WaylandBuffer: failed to query params");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto attrs = buffer->dmabuf();
|
|
|
|
|
|
|
|
for (size_t i = 0; i < attrs.planes; ++i) {
|
|
|
|
params->sendAdd(attrs.fds.at(i), i, attrs.offsets.at(i), attrs.strides.at(i), attrs.modifier >> 32, attrs.modifier & 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.buffer = makeShared<CCWlBuffer>(params->sendCreateImmed(attrs.size.x, attrs.size.y, attrs.format, (zwpLinuxBufferParamsV1Flags)0));
|
2024-06-19 22:40:23 +02:00
|
|
|
|
2024-06-20 19:24:43 +02:00
|
|
|
waylandState.buffer->setRelease([this](CCWlBuffer* r) { pendingRelease = false; });
|
2024-06-19 22:40:23 +02:00
|
|
|
|
|
|
|
params->sendDestroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
Aquamarine::CWaylandBuffer::~CWaylandBuffer() {
|
|
|
|
if (waylandState.buffer && waylandState.buffer->resource())
|
|
|
|
waylandState.buffer->sendDestroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Aquamarine::CWaylandBuffer::good() {
|
|
|
|
return waylandState.buffer && waylandState.buffer->resource();
|
|
|
|
}
|