Blame glib/tests/timeout.c

Packit ae235b
#include <glib.h>
Packit ae235b
#ifdef G_OS_UNIX
Packit ae235b
#include <unistd.h>
Packit ae235b
#endif
Packit ae235b
Packit ae235b
static GMainLoop *loop;
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
stop_waiting (gpointer data)
Packit ae235b
{
Packit ae235b
  g_main_loop_quit (loop);
Packit ae235b
Packit ae235b
  return G_SOURCE_REMOVE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
function (gpointer data)
Packit ae235b
{
Packit ae235b
  g_assert_not_reached ();
Packit ae235b
Packit ae235b
  return G_SOURCE_REMOVE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_seconds (void)
Packit ae235b
{
Packit ae235b
  guint id;
Packit ae235b
Packit ae235b
  /* Bug 642052 mentions that g_timeout_add_seconds(21475) schedules a
Packit ae235b
   * job that runs once per second.
Packit ae235b
   *
Packit ae235b
   * Test that that isn't true anymore by scheduling two jobs:
Packit ae235b
   *   - one, as above
Packit ae235b
   *   - another that runs in 2100ms
Packit ae235b
   *
Packit ae235b
   * If everything is working properly, the 2100ms one should run first
Packit ae235b
   * (and exit the mainloop).  If we ever see the 21475 second job run
Packit ae235b
   * then we have trouble (since it ran in less than 2 seconds).
Packit ae235b
   *
Packit ae235b
   * We need a timeout of at least 2 seconds because
Packit ae235b
   * g_timeout_add_second can add as much as an additional second of
Packit ae235b
   * latency.
Packit ae235b
   */
Packit ae235b
  loop = g_main_loop_new (NULL, FALSE);
Packit ae235b
Packit ae235b
  g_timeout_add (2100, stop_waiting, NULL);
Packit ae235b
  id = g_timeout_add_seconds (21475, function, NULL);
Packit ae235b
Packit ae235b
  g_main_loop_run (loop);
Packit ae235b
  g_main_loop_unref (loop);
Packit ae235b
Packit ae235b
  g_source_remove (id);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gint64 last_time;
Packit ae235b
static gint count;
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
test_func (gpointer data)
Packit ae235b
{
Packit ae235b
  gint64 current_time;
Packit ae235b
Packit ae235b
  current_time = g_get_monotonic_time ();
Packit ae235b
Packit ae235b
  /* We accept 2 on the first iteration because _add_seconds() can
Packit ae235b
   * have an initial latency of 1 second, see its documentation.
Packit ae235b
   */
Packit ae235b
  if (count == 0)
Packit ae235b
    g_assert (current_time / 1000000 - last_time / 1000000 <= 2);
Packit ae235b
  else
Packit ae235b
    g_assert (current_time / 1000000 - last_time / 1000000 == 1);
Packit ae235b
Packit ae235b
  last_time = current_time;
Packit ae235b
  count++;
Packit ae235b
Packit ae235b
  /* Make the timeout take up to 0.1 seconds.
Packit ae235b
   * We should still get scheduled for the next second.
Packit ae235b
   */
Packit ae235b
  g_usleep (count * 10000);
Packit ae235b
Packit ae235b
  if (count < 10)
Packit ae235b
    return TRUE;
Packit ae235b
Packit ae235b
  g_main_loop_quit (loop);
Packit ae235b
Packit ae235b
  return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_rounding (void)
Packit ae235b
{
Packit ae235b
  loop = g_main_loop_new (NULL, FALSE);
Packit ae235b
Packit ae235b
  last_time = g_get_monotonic_time ();
Packit ae235b
  g_timeout_add_seconds (1, test_func, NULL);
Packit ae235b
Packit ae235b
  g_main_loop_run (loop);
Packit ae235b
  g_main_loop_unref (loop);
Packit ae235b
}
Packit ae235b
Packit ae235b
int
Packit ae235b
main (int argc, char *argv[])
Packit ae235b
{
Packit ae235b
  g_test_init (&argc, &argv, NULL);
Packit ae235b
Packit ae235b
  g_test_add_func ("/timeout/seconds", test_seconds);
Packit ae235b
  g_test_add_func ("/timeout/rounding", test_rounding);
Packit ae235b
Packit ae235b
  return g_test_run ();
Packit ae235b
}