stage-cogl: Fix damage tracking with varying buffer ages
With server-side buffer allocation, buffers may be returned out of order (e.g. they may be held onto by external references or hardware). As such we may see older buffers the frame after we discard the history from seeing a very young buffer. To overcome this we want to keep the history in a ring so we can keep track of older entries without keeping an unbounded list. After converting to a ring, the maximum buffer age observed during testing was 5 (expected value of 4), but before we could see ages as high as 9 due to the huge latency spikes caused by doing full buffer redraws (compounded by external listeners doing readback on the damaged areas, for example vnc, drm/udl, prime). For this reason, a maximum age of 16 was chosen to be suitably large enough to prevent these worst cases from taxing the system. v2: Fix off-by-one in combining the damage histroy into the clipping rectangle, and apply copious whitespace fixes. Bugzilla: https://bugzilla.gnome.org/show_bug.cgi?id=745512 References: https://bugzilla.gnome.org/show_bug.cgi?id=724788 References: https://bugzilla.gnome.org/show_bug.cgi?id=669122
This commit is contained in:
parent
55c957267e
commit
239280f855
@ -392,6 +392,15 @@ clutter_stage_cogl_get_redraw_clip_bounds (ClutterStageWindow *stage_window,
|
|||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline gboolean
|
||||||
|
valid_buffer_age (ClutterStageCogl *stage_cogl, int age)
|
||||||
|
{
|
||||||
|
if (age <= 0 || stage_cogl->dirty_backbuffer)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
return age < MIN (stage_cogl->damage_index, DAMAGE_HISTORY_MAX);
|
||||||
|
}
|
||||||
|
|
||||||
/* XXX: This is basically identical to clutter_stage_glx_redraw */
|
/* XXX: This is basically identical to clutter_stage_glx_redraw */
|
||||||
static void
|
static void
|
||||||
clutter_stage_cogl_redraw (ClutterStageWindow *stage_window)
|
clutter_stage_cogl_redraw (ClutterStageWindow *stage_window)
|
||||||
@ -452,64 +461,46 @@ clutter_stage_cogl_redraw (ClutterStageWindow *stage_window)
|
|||||||
|
|
||||||
window_scale = _clutter_stage_window_get_scale_factor (stage_window);
|
window_scale = _clutter_stage_window_get_scale_factor (stage_window);
|
||||||
|
|
||||||
if (use_clipped_redraw)
|
if (has_buffer_age)
|
||||||
{
|
{
|
||||||
if (has_buffer_age)
|
cairo_rectangle_int_t *current_damage =
|
||||||
{
|
&stage_cogl->damage_history[DAMAGE_HISTORY (stage_cogl->damage_index++)];
|
||||||
int age = cogl_onscreen_get_buffer_age (stage_cogl->onscreen);
|
|
||||||
cairo_rectangle_int_t *current_damage;
|
|
||||||
|
|
||||||
current_damage = g_new0 (cairo_rectangle_int_t, 1);
|
if (use_clipped_redraw)
|
||||||
current_damage->x = clip_region->x;
|
{
|
||||||
current_damage->y = clip_region->y;
|
int age = cogl_onscreen_get_buffer_age (stage_cogl->onscreen), i;
|
||||||
current_damage->width = clip_region->width;
|
|
||||||
current_damage->height = clip_region->height;
|
|
||||||
|
|
||||||
stage_cogl->damage_history = g_slist_prepend (stage_cogl->damage_history, current_damage);
|
*current_damage = *clip_region;
|
||||||
|
|
||||||
if (age != 0 && !stage_cogl->dirty_backbuffer && g_slist_length (stage_cogl->damage_history) > age)
|
if (valid_buffer_age (stage_cogl, age))
|
||||||
{
|
{
|
||||||
int i = 0;
|
for (i = 1; i <= age; i++)
|
||||||
GSList *tmp = NULL;
|
_clutter_util_rectangle_union (clip_region,
|
||||||
/* We skip the first entry because it is the clip_region itself */
|
&stage_cogl->damage_history[DAMAGE_HISTORY (stage_cogl->damage_index - i - 1)],
|
||||||
for (tmp = stage_cogl->damage_history->next; tmp; tmp = tmp->next)
|
clip_region);
|
||||||
{
|
|
||||||
_clutter_util_rectangle_union (clip_region, tmp->data, clip_region);
|
|
||||||
i++;
|
|
||||||
if (i == age)
|
|
||||||
{
|
|
||||||
g_slist_free_full (tmp->next, g_free);
|
|
||||||
tmp->next = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
force_swap = TRUE;
|
CLUTTER_NOTE (CLIPPING, "Reusing back buffer(age=%d) - repairing region: x=%d, y=%d, width=%d, height=%d\n",
|
||||||
|
age,
|
||||||
CLUTTER_NOTE (CLIPPING, "Reusing back buffer - repairing region: x=%d, y=%d, width=%d, height=%d\n",
|
clip_region->x,
|
||||||
clip_region->x,
|
clip_region->y,
|
||||||
clip_region->y,
|
clip_region->width,
|
||||||
clip_region->width,
|
clip_region->height);
|
||||||
clip_region->height);
|
force_swap = TRUE;
|
||||||
|
}
|
||||||
}
|
else
|
||||||
else if (age == 0 || stage_cogl->dirty_backbuffer)
|
{
|
||||||
{
|
CLUTTER_NOTE (CLIPPING, "Invalid back buffer(age=%d): forcing full redraw\n", age);
|
||||||
CLUTTER_NOTE (CLIPPING, "Invalid back buffer: Resetting damage history list.\n");
|
use_clipped_redraw = FALSE;
|
||||||
g_slist_free_full (stage_cogl->damage_history, g_free);
|
}
|
||||||
stage_cogl->damage_history = NULL;
|
}
|
||||||
}
|
else
|
||||||
|
{
|
||||||
}
|
current_damage->x = 0;
|
||||||
|
current_damage->y = 0;
|
||||||
|
current_damage->width = geom.width;
|
||||||
|
current_damage->height = geom.height;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (has_buffer_age)
|
|
||||||
{
|
|
||||||
CLUTTER_NOTE (CLIPPING, "Unclipped redraw: Resetting damage history list.\n");
|
|
||||||
g_slist_free_full (stage_cogl->damage_history, g_free);
|
|
||||||
stage_cogl->damage_history = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (has_buffer_age && !force_swap)
|
|
||||||
use_clipped_redraw = FALSE;
|
|
||||||
|
|
||||||
if (use_clipped_redraw)
|
if (use_clipped_redraw)
|
||||||
{
|
{
|
||||||
@ -678,7 +669,7 @@ clutter_stage_cogl_get_dirty_pixel (ClutterStageWindow *stage_window,
|
|||||||
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
||||||
gboolean has_buffer_age = cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_BUFFER_AGE);
|
gboolean has_buffer_age = cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_BUFFER_AGE);
|
||||||
|
|
||||||
if ((stage_cogl->damage_history == NULL && has_buffer_age) || !has_buffer_age)
|
if (!has_buffer_age)
|
||||||
{
|
{
|
||||||
*x = 0;
|
*x = 0;
|
||||||
*y = 0;
|
*y = 0;
|
||||||
@ -686,7 +677,8 @@ clutter_stage_cogl_get_dirty_pixel (ClutterStageWindow *stage_window,
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
cairo_rectangle_int_t *rect;
|
cairo_rectangle_int_t *rect;
|
||||||
rect = (cairo_rectangle_int_t *) (stage_cogl->damage_history->data);
|
|
||||||
|
rect = &stage_cogl->damage_history[DAMAGE_HISTORY (stage_cogl->damage_index-1)];
|
||||||
*x = rect->x;
|
*x = rect->x;
|
||||||
*y = rect->y;
|
*y = rect->y;
|
||||||
}
|
}
|
||||||
|
@ -58,7 +58,10 @@ struct _ClutterStageCogl
|
|||||||
guint dirty_backbuffer : 1;
|
guint dirty_backbuffer : 1;
|
||||||
|
|
||||||
/* Stores a list of previous damaged areas */
|
/* Stores a list of previous damaged areas */
|
||||||
GSList *damage_history;
|
#define DAMAGE_HISTORY_MAX 16
|
||||||
|
#define DAMAGE_HISTORY(x) ((x) & (DAMAGE_HISTORY_MAX - 1))
|
||||||
|
cairo_rectangle_int_t damage_history[DAMAGE_HISTORY_MAX];
|
||||||
|
unsigned damage_index;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct _ClutterStageCoglClass
|
struct _ClutterStageCoglClass
|
||||||
|
Loading…
Reference in New Issue
Block a user