#include #include #include #include #include #include #include #include #include #include #include #include #include #include using Hyprutils::Math::Vector2D; using namespace Hyprutils::Memory; struct SWlState { wl_display* display; CSharedPointer registry; // protocols CSharedPointer wlCompositor; CSharedPointer wlSeat; CSharedPointer wlShm; CSharedPointer xdgShell; CSharedPointer pointerWarp; // shm/buffer stuff CSharedPointer shmPool; CSharedPointer shmBuf; int shmFd; size_t shmBufSize; bool xrgb8888_support = false; // surface/toplevel stuff CSharedPointer surf; CSharedPointer xdgSurf; CSharedPointer xdgToplevel; Vector2D geom; // pointer CSharedPointer pointer; uint32_t enterSerial; }; static bool debug, started, shouldExit; template //NOLINTNEXTLINE static void clientLog(std::format_string fmt, Args&&... args) { std::println("{}", std::vformat(fmt.get(), std::make_format_args(args...))); std::fflush(stdout); } template //NOLINTNEXTLINE static void debugLog(std::format_string fmt, Args&&... args) { if (!debug) return; std::println("{}", std::vformat(fmt.get(), std::make_format_args(args...))); std::fflush(stdout); } static bool bindRegistry(SWlState& state) { state.registry = makeShared((wl_proxy*)wl_display_get_registry(state.display)); state.registry->setGlobal([&](CCWlRegistry* r, uint32_t id, const char* name, uint32_t version) { const std::string NAME = name; if (NAME == "wl_compositor") { debugLog(" > binding to global: {} (version {}) with id {}", name, version, id); state.wlCompositor = makeShared((wl_proxy*)wl_registry_bind((wl_registry*)state.registry->resource(), id, &wl_compositor_interface, 6)); } else if (NAME == "wl_shm") { debugLog(" > binding to global: {} (version {}) with id {}", name, version, id); state.wlShm = makeShared((wl_proxy*)wl_registry_bind((wl_registry*)state.registry->resource(), id, &wl_shm_interface, 1)); } else if (NAME == "wl_seat") { debugLog(" > binding to global: {} (version {}) with id {}", name, version, id); state.wlSeat = makeShared((wl_proxy*)wl_registry_bind((wl_registry*)state.registry->resource(), id, &wl_seat_interface, 9)); } else if (NAME == "xdg_wm_base") { debugLog(" > binding to global: {} (version {}) with id {}", name, version, id); state.xdgShell = makeShared((wl_proxy*)wl_registry_bind((wl_registry*)state.registry->resource(), id, &xdg_wm_base_interface, 1)); } else if (NAME == "wp_pointer_warp_v1") { debugLog(" > binding to global: {} (version {}) with id {}", name, version, id); state.pointerWarp = makeShared((wl_proxy*)wl_registry_bind((wl_registry*)state.registry->resource(), id, &wp_pointer_warp_v1_interface, 1)); } }); state.registry->setGlobalRemove([](CCWlRegistry* r, uint32_t id) { debugLog("Global {} removed", id); }); wl_display_roundtrip(state.display); if (!state.wlCompositor || !state.wlShm || !state.wlSeat || !state.xdgShell || !state.pointerWarp) { clientLog("Failed to get protocols from Hyprland"); return false; } return true; } static bool createShm(SWlState& state, Vector2D geom) { if (!state.xrgb8888_support) return false; size_t stride = geom.x * 4; size_t size = geom.y * stride; if (!state.shmPool) { const char* name = "/wl-shm-pointer-warp"; state.shmFd = shm_open(name, O_RDWR | O_CREAT | O_EXCL, 0600); if (state.shmFd < 0) return false; if (shm_unlink(name) < 0 || ftruncate(state.shmFd, size) < 0) { close(state.shmFd); return false; } state.shmPool = makeShared(state.wlShm->sendCreatePool(state.shmFd, size)); if (!state.shmPool->resource()) { close(state.shmFd); state.shmFd = -1; state.shmPool.reset(); return false; } state.shmBufSize = size; } else if (size > state.shmBufSize) { if (ftruncate(state.shmFd, size) < 0) { close(state.shmFd); state.shmFd = -1; state.shmPool.reset(); return false; } state.shmPool->sendResize(size); state.shmBufSize = size; } auto buf = makeShared(state.shmPool->sendCreateBuffer(0, geom.x, geom.y, stride, WL_SHM_FORMAT_XRGB8888)); if (!buf->resource()) return false; if (state.shmBuf) { state.shmBuf->sendDestroy(); state.shmBuf.reset(); } state.shmBuf = buf; return true; } static bool setupToplevel(SWlState& state) { state.wlShm->setFormat([&](CCWlShm* p, uint32_t format) { if (format == WL_SHM_FORMAT_XRGB8888) state.xrgb8888_support = true; }); state.xdgShell->setPing([&](CCXdgWmBase* p, uint32_t serial) { state.xdgShell->sendPong(serial); }); state.surf = makeShared(state.wlCompositor->sendCreateSurface()); if (!state.surf->resource()) return false; state.xdgSurf = makeShared(state.xdgShell->sendGetXdgSurface(state.surf->resource())); if (!state.xdgSurf->resource()) return false; state.xdgToplevel = makeShared(state.xdgSurf->sendGetToplevel()); if (!state.xdgToplevel->resource()) return false; state.xdgToplevel->setClose([&](CCXdgToplevel* p) { exit(0); }); state.xdgToplevel->setConfigure([&](CCXdgToplevel* p, int32_t w, int32_t h, wl_array* arr) { state.geom = {1280, 720}; if (!createShm(state, state.geom)) exit(-1); }); state.xdgSurf->setConfigure([&](CCXdgSurface* p, uint32_t serial) { if (!state.shmBuf) debugLog("xdgSurf configure but no buf made yet?"); state.xdgSurf->sendSetWindowGeometry(0, 0, state.geom.x, state.geom.y); state.surf->sendAttach(state.shmBuf.get(), 0, 0); state.surf->sendCommit(); state.xdgSurf->sendAckConfigure(serial); if (!started) { started = true; clientLog("started"); } }); state.xdgToplevel->sendSetTitle("pointer-warp test client"); state.xdgToplevel->sendSetAppId("pointer-warp"); state.surf->sendAttach(nullptr, 0, 0); state.surf->sendCommit(); return true; } static bool setupSeat(SWlState& state) { state.pointer = makeShared(state.wlSeat->sendGetPointer()); if (!state.pointer->resource()) return false; state.pointer->setEnter([&](CCWlPointer* p, uint32_t serial, wl_proxy* surf, wl_fixed_t x, wl_fixed_t y) { debugLog("Got pointer enter event, serial {}, x {}, y {}", serial, x, y); state.enterSerial = serial; }); state.pointer->setLeave([&](CCWlPointer* p, uint32_t serial, wl_proxy* surf) { debugLog("Got pointer leave event, serial {}", serial); }); state.pointer->setMotion([&](CCWlPointer* p, uint32_t serial, wl_fixed_t x, wl_fixed_t y) { debugLog("Got pointer motion event, serial {}, x {}, y {}", serial, x, y); }); return true; } // format is like below // "warp 20 20\n" would ask to warp cursor to x=20,y=20 in surface local coords static void parseRequest(SWlState& state, std::string req) { if (req.contains("exit")) { shouldExit = true; return; } if (!req.starts_with("warp ")) return; auto it = req.find_first_of('\n'); if (it == std::string::npos) return; req = req.substr(0, it); it = req.find_first_of(' '); if (it == std::string::npos) return; req = req.substr(it + 1); it = req.find_first_of(' '); int x = std::stoi(req.substr(0, it)); int y = std::stoi(req.substr(it + 1)); state.pointerWarp->sendWarpPointer(state.surf->resource(), state.pointer->resource(), wl_fixed_from_int(x), wl_fixed_from_int(y), state.enterSerial); // sync the request then reply wl_display_roundtrip(state.display); clientLog("parsed request to move to x:{}, y:{}", x, y); } int main(int argc, char** argv) { if (argc != 1 && argc != 2) clientLog("Only the \"--debug\" switch is allowed, it turns on debug logs."); if (argc == 2 && std::string{argv[1]} == "--debug") debug = true; SWlState state; // WAYLAND_DISPLAY env should be set to the correct one state.display = wl_display_connect(nullptr); if (!state.display) { clientLog("Failed to connect to wayland display"); return -1; } if (!bindRegistry(state) || !setupSeat(state) || !setupToplevel(state)) return -1; std::array readBuf; readBuf.fill(0); wl_display_flush(state.display); struct pollfd fds[2] = {{.fd = wl_display_get_fd(state.display), .events = POLLIN | POLLOUT}, {.fd = STDIN_FILENO, .events = POLLIN}}; while (!shouldExit && poll(fds, 2, 0) != -1) { if (fds[0].revents & POLLIN) { wl_display_flush(state.display); if (wl_display_prepare_read(state.display) == 0) { wl_display_read_events(state.display); wl_display_dispatch_pending(state.display); } else wl_display_dispatch(state.display); int ret = 0; do { ret = wl_display_dispatch_pending(state.display); wl_display_flush(state.display); } while (ret > 0); } if (fds[1].revents & POLLIN) { ssize_t bytesRead = read(fds[1].fd, readBuf.data(), 1023); if (bytesRead == -1) continue; readBuf[bytesRead] = 0; parseRequest(state, std::string{readBuf.data()}); } } wl_display* display = state.display; state = {}; wl_display_disconnect(display); return 0; }