Move the code launching the screenshooter client and implementing
the screenshot protocol out of libweston, and make screenshooter.c
a generic way to hook screenshoot protocols into weston.
---
 desktop-shell/shell.c |   2 -
 src/compositor.h      |   5 +-
 src/screenshooter.c   | 176 ++++++++------------------------------------------
 src/weston.c          | 161 +++++++++++++++++++++++++++++++++++++++++++++
 4 files changed, 191 insertions(+), 153 deletions(-)

diff --git a/desktop-shell/shell.c b/desktop-shell/shell.c
index dfc7b65..946ad28 100644
--- a/desktop-shell/shell.c
+++ b/desktop-shell/shell.c
@@ -6595,8 +6595,6 @@ module_init(struct weston_compositor *ec,
        shell->seat_create_listener.notify = handle_seat_created;
        wl_signal_add(&ec->seat_created_signal, &shell->seat_create_listener);
 
-       screenshooter_create(ec);
-
        shell_add_bindings(ec, shell);
 
        shell_fade_init(shell);
diff --git a/src/compositor.h b/src/compositor.h
index e3fcbaf..d07c04a 100644
--- a/src/compositor.h
+++ b/src/compositor.h
@@ -1352,7 +1352,10 @@ int
 tty_activate_vt(struct tty *tty, int vt);
 
 void
-screenshooter_create(struct weston_compositor *ec);
+weston_recorder_start(struct weston_compositor *ec,
+                     struct weston_output *output);
+void
+weston_recorder_stop(struct weston_compositor *ec);
 
 enum weston_screenshooter_outcome {
        WESTON_SCREENSHOOTER_SUCCESS,
diff --git a/src/screenshooter.c b/src/screenshooter.c
index cafbf10..25d52a9 100644
--- a/src/screenshooter.c
+++ b/src/screenshooter.c
@@ -31,18 +31,9 @@
 #include <sys/uio.h>
 
 #include "compositor.h"
-#include "screenshooter-server-protocol.h"
 
 #include "../wcap/wcap-decode.h"
 
-struct screenshooter {
-       struct weston_compositor *ec;
-       struct wl_global *global;
-       struct wl_client *client;
-       struct weston_process process;
-       struct wl_listener destroy_listener;
-};
-
 struct screenshooter_frame_listener {
        struct wl_listener listener;
        struct weston_buffer *buffer;
@@ -212,98 +203,6 @@ weston_screenshooter_shoot(struct weston_output *output,
        return 0;
 }
 
-static void
-screenshooter_done(void *data, enum weston_screenshooter_outcome outcome)
-{
-       struct wl_resource *resource = data;
-
-       switch (outcome) {
-       case WESTON_SCREENSHOOTER_SUCCESS:
-               screenshooter_send_done(resource);
-               break;
-       case WESTON_SCREENSHOOTER_NO_MEMORY:
-               wl_resource_post_no_memory(resource);
-               break;
-       default:
-               break;
-       }
-}
-
-static void
-screenshooter_shoot(struct wl_client *client,
-                   struct wl_resource *resource,
-                   struct wl_resource *output_resource,
-                   struct wl_resource *buffer_resource)
-{
-       struct weston_output *output =
-               wl_resource_get_user_data(output_resource);
-       struct weston_buffer *buffer =
-               weston_buffer_from_resource(buffer_resource);
-
-       if (buffer == NULL) {
-               wl_resource_post_no_memory(resource);
-               return;
-       }
-
-       weston_screenshooter_shoot(output, buffer, screenshooter_done, 
resource);
-}
-
-struct screenshooter_interface screenshooter_implementation = {
-       screenshooter_shoot
-};
-
-static void
-bind_shooter(struct wl_client *client,
-            void *data, uint32_t version, uint32_t id)
-{
-       struct screenshooter *shooter = data;
-       struct wl_resource *resource;
-
-       resource = wl_resource_create(client,
-                                     &screenshooter_interface, 1, id);
-
-       if (client != shooter->client) {
-               wl_resource_post_error(resource, 
WL_DISPLAY_ERROR_INVALID_OBJECT,
-                                      "screenshooter failed: permission 
denied");
-               return;
-       }
-
-       wl_resource_set_implementation(resource, &screenshooter_implementation,
-                                      data, NULL);
-}
-
-static void
-screenshooter_sigchld(struct weston_process *process, int status)
-{
-       struct screenshooter *shooter =
-               container_of(process, struct screenshooter, process);
-
-       shooter->client = NULL;
-}
-
-static void
-screenshooter_binding(struct weston_seat *seat, uint32_t time, uint32_t key,
-                     void *data)
-{
-       struct screenshooter *shooter = data;
-       char *screenshooter_exe;
-       int ret;
-
-       ret = asprintf(&screenshooter_exe, "%s/%s",
-                      weston_config_get_libexec_dir(),
-                      "/weston-screenshooter");
-       if (ret < 0) {
-               weston_log("Could not construct screenshooter path.\n");
-               return;
-       }
-
-       if (!shooter->client)
-               shooter->client = weston_client_launch(shooter->ec,
-                                       &shooter->process,
-                                       screenshooter_exe, 
screenshooter_sigchld);
-       free(screenshooter_exe);
-}
-
 struct weston_recorder {
        struct weston_output *output;
        uint32_t *frame, *rect;
@@ -551,17 +450,13 @@ weston_recorder_destroy(struct weston_recorder *recorder)
        weston_recorder_free(recorder);
 }
 
-static void
-recorder_binding(struct weston_seat *seat, uint32_t time, uint32_t key, void 
*data)
+static struct weston_recorder *
+get_recorder(struct weston_compositor *ec)
 {
-       struct weston_seat *ws = (struct weston_seat *) seat;
-       struct weston_compositor *ec = ws->compositor;
        struct weston_output *output;
        struct wl_listener *listener = NULL;
-       struct weston_recorder *recorder;
-       static const char filename[] = "capture.wcap";
 
-       wl_list_for_each(output, &seat->compositor->output_list, link) {
+       wl_list_for_each(output, &ec->output_list, link) {
                listener = wl_signal_get(&output->frame_signal,
                                         weston_recorder_frame_notify);
                if (listener)
@@ -569,59 +464,40 @@ recorder_binding(struct weston_seat *seat, uint32_t time, 
uint32_t key, void *da
        }
 
        if (listener) {
-               recorder = container_of(listener, struct weston_recorder,
-                                       frame_listener);
+               return container_of(listener, struct weston_recorder,
+                                   frame_listener);
+       }
+       return NULL;
+}
 
-               weston_log(
-                       "stopping recorder, total file size %dM, %d frames\n",
-                       recorder->total / (1024 * 1024), recorder->count);
+WL_EXPORT void
+weston_recorder_start(struct weston_compositor *ec,
+                     struct weston_output *output)
+{
+       struct weston_recorder *recorder = get_recorder(ec);
+       static const char filename[] = "capture.wcap";
 
-               recorder->destroying = 1;
-               weston_output_schedule_repaint(recorder->output);
+       if (recorder) {
+               // Already running, do nothing;
+               return;
        } else {
-               if (seat->keyboard && seat->keyboard->focus &&
-                   seat->keyboard->focus->output)
-                       output = seat->keyboard->focus->output;
-               else
-                       output = container_of(ec->output_list.next,
-                                             struct weston_output, link);
-
                weston_log("starting recorder for output %s, file %s\n",
                           output->name, filename);
                weston_recorder_create(output, filename);
        }
 }
 
-static void
-screenshooter_destroy(struct wl_listener *listener, void *data)
-{
-       struct screenshooter *shooter =
-               container_of(listener, struct screenshooter, destroy_listener);
-
-       wl_global_destroy(shooter->global);
-       free(shooter);
-}
-
 WL_EXPORT void
-screenshooter_create(struct weston_compositor *ec)
+weston_recorder_stop(struct weston_compositor *ec)
 {
-       struct screenshooter *shooter;
+       struct weston_recorder *recorder = get_recorder(ec);
 
-       shooter = malloc(sizeof *shooter);
-       if (shooter == NULL)
-               return;
-
-       shooter->ec = ec;
-       shooter->client = NULL;
-
-       shooter->global = wl_global_create(ec->wl_display,
-                                          &screenshooter_interface, 1,
-                                          shooter, bind_shooter);
-       weston_compositor_add_key_binding(ec, KEY_S, MODIFIER_SUPER,
-                                         screenshooter_binding, shooter);
-       weston_compositor_add_key_binding(ec, KEY_R, MODIFIER_SUPER,
-                                         recorder_binding, shooter);
+       if (recorder) {
+               weston_log(
+                       "stopping recorder, total file size %dM, %d frames\n",
+                       recorder->total / (1024 * 1024), recorder->count);
 
-       shooter->destroy_listener.notify = screenshooter_destroy;
-       wl_signal_add(&ec->destroy_signal, &shooter->destroy_listener);
+               recorder->destroying = 1;
+               weston_output_schedule_repaint(recorder->output);
+       }
 }
diff --git a/src/weston.c b/src/weston.c
index 566bee0..c192354 100644
--- a/src/weston.c
+++ b/src/weston.c
@@ -34,6 +34,7 @@
 #include <sys/wait.h>
 #include <sys/time.h>
 #include <sys/socket.h>
+#include <linux/input.h>
 
 #ifdef HAVE_LIBUNWIND
 #define UNW_LOCAL_ONLY
@@ -44,6 +45,15 @@
 #include "../shared/os-compatibility.h"
 #include "git-version.h"
 #include "version.h"
+#include "screenshooter-server-protocol.h"
+
+struct screenshooter {
+       struct weston_compositor *ec;
+       struct wl_global *global;
+       struct wl_client *client;
+       struct weston_process process;
+       struct wl_listener destroy_listener;
+};
 
 static struct wl_list child_process_list;
 static struct weston_compositor *segv_compositor;
@@ -151,6 +161,155 @@ vlog_continue(const char *fmt, va_list argp)
        return vfprintf(weston_logfile, fmt, argp);
 }
 
+static void
+screenshooter_done(void *data, enum weston_screenshooter_outcome outcome)
+{
+       struct wl_resource *resource = data;
+
+       switch (outcome) {
+       case WESTON_SCREENSHOOTER_SUCCESS:
+               screenshooter_send_done(resource);
+               break;
+       case WESTON_SCREENSHOOTER_NO_MEMORY:
+               wl_resource_post_no_memory(resource);
+               break;
+       default:
+               break;
+       }
+}
+
+static void
+screenshooter_shoot(struct wl_client *client,
+                   struct wl_resource *resource,
+                   struct wl_resource *output_resource,
+                   struct wl_resource *buffer_resource)
+{
+       struct weston_output *output =
+               wl_resource_get_user_data(output_resource);
+       struct weston_buffer *buffer =
+               weston_buffer_from_resource(buffer_resource);
+
+       if (buffer == NULL) {
+               wl_resource_post_no_memory(resource);
+               return;
+       }
+
+       weston_screenshooter_shoot(output, buffer, screenshooter_done, 
resource);
+}
+
+struct screenshooter_interface screenshooter_implementation = {
+       screenshooter_shoot
+};
+
+static void
+bind_shooter(struct wl_client *client,
+            void *data, uint32_t version, uint32_t id)
+{
+       struct screenshooter *shooter = data;
+       struct wl_resource *resource;
+
+       resource = wl_resource_create(client,
+                                     &screenshooter_interface, 1, id);
+
+       if (client != shooter->client) {
+               wl_resource_post_error(resource, 
WL_DISPLAY_ERROR_INVALID_OBJECT,
+                                      "screenshooter failed: permission 
denied");
+               return;
+       }
+
+       wl_resource_set_implementation(resource, &screenshooter_implementation,
+                                      data, NULL);
+}
+
+static void
+screenshooter_sigchld(struct weston_process *process, int status)
+{
+       struct screenshooter *shooter =
+               container_of(process, struct screenshooter, process);
+
+       shooter->client = NULL;
+}
+
+static void
+screenshooter_binding(struct weston_seat *seat, uint32_t time, uint32_t key,
+                     void *data)
+{
+       struct screenshooter *shooter = data;
+       char *screenshooter_exe;
+       int ret;
+
+       ret = asprintf(&screenshooter_exe, "%s/%s",
+                      weston_config_get_libexec_dir(),
+                      "weston-screenshooter");
+       if (ret < 0) {
+               weston_log("Could not construct screenshooter path.\n");
+               return;
+       }
+
+       if (!shooter->client)
+               shooter->client = weston_client_launch(shooter->ec,
+                                       &shooter->process,
+                                       screenshooter_exe, 
screenshooter_sigchld);
+       free(screenshooter_exe);
+}
+
+static void
+recorder_binding(struct weston_seat *seat, uint32_t time, uint32_t key, void 
*data)
+{
+       struct weston_compositor *ec = seat->compositor;
+       struct weston_output *output;
+       int *running = data;
+
+       if (*running == 1) {
+               weston_recorder_stop(ec);
+               *running = 0;
+       } else {
+               if (seat->keyboard && seat->keyboard->focus &&
+                   seat->keyboard->focus->output)
+                       output = seat->keyboard->focus->output;
+               else
+                       output = container_of(ec->output_list.next,
+                                             struct weston_output, link);
+
+               weston_recorder_start(ec, output);
+               *running = 1;
+       }
+}
+
+static void
+screenshooter_destroy(struct wl_listener *listener, void *data)
+{
+       struct screenshooter *shooter =
+               container_of(listener, struct screenshooter, destroy_listener);
+
+       wl_global_destroy(shooter->global);
+       free(shooter);
+}
+
+static void
+screenshooter_create(struct weston_compositor *ec)
+{
+       static int recorder_running = 0;
+       struct screenshooter *shooter;
+
+       shooter = malloc(sizeof *shooter);
+       if (shooter == NULL)
+               return;
+
+       shooter->ec = ec;
+       shooter->client = NULL;
+
+       shooter->global = wl_global_create(ec->wl_display,
+                                          &screenshooter_interface, 1,
+                                          shooter, bind_shooter);
+       weston_compositor_add_key_binding(ec, KEY_S, MODIFIER_SUPER,
+                                         screenshooter_binding, shooter);
+       weston_compositor_add_key_binding(ec, KEY_R, MODIFIER_SUPER,
+                                         recorder_binding, &recorder_running);
+
+       shooter->destroy_listener.notify = screenshooter_destroy;
+       wl_signal_add(&ec->destroy_signal, &shooter->destroy_listener);
+}
 
 #ifdef HAVE_LIBUNWIND
 
@@ -913,6 +1072,8 @@ int main(int argc, char *argv[])
                }
        }
 
+       screenshooter_create(ec);
+
        weston_compositor_wake(ec);
 
        wl_display_run(display);
-- 
2.1.3

_______________________________________________
wayland-devel mailing list
wayland-devel@lists.freedesktop.org
http://lists.freedesktop.org/mailman/listinfo/wayland-devel

Reply via email to