2023-10-28 18:46:43 +00:00
|
|
|
#include "main.h"
|
|
|
|
|
|
|
|
void rendermon(struct wl_listener *listener, void *data) {
|
2023-10-30 12:00:27 +00:00
|
|
|
struct output *output = wl_container_of(listener, output, frame);
|
|
|
|
struct server *server = output->server;
|
|
|
|
struct wlr_output *wlr_output = data;
|
|
|
|
|
|
|
|
struct timespec now;
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
|
|
output->last_frame = now;
|
|
|
|
|
|
|
|
int width, height;
|
|
|
|
wlr_output_effective_resolution(output->wlr_output, &width, &height);
|
|
|
|
wlr_renderer_begin(server->renderer, width, height);
|
|
|
|
|
|
|
|
float color[4] = {0.3, 0.3, 0.3, 1.0};
|
|
|
|
wlr_renderer_clear(server->renderer, color);
|
|
|
|
|
|
|
|
wlr_renderer_end(server->renderer);
|
|
|
|
wlr_output_commit(output->wlr_output);
|
2023-10-28 18:46:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void cleanupmon(struct wl_listener *listener, void *data) {
|
2023-10-30 12:00:27 +00:00
|
|
|
struct output *output = wl_container_of(listener, output, destroy);
|
|
|
|
|
|
|
|
wl_list_remove(&output->link);
|
|
|
|
wl_list_remove(&output->destroy.link);
|
|
|
|
wl_list_remove(&output->frame.link);
|
|
|
|
|
|
|
|
free(output);
|
2023-10-28 18:46:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void createmon(struct wl_listener *listener, void *data) {
|
|
|
|
struct server *server = wl_container_of(listener, server, new_output);
|
|
|
|
struct wlr_output *wlr_output = data;
|
|
|
|
|
2023-10-30 12:00:27 +00:00
|
|
|
struct output *output = malloc(sizeof(struct output));
|
|
|
|
if (output == NULL) {
|
|
|
|
wlr_backend_destroy(server->backend);
|
|
|
|
wl_display_destroy(server->wl_display);
|
|
|
|
die("createmon", 1);
|
|
|
|
}
|
2023-10-28 18:46:43 +00:00
|
|
|
|
2023-10-30 12:00:27 +00:00
|
|
|
if (wl_list_length(&wlr_output->modes) > 0) {
|
|
|
|
struct wlr_output_mode *mode = wl_container_of((&wlr_output->modes)->prev, mode, link);
|
|
|
|
wlr_output_set_mode(wlr_output, mode);
|
|
|
|
}
|
2023-10-28 18:46:43 +00:00
|
|
|
|
2023-10-30 12:00:27 +00:00
|
|
|
clock_gettime(CLOCK_MONOTONIC, &output->last_frame);
|
|
|
|
|
|
|
|
output->server = server;
|
|
|
|
output->wlr_output = wlr_output;
|
|
|
|
wl_list_insert(&server->outputs, &output->link);
|
|
|
|
|
|
|
|
output->destroy.notify = cleanupmon;
|
|
|
|
wl_signal_add(&wlr_output->events.destroy, &output->destroy);
|
|
|
|
|
|
|
|
output->frame.notify = rendermon;
|
|
|
|
wl_signal_add(&wlr_output->events.frame, &output->frame);
|
|
|
|
|
|
|
|
wlr_output_schedule_frame(wlr_output);
|
2023-10-28 18:46:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void server_init(struct server *server) {
|
|
|
|
server->wl_display = wl_display_create();
|
|
|
|
assert(server->wl_display);
|
|
|
|
|
2023-10-30 12:00:27 +00:00
|
|
|
server->wl_event_loop = wl_display_get_event_loop(server->wl_display);
|
|
|
|
assert(server->wl_event_loop);
|
|
|
|
|
2023-10-28 18:46:43 +00:00
|
|
|
server->backend = wlr_backend_autocreate(server->wl_display);
|
|
|
|
assert(server->backend);
|
|
|
|
|
|
|
|
server->renderer = wlr_renderer_autocreate(server->backend);
|
|
|
|
assert(server->renderer);
|
|
|
|
|
|
|
|
wlr_data_device_manager_create(server->wl_display);
|
|
|
|
wlr_export_dmabuf_manager_v1_create(server->wl_display);
|
|
|
|
wlr_screencopy_manager_v1_create(server->wl_display);
|
|
|
|
wlr_data_control_manager_v1_create(server->wl_display);
|
|
|
|
wlr_gamma_control_manager_v1_create(server->wl_display);
|
|
|
|
wlr_primary_selection_v1_device_manager_create(server->wl_display);
|
|
|
|
|
|
|
|
/* Init monitors */
|
2023-10-30 12:00:27 +00:00
|
|
|
wl_list_init(&server->outputs);
|
2023-10-28 18:46:43 +00:00
|
|
|
server->new_output.notify = createmon;
|
|
|
|
wl_signal_add(&server->backend->events.new_output, &server->new_output);
|
|
|
|
|
|
|
|
const char *socket = wl_display_add_socket_auto(server->wl_display);
|
|
|
|
assert(socket);
|
|
|
|
|
|
|
|
printf("Running compositor on wayland display '%s'\n", socket);
|
|
|
|
setenv("WAYLAND_DISPLAY", socket, 1);
|
|
|
|
|
|
|
|
if (!wlr_backend_start(server->backend)) {
|
|
|
|
wl_display_destroy(server->wl_display);
|
|
|
|
die("wlr_backend_start", 1);
|
|
|
|
}
|
2023-10-28 15:14:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(void) {
|
|
|
|
if (!getenv("XDG_RUNTIME_DIR"))
|
|
|
|
die("XDG_RUNTIME_DIR must be set", 0);
|
|
|
|
|
|
|
|
int sig[] = {SIGCHLD, SIGINT, SIGTERM, SIGPIPE};
|
|
|
|
for (size_t i = 0; i < sizeof(sig) / sizeof(sig[0]); i++)
|
2023-10-28 18:46:43 +00:00
|
|
|
signal(sig[i], NULL);
|
2023-10-28 15:14:49 +00:00
|
|
|
|
|
|
|
wlr_log_init(LOG_LEVEL, NULL);
|
|
|
|
|
|
|
|
struct server server;
|
|
|
|
server_init(&server);
|
|
|
|
|
|
|
|
wl_display_run(server.wl_display);
|
2023-10-28 18:46:43 +00:00
|
|
|
wlr_backend_destroy(server.backend);
|
|
|
|
wl_display_destroy(server.wl_display);
|
2023-10-28 15:14:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|