data-device: Move all drag-related fields out of weston_seat

We can now allocate a temporary weston_drag structure that we keep all
this drag-and-drop related state in.
diff --git a/src/data-device.c b/src/data-device.c
index 8c20d89..344fdff 100644
--- a/src/data-device.c
+++ b/src/data-device.c
@@ -27,6 +27,19 @@
 
 #include "compositor.h"
 
+struct weston_drag {
+	struct wl_client *client;
+	struct wl_data_source *data_source;
+	struct wl_listener data_source_listener;
+	struct weston_surface *focus;
+	struct wl_resource *focus_resource;
+	struct wl_listener focus_listener;
+	struct weston_pointer_grab grab;
+	struct weston_surface *surface;
+	struct wl_listener surface_destroy_listener;
+	int32_t dx, dy;
+};
+
 static void
 empty_region(pixman_region32_t *region)
 {
@@ -165,8 +178,8 @@
 static void
 drag_surface_configure(struct weston_surface *es, int32_t sx, int32_t sy, int32_t width, int32_t height)
 {
-	struct weston_seat *seat = es->configure_private;
-	struct weston_pointer *pointer = seat->pointer;
+	struct weston_drag *drag = es->configure_private;
+	struct weston_pointer *pointer = drag->grab.pointer;
 	struct wl_list *list;
 	float fx, fy;
 
@@ -174,23 +187,23 @@
 		if (pointer->sprite && weston_surface_is_mapped(pointer->sprite))
 			list = &pointer->sprite->layer_link;
 		else
-			list = &seat->compositor->cursor_layer.surface_list;
+			list = &es->compositor->cursor_layer.surface_list;
 
 		wl_list_insert(list, &es->layer_link);
 		weston_surface_update_transform(es);
 		empty_region(&es->pending.input);
 	}
 
-	seat->drag_dx += sx;
-	seat->drag_dy += sy;
+	drag->dx += sx;
+	drag->dy += sy;
 
-	fx = wl_fixed_to_double(seat->pointer->x) + seat->drag_dx;
-	fy = wl_fixed_to_double(seat->pointer->y) + seat->drag_dy;
+	fx = wl_fixed_to_double(pointer->x) + drag->dx;
+	fy = wl_fixed_to_double(pointer->y) + drag->dy;
 	weston_surface_configure(es, fx, fy, width, height);
 }
 
 static int
-device_setup_new_drag_surface(struct weston_seat *seat,
+device_setup_new_drag_surface(struct weston_drag *drag,
 			      struct weston_surface *surface)
 {
 	if (surface->configure) {
@@ -200,65 +213,64 @@
 		return 0;
 	}
 
-	seat->drag_surface = surface;
-	seat->drag_dx = 0;
-	seat->drag_dy = 0;
+	drag->surface = surface;
+	drag->dx = 0;
+	drag->dy = 0;
 
 	surface->configure = drag_surface_configure;
-	surface->configure_private = seat;
+	surface->configure_private = drag;
 
 	wl_signal_add(&surface->resource.destroy_signal,
-		       &seat->drag_surface_destroy_listener);
+		       &drag->surface_destroy_listener);
 
 	return 1;
 }
 
 static void
-device_release_drag_surface(struct weston_seat *seat)
+device_release_drag_surface(struct weston_drag *drag)
 {
-	if (weston_surface_is_mapped(seat->drag_surface))
-		weston_surface_unmap(seat->drag_surface);
+	if (weston_surface_is_mapped(drag->surface))
+		weston_surface_unmap(drag->surface);
 
-	seat->drag_surface->configure = NULL;
-	empty_region(&seat->drag_surface->pending.input);
-	wl_list_remove(&seat->drag_surface_destroy_listener.link);
-	seat->drag_surface = NULL;
+	drag->surface->configure = NULL;
+	empty_region(&drag->surface->pending.input);
+	wl_list_remove(&drag->surface_destroy_listener.link);
+	drag->surface = NULL;
 }
 
 static void
 destroy_drag_focus(struct wl_listener *listener, void *data)
 {
-	struct weston_seat *seat =
-		container_of(listener, struct weston_seat, drag_focus_listener);
+	struct weston_drag *drag =
+		container_of(listener, struct weston_drag, focus_listener);
 
-	seat->drag_focus_resource = NULL;
+	drag->focus_resource = NULL;
 }
 
 static void
 drag_grab_focus(struct weston_pointer_grab *grab,
 		struct weston_surface *surface, wl_fixed_t x, wl_fixed_t y)
 {
-	struct weston_seat *seat =
-		container_of(grab, struct weston_seat, drag_grab);
+	struct weston_drag *drag =
+		container_of(grab, struct weston_drag, grab);
 	struct wl_resource *resource, *offer = NULL;
 	struct wl_display *display;
 	uint32_t serial;
 
-	if (seat->drag_focus_resource) {
-		wl_data_device_send_leave(seat->drag_focus_resource);
-		wl_list_remove(&seat->drag_focus_listener.link);
-		seat->drag_focus_resource = NULL;
-		seat->drag_focus = NULL;
+	if (drag->focus_resource) {
+		wl_data_device_send_leave(drag->focus_resource);
+		wl_list_remove(&drag->focus_listener.link);
+		drag->focus_resource = NULL;
+		drag->focus = NULL;
 	}
 
 	if (!surface)
 		return;
 
-	if (!seat->drag_data_source &&
-	    surface->resource.client != seat->drag_client)
+	if (!drag->data_source && surface->resource.client != drag->client)
 		return;
 
-	resource = find_resource(&seat->drag_resource_list,
+	resource = find_resource(&drag->grab.pointer->seat->drag_resource_list,
 				 surface->resource.client);
 	if (!resource)
 		return;
@@ -266,18 +278,16 @@
 	display = wl_client_get_display(resource->client);
 	serial = wl_display_next_serial(display);
 
-	if (seat->drag_data_source)
-		offer = wl_data_source_send_offer(seat->drag_data_source,
-						  resource);
+	if (drag->data_source)
+		offer = wl_data_source_send_offer(drag->data_source, resource);
 
 	wl_data_device_send_enter(resource, serial, &surface->resource,
 				  x, y, offer);
 
-	seat->drag_focus = surface;
-	seat->drag_focus_listener.notify = destroy_drag_focus;
-	wl_signal_add(&resource->destroy_signal,
-		      &seat->drag_focus_listener);
-	seat->drag_focus_resource = resource;
+	drag->focus = surface;
+	drag->focus_listener.notify = destroy_drag_focus;
+	wl_signal_add(&resource->destroy_signal, &drag->focus_listener);
+	drag->focus_resource = resource;
 	grab->focus = surface;
 }
 
@@ -285,55 +295,55 @@
 drag_grab_motion(struct weston_pointer_grab *grab,
 		 uint32_t time, wl_fixed_t x, wl_fixed_t y)
 {
-	struct weston_seat *seat =
-		container_of(grab, struct weston_seat, drag_grab);
+	struct weston_drag *drag =
+		container_of(grab, struct weston_drag, grab);
+	struct weston_pointer *pointer = drag->grab.pointer;
 	float fx, fy;
 
-	if (seat->drag_surface) {
-		fx = wl_fixed_to_double(seat->pointer->x) + seat->drag_dx;
-		fy = wl_fixed_to_double(seat->pointer->y) + seat->drag_dy;
-		weston_surface_set_position(seat->drag_surface, fx, fy);
-		weston_surface_schedule_repaint(seat->drag_surface);
+	if (drag->surface) {
+		fx = wl_fixed_to_double(pointer->x) + drag->dx;
+		fy = wl_fixed_to_double(pointer->y) + drag->dy;
+		weston_surface_set_position(drag->surface, fx, fy);
+		weston_surface_schedule_repaint(drag->surface);
 	}
 
-	if (seat->drag_focus_resource)
-		wl_data_device_send_motion(seat->drag_focus_resource,
-					   time, x, y);
+	if (drag->focus_resource)
+		wl_data_device_send_motion(drag->focus_resource, time, x, y);
 }
 
 static void
-data_device_end_drag_grab(struct weston_seat *seat)
+data_device_end_drag_grab(struct weston_drag *drag)
 {
-	if (seat->drag_surface)
-		device_release_drag_surface(seat);
+	if (drag->surface)
+		device_release_drag_surface(drag);
 
-	drag_grab_focus(&seat->drag_grab, NULL,
+	drag_grab_focus(&drag->grab, NULL,
 	                wl_fixed_from_int(0), wl_fixed_from_int(0));
 
-	weston_pointer_end_grab(seat->pointer);
+	weston_pointer_end_grab(drag->grab.pointer);
 
-	seat->drag_data_source = NULL;
-	seat->drag_client = NULL;
+	free(drag);
 }
 
 static void
 drag_grab_button(struct weston_pointer_grab *grab,
 		 uint32_t time, uint32_t button, uint32_t state_w)
 {
-	struct weston_seat *seat =
-		container_of(grab, struct weston_seat, drag_grab);
+	struct weston_drag *drag =
+		container_of(grab, struct weston_drag, grab);
+	struct weston_pointer *pointer = drag->grab.pointer;
 	enum wl_pointer_button_state state = state_w;
 
-	if (seat->drag_focus_resource &&
-	    seat->pointer->grab_button == button &&
+	if (drag->focus_resource &&
+	    pointer->grab_button == button &&
 	    state == WL_POINTER_BUTTON_STATE_RELEASED)
-		wl_data_device_send_drop(seat->drag_focus_resource);
+		wl_data_device_send_drop(drag->focus_resource);
 
-	if (seat->pointer->button_count == 0 &&
+	if (pointer->button_count == 0 &&
 	    state == WL_POINTER_BUTTON_STATE_RELEASED) {
-		if (seat->drag_data_source)
-			wl_list_remove(&seat->drag_data_source_listener.link);
-		data_device_end_drag_grab(seat);
+		if (drag->data_source)
+			wl_list_remove(&drag->data_source_listener.link);
+		data_device_end_drag_grab(drag);
 	}
 }
 
@@ -346,21 +356,19 @@
 static void
 destroy_data_device_source(struct wl_listener *listener, void *data)
 {
-	struct weston_seat *seat = container_of(listener, struct weston_seat,
-						drag_data_source_listener);
+	struct weston_drag *drag = container_of(listener, struct weston_drag,
+						data_source_listener);
 
-	data_device_end_drag_grab(seat);
+	data_device_end_drag_grab(drag);
 }
 
 static void
 handle_drag_surface_destroy(struct wl_listener *listener, void *data)
 {
-	struct weston_seat *seat;
+	struct weston_drag *drag = container_of(listener, struct weston_drag,
+						surface_destroy_listener);
 
-	seat = container_of(listener, struct weston_seat,
-			    drag_surface_destroy_listener);
-
-	seat->drag_surface = NULL;
+	drag->surface = NULL;
 }
 
 static void
@@ -370,35 +378,39 @@
 		       struct wl_resource *icon_resource, uint32_t serial)
 {
 	struct weston_seat *seat = resource->data;
+	struct weston_drag *drag = resource->data;
 
 	/* FIXME: Check that client has implicit grab on the origin
 	 * surface that matches the given time. */
 
 	/* FIXME: Check that the data source type array isn't empty. */
 
-	seat->drag_grab.interface = &drag_grab_interface;
+	drag = malloc(sizeof *drag);
+	if (drag == NULL) {
+		wl_resource_post_no_memory(resource);
+		return;
+	}
 
-	seat->drag_client = client;
-	seat->drag_data_source = NULL;
-	seat->drag_surface_destroy_listener.notify =
-		handle_drag_surface_destroy;
+	memset(drag, 0, sizeof *drag);
+	drag->grab.interface = &drag_grab_interface;
+	drag->client = client;
+	drag->surface_destroy_listener.notify = handle_drag_surface_destroy;
 
 	if (source_resource) {
-		seat->drag_data_source = source_resource->data;
-		seat->drag_data_source_listener.notify =
-			destroy_data_device_source;
+		drag->data_source = source_resource->data;
+		drag->data_source_listener.notify = destroy_data_device_source;
 		wl_signal_add(&source_resource->destroy_signal,
-			      &seat->drag_data_source_listener);
+			      &drag->data_source_listener);
 	}
 
 	if (icon_resource) {
-		if (!device_setup_new_drag_surface(seat, icon_resource->data))
+		if (!device_setup_new_drag_surface(drag, icon_resource->data))
 			return;
 	}
 
 	weston_pointer_set_focus(seat->pointer, NULL,
 				 wl_fixed_from_int(0), wl_fixed_from_int(0));
-	weston_pointer_start_grab(seat->pointer, &seat->drag_grab);
+	weston_pointer_start_grab(seat->pointer, &drag->grab);
 }
 
 static void