Add a detailed test case for interface initialization, testing the ability
authorOwen Taylor <otaylor@redhat.com>
Wed, 27 Aug 2003 05:28:39 +0000 (05:28 +0000)
committerOwen Taylor <otaylor@src.gnome.org>
Wed, 27 Aug 2003 05:28:39 +0000 (05:28 +0000)
Wed Aug 27 01:25:40 2003  Owen Taylor  <otaylor@redhat.com>

        * Makefile.am testifaceinit.c: Add a detailed test case
        for interface initialization, testing the ability to
        add interfaces during class initialization and the ordering
        of interface base_init, class init, and interface_init.
        (Expected to fail at the moment.)

gobject/ChangeLog
gobject/Makefile.am
gobject/testifaceinit.c [new file with mode: 0644]
tests/gobject/ifaceinit.c [new file with mode: 0644]

index e59dbfa..e9d7bbf 100644 (file)
@@ -1,3 +1,11 @@
+Wed Aug 27 01:25:40 2003  Owen Taylor  <otaylor@redhat.com>
+
+       * Makefile.am testifaceinit.c: Add a detailed test case
+       for interface initialization, testing the ability to
+       add interfaces during class initialization and the ordering
+       of interface base_init, class init, and interface_init.
+       (Expected to fail at the moment.)
+
 Mon Aug 25 14:51:46 2003  Manish Singh  <yosh@gimp.org>
 
        * gtypemodule.c (g_type_module_register_type): fix typo in last
index 3f299c4..200ad0f 100644 (file)
@@ -177,17 +177,19 @@ libgobject_2_0_la_SOURCES = $(gobject_target_sources)
 #
 bin_PROGRAMS = gobject-query glib-genmarshal
 bin_SCRIPTS = glib-mkenums
-noinst_PROGRAMS =  testgobject testoverride
+noinst_PROGRAMS = testgobject testifaceinit testoverride
 # source files
 gobject_query_SOURCES = gobject-query.c
 glib_genmarshal_SOURCES = glib-genmarshal.c
 testgobject_SOURCES = testgobject.c
+testifaceinit_SOURCES = testifaceinit.c
 testoverride_SOURCES = testoverride.c
 # link programs against libgobject
 progs_LDADD = ./libgobject-2.0.la $(libglib)
 glib_genmarshal_LDADD = $(libglib)
 gobject_query_LDADD = $(progs_LDADD)
 testgobject_LDADD = $(progs_LDADD)
+testifaceinit_LDADD = $(progs_LDADD)
 testoverride_LDADD = $(progs_LDADD)
 
 #
diff --git a/gobject/testifaceinit.c b/gobject/testifaceinit.c
new file mode 100644 (file)
index 0000000..94154ce
--- /dev/null
@@ -0,0 +1,428 @@
+/* GObject - GLib Type, Object, Parameter and Signal Library
+ * Copyright (C) 2001, 2003 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#undef G_LOG_DOMAIN
+#define        G_LOG_DOMAIN "TestIfaceInit"
+#include       <glib-object.h>
+
+/* What this test tests is the ability to add interfaces dynamically; in
+ * particular adding interfaces to a class while that class is being
+ * initialized.
+ *
+ * The test defines 5 interfaces:
+ * 
+ * - TestIface1 is added before the class is initialized
+ * - TestIface2 is added in base_object_baes_init()
+ * - TestIface3 is added in test_iface1_base_init()
+ * - TestIface4 is added in test_object_class_init()
+ * - TestIface5 is added in test_object_test_iface1_init()
+ * - TestIface6 is added after the class is initialized
+ */
+
+#define DEFINE_TYPE(name, prefix,                              \
+                   class_init, base_init, instance_init,       \
+                   parent_type)                                \
+GType                                                          \
+prefix ## _get_type (void)                                     \
+{                                                              \
+  static GType object_type = 0;                                        \
+                                                               \
+  if (!object_type)                                            \
+    {                                                          \
+      static const GTypeInfo object_info =                     \
+       {                                                       \
+         sizeof (name ## Class),                               \
+         (GBaseInitFunc) base_init,                            \
+         (GBaseFinalizeFunc) NULL,                             \
+         (GClassInitFunc) class_init,                          \
+         (GClassFinalizeFunc) NULL,                            \
+         NULL,           /* class_data */                      \
+         sizeof (name),                                        \
+         0,             /* n_prelocs */                        \
+         (GInstanceInitFunc) instance_init                     \
+       };                                                      \
+                                                               \
+      object_type = g_type_register_static (parent_type,       \
+                                           # name,             \
+                                           &object_info, 0);   \
+    }                                                          \
+                                                               \
+  return object_type;                                          \
+}
+
+#define DEFINE_IFACE(name, prefix, base_init)                  \
+GType                                                          \
+prefix ## _get_type (void)                                     \
+{                                                              \
+  static GType iface_type = 0;                                 \
+                                                               \
+  if (!iface_type)                                             \
+    {                                                          \
+      static const GTypeInfo iface_info =                      \
+      {                                                                \
+       sizeof (name ## Class),                                 \
+       (GBaseInitFunc) base_init,                              \
+       (GBaseFinalizeFunc) NULL,                               \
+      };                                                       \
+                                                               \
+      iface_type = g_type_register_static (G_TYPE_INTERFACE,   \
+                                           # name,             \
+                                           &iface_info, 0);    \
+    }                                                          \
+  return iface_type;                                           \
+}
+
+/* All 6 interfaces actually share the same class structure, though
+ * we use separate typedefs
+ */
+typedef struct _TestIfaceClass TestIfaceClass;
+
+struct _TestIfaceClass
+{
+  GTypeInterface base_iface;
+  guint val;
+  guint base_val;
+};
+
+#define TEST_TYPE_IFACE1           (test_iface1_get_type ())
+#define TEST_IFACE1_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE1, TestIface1Class))
+typedef struct _TestIface1      TestIface1;
+typedef struct _TestIfaceClass  TestIface1Class;
+
+static void test_iface1_base_init (TestIface1Class *iface);
+
+static DEFINE_IFACE(TestIface1, test_iface1, test_iface1_base_init)
+
+#define TEST_TYPE_IFACE2           (test_iface2_get_type ())
+#define TEST_IFACE2_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE2, TestIface2Class))
+typedef struct _TestIface2      TestIface2;
+typedef struct _TestIfaceClass  TestIface2Class;
+
+static void test_iface2_base_init (TestIface2Class *iface);
+
+static DEFINE_IFACE(TestIface2, test_iface2, test_iface2_base_init)
+
+#define TEST_TYPE_IFACE3           (test_iface3_get_type ())
+#define TEST_IFACE3_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE3, TestIface3Class))
+typedef struct _TestIface3      TestIface3;
+typedef struct _TestIfaceClass  TestIface3Class;
+
+static void  test_iface3_base_init (TestIface3Class *iface);
+
+static DEFINE_IFACE(TestIface3, test_iface3, test_iface3_base_init)
+
+#define TEST_TYPE_IFACE4           (test_iface4_get_type ())
+#define TEST_IFACE4_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE4, TestIface4Class))
+typedef struct _TestIface4      TestIface4;
+typedef struct _TestIfaceClass  TestIface4Class;
+
+static void  test_iface4_base_init (TestIface4Class *iface);
+
+static DEFINE_IFACE(TestIface4, test_iface4, test_iface4_base_init)
+
+#define TEST_TYPE_IFACE5           (test_iface5_get_type ())
+#define TEST_IFACE5_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE5, TestIface5Class))
+typedef struct _TestIface5      TestIface5;
+typedef struct _TestIfaceClass  TestIface5Class;
+
+static void  test_iface5_base_init (TestIface5Class *iface);
+
+static DEFINE_IFACE(TestIface5, test_iface5, test_iface5_base_init)
+
+#define TEST_TYPE_IFACE6           (test_iface6_get_type ())
+#define TEST_IFACE6_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE6, TestIface6Class))
+typedef struct _TestIface6      TestIface6;
+typedef struct _TestIfaceClass  TestIface6Class;
+
+static void  test_iface6_base_init (TestIface6Class *iface);
+
+static DEFINE_IFACE(TestIface6, test_iface6, test_iface6_base_init)
+
+/*
+ * BaseObject, a parent class for TestObject
+ */
+#define BASE_TYPE_OBJECT          (base_object_get_type ())
+typedef struct _BaseObject        BaseObject;
+typedef struct _BaseObjectClass   BaseObjectClass;
+
+struct _BaseObject
+{
+  GObject parent_instance;
+};
+struct _BaseObjectClass
+{
+  GObjectClass parent_class;
+};
+
+/*
+ * TestObject, a parent class for TestObject
+ */
+#define TEST_TYPE_OBJECT          (test_object_get_type ())
+typedef struct _TestObject        TestObject;
+typedef struct _TestObjectClass   TestObjectClass;
+
+struct _TestObject
+{
+  BaseObject parent_instance;
+};
+struct _TestObjectClass
+{
+  BaseObjectClass parent_class;
+};
+
+#define TEST_CALLED_ONCE() G_STMT_START { \
+  static gboolean called = FALSE;         \
+  g_assert (!called);                     \
+  called = TRUE;                          \
+} G_STMT_END
+
+#define ADD_IFACE(n)  G_STMT_START {                           \
+  static GInterfaceInfo iface_info = {                         \
+    (GInterfaceInitFunc)test_object_test_iface##n##_init,      \
+    NULL, NULL };                                              \
+                                                               \
+  g_type_add_interface_static (TEST_TYPE_OBJECT,               \
+                              test_iface##n##_get_type (),     \
+                              &iface_info);                    \
+                                                               \
+} G_STMT_END
+
+static gboolean base1, base2, base3, base4, base5, base6;
+static gboolean iface1, iface2, iface3, iface4, iface5, iface6;
+
+static void test_object_test_iface1_init (TestIface1Class *iface);
+static void test_object_test_iface2_init (TestIface1Class *iface);
+static void test_object_test_iface3_init (TestIface3Class *iface);
+static void test_object_test_iface4_init (TestIface4Class *iface);
+static void test_object_test_iface5_init (TestIface5Class *iface);
+static void test_object_test_iface6_init (TestIface6Class *iface);
+
+static GType test_object_get_type (void);
+
+static void
+test_object_test_iface1_init (TestIface1Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x10001;
+
+  ADD_IFACE(5);
+
+  iface1 = TRUE;
+}
+
+static void
+test_object_test_iface2_init (TestIface2Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x20002;
+  
+  iface2 = TRUE;
+}
+
+static void
+test_object_test_iface3_init (TestIface3Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x30003;
+  
+  iface3 = TRUE;
+}
+
+static void
+test_object_test_iface4_init (TestIface4Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x40004;
+  
+  iface4 = TRUE;
+}
+
+static void
+test_object_test_iface5_init (TestIface5Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x50005;
+  
+  iface5 = TRUE;
+}
+
+static void
+test_object_test_iface6_init (TestIface6Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x60006;
+  
+  iface6 = TRUE;
+}
+
+static void
+test_iface1_base_init (TestIface1Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x110011;
+  
+  ADD_IFACE(3);
+  
+  base1 = TRUE;
+}
+
+static void
+test_iface2_base_init (TestIface2Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x220022;
+  
+  base2 = TRUE;
+}
+
+static void
+test_iface3_base_init (TestIface3Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x330033;
+  
+  base3 = TRUE;
+}
+
+static void
+test_iface4_base_init (TestIface4Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x440044;
+
+  base4 = TRUE;
+}
+
+static void
+test_iface5_base_init (TestIface5Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x550055;
+
+  base5 = TRUE;
+}
+
+static void
+test_iface6_base_init (TestIface6Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x660066;
+  
+  base6 = TRUE;
+}
+
+static void
+base_object_base_init (BaseObjectClass *class)
+{
+  static int n_called = 0;
+  n_called++;
+  
+  /* The second time this is called is for TestObject */
+  if (n_called == 2)
+    {
+      ADD_IFACE(2);
+      
+      /* No interface base init functions should have been called yet
+       */
+      g_assert (!base1 && !base2 && !base3 && !base4 && !base5 && !base6);
+      g_assert (!iface1 && !iface2 && !iface3 && !iface4 && !iface5 && !iface6);
+    }
+}
+
+static void
+test_object_class_init (TestObjectClass *class)
+{
+  ADD_IFACE(4);
+
+  /* At this point, the base init functions for all interfaces that have
+   * been added should be called, but no interface init functions.
+   */
+  g_assert (base1 && base2 && base3 && base4 && !base5 && !base6);
+  g_assert (!iface1 && !iface2 && !iface3 && !iface4 && !iface5 && !iface6);
+}
+
+static DEFINE_TYPE(BaseObject, base_object,
+                  NULL, base_object_base_init, NULL,
+                  G_TYPE_OBJECT);
+static DEFINE_TYPE(TestObject, test_object,
+                  test_object_class_init, NULL, NULL,
+                  BASE_TYPE_OBJECT);
+
+int
+main (int   argc,
+      char *argv[])
+{
+  TestObject *object;
+  TestObjectClass *object_class;
+  TestIfaceClass *iface;
+       
+  g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
+                         G_LOG_LEVEL_WARNING |
+                         G_LOG_LEVEL_CRITICAL);
+  g_type_init ();
+
+  /* We force the interfaces to be registered in a different order
+   * than we add them, so our logic doesn't always deal with interfaces
+   * added at the end.
+   */
+  (void)TEST_TYPE_IFACE4;
+  (void)TEST_TYPE_IFACE2;
+  (void)TEST_TYPE_IFACE6;
+  (void)TEST_TYPE_IFACE5;
+  (void)TEST_TYPE_IFACE3;
+  (void)TEST_TYPE_IFACE1;
+
+  ADD_IFACE(1);
+
+  object_class = g_type_class_ref (TEST_TYPE_OBJECT);
+
+  ADD_IFACE(6);
+
+  /* All base and interface init functions should have been called
+   */
+  g_assert (base1 && base2 && base3 && base4 && base5 && base6);
+  g_assert (iface1 && iface2 && iface3 && iface4 && iface5 && iface6);
+  
+  object = g_object_new (TEST_TYPE_OBJECT, NULL);
+
+  iface = TEST_IFACE1_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x10001 && iface->base_val == 0x110011);
+  iface = TEST_IFACE3_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x30003 && iface->base_val == 0x330033);
+  iface = TEST_IFACE4_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x40004 && iface->base_val == 0x440044);
+  iface = TEST_IFACE5_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x50005 && iface->base_val == 0x550055);
+  iface = TEST_IFACE6_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x60006 && iface->base_val == 0x660066);
+
+  return 0;
+}
diff --git a/tests/gobject/ifaceinit.c b/tests/gobject/ifaceinit.c
new file mode 100644 (file)
index 0000000..94154ce
--- /dev/null
@@ -0,0 +1,428 @@
+/* GObject - GLib Type, Object, Parameter and Signal Library
+ * Copyright (C) 2001, 2003 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#undef G_LOG_DOMAIN
+#define        G_LOG_DOMAIN "TestIfaceInit"
+#include       <glib-object.h>
+
+/* What this test tests is the ability to add interfaces dynamically; in
+ * particular adding interfaces to a class while that class is being
+ * initialized.
+ *
+ * The test defines 5 interfaces:
+ * 
+ * - TestIface1 is added before the class is initialized
+ * - TestIface2 is added in base_object_baes_init()
+ * - TestIface3 is added in test_iface1_base_init()
+ * - TestIface4 is added in test_object_class_init()
+ * - TestIface5 is added in test_object_test_iface1_init()
+ * - TestIface6 is added after the class is initialized
+ */
+
+#define DEFINE_TYPE(name, prefix,                              \
+                   class_init, base_init, instance_init,       \
+                   parent_type)                                \
+GType                                                          \
+prefix ## _get_type (void)                                     \
+{                                                              \
+  static GType object_type = 0;                                        \
+                                                               \
+  if (!object_type)                                            \
+    {                                                          \
+      static const GTypeInfo object_info =                     \
+       {                                                       \
+         sizeof (name ## Class),                               \
+         (GBaseInitFunc) base_init,                            \
+         (GBaseFinalizeFunc) NULL,                             \
+         (GClassInitFunc) class_init,                          \
+         (GClassFinalizeFunc) NULL,                            \
+         NULL,           /* class_data */                      \
+         sizeof (name),                                        \
+         0,             /* n_prelocs */                        \
+         (GInstanceInitFunc) instance_init                     \
+       };                                                      \
+                                                               \
+      object_type = g_type_register_static (parent_type,       \
+                                           # name,             \
+                                           &object_info, 0);   \
+    }                                                          \
+                                                               \
+  return object_type;                                          \
+}
+
+#define DEFINE_IFACE(name, prefix, base_init)                  \
+GType                                                          \
+prefix ## _get_type (void)                                     \
+{                                                              \
+  static GType iface_type = 0;                                 \
+                                                               \
+  if (!iface_type)                                             \
+    {                                                          \
+      static const GTypeInfo iface_info =                      \
+      {                                                                \
+       sizeof (name ## Class),                                 \
+       (GBaseInitFunc) base_init,                              \
+       (GBaseFinalizeFunc) NULL,                               \
+      };                                                       \
+                                                               \
+      iface_type = g_type_register_static (G_TYPE_INTERFACE,   \
+                                           # name,             \
+                                           &iface_info, 0);    \
+    }                                                          \
+  return iface_type;                                           \
+}
+
+/* All 6 interfaces actually share the same class structure, though
+ * we use separate typedefs
+ */
+typedef struct _TestIfaceClass TestIfaceClass;
+
+struct _TestIfaceClass
+{
+  GTypeInterface base_iface;
+  guint val;
+  guint base_val;
+};
+
+#define TEST_TYPE_IFACE1           (test_iface1_get_type ())
+#define TEST_IFACE1_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE1, TestIface1Class))
+typedef struct _TestIface1      TestIface1;
+typedef struct _TestIfaceClass  TestIface1Class;
+
+static void test_iface1_base_init (TestIface1Class *iface);
+
+static DEFINE_IFACE(TestIface1, test_iface1, test_iface1_base_init)
+
+#define TEST_TYPE_IFACE2           (test_iface2_get_type ())
+#define TEST_IFACE2_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE2, TestIface2Class))
+typedef struct _TestIface2      TestIface2;
+typedef struct _TestIfaceClass  TestIface2Class;
+
+static void test_iface2_base_init (TestIface2Class *iface);
+
+static DEFINE_IFACE(TestIface2, test_iface2, test_iface2_base_init)
+
+#define TEST_TYPE_IFACE3           (test_iface3_get_type ())
+#define TEST_IFACE3_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE3, TestIface3Class))
+typedef struct _TestIface3      TestIface3;
+typedef struct _TestIfaceClass  TestIface3Class;
+
+static void  test_iface3_base_init (TestIface3Class *iface);
+
+static DEFINE_IFACE(TestIface3, test_iface3, test_iface3_base_init)
+
+#define TEST_TYPE_IFACE4           (test_iface4_get_type ())
+#define TEST_IFACE4_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE4, TestIface4Class))
+typedef struct _TestIface4      TestIface4;
+typedef struct _TestIfaceClass  TestIface4Class;
+
+static void  test_iface4_base_init (TestIface4Class *iface);
+
+static DEFINE_IFACE(TestIface4, test_iface4, test_iface4_base_init)
+
+#define TEST_TYPE_IFACE5           (test_iface5_get_type ())
+#define TEST_IFACE5_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE5, TestIface5Class))
+typedef struct _TestIface5      TestIface5;
+typedef struct _TestIfaceClass  TestIface5Class;
+
+static void  test_iface5_base_init (TestIface5Class *iface);
+
+static DEFINE_IFACE(TestIface5, test_iface5, test_iface5_base_init)
+
+#define TEST_TYPE_IFACE6           (test_iface6_get_type ())
+#define TEST_IFACE6_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE6, TestIface6Class))
+typedef struct _TestIface6      TestIface6;
+typedef struct _TestIfaceClass  TestIface6Class;
+
+static void  test_iface6_base_init (TestIface6Class *iface);
+
+static DEFINE_IFACE(TestIface6, test_iface6, test_iface6_base_init)
+
+/*
+ * BaseObject, a parent class for TestObject
+ */
+#define BASE_TYPE_OBJECT          (base_object_get_type ())
+typedef struct _BaseObject        BaseObject;
+typedef struct _BaseObjectClass   BaseObjectClass;
+
+struct _BaseObject
+{
+  GObject parent_instance;
+};
+struct _BaseObjectClass
+{
+  GObjectClass parent_class;
+};
+
+/*
+ * TestObject, a parent class for TestObject
+ */
+#define TEST_TYPE_OBJECT          (test_object_get_type ())
+typedef struct _TestObject        TestObject;
+typedef struct _TestObjectClass   TestObjectClass;
+
+struct _TestObject
+{
+  BaseObject parent_instance;
+};
+struct _TestObjectClass
+{
+  BaseObjectClass parent_class;
+};
+
+#define TEST_CALLED_ONCE() G_STMT_START { \
+  static gboolean called = FALSE;         \
+  g_assert (!called);                     \
+  called = TRUE;                          \
+} G_STMT_END
+
+#define ADD_IFACE(n)  G_STMT_START {                           \
+  static GInterfaceInfo iface_info = {                         \
+    (GInterfaceInitFunc)test_object_test_iface##n##_init,      \
+    NULL, NULL };                                              \
+                                                               \
+  g_type_add_interface_static (TEST_TYPE_OBJECT,               \
+                              test_iface##n##_get_type (),     \
+                              &iface_info);                    \
+                                                               \
+} G_STMT_END
+
+static gboolean base1, base2, base3, base4, base5, base6;
+static gboolean iface1, iface2, iface3, iface4, iface5, iface6;
+
+static void test_object_test_iface1_init (TestIface1Class *iface);
+static void test_object_test_iface2_init (TestIface1Class *iface);
+static void test_object_test_iface3_init (TestIface3Class *iface);
+static void test_object_test_iface4_init (TestIface4Class *iface);
+static void test_object_test_iface5_init (TestIface5Class *iface);
+static void test_object_test_iface6_init (TestIface6Class *iface);
+
+static GType test_object_get_type (void);
+
+static void
+test_object_test_iface1_init (TestIface1Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x10001;
+
+  ADD_IFACE(5);
+
+  iface1 = TRUE;
+}
+
+static void
+test_object_test_iface2_init (TestIface2Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x20002;
+  
+  iface2 = TRUE;
+}
+
+static void
+test_object_test_iface3_init (TestIface3Class *iface)
+{
+  TEST_CALLED_ONCE();
+  
+  iface->val = 0x30003;
+  
+  iface3 = TRUE;
+}
+
+static void
+test_object_test_iface4_init (TestIface4Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x40004;
+  
+  iface4 = TRUE;
+}
+
+static void
+test_object_test_iface5_init (TestIface5Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x50005;
+  
+  iface5 = TRUE;
+}
+
+static void
+test_object_test_iface6_init (TestIface6Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->val = 0x60006;
+  
+  iface6 = TRUE;
+}
+
+static void
+test_iface1_base_init (TestIface1Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x110011;
+  
+  ADD_IFACE(3);
+  
+  base1 = TRUE;
+}
+
+static void
+test_iface2_base_init (TestIface2Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x220022;
+  
+  base2 = TRUE;
+}
+
+static void
+test_iface3_base_init (TestIface3Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x330033;
+  
+  base3 = TRUE;
+}
+
+static void
+test_iface4_base_init (TestIface4Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x440044;
+
+  base4 = TRUE;
+}
+
+static void
+test_iface5_base_init (TestIface5Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x550055;
+
+  base5 = TRUE;
+}
+
+static void
+test_iface6_base_init (TestIface6Class *iface)
+{
+  TEST_CALLED_ONCE();
+
+  iface->base_val = 0x660066;
+  
+  base6 = TRUE;
+}
+
+static void
+base_object_base_init (BaseObjectClass *class)
+{
+  static int n_called = 0;
+  n_called++;
+  
+  /* The second time this is called is for TestObject */
+  if (n_called == 2)
+    {
+      ADD_IFACE(2);
+      
+      /* No interface base init functions should have been called yet
+       */
+      g_assert (!base1 && !base2 && !base3 && !base4 && !base5 && !base6);
+      g_assert (!iface1 && !iface2 && !iface3 && !iface4 && !iface5 && !iface6);
+    }
+}
+
+static void
+test_object_class_init (TestObjectClass *class)
+{
+  ADD_IFACE(4);
+
+  /* At this point, the base init functions for all interfaces that have
+   * been added should be called, but no interface init functions.
+   */
+  g_assert (base1 && base2 && base3 && base4 && !base5 && !base6);
+  g_assert (!iface1 && !iface2 && !iface3 && !iface4 && !iface5 && !iface6);
+}
+
+static DEFINE_TYPE(BaseObject, base_object,
+                  NULL, base_object_base_init, NULL,
+                  G_TYPE_OBJECT);
+static DEFINE_TYPE(TestObject, test_object,
+                  test_object_class_init, NULL, NULL,
+                  BASE_TYPE_OBJECT);
+
+int
+main (int   argc,
+      char *argv[])
+{
+  TestObject *object;
+  TestObjectClass *object_class;
+  TestIfaceClass *iface;
+       
+  g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
+                         G_LOG_LEVEL_WARNING |
+                         G_LOG_LEVEL_CRITICAL);
+  g_type_init ();
+
+  /* We force the interfaces to be registered in a different order
+   * than we add them, so our logic doesn't always deal with interfaces
+   * added at the end.
+   */
+  (void)TEST_TYPE_IFACE4;
+  (void)TEST_TYPE_IFACE2;
+  (void)TEST_TYPE_IFACE6;
+  (void)TEST_TYPE_IFACE5;
+  (void)TEST_TYPE_IFACE3;
+  (void)TEST_TYPE_IFACE1;
+
+  ADD_IFACE(1);
+
+  object_class = g_type_class_ref (TEST_TYPE_OBJECT);
+
+  ADD_IFACE(6);
+
+  /* All base and interface init functions should have been called
+   */
+  g_assert (base1 && base2 && base3 && base4 && base5 && base6);
+  g_assert (iface1 && iface2 && iface3 && iface4 && iface5 && iface6);
+  
+  object = g_object_new (TEST_TYPE_OBJECT, NULL);
+
+  iface = TEST_IFACE1_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x10001 && iface->base_val == 0x110011);
+  iface = TEST_IFACE3_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x30003 && iface->base_val == 0x330033);
+  iface = TEST_IFACE4_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x40004 && iface->base_val == 0x440044);
+  iface = TEST_IFACE5_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x50005 && iface->base_val == 0x550055);
+  iface = TEST_IFACE6_GET_CLASS (object);
+  g_assert (iface && iface->val == 0x60006 && iface->base_val == 0x660066);
+
+  return 0;
+}