Blame glib/tests/tree.c

Packit ae235b
/* GLIB - Library of useful routines for C programming
Packit ae235b
 * Copyright (C) 1995-1997  Peter Mattis, Spencer Kimball and Josh MacDonald
Packit ae235b
 *
Packit ae235b
 * This library is free software; you can redistribute it and/or
Packit ae235b
 * modify it under the terms of the GNU Lesser General Public
Packit ae235b
 * License as published by the Free Software Foundation; either
Packit ae235b
 * version 2.1 of the License, or (at your option) any later version.
Packit ae235b
 *
Packit ae235b
 * This library is distributed in the hope that it will be useful,
Packit ae235b
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit ae235b
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit ae235b
 * Lesser General Public License for more details.
Packit ae235b
 *
Packit ae235b
 * You should have received a copy of the GNU Lesser General Public
Packit ae235b
 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
Packit ae235b
 */
Packit ae235b
Packit ae235b
/*
Packit ae235b
 * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
Packit ae235b
 * file for a list of people on the GLib Team.  See the ChangeLog
Packit ae235b
 * files for a list of changes.  These files are distributed with
Packit ae235b
 * GLib at ftp://ftp.gtk.org/pub/gtk/.
Packit ae235b
 */
Packit ae235b
Packit ae235b
#undef G_DISABLE_ASSERT
Packit ae235b
#undef G_LOG_DOMAIN
Packit ae235b
Packit ae235b
/* We are testing some deprecated APIs here */
Packit ae235b
#define GLIB_DISABLE_DEPRECATION_WARNINGS
Packit ae235b
Packit ae235b
#include <stdio.h>
Packit ae235b
#include <string.h>
Packit ae235b
#include "glib.h"
Packit ae235b
Packit ae235b
Packit ae235b
static gint
Packit ae235b
my_compare (gconstpointer a,
Packit ae235b
            gconstpointer b)
Packit ae235b
{
Packit ae235b
  const char *cha = a;
Packit ae235b
  const char *chb = b;
Packit ae235b
Packit ae235b
  return *cha - *chb;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gint
Packit ae235b
my_compare_with_data (gconstpointer a,
Packit ae235b
                      gconstpointer b,
Packit ae235b
                      gpointer      user_data)
Packit ae235b
{
Packit ae235b
  const char *cha = a;
Packit ae235b
  const char *chb = b;
Packit ae235b
Packit ae235b
  /* just check that we got the right data */
Packit ae235b
  g_assert (GPOINTER_TO_INT(user_data) == 123);
Packit ae235b
Packit ae235b
  return *cha - *chb;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gint
Packit ae235b
my_search (gconstpointer a,
Packit ae235b
           gconstpointer b)
Packit ae235b
{
Packit ae235b
  return my_compare (b, a);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gpointer destroyed_key = NULL;
Packit ae235b
static gpointer destroyed_value = NULL;
Packit ae235b
Packit ae235b
static void
Packit ae235b
my_key_destroy (gpointer key)
Packit ae235b
{
Packit ae235b
  destroyed_key = key;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
my_value_destroy (gpointer value)
Packit ae235b
{
Packit ae235b
  destroyed_value = value;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gint
Packit ae235b
my_traverse (gpointer key,
Packit ae235b
             gpointer value,
Packit ae235b
             gpointer data)
Packit ae235b
{
Packit ae235b
  char *ch = key;
Packit ae235b
Packit ae235b
  g_assert ((*ch) > 0);
Packit ae235b
Packit ae235b
  if (*ch == 'd')
Packit ae235b
    return TRUE;
Packit ae235b
Packit ae235b
  return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
char chars[] =
Packit ae235b
  "0123456789"
Packit ae235b
  "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
Packit ae235b
  "abcdefghijklmnopqrstuvwxyz";
Packit ae235b
Packit ae235b
char chars2[] =
Packit ae235b
  "0123456789"
Packit ae235b
  "abcdefghijklmnopqrstuvwxyz";
Packit ae235b
Packit ae235b
static gint
Packit ae235b
check_order (gpointer key,
Packit ae235b
             gpointer value,
Packit ae235b
             gpointer data)
Packit ae235b
{
Packit ae235b
  char **p = data;
Packit ae235b
  char *ch = key;
Packit ae235b
 
Packit ae235b
  g_assert (**p == *ch);
Packit ae235b
Packit ae235b
  (*p)++;
Packit ae235b
Packit ae235b
  return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_tree_search (void)
Packit ae235b
{
Packit ae235b
  gint i;
Packit ae235b
  GTree *tree;
Packit ae235b
  gboolean removed;
Packit ae235b
  gchar c;
Packit ae235b
  gchar *p, *d;
Packit ae235b
Packit ae235b
  tree = g_tree_new_with_data (my_compare_with_data, GINT_TO_POINTER(123));
Packit ae235b
Packit ae235b
  for (i = 0; chars[i]; i++)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
Packit ae235b
  g_tree_foreach (tree, my_traverse, NULL);
Packit ae235b
Packit ae235b
  g_assert_cmpint (g_tree_nnodes (tree), ==, strlen (chars));
Packit ae235b
  g_assert_cmpint (g_tree_height (tree), ==, 6);
Packit ae235b
 
Packit ae235b
  p = chars;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  for (i = 0; i < 26; i++)
Packit ae235b
    {
Packit ae235b
      removed = g_tree_remove (tree, &chars[i + 10]);
Packit ae235b
      g_assert (removed);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  c = '\0';
Packit ae235b
  removed = g_tree_remove (tree, &c);
Packit ae235b
  g_assert (!removed);
Packit ae235b
Packit ae235b
  g_tree_foreach (tree, my_traverse, NULL);
Packit ae235b
Packit ae235b
  g_assert_cmpint (g_tree_nnodes (tree), ==, strlen (chars2));
Packit ae235b
  g_assert_cmpint (g_tree_height (tree), ==, 6);
Packit ae235b
Packit ae235b
  p = chars2;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  for (i = 25; i >= 0; i--)
Packit ae235b
    g_tree_insert (tree, &chars[i + 10], &chars[i + 10]);
Packit ae235b
Packit ae235b
  p = chars;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  c = '0';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
  g_assert (g_tree_lookup_extended (tree, &c, (gpointer *)&d, (gpointer *)&p);;
Packit ae235b
  g_assert (c == *d && c == *p);
Packit ae235b
Packit ae235b
  c = 'A';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = 'a';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = 'z';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = '!';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  c = '=';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  c = '|';
Packit ae235b
  p = g_tree_lookup (tree, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  c = '0';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = 'A';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = 'a';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p &&*p == c);
Packit ae235b
Packit ae235b
  c = 'z';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p && *p == c);
Packit ae235b
Packit ae235b
  c = '!';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  c = '=';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  c = '|';
Packit ae235b
  p = g_tree_search (tree, my_search, &c);
Packit ae235b
  g_assert (p == NULL);
Packit ae235b
Packit ae235b
  g_tree_destroy (tree);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_tree_remove (void)
Packit ae235b
{
Packit ae235b
  GTree *tree;
Packit ae235b
  char c, d;
Packit ae235b
  gint i;
Packit ae235b
  gboolean removed;
Packit ae235b
  gchar *remove;
Packit ae235b
Packit ae235b
  tree = g_tree_new_full ((GCompareDataFunc)my_compare, NULL,
Packit ae235b
                          my_key_destroy,
Packit ae235b
                          my_value_destroy);
Packit ae235b
Packit ae235b
  for (i = 0; chars[i]; i++)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
Packit ae235b
  c = '0';
Packit ae235b
  g_tree_insert (tree, &c, &c);
Packit ae235b
  g_assert (destroyed_key == &c);
Packit ae235b
  g_assert (destroyed_value == &chars[0]);
Packit ae235b
  destroyed_key = NULL;
Packit ae235b
  destroyed_value = NULL;
Packit ae235b
Packit ae235b
  d = '1';
Packit ae235b
  g_tree_replace (tree, &d, &d);
Packit ae235b
  g_assert (destroyed_key == &chars[1]);
Packit ae235b
  g_assert (destroyed_value == &chars[1]);
Packit ae235b
  destroyed_key = NULL;
Packit ae235b
  destroyed_value = NULL;
Packit ae235b
Packit ae235b
  c = '2';
Packit ae235b
  removed = g_tree_remove (tree, &c);
Packit ae235b
  g_assert (removed);
Packit ae235b
  g_assert (destroyed_key == &chars[2]);
Packit ae235b
  g_assert (destroyed_value == &chars[2]);
Packit ae235b
  destroyed_key = NULL;
Packit ae235b
  destroyed_value = NULL;
Packit ae235b
Packit ae235b
  c = '3';
Packit ae235b
  removed = g_tree_steal (tree, &c);
Packit ae235b
  g_assert (removed);
Packit ae235b
  g_assert (destroyed_key == NULL);
Packit ae235b
  g_assert (destroyed_value == NULL);
Packit ae235b
Packit ae235b
  remove = "omkjigfedba";
Packit ae235b
  for (i = 0; remove[i]; i++)
Packit ae235b
    {
Packit ae235b
      removed = g_tree_remove (tree, &remove[i]);
Packit ae235b
      g_assert (removed);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  g_tree_destroy (tree);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_tree_destroy (void)
Packit ae235b
{
Packit ae235b
  GTree *tree;
Packit ae235b
  gint i;
Packit ae235b
Packit ae235b
  tree = g_tree_new (my_compare);
Packit ae235b
Packit ae235b
  for (i = 0; chars[i]; i++)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
Packit ae235b
  g_assert_cmpint (g_tree_nnodes (tree), ==, strlen (chars));
Packit ae235b
Packit ae235b
  g_tree_ref (tree);
Packit ae235b
  g_tree_destroy (tree);
Packit ae235b
Packit ae235b
  g_assert_cmpint (g_tree_nnodes (tree), ==, 0);
Packit ae235b
Packit ae235b
  g_tree_unref (tree);
Packit ae235b
}
Packit ae235b
Packit ae235b
typedef struct {
Packit ae235b
  GString *s;
Packit ae235b
  gint count;
Packit ae235b
} CallbackData;
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
traverse_func (gpointer key, gpointer value, gpointer data)
Packit ae235b
{
Packit ae235b
  CallbackData *d = data;
Packit ae235b
Packit ae235b
  gchar *c = value;
Packit ae235b
  g_string_append_c (d->s, *c);
Packit ae235b
Packit ae235b
  d->count--;
Packit ae235b
Packit ae235b
  if (d->count == 0)
Packit ae235b
    return TRUE;
Packit ae235b
Packit ae235b
  return FALSE;
Packit ae235b
}
Packit ae235b
Packit ae235b
typedef struct {
Packit ae235b
  GTraverseType  traverse;
Packit ae235b
  gint           limit;
Packit ae235b
  const gchar   *expected;
Packit ae235b
} TraverseData;
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_tree_traverse (void)
Packit ae235b
{
Packit ae235b
  GTree *tree;
Packit ae235b
  gint i;
Packit ae235b
  TraverseData orders[] = {
Packit ae235b
    { G_IN_ORDER,   -1, "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz" },
Packit ae235b
    { G_IN_ORDER,    1, "0" },
Packit ae235b
    { G_IN_ORDER,    2, "01" },
Packit ae235b
    { G_IN_ORDER,    3, "012" },
Packit ae235b
    { G_IN_ORDER,    4, "0123" },
Packit ae235b
    { G_IN_ORDER,    5, "01234" },
Packit ae235b
    { G_IN_ORDER,    6, "012345" },
Packit ae235b
    { G_IN_ORDER,    7, "0123456" },
Packit ae235b
    { G_IN_ORDER,    8, "01234567" },
Packit ae235b
    { G_IN_ORDER,    9, "012345678" },
Packit ae235b
    { G_IN_ORDER,   10, "0123456789" },
Packit ae235b
    { G_IN_ORDER,   11, "0123456789A" },
Packit ae235b
    { G_IN_ORDER,   12, "0123456789AB" },
Packit ae235b
    { G_IN_ORDER,   13, "0123456789ABC" },
Packit ae235b
    { G_IN_ORDER,   14, "0123456789ABCD" },
Packit ae235b
Packit ae235b
    { G_PRE_ORDER,  -1, "VF73102546B98ADCENJHGILKMRPOQTSUldZXWYbachfegjiktpnmorqsxvuwyz" },
Packit ae235b
    { G_PRE_ORDER,   1, "V" },
Packit ae235b
    { G_PRE_ORDER,   2, "VF" },
Packit ae235b
    { G_PRE_ORDER,   3, "VF7" },
Packit ae235b
    { G_PRE_ORDER,   4, "VF73" },
Packit ae235b
    { G_PRE_ORDER,   5, "VF731" },
Packit ae235b
    { G_PRE_ORDER,   6, "VF7310" },
Packit ae235b
    { G_PRE_ORDER,   7, "VF73102" },
Packit ae235b
    { G_PRE_ORDER,   8, "VF731025" },
Packit ae235b
    { G_PRE_ORDER,   9, "VF7310254" },
Packit ae235b
    { G_PRE_ORDER,  10, "VF73102546" },
Packit ae235b
    { G_PRE_ORDER,  11, "VF73102546B" },
Packit ae235b
    { G_PRE_ORDER,  12, "VF73102546B9" },
Packit ae235b
    { G_PRE_ORDER,  13, "VF73102546B98" },
Packit ae235b
    { G_PRE_ORDER,  14, "VF73102546B98A" },
Packit ae235b
Packit ae235b
    { G_POST_ORDER, -1, "02146538A9CEDB7GIHKMLJOQPSUTRNFWYXacbZegfikjhdmonqsrpuwvzyxtlV" },
Packit ae235b
    { G_POST_ORDER,  1, "0" },
Packit ae235b
    { G_POST_ORDER,  2, "02" },
Packit ae235b
    { G_POST_ORDER,  3, "021" },
Packit ae235b
    { G_POST_ORDER,  4, "0214" },
Packit ae235b
    { G_POST_ORDER,  5, "02146" },
Packit ae235b
    { G_POST_ORDER,  6, "021465" },
Packit ae235b
    { G_POST_ORDER,  7, "0214653" },
Packit ae235b
    { G_POST_ORDER,  8, "02146538" },
Packit ae235b
    { G_POST_ORDER,  9, "02146538A" },
Packit ae235b
    { G_POST_ORDER, 10, "02146538A9" },
Packit ae235b
    { G_POST_ORDER, 11, "02146538A9C" },
Packit ae235b
    { G_POST_ORDER, 12, "02146538A9CE" },
Packit ae235b
    { G_POST_ORDER, 13, "02146538A9CED" },
Packit ae235b
    { G_POST_ORDER, 14, "02146538A9CEDB" }
Packit ae235b
  };
Packit ae235b
  CallbackData data;
Packit ae235b
Packit ae235b
  tree = g_tree_new (my_compare);
Packit ae235b
Packit ae235b
  for (i = 0; chars[i]; i++)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
Packit ae235b
  data.s = g_string_new ("");
Packit ae235b
  for (i = 0; i < G_N_ELEMENTS (orders); i++)
Packit ae235b
    {
Packit ae235b
      g_string_set_size (data.s, 0);
Packit ae235b
      data.count = orders[i].limit;
Packit ae235b
      g_tree_traverse (tree, traverse_func, orders[i].traverse, &data);
Packit ae235b
      g_assert_cmpstr (data.s->str, ==, orders[i].expected);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  g_tree_unref (tree);
Packit ae235b
  g_string_free (data.s, TRUE); 
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_tree_insert (void)
Packit ae235b
{
Packit ae235b
  GTree *tree;
Packit ae235b
  gchar *p;
Packit ae235b
  gint i;
Packit ae235b
  gchar *scrambled;
Packit ae235b
Packit ae235b
  tree = g_tree_new (my_compare);
Packit ae235b
Packit ae235b
  for (i = 0; chars[i]; i++)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
  p = chars;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  g_tree_unref (tree);
Packit ae235b
  tree = g_tree_new (my_compare);
Packit ae235b
Packit ae235b
  for (i = strlen (chars) - 1; i >= 0; i--)
Packit ae235b
    g_tree_insert (tree, &chars[i], &chars[i]);
Packit ae235b
  p = chars;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  g_tree_unref (tree);
Packit ae235b
  tree = g_tree_new (my_compare);
Packit ae235b
Packit ae235b
  scrambled = g_strdup (chars);
Packit ae235b
Packit ae235b
  for (i = 0; i < 30; i++)
Packit ae235b
    {
Packit ae235b
      gchar tmp;
Packit ae235b
      gint a, b;
Packit ae235b
Packit ae235b
      a = g_random_int_range (0, strlen (scrambled));
Packit ae235b
      b = g_random_int_range (0, strlen (scrambled));
Packit ae235b
      tmp = scrambled[a];
Packit ae235b
      scrambled[a] = scrambled[b];
Packit ae235b
      scrambled[b] = tmp;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  for (i = 0; scrambled[i]; i++)
Packit ae235b
    g_tree_insert (tree, &scrambled[i], &scrambled[i]);
Packit ae235b
  p = chars;
Packit ae235b
  g_tree_foreach (tree, check_order, &p);
Packit ae235b
Packit ae235b
  g_free (scrambled);
Packit ae235b
  g_tree_unref (tree);
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 ("/tree/search", test_tree_search);
Packit ae235b
  g_test_add_func ("/tree/remove", test_tree_remove);
Packit ae235b
  g_test_add_func ("/tree/destroy", test_tree_destroy);
Packit ae235b
  g_test_add_func ("/tree/traverse", test_tree_traverse);
Packit ae235b
  g_test_add_func ("/tree/insert", test_tree_insert);
Packit ae235b
Packit ae235b
  return g_test_run ();
Packit ae235b
}
Packit ae235b