aboutsummaryrefslogtreecommitdiffstats
path: root/sway/desktop/transaction.c
diff options
context:
space:
mode:
authorLibravatar Ryan Dwyer <ryandwyer1@gmail.com>2018-08-02 20:54:03 +1000
committerLibravatar Ryan Dwyer <ryandwyer1@gmail.com>2018-08-02 20:54:03 +1000
commit8314019f660cd28fc8cdb634f82b437105074258 (patch)
treed788e6fc6cc5855cc2ffea909ddb22fb8b9f37ea /sway/desktop/transaction.c
parentLink xcb dependency to meson options "enable_xwayland" (#2393) (diff)
downloadsway-8314019f660cd28fc8cdb634f82b437105074258.tar.gz
sway-8314019f660cd28fc8cdb634f82b437105074258.tar.zst
sway-8314019f660cd28fc8cdb634f82b437105074258.zip
Fix race condition crashes when unmapping views
This fixes two issues which were both introduced in #2396. First issue: The PR changes the location of the buffer save to transaction_apply, but puts it inside the should_configure block. For unmapping (destroying) views, should_configure returns false so it wasn't saving the buffer. If a frame was rendered between the unmap and the transaction applying then it would result in a crash. Second issue: If a destroying view is involved in two transactions, we must not release the buffer between the transactions because there is no live buffer to grab any more.
Diffstat (limited to 'sway/desktop/transaction.c')
-rw-r--r--sway/desktop/transaction.c23
1 files changed, 15 insertions, 8 deletions
diff --git a/sway/desktop/transaction.c b/sway/desktop/transaction.c
index 94070363..4e6af86a 100644
--- a/sway/desktop/transaction.c
+++ b/sway/desktop/transaction.c
@@ -195,11 +195,18 @@ static void transaction_apply(struct sway_transaction *transaction) {
195 sizeof(struct sway_container_state)); 195 sizeof(struct sway_container_state));
196 196
197 if (container->type == C_VIEW) { 197 if (container->type == C_VIEW) {
198 if (container->sway_view->saved_buffer) { 198 if (container->destroying) {
199 view_remove_saved_buffer(container->sway_view); 199 if (container->instructions->length == 1 &&
200 } 200 container->sway_view->saved_buffer) {
201 if (container->instructions->length > 1) { 201 view_remove_saved_buffer(container->sway_view);
202 view_save_buffer(container->sway_view); 202 }
203 } else {
204 if (container->sway_view->saved_buffer) {
205 view_remove_saved_buffer(container->sway_view);
206 }
207 if (container->instructions->length > 1) {
208 view_save_buffer(container->sway_view);
209 }
203 } 210 }
204 } 211 }
205 } 212 }
@@ -276,9 +283,9 @@ static void transaction_commit(struct sway_transaction *transaction) {
276 // mapping and its default geometry doesn't intersect an output. 283 // mapping and its default geometry doesn't intersect an output.
277 struct timespec when; 284 struct timespec when;
278 wlr_surface_send_frame_done(con->sway_view->surface, &when); 285 wlr_surface_send_frame_done(con->sway_view->surface, &when);
279 if (!con->sway_view->saved_buffer) { 286 }
280 view_save_buffer(con->sway_view); 287 if (con->type == C_VIEW && !con->sway_view->saved_buffer) {
281 } 288 view_save_buffer(con->sway_view);
282 } 289 }
283 list_add(con->instructions, instruction); 290 list_add(con->instructions, instruction);
284 } 291 }