wlr_output: Introduce struct wlr_output_commit

This commit is contained in:
Alexander Orzechowski 2023-10-03 05:40:50 -04:00
parent 1b0694b794
commit c299c13d13
12 changed files with 145 additions and 49 deletions

View file

@ -699,12 +699,12 @@ bool drm_connector_supports_vrr(struct wlr_drm_connector *conn) {
return true; return true;
} }
bool drm_connector_commit_state(struct wlr_drm_connector *conn, struct wlr_output_commit *drm_connector_commit_state(struct wlr_drm_connector *conn,
const struct wlr_output_state *base) { const struct wlr_output_state *base) {
struct wlr_drm_backend *drm = conn->backend; struct wlr_drm_backend *drm = conn->backend;
if (!drm->session->active) { if (!drm->session->active) {
return false; return NULL;
} }
bool ok = false; bool ok = false;
@ -780,15 +780,21 @@ bool drm_connector_commit_state(struct wlr_drm_connector *conn,
out: out:
drm_connector_state_finish(&pending); drm_connector_state_finish(&pending);
return ok;
if (!ok) {
return NULL;
} }
static bool drm_connector_commit(struct wlr_output *output, wlr_output_commit_init(&conn->commit, &conn->output);
return &conn->commit;
}
static struct wlr_output_commit *drm_connector_commit(struct wlr_output *output,
const struct wlr_output_state *state) { const struct wlr_output_state *state) {
struct wlr_drm_connector *conn = get_drm_connector_from_output(output); struct wlr_drm_connector *conn = get_drm_connector_from_output(output);
if (!drm_connector_test(output, state)) { if (!drm_connector_test(output, state)) {
return false; return NULL;
} }
return drm_connector_commit_state(conn, state); return drm_connector_commit_state(conn, state);
@ -1711,6 +1717,7 @@ static void handle_page_flip(int fd, unsigned seq,
struct wlr_output_event_present present_event = { struct wlr_output_event_present present_event = {
/* The DRM backend guarantees that the presentation event will be for /* The DRM backend guarantees that the presentation event will be for
* the last submitted frame. */ * the last submitted frame. */
.output = &conn->output,
.commit_seq = conn->output.commit_seq, .commit_seq = conn->output.commit_seq,
.presented = drm->session->active, .presented = drm->session->active,
.when = &present_time, .when = &present_time,
@ -1719,6 +1726,7 @@ static void handle_page_flip(int fd, unsigned seq,
.flags = present_flags, .flags = present_flags,
}; };
wlr_output_send_present(&conn->output, &present_event); wlr_output_send_present(&conn->output, &present_event);
wl_signal_emit_mutable(&conn->commit.events.present, &present_event);
if (drm->session->active) { if (drm->session->active) {
wlr_output_send_frame(&conn->output); wlr_output_send_frame(&conn->output);

View file

@ -1,6 +1,8 @@
#define _POSIX_C_SOURCE 200809L
#include <assert.h> #include <assert.h>
#include <stdlib.h> #include <stdlib.h>
#include <stdio.h> #include <stdio.h>
#include <time.h>
#include <wlr/interfaces/wlr_output.h> #include <wlr/interfaces/wlr_output.h>
#include <wlr/types/wlr_output_layer.h> #include <wlr/types/wlr_output_layer.h>
#include <wlr/util/log.h> #include <wlr/util/log.h>
@ -53,7 +55,7 @@ static bool output_test(struct wlr_output *wlr_output,
return true; return true;
} }
static bool output_commit(struct wlr_output *wlr_output, static struct wlr_output_commit *output_commit(struct wlr_output *wlr_output,
const struct wlr_output_state *state) { const struct wlr_output_state *state) {
struct wlr_headless_output *output = struct wlr_headless_output *output =
headless_output_from_output(wlr_output); headless_output_from_output(wlr_output);
@ -62,21 +64,20 @@ static bool output_commit(struct wlr_output *wlr_output,
return false; return false;
} }
if (state->committed & WLR_OUTPUT_STATE_MODE) { if (state->committed & WLR_OUTPUT_STATE_MODE) {
output_update_refresh(output, state->custom_mode.refresh); output_update_refresh(output, state->custom_mode.refresh);
} }
wlr_output_commit_init(&output->commit, wlr_output);
if (output_pending_enabled(wlr_output, state)) { if (output_pending_enabled(wlr_output, state)) {
struct wlr_output_event_present present_event = { output_commit_defer_present(&output->commit);
.commit_seq = wlr_output->commit_seq + 1,
.presented = true,
};
output_defer_present(wlr_output, present_event);
wl_event_source_timer_update(output->frame_timer, output->frame_delay); wl_event_source_timer_update(output->frame_timer, output->frame_delay);
} }
return true; return &output->commit;
} }
static void output_destroy(struct wlr_output *wlr_output) { static void output_destroy(struct wlr_output *wlr_output) {

View file

@ -1,3 +1,4 @@
#define _POSIX_C_SOURCE 200809L
#include <assert.h> #include <assert.h>
#include <stdint.h> #include <stdint.h>
#include <stdio.h> #include <stdio.h>
@ -6,6 +7,7 @@
#include <sys/mman.h> #include <sys/mman.h>
#include <sys/types.h> #include <sys/types.h>
#include <unistd.h> #include <unistd.h>
#include <time.h>
#include <drm_fourcc.h> #include <drm_fourcc.h>
#include <wayland-client.h> #include <wayland-client.h>
@ -103,12 +105,14 @@ static void presentation_feedback_handle_presented(void *data,
struct wlr_output_event_present event = { struct wlr_output_event_present event = {
.commit_seq = feedback->commit_seq, .commit_seq = feedback->commit_seq,
.presented = true, .presented = true,
.output = &feedback->output->wlr_output,
.when = &t, .when = &t,
.seq = ((uint64_t)seq_hi << 32) | seq_lo, .seq = ((uint64_t)seq_hi << 32) | seq_lo,
.refresh = refresh_ns, .refresh = refresh_ns,
.flags = flags, .flags = flags,
}; };
wlr_output_send_present(&feedback->output->wlr_output, &event); wlr_output_send_present(&feedback->output->wlr_output, &event);
wl_signal_emit_mutable(&feedback->commit.events.present, &event);
presentation_feedback_destroy(feedback); presentation_feedback_destroy(feedback);
} }
@ -120,8 +124,10 @@ static void presentation_feedback_handle_discarded(void *data,
struct wlr_output_event_present event = { struct wlr_output_event_present event = {
.commit_seq = feedback->commit_seq, .commit_seq = feedback->commit_seq,
.presented = false, .presented = false,
.output = &feedback->output->wlr_output,
}; };
wlr_output_send_present(&feedback->output->wlr_output, &event); wlr_output_send_present(&feedback->output->wlr_output, &event);
wl_signal_emit_mutable(&feedback->commit.events.present, &event);
presentation_feedback_destroy(feedback); presentation_feedback_destroy(feedback);
} }
@ -520,13 +526,13 @@ static bool commit_layers(struct wlr_wl_output *output,
return true; return true;
} }
static bool output_commit(struct wlr_output *wlr_output, static struct wlr_output_commit *output_commit(struct wlr_output *wlr_output,
const struct wlr_output_state *state) { const struct wlr_output_state *state) {
struct wlr_wl_output *output = struct wlr_wl_output *output =
get_wl_output_from_output(wlr_output); get_wl_output_from_output(wlr_output);
if (!output_test(wlr_output, state)) { if (!output_test(wlr_output, state)) {
return false; return NULL;
} }
if ((state->committed & WLR_OUTPUT_STATE_ENABLED) && !state->enabled) { if ((state->committed & WLR_OUTPUT_STATE_ENABLED) && !state->enabled) {
@ -544,7 +550,7 @@ static bool output_commit(struct wlr_output *wlr_output,
struct wlr_wl_buffer *buffer = struct wlr_wl_buffer *buffer =
get_or_create_wl_buffer(output->backend, wlr_buffer); get_or_create_wl_buffer(output->backend, wlr_buffer);
if (buffer == NULL) { if (buffer == NULL) {
return false; return NULL;
} }
wl_surface_attach(output->surface, buffer->wl_buffer, 0, 0); wl_surface_attach(output->surface, buffer->wl_buffer, 0, 0);
@ -553,9 +559,11 @@ static bool output_commit(struct wlr_output *wlr_output,
if ((state->committed & WLR_OUTPUT_STATE_LAYERS) && if ((state->committed & WLR_OUTPUT_STATE_LAYERS) &&
!commit_layers(output, state->layers, state->layers_len)) { !commit_layers(output, state->layers, state->layers_len)) {
return false; return NULL;
} }
struct wlr_output_commit *commit = NULL;
if (output_pending_enabled(wlr_output, state)) { if (output_pending_enabled(wlr_output, state)) {
if (output->frame_callback != NULL) { if (output->frame_callback != NULL) {
wl_callback_destroy(output->frame_callback); wl_callback_destroy(output->frame_callback);
@ -576,7 +584,7 @@ static bool output_commit(struct wlr_output *wlr_output,
calloc(1, sizeof(*feedback)); calloc(1, sizeof(*feedback));
if (feedback == NULL) { if (feedback == NULL) {
wp_presentation_feedback_destroy(wp_feedback); wp_presentation_feedback_destroy(wp_feedback);
return false; return NULL;
} }
feedback->output = output; feedback->output = output;
feedback->feedback = wp_feedback; feedback->feedback = wp_feedback;
@ -585,18 +593,21 @@ static bool output_commit(struct wlr_output *wlr_output,
wp_presentation_feedback_add_listener(wp_feedback, wp_presentation_feedback_add_listener(wp_feedback,
&presentation_feedback_listener, feedback); &presentation_feedback_listener, feedback);
commit = &feedback->commit;
wlr_output_commit_init(commit, wlr_output);
} else { } else {
struct wlr_output_event_present present_event = { wlr_output_commit_init(commit, wlr_output);
.commit_seq = wlr_output->commit_seq + 1, output_commit_defer_present(commit);
.presented = true,
};
output_defer_present(wlr_output, present_event);
} }
} else {
commit = &output->commit;
wlr_output_commit_init(commit, wlr_output);
} }
wl_display_flush(output->backend->remote_display); wl_display_flush(output->backend->remote_display);
return true; return commit;
} }
static bool output_set_cursor(struct wlr_output *wlr_output, static bool output_set_cursor(struct wlr_output *wlr_output,

View file

@ -105,6 +105,12 @@ static void output_destroy(struct wlr_output *wlr_output) {
xcb_render_free_picture(x11->xcb, output->cursor.pic); xcb_render_free_picture(x11->xcb, output->cursor.pic);
} }
struct wlr_x11_output_commit *commit, *tmp_commit;
wl_list_for_each_safe(commit, tmp_commit, &output->commits, link) {
wl_list_remove(&commit->link);
free(commit);
}
// A zero event mask deletes the event context // A zero event mask deletes the event context
xcb_present_select_input(x11->xcb, output->present_event_id, output->win, 0); xcb_present_select_input(x11->xcb, output->present_event_id, output->win, 0);
xcb_destroy_window(x11->xcb, output->win); xcb_destroy_window(x11->xcb, output->win);
@ -288,7 +294,7 @@ static struct wlr_x11_buffer *get_or_create_x11_buffer(
} }
static bool output_commit_buffer(struct wlr_x11_output *output, static bool output_commit_buffer(struct wlr_x11_output *output,
const struct wlr_output_state *state) { const struct wlr_output_state *state, uint32_t serial) {
struct wlr_x11_backend *x11 = output->x11; struct wlr_x11_backend *x11 = output->x11;
struct wlr_buffer *buffer = state->buffer; struct wlr_buffer *buffer = state->buffer;
@ -328,7 +334,6 @@ static bool output_commit_buffer(struct wlr_x11_output *output,
pixman_region32_clear(&output->exposed); pixman_region32_clear(&output->exposed);
uint32_t serial = output->wlr_output.commit_seq;
uint32_t options = 0; uint32_t options = 0;
uint64_t target_msc = output->last_msc ? output->last_msc + 1 : 0; uint64_t target_msc = output->last_msc ? output->last_msc + 1 : 0;
xcb_present_pixmap(x11->xcb, output->win, x11_buffer->pixmap, serial, xcb_present_pixmap(x11->xcb, output->win, x11_buffer->pixmap, serial,
@ -346,13 +351,18 @@ error:
return false; return false;
} }
static bool output_commit(struct wlr_output *wlr_output, static struct wlr_output_commit *output_commit(struct wlr_output *wlr_output,
const struct wlr_output_state *state) { const struct wlr_output_state *state) {
struct wlr_x11_output *output = get_x11_output_from_output(wlr_output); struct wlr_x11_output *output = get_x11_output_from_output(wlr_output);
struct wlr_x11_backend *x11 = output->x11; struct wlr_x11_backend *x11 = output->x11;
if (!output_test(wlr_output, state)) { if (!output_test(wlr_output, state)) {
return false; return NULL;
}
struct wlr_x11_output_commit *commit = calloc(1, sizeof(*commit));
if (!commit){
return NULL;
} }
if (state->committed & WLR_OUTPUT_STATE_ENABLED) { if (state->committed & WLR_OUTPUT_STATE_ENABLED) {
@ -368,23 +378,29 @@ static bool output_commit(struct wlr_output *wlr_output,
state->custom_mode.width, state->custom_mode.width,
state->custom_mode.height, state->custom_mode.height,
state->custom_mode.refresh)) { state->custom_mode.refresh)) {
return false; free(commit);
return NULL;
} }
} }
uint32_t serial = output->wlr_output.commit_seq;
wlr_output_commit_init(&commit->commit, wlr_output);
wl_list_insert(&output->commits, &commit->link);
commit->serial = serial;
if (state->committed & WLR_OUTPUT_STATE_BUFFER) { if (state->committed & WLR_OUTPUT_STATE_BUFFER) {
if (!output_commit_buffer(output, state)) { if (!output_commit_buffer(output, state, serial)) {
return false; return NULL;
} }
} else if (output_pending_enabled(wlr_output, state)) { } else if (output_pending_enabled(wlr_output, state)) {
uint32_t serial = output->wlr_output.commit_seq;
uint64_t target_msc = output->last_msc ? output->last_msc + 1 : 0; uint64_t target_msc = output->last_msc ? output->last_msc + 1 : 0;
xcb_present_notify_msc(x11->xcb, output->win, serial, target_msc, 0, 0); xcb_present_notify_msc(x11->xcb, output->win, serial, target_msc, 0, 0);
} }
xcb_flush(x11->xcb); xcb_flush(x11->xcb);
return true; return &commit->commit;
} }
static void update_x11_output_cursor(struct wlr_x11_output *output, static void update_x11_output_cursor(struct wlr_x11_output *output,
@ -541,6 +557,7 @@ struct wlr_output *wlr_x11_output_create(struct wlr_backend *backend) {
} }
output->x11 = x11; output->x11 = x11;
wl_list_init(&output->buffers); wl_list_init(&output->buffers);
wl_list_init(&output->commits);
pixman_region32_init(&output->exposed); pixman_region32_init(&output->exposed);
struct wlr_output *wlr_output = &output->wlr_output; struct wlr_output *wlr_output = &output->wlr_output;
@ -724,6 +741,16 @@ void handle_x11_present_event(struct wlr_x11_backend *x11,
flags |= WLR_OUTPUT_PRESENT_ZERO_COPY; flags |= WLR_OUTPUT_PRESENT_ZERO_COPY;
} }
uint32_t serial = complete_notify->serial;
struct wlr_x11_output_commit *commit = NULL;
wl_list_for_each(commit, &output->commits, link) {
if (commit->serial == serial) {
break;
}
}
assert(commit && commit->serial == serial);
bool presented = complete_notify->mode != XCB_PRESENT_COMPLETE_MODE_SKIP; bool presented = complete_notify->mode != XCB_PRESENT_COMPLETE_MODE_SKIP;
struct wlr_output_event_present present_event = { struct wlr_output_event_present present_event = {
.output = &output->wlr_output, .output = &output->wlr_output,
@ -734,6 +761,10 @@ void handle_x11_present_event(struct wlr_x11_backend *x11,
.flags = flags, .flags = flags,
}; };
wlr_output_send_present(&output->wlr_output, &present_event); wlr_output_send_present(&output->wlr_output, &present_event);
wl_signal_emit_mutable(&commit->commit.events.present, &present_event);
wl_list_remove(&commit->link);
free(commit);
wlr_output_send_frame(&output->wlr_output); wlr_output_send_frame(&output->wlr_output);
break; break;

View file

@ -133,6 +133,7 @@ struct wlr_drm_connector_state {
struct wlr_drm_connector { struct wlr_drm_connector {
struct wlr_output output; // only valid if status != DISCONNECTED struct wlr_output output; // only valid if status != DISCONNECTED
struct wlr_output_commit commit;
struct wlr_drm_backend *backend; struct wlr_drm_backend *backend;
char name[24]; char name[24];
@ -175,7 +176,7 @@ void scan_drm_connectors(struct wlr_drm_backend *state,
void scan_drm_leases(struct wlr_drm_backend *drm); void scan_drm_leases(struct wlr_drm_backend *drm);
int handle_drm_event(int fd, uint32_t mask, void *data); int handle_drm_event(int fd, uint32_t mask, void *data);
void destroy_drm_connector(struct wlr_drm_connector *conn); void destroy_drm_connector(struct wlr_drm_connector *conn);
bool drm_connector_commit_state(struct wlr_drm_connector *conn, struct wlr_output_commit *drm_connector_commit_state(struct wlr_drm_connector *conn,
const struct wlr_output_state *state); const struct wlr_output_state *state);
bool drm_connector_is_cursor_visible(struct wlr_drm_connector *conn); bool drm_connector_is_cursor_visible(struct wlr_drm_connector *conn);
bool drm_connector_supports_vrr(struct wlr_drm_connector *conn); bool drm_connector_supports_vrr(struct wlr_drm_connector *conn);

View file

@ -16,6 +16,7 @@ struct wlr_headless_backend {
struct wlr_headless_output { struct wlr_headless_output {
struct wlr_output wlr_output; struct wlr_output wlr_output;
struct wlr_output_commit commit;
struct wlr_headless_backend *backend; struct wlr_headless_backend *backend;
struct wl_list link; struct wl_list link;

View file

@ -62,8 +62,10 @@ struct wlr_wl_buffer {
}; };
struct wlr_wl_presentation_feedback { struct wlr_wl_presentation_feedback {
struct wlr_output_commit commit;
struct wlr_wl_output *output; struct wlr_wl_output *output;
struct wl_list link; struct wl_list link; // wlr_wl_output.presentation_feedbacks
struct wp_presentation_feedback *feedback; struct wp_presentation_feedback *feedback;
uint32_t commit_seq; uint32_t commit_seq;
}; };
@ -80,8 +82,11 @@ struct wlr_wl_output_layer {
struct wlr_wl_output { struct wlr_wl_output {
struct wlr_output wlr_output; struct wlr_output wlr_output;
// commit used when output is disabled or presentation events are not supported
struct wlr_output_commit commit;
struct wlr_wl_backend *backend; struct wlr_wl_backend *backend;
struct wl_list link; struct wl_list link; // wlr_wl_backend.outputs
struct wl_surface *surface; struct wl_surface *surface;
bool own_surface; bool own_surface;
@ -89,7 +94,7 @@ struct wlr_wl_output {
struct xdg_surface *xdg_surface; struct xdg_surface *xdg_surface;
struct xdg_toplevel *xdg_toplevel; struct xdg_toplevel *xdg_toplevel;
struct zxdg_toplevel_decoration_v1 *zxdg_toplevel_decoration_v1; struct zxdg_toplevel_decoration_v1 *zxdg_toplevel_decoration_v1;
struct wl_list presentation_feedbacks; struct wl_list presentation_feedbacks; // wlr_wl_presentation_feedback.link
bool configured; bool configured;
uint32_t enter_serial; uint32_t enter_serial;

View file

@ -27,6 +27,12 @@
struct wlr_x11_backend; struct wlr_x11_backend;
struct wlr_x11_output_commit {
struct wl_list link; // wlr_x11_output::commits;
uint32_t serial;
struct wlr_output_commit commit;
};
struct wlr_x11_output { struct wlr_x11_output {
struct wlr_output wlr_output; struct wlr_output wlr_output;
struct wlr_x11_backend *x11; struct wlr_x11_backend *x11;
@ -50,6 +56,8 @@ struct wlr_x11_output {
struct wlr_swapchain *swapchain; struct wlr_swapchain *swapchain;
xcb_render_picture_t pic; xcb_render_picture_t pic;
} cursor; } cursor;
struct wl_list commits; // wlr_x11_output_commit::link
}; };
struct wlr_x11_touchpoint { struct wlr_x11_touchpoint {

View file

@ -21,6 +21,6 @@ bool output_cursor_set_texture(struct wlr_output_cursor *cursor,
int dst_width, int dst_height, enum wl_output_transform transform, int dst_width, int dst_height, enum wl_output_transform transform,
int32_t hotspot_x, int32_t hotspot_y); int32_t hotspot_x, int32_t hotspot_y);
void output_defer_present(struct wlr_output *output, struct wlr_output_event_present event); void output_commit_defer_present(struct wlr_output_commit *commit);
#endif #endif

View file

@ -65,7 +65,8 @@ struct wlr_output_impl {
* *
* If a buffer has been attached, a frame event is scheduled. * If a buffer has been attached, a frame event is scheduled.
*/ */
bool (*commit)(struct wlr_output *output, const struct wlr_output_state *state); struct wlr_output_commit *(*commit)(struct wlr_output *output,
const struct wlr_output_state *state);
/** /**
* Get the maximum number of gamma LUT elements for each channel. * Get the maximum number of gamma LUT elements for each channel.
* *

View file

@ -227,6 +227,7 @@ struct wlr_output_event_commit {
uint32_t committed; // bitmask of enum wlr_output_state_field uint32_t committed; // bitmask of enum wlr_output_state_field
struct timespec *when; struct timespec *when;
struct wlr_buffer *buffer; // NULL if no buffer is committed struct wlr_buffer *buffer; // NULL if no buffer is committed
struct wlr_output_commit *commit;
}; };
enum wlr_output_present_flag { enum wlr_output_present_flag {
@ -270,8 +271,20 @@ struct wlr_output_event_request_state {
const struct wlr_output_state *state; const struct wlr_output_state *state;
}; };
struct wlr_output_commit {
struct wlr_output *output;
struct {
struct wl_signal present; // struct wlr_output_event_present
} events;
};
struct wlr_surface; struct wlr_surface;
void wlr_output_commit_init(struct wlr_output_commit *commit,
struct wlr_output *output);
/** /**
* Enables or disables the output. A disabled output is turned off and doesn't * Enables or disables the output. A disabled output is turned off and doesn't
* emit `frame` events. * emit `frame` events.

View file

@ -780,6 +780,12 @@ bool wlr_output_test(struct wlr_output *output) {
return wlr_output_test_state(output, &state); return wlr_output_test_state(output, &state);
} }
void wlr_output_commit_init(struct wlr_output_commit *commit,
struct wlr_output *output) {
wl_signal_init(&commit->events.present);
commit->output = output;
}
bool wlr_output_commit_state(struct wlr_output *output, bool wlr_output_commit_state(struct wlr_output *output,
const struct wlr_output_state *state) { const struct wlr_output_state *state) {
uint32_t unchanged = output_compare_state(output, state); uint32_t unchanged = output_compare_state(output, state);
@ -815,7 +821,8 @@ bool wlr_output_commit_state(struct wlr_output *output,
}; };
wl_signal_emit_mutable(&output->events.precommit, &pre_event); wl_signal_emit_mutable(&output->events.precommit, &pre_event);
if (!output->impl->commit(output, &pending)) { struct wlr_output_commit *commit = output->impl->commit(output, &pending);
if (!commit) {
if (new_back_buffer) { if (new_back_buffer) {
wlr_buffer_unlock(pending.buffer); wlr_buffer_unlock(pending.buffer);
} }
@ -836,6 +843,7 @@ bool wlr_output_commit_state(struct wlr_output *output,
.committed = pending.committed, .committed = pending.committed,
.when = &now, .when = &now,
.buffer = (pending.committed & WLR_OUTPUT_STATE_BUFFER) ? pending.buffer : NULL, .buffer = (pending.committed & WLR_OUTPUT_STATE_BUFFER) ? pending.buffer : NULL,
.commit = commit,
}; };
wl_signal_emit_mutable(&output->events.commit, &event); wl_signal_emit_mutable(&output->events.commit, &event);
@ -928,9 +936,8 @@ void wlr_output_send_present(struct wlr_output *output,
} }
struct deferred_present_event { struct deferred_present_event {
struct wlr_output *output; struct wlr_output_commit *commit;
struct wl_event_source *idle_source; struct wl_event_source *idle_source;
struct wlr_output_event_present event;
struct wl_listener output_destroy; struct wl_listener output_destroy;
}; };
@ -941,7 +948,19 @@ static void deferred_present_event_destroy(struct deferred_present_event *deferr
static void deferred_present_event_handle_idle(void *data) { static void deferred_present_event_handle_idle(void *data) {
struct deferred_present_event *deferred = data; struct deferred_present_event *deferred = data;
wlr_output_send_present(deferred->output, &deferred->event);
clockid_t clock = wlr_backend_get_presentation_clock(deferred->commit->output->backend);
struct timespec now;
clock_gettime(clock, &now);
struct wlr_output_event_present present_event = {
.presented = true,
.output = deferred->commit->output,
.when = &now,
};
wl_signal_emit_mutable(&deferred->commit->events.present, &present_event);
deferred_present_event_destroy(deferred); deferred_present_event_destroy(deferred);
} }
@ -951,19 +970,16 @@ static void deferred_present_event_handle_output_destroy(struct wl_listener *lis
deferred_present_event_destroy(deferred); deferred_present_event_destroy(deferred);
} }
void output_defer_present(struct wlr_output *output, struct wlr_output_event_present event) { void output_commit_defer_present(struct wlr_output_commit *commit) {
struct deferred_present_event *deferred = calloc(1, sizeof(*deferred)); struct deferred_present_event *deferred = calloc(1, sizeof(*deferred));
if (!deferred) { if (!deferred) {
return; return;
} }
*deferred = (struct deferred_present_event){ deferred->commit = commit;
.output = output,
.event = event,
};
deferred->output_destroy.notify = deferred_present_event_handle_output_destroy; deferred->output_destroy.notify = deferred_present_event_handle_output_destroy;
wl_signal_add(&output->events.destroy, &deferred->output_destroy); wl_signal_add(&commit->output->events.destroy, &deferred->output_destroy);
struct wl_event_loop *ev = wl_display_get_event_loop(output->display); struct wl_event_loop *ev = wl_display_get_event_loop(commit->output->display);
deferred->idle_source = wl_event_loop_add_idle(ev, deferred_present_event_handle_idle, deferred); deferred->idle_source = wl_event_loop_add_idle(ev, deferred_present_event_handle_idle, deferred);
} }