/*
* wlopm - Wayland output power manager
*
* Copyright (C) 2021 Leon Henrik Plickat
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include
#include
#include
#include
#include
#include
#include
#include "xdg-output-unstable-v1.h"
#include "wlr-output-power-management-unstable-v1.h"
#define VERSION "0.0.1"
const char usage[] =
"Usage: wlopm [options...]\n"
" -j, --json Use JSON format.\n"
" -h, --help Print this help text and exit.\n"
" --on Set the power mode of the specified output to on.\n"
" --off Set the power mode of the specified output to off.\n"
" --toggle Toggle the power mode of the specified output.\n"
"\n";
struct Output
{
struct wl_list link;
struct wl_output *wl_output;
struct zxdg_output_v1 *xdg_output;
struct zwlr_output_power_v1 *wlr_output_power;
enum zwlr_output_power_v1_mode mode;
char *name;
bool operation_failed;
uint32_t global_name;
};
enum Action
{
LIST,
OPERATIONS,
};
enum Power_mode
{
ON,
OFF,
TOGGLE,
};
struct Operation
{
struct wl_list link;
char *name;
enum Power_mode power_mode;
};
bool json = false;
bool json_prev = false;
struct wl_display *wl_display = NULL;
struct wl_registry *wl_registry = NULL;
struct wl_callback *sync_callback = NULL;
struct wl_list outputs;
struct wl_list operations;
struct zxdg_output_manager_v1 *xdg_output_manager = NULL;
struct zwlr_output_power_manager_v1 *wlr_output_power_manager = NULL;
int ret = EXIT_SUCCESS;
bool loop = true;
static void noop () {}
static void wlr_output_power_handle_mode (void *data, struct zwlr_output_power_v1 *wlr_output_power,
enum zwlr_output_power_v1_mode mode)
{
struct Output *output = (struct Output *)data;
output->mode = mode;
}
static void wlr_output_power_handle_failed (void *data, struct zwlr_output_power_v1 *wlr_output_power)
{
struct Output *output = (struct Output *)data;
output->operation_failed = true;
}
static const struct zwlr_output_power_v1_listener wlr_output_power_listener = {
.mode = wlr_output_power_handle_mode,
.failed = wlr_output_power_handle_failed,
};
static void xdg_output_handle_name (void *data, struct zxdg_output_v1 *xdg_output,
const char *name)
{
struct Output *output = (struct Output *)data;
if ( output->name != NULL )
free(output->name);
output->name = strdup(name);
}
static const struct zxdg_output_v1_listener xdg_output_listener = {
.logical_size = noop,
.name = xdg_output_handle_name,
.logical_position = noop,
.description = noop,
.done = noop, /* Deprecated since version 3. */
};
static void registry_handle_global (void *data, struct wl_registry *registry,
uint32_t name, const char *interface, uint32_t version)
{
if ( strcmp(interface, wl_output_interface.name) == 0 )
{
struct Output *output = calloc(1, sizeof(struct Output));
if ( output == NULL )
{
fputs("ERROR: Failed to allocate.\n", stderr);
return;
}
output->wl_output = wl_registry_bind(registry, name,
&wl_output_interface, version);
output->xdg_output = NULL;
output->wlr_output_power = NULL;
output->name = NULL;
output->global_name = name;
wl_list_insert(&outputs, &output->link);
}
else if ( strcmp(interface, zxdg_output_manager_v1_interface.name) == 0 )
xdg_output_manager = wl_registry_bind(registry, name,
&zxdg_output_manager_v1_interface, version);
else if ( strcmp(interface, zwlr_output_power_manager_v1_interface.name) == 0 )
wlr_output_power_manager = wl_registry_bind(registry, name,
&zwlr_output_power_manager_v1_interface, version);
}
static const struct wl_registry_listener registry_listener = {
.global = registry_handle_global,
.global_remove = noop, /* We don't run long enough to care. */
};
static void sync_handle_done (void *data, struct wl_callback *wl_callback, uint32_t other_data);
static const struct wl_callback_listener sync_callback_listener = {
.done = sync_handle_done,
};
static struct Output *output_from_name (const char *str)
{
struct Output *output;
wl_list_for_each(output, &outputs, link)
if ( strcmp(output->name, str) == 0 )
return output;
return NULL;
}
static char *power_mode_to_string (enum zwlr_output_power_v1_mode mode)
{
return mode == ZWLR_OUTPUT_POWER_V1_MODE_ON ? "on" : "off";
}
static void sync_handle_done (void *data, struct wl_callback *wl_callback, uint32_t other_data)
{
wl_callback_destroy(wl_callback);
sync_callback = NULL;
static int sync = 0;
if ( sync == 0 )
{
if ( wlr_output_power_manager == NULL )
{
fputs("ERROR: Wayland server does not support wlr-output-power-management-v1.\n", stderr);
loop = false;
ret = EXIT_FAILURE;
return;
}
if ( xdg_output_manager == NULL )
{
fputs("ERROR: Wayland server does not support xdg-output.\n", stderr);
loop = false;
ret = EXIT_FAILURE;
return;
}
struct Output *output;
wl_list_for_each(output, &outputs, link)
{
output->xdg_output = zxdg_output_manager_v1_get_xdg_output(
xdg_output_manager, output->wl_output);
zxdg_output_v1_add_listener(output->xdg_output,
&xdg_output_listener, output);
output->wlr_output_power = zwlr_output_power_manager_v1_get_output_power(
wlr_output_power_manager, output->wl_output);
zwlr_output_power_v1_add_listener(output->wlr_output_power,
&wlr_output_power_listener, output);
}
sync_callback = wl_display_sync(wl_display);
wl_callback_add_listener(sync_callback, &sync_callback_listener, NULL);
}
else if ( sync == 1 )
{
if (wl_list_empty(&operations))
{
/* The operations list is empty, so let's just list all
* outputs and their current power mode.
*/
struct Output *output;
if (json)
{
fputs("[", stdout);
wl_list_for_each(output, &outputs, link)
{
fprintf(stdout,
"%s\n {\n"
" \"output\": \"%s\",\n"
" \"power-mode\": \"%s\"\n"
" }",
json_prev ? "," : "",
output->name,
power_mode_to_string(output->mode));
json_prev = true;
}
fputs("\n]\n", stdout);
}
else
wl_list_for_each(output, &outputs, link)
fprintf(stdout, "%s %s\n", output->name,
power_mode_to_string(output->mode));
loop = false;
}
else
{
/* There are operations in the operations list. We have
* things to do!
*/
if (json)
fputs(
"{\n"
" \"errors\": [",
stdout);
struct Operation *operation;
wl_list_for_each(operation, &operations, link)
{
const struct Output *output = output_from_name(operation->name);
if ( output == NULL )
{
if (json)
{
fprintf(stdout,
"%s\n {\n"
" \"output\": \"%s\",\n"
" \"error\": \"output does not exist\"\n"
" }",
json_prev ? "," : "",
operation->name);
json_prev = true;
}
else
fprintf(stderr, "ERROR: Output '%s' does not exist.\n",
operation->name);
continue;
}
enum zwlr_output_power_v1_mode new_mode;
switch (operation->power_mode)
{
case ON:
new_mode = ZWLR_OUTPUT_POWER_V1_MODE_ON;
break;
case OFF:
new_mode = ZWLR_OUTPUT_POWER_V1_MODE_OFF;
break;
case TOGGLE:
if ( output->mode == ZWLR_OUTPUT_POWER_V1_MODE_ON )
new_mode = ZWLR_OUTPUT_POWER_V1_MODE_OFF;
else
new_mode = ZWLR_OUTPUT_POWER_V1_MODE_ON;
break;
}
zwlr_output_power_v1_set_mode(output->wlr_output_power, new_mode);
}
/* We need to sync yet another time because setting the
* power mode might fail and we want to display those
* error messages.
*/
sync_callback = wl_display_sync(wl_display);
wl_callback_add_listener(sync_callback, &sync_callback_listener, NULL);
}
}
else
{
struct Output *output;
wl_list_for_each(output, &outputs, link)
if (output->operation_failed)
{
if (json)
{
fprintf(stdout,
"%s\n {\n"
" \"output\": \"%s\","
" \"error\": \"setting power mode failed\"\n"
" }",
json_prev ? "," : "",
output->name);
json_prev = true;
}
else
fprintf(stderr, "ERROR: Setting power mode for output '%s' failed.\n",
output->name);
}
if (json)
fputs(
"\n ]\n"
"}\n",
stdout);
loop = false;
}
sync++;
}
static void destroy_all_outputs (void)
{
struct Output *output, *tmp;
wl_list_for_each_safe(output, tmp, &outputs, link)
{
if ( output->wlr_output_power != NULL )
zwlr_output_power_v1_destroy(output->wlr_output_power);
if ( output->xdg_output != NULL )
zxdg_output_v1_destroy(output->xdg_output);
wl_output_destroy(output->wl_output);
wl_list_remove(&output->link);
free(output->name);
free(output);
}
}
static void destroy_all_operations (void)
{
struct Operation *operation, *tmp;
wl_list_for_each_safe(operation, tmp, &operations, link)
{
wl_list_remove(&operation->link);
free(operation->name);
free(operation);
}
}
static bool create_operation (const char *name, enum Power_mode power_mode)
{
struct Operation *operation = calloc(1, sizeof(struct Operation));
if ( operation == NULL )
{
fprintf(stderr, "ERROR: calloc: %s\n", strerror(errno));
return false;
}
operation->name = strdup(name);
if ( operation->name == NULL )
{
fprintf(stderr, "ERROR: calloc: %s\n", strerror(errno));
free(operation);
return false;
}
operation->power_mode = power_mode;
wl_list_insert(&operations, &operation->link);
return true;
}
int main(int argc, char *argv[])
{
wl_list_init(&operations);
for (int i = 1; i < argc; i++)
{
if ( strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0 )
{
destroy_all_operations();
fputs(usage, stderr);
return EXIT_SUCCESS;
}
else if ( strcmp(argv[i], "-j") == 0 || strcmp(argv[i], "--json") == 0 )
json = true;
else if ( strcmp(argv[i], "--on") == 0 )
{
if ( i == argc - 1 )
{
fputs("ERROR: '--on' needs an output name.\n", stderr);
destroy_all_operations();
return EXIT_FAILURE;
}
if (! create_operation(argv[i+1], ON))
{
destroy_all_operations();
return EXIT_FAILURE;
}
i++;
}
else if ( strcmp(argv[i], "--off") == 0 )
{
if ( i == argc - 1 )
{
fputs("ERROR: '--off' needs an output name.\n", stderr);
destroy_all_operations();
return EXIT_FAILURE;
}
if (! create_operation(argv[i+1], OFF))
{
destroy_all_operations();
return EXIT_FAILURE;
}
i++;
}
else if ( strcmp(argv[i], "--toggle") == 0 )
{
if ( i == argc - 1 )
{
fputs("ERROR: '--toggle' needs an output name.\n", stderr);
destroy_all_operations();
return EXIT_FAILURE;
}
if (! create_operation(argv[i+1], TOGGLE))
{
destroy_all_operations();
return EXIT_FAILURE;
}
i++;
}
else
{
fprintf(stderr, "ERROR: Unknown option '%s'\n", argv[i]);
destroy_all_operations();
return EXIT_FAILURE;
}
}
/* We query the display name here instead of letting wl_display_connect()
* figure it out itself, because libwayland (for legacy reasons) falls
* back to using "wayland-0" when $WAYLAND_DISPLAY is not set, which is
* generally not desirable.
*/
const char *display_name = getenv("WAYLAND_DISPLAY");
if ( display_name == NULL )
{
fputs("ERROR: WAYLAND_DISPLAY is not set.\n", stderr);
return EXIT_FAILURE;
}
wl_display = wl_display_connect(display_name);
if ( wl_display == NULL )
{
fputs("ERROR: Can not connect to wayland display.\n", stderr);
return EXIT_FAILURE;
}
wl_list_init(&outputs);
wl_registry = wl_display_get_registry(wl_display);
wl_registry_add_listener(wl_registry, ®istry_listener, NULL);
sync_callback = wl_display_sync(wl_display);
wl_callback_add_listener(sync_callback, &sync_callback_listener, NULL);
while ( loop && wl_display_dispatch(wl_display) > 0 );
destroy_all_operations();
destroy_all_outputs();
if ( sync_callback != NULL )
wl_callback_destroy(sync_callback);
if ( wlr_output_power_manager != NULL )
zwlr_output_power_manager_v1_destroy(wlr_output_power_manager);
if ( xdg_output_manager != NULL )
zxdg_output_manager_v1_destroy(xdg_output_manager);
if ( wl_registry != NULL )
wl_registry_destroy(wl_registry);
wl_display_disconnect(wl_display);
return ret;
}