From b98563d2d7fd2481188c60c0dafc036a9ca4dd15 Mon Sep 17 00:00:00 2001
From: emersion <contact@emersion.fr>
Date: Wed, 28 Nov 2018 14:08:20 +0100
Subject: Fix segfault when destroying unmapped child view

---
 sway/tree/view.c | 12 +++++++++---
 1 file changed, 9 insertions(+), 3 deletions(-)

(limited to 'sway/tree')

diff --git a/sway/tree/view.c b/sway/tree/view.c
index 0edefc8e..4c46d0e9 100644
--- a/sway/tree/view.c
+++ b/sway/tree/view.c
@@ -711,8 +711,6 @@ static const struct sway_view_child_impl subsurface_impl = {
 	.destroy = subsurface_destroy,
 };
 
-static void view_child_damage(struct sway_view_child *child, bool whole);
-
 static void subsurface_handle_destroy(struct wl_listener *listener,
 		void *data) {
 	struct sway_subsurface *subsurface =
@@ -721,6 +719,8 @@ static void subsurface_handle_destroy(struct wl_listener *listener,
 	view_child_destroy(child);
 }
 
+static void view_child_damage(struct sway_view_child *child, bool whole);
+
 static void view_subsurface_create(struct sway_view *view,
 		struct wlr_subsurface *wlr_subsurface) {
 	struct sway_subsurface *subsurface =
@@ -734,6 +734,9 @@ static void view_subsurface_create(struct sway_view *view,
 
 	wl_signal_add(&wlr_subsurface->events.destroy, &subsurface->destroy);
 	subsurface->destroy.notify = subsurface_handle_destroy;
+
+	subsurface->child.mapped = true;
+	view_child_damage(&subsurface->child, true);
 }
 
 static void view_child_damage(struct sway_view_child *child, bool whole) {
@@ -778,6 +781,7 @@ static void view_child_handle_surface_map(struct wl_listener *listener,
 		void *data) {
 	struct sway_view_child *child =
 		wl_container_of(listener, child, surface_map);
+	child->mapped = true;
 	view_child_damage(child, true);
 }
 
@@ -786,6 +790,7 @@ static void view_child_handle_surface_unmap(struct wl_listener *listener,
 	struct sway_view_child *child =
 		wl_container_of(listener, child, surface_unmap);
 	view_child_damage(child, true);
+	child->mapped = false;
 }
 
 void view_child_init(struct sway_view_child *child,
@@ -804,6 +809,7 @@ void view_child_init(struct sway_view_child *child,
 	wl_signal_add(&surface->events.destroy, &child->surface_destroy);
 	child->surface_destroy.notify = view_child_handle_surface_destroy;
 
+	// Not all child views have a map/unmap event
 	child->surface_map.notify = view_child_handle_surface_map;
 	child->surface_unmap.notify = view_child_handle_surface_unmap;
 
@@ -814,7 +820,7 @@ void view_child_init(struct sway_view_child *child,
 }
 
 void view_child_destroy(struct sway_view_child *child) {
-	if (child->view->container != NULL) {
+	if (child->mapped && child->view->container != NULL) {
 		view_child_damage(child, true);
 	}
 
-- 
cgit v1.2.3