153a5a1b3Sopenharmony_ci/***
253a5a1b3Sopenharmony_ci  This file is part of PulseAudio.
353a5a1b3Sopenharmony_ci
453a5a1b3Sopenharmony_ci  Copyright 2009 Lennart Poettering
553a5a1b3Sopenharmony_ci
653a5a1b3Sopenharmony_ci  PulseAudio is free software; you can redistribute it and/or modify
753a5a1b3Sopenharmony_ci  it under the terms of the GNU Lesser General Public License as published
853a5a1b3Sopenharmony_ci  by the Free Software Foundation; either version 2.1 of the License,
953a5a1b3Sopenharmony_ci  or (at your option) any later version.
1053a5a1b3Sopenharmony_ci
1153a5a1b3Sopenharmony_ci  PulseAudio is distributed in the hope that it will be useful, but
1253a5a1b3Sopenharmony_ci  WITHOUT ANY WARRANTY; without even the implied warranty of
1353a5a1b3Sopenharmony_ci  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1453a5a1b3Sopenharmony_ci  General Public License for more details.
1553a5a1b3Sopenharmony_ci
1653a5a1b3Sopenharmony_ci  You should have received a copy of the GNU Lesser General Public License
1753a5a1b3Sopenharmony_ci  along with PulseAudio; if not, see <http://www.gnu.org/licenses/>.
1853a5a1b3Sopenharmony_ci***/
1953a5a1b3Sopenharmony_ci
2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H
2153a5a1b3Sopenharmony_ci#include <config.h>
2253a5a1b3Sopenharmony_ci#endif
2353a5a1b3Sopenharmony_ci
2453a5a1b3Sopenharmony_ci#include <stdio.h>
2553a5a1b3Sopenharmony_ci#include <stdlib.h>
2653a5a1b3Sopenharmony_ci#include <unistd.h>
2753a5a1b3Sopenharmony_ci
2853a5a1b3Sopenharmony_ci#include <X11/Xlib.h>
2953a5a1b3Sopenharmony_ci#include <X11/extensions/XTest.h>
3053a5a1b3Sopenharmony_ci#include <X11/XF86keysym.h>
3153a5a1b3Sopenharmony_ci#include <X11/keysym.h>
3253a5a1b3Sopenharmony_ci
3353a5a1b3Sopenharmony_ci#include <pulse/xmalloc.h>
3453a5a1b3Sopenharmony_ci
3553a5a1b3Sopenharmony_ci#include <pulsecore/module.h>
3653a5a1b3Sopenharmony_ci#include <pulsecore/modargs.h>
3753a5a1b3Sopenharmony_ci#include <pulsecore/log.h>
3853a5a1b3Sopenharmony_ci#include <pulsecore/x11wrap.h>
3953a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h>
4053a5a1b3Sopenharmony_ci
4153a5a1b3Sopenharmony_ciPA_MODULE_AUTHOR("Lennart Poettering");
4253a5a1b3Sopenharmony_ciPA_MODULE_DESCRIPTION("Synthesize X11 media key events when cork/uncork is requested");
4353a5a1b3Sopenharmony_ciPA_MODULE_VERSION(PACKAGE_VERSION);
4453a5a1b3Sopenharmony_ciPA_MODULE_LOAD_ONCE(false);
4553a5a1b3Sopenharmony_ciPA_MODULE_USAGE("display=<X11 display>");
4653a5a1b3Sopenharmony_ci
4753a5a1b3Sopenharmony_cistatic const char* const valid_modargs[] = {
4853a5a1b3Sopenharmony_ci    "display",
4953a5a1b3Sopenharmony_ci    "xauthority",
5053a5a1b3Sopenharmony_ci    NULL
5153a5a1b3Sopenharmony_ci};
5253a5a1b3Sopenharmony_ci
5353a5a1b3Sopenharmony_cistruct userdata {
5453a5a1b3Sopenharmony_ci    pa_module *module;
5553a5a1b3Sopenharmony_ci
5653a5a1b3Sopenharmony_ci    pa_x11_wrapper *x11_wrapper;
5753a5a1b3Sopenharmony_ci    pa_x11_client *x11_client;
5853a5a1b3Sopenharmony_ci
5953a5a1b3Sopenharmony_ci    pa_hook_slot *hook_slot;
6053a5a1b3Sopenharmony_ci};
6153a5a1b3Sopenharmony_ci
6253a5a1b3Sopenharmony_cistatic void x11_kill_cb(pa_x11_wrapper *w, void *userdata) {
6353a5a1b3Sopenharmony_ci    struct userdata *u = userdata;
6453a5a1b3Sopenharmony_ci
6553a5a1b3Sopenharmony_ci    pa_assert(w);
6653a5a1b3Sopenharmony_ci    pa_assert(u);
6753a5a1b3Sopenharmony_ci    pa_assert(u->x11_wrapper == w);
6853a5a1b3Sopenharmony_ci
6953a5a1b3Sopenharmony_ci    pa_log_debug("X11 client kill callback called");
7053a5a1b3Sopenharmony_ci
7153a5a1b3Sopenharmony_ci    if (u->x11_client) {
7253a5a1b3Sopenharmony_ci        pa_x11_client_free(u->x11_client);
7353a5a1b3Sopenharmony_ci        u->x11_client = NULL;
7453a5a1b3Sopenharmony_ci    }
7553a5a1b3Sopenharmony_ci
7653a5a1b3Sopenharmony_ci    if (u->x11_wrapper) {
7753a5a1b3Sopenharmony_ci        pa_x11_wrapper_unref(u->x11_wrapper);
7853a5a1b3Sopenharmony_ci        u->x11_wrapper = NULL;
7953a5a1b3Sopenharmony_ci    }
8053a5a1b3Sopenharmony_ci
8153a5a1b3Sopenharmony_ci    pa_module_unload_request(u->module, true);
8253a5a1b3Sopenharmony_ci}
8353a5a1b3Sopenharmony_ci
8453a5a1b3Sopenharmony_cistatic pa_hook_result_t sink_input_send_event_hook_cb(
8553a5a1b3Sopenharmony_ci        pa_core *c,
8653a5a1b3Sopenharmony_ci        pa_sink_input_send_event_hook_data *data,
8753a5a1b3Sopenharmony_ci        struct userdata *u) {
8853a5a1b3Sopenharmony_ci
8953a5a1b3Sopenharmony_ci    KeySym sym;
9053a5a1b3Sopenharmony_ci    KeyCode code;
9153a5a1b3Sopenharmony_ci    Display *display;
9253a5a1b3Sopenharmony_ci
9353a5a1b3Sopenharmony_ci    pa_assert(c);
9453a5a1b3Sopenharmony_ci    pa_assert(data);
9553a5a1b3Sopenharmony_ci    pa_assert(u);
9653a5a1b3Sopenharmony_ci
9753a5a1b3Sopenharmony_ci    if (pa_streq(data->event, PA_STREAM_EVENT_REQUEST_CORK))
9853a5a1b3Sopenharmony_ci        sym = XF86XK_AudioPause;
9953a5a1b3Sopenharmony_ci    else if (pa_streq(data->event, PA_STREAM_EVENT_REQUEST_UNCORK))
10053a5a1b3Sopenharmony_ci        sym = XF86XK_AudioPlay;
10153a5a1b3Sopenharmony_ci    else
10253a5a1b3Sopenharmony_ci        return PA_HOOK_OK;
10353a5a1b3Sopenharmony_ci
10453a5a1b3Sopenharmony_ci    pa_log_debug("Triggering X11 keysym: %s", XKeysymToString(sym));
10553a5a1b3Sopenharmony_ci
10653a5a1b3Sopenharmony_ci    display = pa_x11_wrapper_get_display(u->x11_wrapper);
10753a5a1b3Sopenharmony_ci    code = XKeysymToKeycode(display, sym);
10853a5a1b3Sopenharmony_ci
10953a5a1b3Sopenharmony_ci    XTestFakeKeyEvent(display, code, True, CurrentTime);
11053a5a1b3Sopenharmony_ci    XSync(display, False);
11153a5a1b3Sopenharmony_ci
11253a5a1b3Sopenharmony_ci    XTestFakeKeyEvent(display, code, False, CurrentTime);
11353a5a1b3Sopenharmony_ci    XSync(display, False);
11453a5a1b3Sopenharmony_ci
11553a5a1b3Sopenharmony_ci    return PA_HOOK_OK;
11653a5a1b3Sopenharmony_ci}
11753a5a1b3Sopenharmony_ci
11853a5a1b3Sopenharmony_ciint pa__init(pa_module *m) {
11953a5a1b3Sopenharmony_ci    struct userdata *u;
12053a5a1b3Sopenharmony_ci    pa_modargs *ma;
12153a5a1b3Sopenharmony_ci    int xtest_event_base, xtest_error_base;
12253a5a1b3Sopenharmony_ci    int major_version, minor_version;
12353a5a1b3Sopenharmony_ci
12453a5a1b3Sopenharmony_ci    pa_assert(m);
12553a5a1b3Sopenharmony_ci
12653a5a1b3Sopenharmony_ci    if (!(ma = pa_modargs_new(m->argument, valid_modargs))) {
12753a5a1b3Sopenharmony_ci        pa_log("failed to parse module arguments");
12853a5a1b3Sopenharmony_ci        goto fail;
12953a5a1b3Sopenharmony_ci    }
13053a5a1b3Sopenharmony_ci
13153a5a1b3Sopenharmony_ci    m->userdata = u = pa_xnew0(struct userdata, 1);
13253a5a1b3Sopenharmony_ci    u->module = m;
13353a5a1b3Sopenharmony_ci
13453a5a1b3Sopenharmony_ci    if (pa_modargs_get_value(ma, "xauthority", NULL)) {
13553a5a1b3Sopenharmony_ci        if (setenv("XAUTHORITY", pa_modargs_get_value(ma, "xauthority", NULL), 1)) {
13653a5a1b3Sopenharmony_ci            pa_log("setenv() for $XAUTHORITY failed");
13753a5a1b3Sopenharmony_ci            goto fail;
13853a5a1b3Sopenharmony_ci        }
13953a5a1b3Sopenharmony_ci    }
14053a5a1b3Sopenharmony_ci
14153a5a1b3Sopenharmony_ci    if (!(u->x11_wrapper = pa_x11_wrapper_get(m->core, pa_modargs_get_value(ma, "display", NULL))))
14253a5a1b3Sopenharmony_ci        goto fail;
14353a5a1b3Sopenharmony_ci
14453a5a1b3Sopenharmony_ci    if (!XTestQueryExtension(
14553a5a1b3Sopenharmony_ci                pa_x11_wrapper_get_display(u->x11_wrapper),
14653a5a1b3Sopenharmony_ci                &xtest_event_base, &xtest_error_base,
14753a5a1b3Sopenharmony_ci                &major_version, &minor_version)) {
14853a5a1b3Sopenharmony_ci
14953a5a1b3Sopenharmony_ci        pa_log("XTest extension not supported.");
15053a5a1b3Sopenharmony_ci        goto fail;
15153a5a1b3Sopenharmony_ci    }
15253a5a1b3Sopenharmony_ci
15353a5a1b3Sopenharmony_ci    pa_log_debug("XTest %i.%i supported.", major_version, minor_version);
15453a5a1b3Sopenharmony_ci
15553a5a1b3Sopenharmony_ci    u->x11_client = pa_x11_client_new(u->x11_wrapper, NULL, x11_kill_cb, u);
15653a5a1b3Sopenharmony_ci
15753a5a1b3Sopenharmony_ci    u->hook_slot = pa_hook_connect(
15853a5a1b3Sopenharmony_ci            &m->core->hooks[PA_CORE_HOOK_SINK_INPUT_SEND_EVENT],
15953a5a1b3Sopenharmony_ci            PA_HOOK_NORMAL,
16053a5a1b3Sopenharmony_ci            (pa_hook_cb_t) sink_input_send_event_hook_cb, u);
16153a5a1b3Sopenharmony_ci
16253a5a1b3Sopenharmony_ci    pa_modargs_free(ma);
16353a5a1b3Sopenharmony_ci
16453a5a1b3Sopenharmony_ci    return 0;
16553a5a1b3Sopenharmony_ci
16653a5a1b3Sopenharmony_cifail:
16753a5a1b3Sopenharmony_ci    if (ma)
16853a5a1b3Sopenharmony_ci        pa_modargs_free(ma);
16953a5a1b3Sopenharmony_ci
17053a5a1b3Sopenharmony_ci    pa__done(m);
17153a5a1b3Sopenharmony_ci
17253a5a1b3Sopenharmony_ci    return -1;
17353a5a1b3Sopenharmony_ci}
17453a5a1b3Sopenharmony_ci
17553a5a1b3Sopenharmony_civoid pa__done(pa_module*m) {
17653a5a1b3Sopenharmony_ci    struct userdata*u;
17753a5a1b3Sopenharmony_ci
17853a5a1b3Sopenharmony_ci    pa_assert(m);
17953a5a1b3Sopenharmony_ci
18053a5a1b3Sopenharmony_ci    if (!(u = m->userdata))
18153a5a1b3Sopenharmony_ci        return;
18253a5a1b3Sopenharmony_ci
18353a5a1b3Sopenharmony_ci    if (u->x11_client)
18453a5a1b3Sopenharmony_ci        pa_x11_client_free(u->x11_client);
18553a5a1b3Sopenharmony_ci
18653a5a1b3Sopenharmony_ci    if (u->x11_wrapper)
18753a5a1b3Sopenharmony_ci        pa_x11_wrapper_unref(u->x11_wrapper);
18853a5a1b3Sopenharmony_ci
18953a5a1b3Sopenharmony_ci    if (u->hook_slot)
19053a5a1b3Sopenharmony_ci        pa_hook_slot_free(u->hook_slot);
19153a5a1b3Sopenharmony_ci
19253a5a1b3Sopenharmony_ci    pa_xfree(u);
19353a5a1b3Sopenharmony_ci}
194