forked from AuroraMiddleware/gtk
15ee04c66f
In order to be able to track statistics about how well we are drawing, and in order to be able to do sophisticated things with frame timing like predicting per-frame latencies and synchronizing audio with video, we need to be able to track exactly when previous frames were drawn to the screen. Information about each frame is stored in a new GdkFrameTimings object. A new GdkFrameHistory object is added which keeps a queue of recent GdkFrameTimings (this is added to avoid further complicating the implementation of GdkFrameClock.) https://bugzilla.gnome.org/show_bug.cgi?id=685460
144 lines
3.6 KiB
C
144 lines
3.6 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 2012 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gdkframehistory.h"
|
|
|
|
#define FRAME_HISTORY_MAX_LENGTH 16
|
|
|
|
struct _GdkFrameHistory
|
|
{
|
|
GObject parent_instance;
|
|
|
|
gint64 frame_counter;
|
|
gint n_timings;
|
|
gint current;
|
|
GdkFrameTimings *timings[FRAME_HISTORY_MAX_LENGTH];
|
|
};
|
|
|
|
struct _GdkFrameHistoryClass
|
|
{
|
|
GObjectClass parent_class;
|
|
};
|
|
|
|
G_DEFINE_TYPE (GdkFrameHistory, gdk_frame_history, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
gdk_frame_history_finalize (GObject *object)
|
|
{
|
|
GdkFrameHistory *history = GDK_FRAME_HISTORY (object);
|
|
int i;
|
|
|
|
for (i = 0; i < FRAME_HISTORY_MAX_LENGTH; i++)
|
|
if (history->timings[i] != 0)
|
|
gdk_frame_timings_unref (history->timings[i]);
|
|
|
|
G_OBJECT_CLASS (gdk_frame_history_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gdk_frame_history_class_init (GdkFrameHistoryClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->finalize = gdk_frame_history_finalize;
|
|
}
|
|
|
|
static void
|
|
gdk_frame_history_init (GdkFrameHistory *history)
|
|
{
|
|
history->frame_counter = -1;
|
|
history->current = FRAME_HISTORY_MAX_LENGTH - 1;
|
|
}
|
|
|
|
GdkFrameHistory *
|
|
gdk_frame_history_new (void)
|
|
{
|
|
return g_object_new (GDK_TYPE_FRAME_HISTORY, NULL);
|
|
}
|
|
|
|
gint64
|
|
gdk_frame_history_get_frame_counter (GdkFrameHistory *history)
|
|
{
|
|
g_return_val_if_fail (GDK_IS_FRAME_HISTORY (history), 0);
|
|
|
|
return history->frame_counter;
|
|
}
|
|
|
|
gint64
|
|
gdk_frame_history_get_start (GdkFrameHistory *history)
|
|
{
|
|
g_return_val_if_fail (GDK_IS_FRAME_HISTORY (history), 0);
|
|
|
|
return history->frame_counter + 1 - history->n_timings;
|
|
}
|
|
|
|
void
|
|
gdk_frame_history_begin_frame (GdkFrameHistory *history)
|
|
{
|
|
g_return_if_fail (GDK_IS_FRAME_HISTORY (history));
|
|
|
|
history->frame_counter++;
|
|
history->current = (history->current + 1) % FRAME_HISTORY_MAX_LENGTH;
|
|
|
|
if (history->n_timings < FRAME_HISTORY_MAX_LENGTH)
|
|
history->n_timings++;
|
|
else
|
|
{
|
|
gdk_frame_timings_unref(history->timings[history->current]);
|
|
}
|
|
|
|
history->timings[history->current] = gdk_frame_timings_new (history->frame_counter);
|
|
}
|
|
|
|
GdkFrameTimings *
|
|
gdk_frame_history_get_timings (GdkFrameHistory *history,
|
|
gint64 frame_counter)
|
|
{
|
|
gint pos;
|
|
|
|
g_return_val_if_fail (GDK_IS_FRAME_HISTORY (history), NULL);
|
|
|
|
if (frame_counter > history->frame_counter)
|
|
return NULL;
|
|
|
|
if (frame_counter <= history->frame_counter - history->n_timings)
|
|
return NULL;
|
|
|
|
pos = (history->current - (history->frame_counter - frame_counter) + FRAME_HISTORY_MAX_LENGTH) % FRAME_HISTORY_MAX_LENGTH;
|
|
|
|
return history->timings[pos];
|
|
}
|
|
|
|
GdkFrameTimings *
|
|
gdk_frame_history_get_last_complete (GdkFrameHistory *history)
|
|
{
|
|
gint i;
|
|
|
|
g_return_val_if_fail (GDK_IS_FRAME_HISTORY (history), NULL);
|
|
|
|
for (i = 0; i < history->n_timings; i++)
|
|
{
|
|
gint pos = ((history->current - i) + FRAME_HISTORY_MAX_LENGTH) % FRAME_HISTORY_MAX_LENGTH;
|
|
if (gdk_frame_timings_get_complete (history->timings[pos]))
|
|
return history->timings[pos];
|
|
}
|
|
|
|
return NULL;
|
|
}
|