// Copyright 2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Shared frame timing code. * @author Jakob Bornecrantz * @ingroup aux_util */ #include "util/u_misc.h" #include "util/u_logging.h" #include "util/u_render_timing.h" #include #include #include /* * * Helpers * */ #if 0 #define DEBUG_PRINT_FRAME_ID() U_LOG_RAW("%" PRIi64 " %s", frame_id, __func__) #else #define DEBUG_PRINT_FRAME_ID() \ do { \ } while (false) #endif static uint64_t min_period(const struct u_rt_helper *urth) { return urth->last_input.predicted_display_period_ns; } static uint64_t last_displayed(const struct u_rt_helper *urth) { return urth->last_input.predicted_display_time_ns; } static uint64_t get_last_input_plus_period_at_least_greater_then(struct u_rt_helper *urth, uint64_t then_ns) { uint64_t val = last_displayed(urth); if (min_period(urth) == 0) { return then_ns; } while (val <= then_ns) { val += min_period(urth); assert(val != 0); } return val; } /* * * 'Exported' functions. * */ void u_rt_helper_client_clear(struct u_rt_helper *urth) { for (size_t i = 0; i < ARRAY_SIZE(urth->frames); i++) { urth->frames[i].state = U_RT_READY; urth->frames[i].frame_id = -1; } } void u_rt_helper_init(struct u_rt_helper *urth) { U_ZERO(urth); u_rt_helper_client_clear(urth); } void u_rt_helper_predict(struct u_rt_helper *urth, int64_t *out_frame_id, uint64_t *predicted_display_time, uint64_t *wake_up_time, uint64_t *predicted_display_period, uint64_t *min_display_period) { int64_t frame_id = ++urth->frame_counter; *out_frame_id = frame_id; DEBUG_PRINT_FRAME_ID(); uint64_t at_least_ns = os_monotonic_get_ns(); // Don't return a time before the last returned type. if (at_least_ns < urth->last_returned_ns) { at_least_ns = urth->last_returned_ns; } uint64_t predict_ns = get_last_input_plus_period_at_least_greater_then(urth, at_least_ns); urth->last_returned_ns = predict_ns; *wake_up_time = predict_ns - min_period(urth); *predicted_display_time = predict_ns; *predicted_display_period = min_period(urth); *min_display_period = min_period(urth); size_t index = (uint64_t)frame_id % ARRAY_SIZE(urth->frames); assert(urth->frames[index].frame_id == -1); assert(urth->frames[index].state == U_RT_READY); urth->frames[index].predicted = os_monotonic_get_ns(); urth->frames[index].state = U_RT_PREDICTED; urth->frames[index].frame_id = frame_id; } void u_rt_helper_mark_wait_woke(struct u_rt_helper *urth, int64_t frame_id) { DEBUG_PRINT_FRAME_ID(); size_t index = (uint64_t)frame_id % ARRAY_SIZE(urth->frames); assert(urth->frames[index].frame_id == frame_id); assert(urth->frames[index].state == U_RT_PREDICTED); urth->frames[index].wait_woke = os_monotonic_get_ns(); urth->frames[index].state = U_RT_WAIT_LEFT; } void u_rt_helper_mark_begin(struct u_rt_helper *urth, int64_t frame_id) { DEBUG_PRINT_FRAME_ID(); size_t index = (uint64_t)frame_id % ARRAY_SIZE(urth->frames); assert(urth->frames[index].frame_id == frame_id); assert(urth->frames[index].state == U_RT_WAIT_LEFT); urth->frames[index].begin = os_monotonic_get_ns(); urth->frames[index].state = U_RT_BEGUN; } void u_rt_helper_mark_discarded(struct u_rt_helper *urth, int64_t frame_id) { DEBUG_PRINT_FRAME_ID(); size_t index = (uint64_t)frame_id % ARRAY_SIZE(urth->frames); assert(urth->frames[index].frame_id == frame_id); assert(urth->frames[index].state == U_RT_WAIT_LEFT || urth->frames[index].state == U_RT_BEGUN); urth->frames[index].end_frame = os_monotonic_get_ns(); urth->frames[index].state = U_RT_READY; urth->frames[index].frame_id = -1; } void u_rt_helper_mark_delivered(struct u_rt_helper *urth, int64_t frame_id) { DEBUG_PRINT_FRAME_ID(); size_t index = (uint64_t)frame_id % ARRAY_SIZE(urth->frames); assert(urth->frames[index].frame_id == frame_id); assert(urth->frames[index].state == U_RT_BEGUN); uint64_t now_ns = os_monotonic_get_ns(); urth->frames[index].end_frame = now_ns; urth->frames[index].state = U_RT_READY; urth->frames[index].frame_id = -1; #if 0 uint64_t then_ns = urth->frames[index].wait_woke; uint64_t diff_ns = now_ns - then_ns; uint64_t ms100 = diff_ns / (1000 * 10); U_LOG_RAW("Diff %i.%02ims", (int)ms100 / 100, (int)ms100 % 100); #endif } void u_rt_helper_new_sample(struct u_rt_helper *urth, uint64_t predicted_display_time_ns, uint64_t predicted_display_period_ns, uint64_t extra_ns) { urth->last_input.predicted_display_time_ns = predicted_display_time_ns; urth->last_input.predicted_display_period_ns = predicted_display_period_ns; urth->last_input.extra_ns = extra_ns; }