diff options
| author | Isaac Freund <ifreund@ifreund.xyz> | 2021-09-18 18:25:21 +0200 |
|---|---|---|
| committer | Isaac Freund <ifreund@ifreund.xyz> | 2021-09-18 18:45:30 +0200 |
| commit | b8ebbc29cf42a36e97666bf9757a9bc354e31925 (patch) | |
| tree | 5daaf5ead259075d19368e90a7fe24291b14230c | |
| parent | ab55ab8fc2a7f795ec586ad829d040a9820a4486 (diff) | |
| download | river-b8ebbc29cf42a36e97666bf9757a9bc354e31925.tar.gz river-b8ebbc29cf42a36e97666bf9757a9bc354e31925.tar.xz | |
xdg-toplevel: fix configure serial checking
Currently if another configure is in flight after the one we are
tracking the serial of and the client acks the second configure as well
(or only the second configure) before committing, we will never realize
the configure we are tracking has been acked.
Instead, listen for the ack_configure signal and set a bool that we can
check on surface commit.
This probably isn't an issue that would actually be hit by well behaved
clients as river doesn't send redundant configure events. However,
having correct code is always better even if it's slightly more complex.
| -rw-r--r-- | river/View.zig | 6 | ||||
| -rw-r--r-- | river/XdgToplevel.zig | 27 |
2 files changed, 26 insertions, 7 deletions
diff --git a/river/View.zig b/river/View.zig index a27bac4..af380d3 100644 --- a/river/View.zig +++ b/river/View.zig @@ -229,10 +229,10 @@ pub fn needsConfigure(self: Self) bool { }; } -pub fn configure(self: Self) void { +pub fn configure(self: *Self) void { switch (self.impl) { - .xdg_toplevel => |xdg_toplevel| xdg_toplevel.configure(), - .xwayland_view => |xwayland_view| xwayland_view.configure(), + .xdg_toplevel => |*xdg_toplevel| xdg_toplevel.configure(), + .xwayland_view => |*xwayland_view| xwayland_view.configure(), } } diff --git a/river/XdgToplevel.zig b/river/XdgToplevel.zig index 1606e6a..adbc7c7 100644 --- a/river/XdgToplevel.zig +++ b/river/XdgToplevel.zig @@ -40,6 +40,9 @@ view: *View, /// The corresponding wlroots object xdg_surface: *wlr.XdgSurface, +/// Set to true when the client acks the configure with serial View.pending_serial. +acked_pending_serial: bool = false, + // Listeners that are always active over the view's lifetime destroy: wl.Listener(*wlr.XdgSurface) = wl.Listener(*wlr.XdgSurface).init(handleDestroy), map: wl.Listener(*wlr.XdgSurface) = wl.Listener(*wlr.XdgSurface).init(handleMap), @@ -48,6 +51,8 @@ new_popup: wl.Listener(*wlr.XdgPopup) = wl.Listener(*wlr.XdgPopup).init(handleNe new_subsurface: wl.Listener(*wlr.Subsurface) = wl.Listener(*wlr.Subsurface).init(handleNewSubsurface), // Listeners that are only active while the view is mapped +ack_configure: wl.Listener(*wlr.XdgSurface.Configure) = + wl.Listener(*wlr.XdgSurface.Configure).init(handleAckConfigure), commit: wl.Listener(*wlr.Surface) = wl.Listener(*wlr.Surface).init(handleCommit), request_fullscreen: wl.Listener(*wlr.XdgToplevel.event.SetFullscreen) = wl.Listener(*wlr.XdgToplevel.event.SetFullscreen).init(handleRequestFullscreen), @@ -104,10 +109,11 @@ pub fn needsConfigure(self: Self) bool { } /// Send a configure event, applying the pending state of the view. -pub fn configure(self: Self) void { +pub fn configure(self: *Self) void { const toplevel = self.xdg_surface.role_data.toplevel; const state = &self.view.pending; self.view.pending_serial = toplevel.setSize(state.box.width, state.box.height); + self.acked_pending_serial = false; } /// Close the view. This will lead to the unmap and destroy events being sent @@ -173,6 +179,7 @@ fn handleMap(listener: *wl.Listener(*wlr.XdgSurface), xdg_surface: *wlr.XdgSurfa const toplevel = self.xdg_surface.role_data.toplevel; // Add listeners that are only active while mapped + self.xdg_surface.events.ack_configure.add(&self.ack_configure); self.xdg_surface.surface.events.commit.add(&self.commit); toplevel.events.request_fullscreen.add(&self.request_fullscreen); toplevel.events.request_move.add(&self.request_move); @@ -237,6 +244,7 @@ fn handleUnmap(listener: *wl.Listener(*wlr.XdgSurface), xdg_surface: *wlr.XdgSur const self = @fieldParentPtr(Self, "unmap", listener); // Remove listeners that are only active while mapped + self.ack_configure.link.remove(); self.commit.link.remove(); self.request_fullscreen.link.remove(); self.request_move.link.remove(); @@ -247,7 +255,18 @@ fn handleUnmap(listener: *wl.Listener(*wlr.XdgSurface), xdg_surface: *wlr.XdgSur self.view.unmap(); } -/// Called when the surface is comitted +fn handleAckConfigure( + listener: *wl.Listener(*wlr.XdgSurface.Configure), + acked_configure: *wlr.XdgSurface.Configure, +) void { + const self = @fieldParentPtr(Self, "ack_configure", listener); + if (self.view.pending_serial) |serial| { + if (serial == acked_configure.serial) { + self.acked_pending_serial = true; + } + } +} + fn handleCommit(listener: *wl.Listener(*wlr.Surface), surface: *wlr.Surface) void { const self = @fieldParentPtr(Self, "commit", listener); const view = self.view; @@ -257,11 +276,11 @@ fn handleCommit(listener: *wl.Listener(*wlr.Surface), surface: *wlr.Surface) voi const new_box = Box.fromWlrBox(wlr_box); // If we have sent a configure changing the size - if (view.pending_serial) |s| { + if (view.pending_serial != null) { // Update the stored dimensions of the surface view.surface_box = new_box; - if (s == self.xdg_surface.configure_serial) { + if (self.acked_pending_serial) { // If this commit is in response to our configure and the // transaction code is tracking this configure, notify it. // Otherwise, apply the pending state immediately. |
