mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-26 05:56:14 +01:00
d9b23c5466
There’s no reason that anyone can think of that this should be disallowed. It’s useful for language runtimes like GJS to be able to find out the allocation size of dynamic GObjects. Signed-off-by: Philip Withnall <pwithnall@endlessos.org> Fixes: #623
204 lines
5.3 KiB
C
204 lines
5.3 KiB
C
/* GObject - GLib Type, Object, Parameter and Signal Library
|
|
* Copyright (C) 2001, 2003 Red Hat, Inc.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*
|
|
* 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.1 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 <glib-object.h>
|
|
|
|
#include "testcommon.h"
|
|
#include "testmodule.h"
|
|
|
|
/* This test tests the macros for defining dynamic types */
|
|
|
|
static gboolean loaded = FALSE;
|
|
|
|
struct _TestIfaceClass
|
|
{
|
|
GTypeInterface base_iface;
|
|
guint val;
|
|
};
|
|
|
|
static GType test_iface_get_type (void);
|
|
|
|
#define TEST_TYPE_IFACE (test_iface_get_type ())
|
|
#define TEST_IFACE_GET_CLASS(obj) \
|
|
(G_TYPE_INSTANCE_GET_INTERFACE ((obj), TEST_TYPE_IFACE, TestIfaceClass))
|
|
|
|
typedef struct _TestIface TestIface;
|
|
typedef struct _TestIfaceClass TestIfaceClass;
|
|
|
|
static void test_iface_base_init (TestIfaceClass *iface);
|
|
static void test_iface_default_init (TestIfaceClass *iface, gpointer class_data);
|
|
|
|
static DEFINE_IFACE(TestIface, test_iface, test_iface_base_init, test_iface_default_init)
|
|
|
|
static void
|
|
test_iface_default_init (TestIfaceClass *iface,
|
|
gpointer class_data)
|
|
{
|
|
}
|
|
|
|
static void
|
|
test_iface_base_init (TestIfaceClass *iface)
|
|
{
|
|
}
|
|
|
|
GType dynamic_object_get_type (void);
|
|
#define DYNAMIC_OBJECT_TYPE (dynamic_object_get_type ())
|
|
|
|
typedef GObject DynamicObject;
|
|
typedef struct _DynamicObjectClass DynamicObjectClass;
|
|
|
|
struct _DynamicObjectClass
|
|
{
|
|
GObjectClass parent_class;
|
|
guint val;
|
|
};
|
|
|
|
static void dynamic_object_iface_init (TestIface *iface);
|
|
|
|
G_DEFINE_DYNAMIC_TYPE_EXTENDED(DynamicObject, dynamic_object, G_TYPE_OBJECT, 0,
|
|
G_IMPLEMENT_INTERFACE_DYNAMIC (TEST_TYPE_IFACE,
|
|
dynamic_object_iface_init));
|
|
|
|
static void
|
|
dynamic_object_class_init (DynamicObjectClass *class)
|
|
{
|
|
class->val = 42;
|
|
loaded = TRUE;
|
|
}
|
|
|
|
static void
|
|
dynamic_object_class_finalize (DynamicObjectClass *class)
|
|
{
|
|
loaded = FALSE;
|
|
}
|
|
|
|
static void
|
|
dynamic_object_iface_init (TestIface *iface)
|
|
{
|
|
}
|
|
|
|
static void
|
|
dynamic_object_init (DynamicObject *dynamic_object)
|
|
{
|
|
}
|
|
|
|
static void
|
|
module_register (GTypeModule *module)
|
|
{
|
|
dynamic_object_register_type (module);
|
|
}
|
|
|
|
static void
|
|
test_dynamic_type (void)
|
|
{
|
|
DynamicObjectClass *class;
|
|
|
|
/* Not loaded until we call ref for the first time */
|
|
class = g_type_class_peek (DYNAMIC_OBJECT_TYPE);
|
|
g_assert_null (class);
|
|
g_assert_false (loaded);
|
|
|
|
/* Make sure interfaces work */
|
|
g_assert_true (g_type_is_a (DYNAMIC_OBJECT_TYPE,
|
|
TEST_TYPE_IFACE));
|
|
|
|
/* Ref loads */
|
|
class = g_type_class_ref (DYNAMIC_OBJECT_TYPE);
|
|
g_assert_nonnull (class);
|
|
g_assert_cmpint (class->val, ==, 42);
|
|
g_assert_true (loaded);
|
|
|
|
/* Peek then works */
|
|
class = g_type_class_peek (DYNAMIC_OBJECT_TYPE);
|
|
g_assert_nonnull (class);
|
|
g_assert_cmpint (class->val, ==, 42);
|
|
g_assert_true (loaded);
|
|
|
|
/* Make sure interfaces still work */
|
|
g_assert_true (g_type_is_a (DYNAMIC_OBJECT_TYPE,
|
|
TEST_TYPE_IFACE));
|
|
|
|
/* Unref causes finalize */
|
|
g_type_class_unref (class);
|
|
|
|
/* Peek returns NULL */
|
|
class = g_type_class_peek (DYNAMIC_OBJECT_TYPE);
|
|
#if 0
|
|
/* Disabled as unloading dynamic types is disabled.
|
|
* See https://gitlab.gnome.org/GNOME/glib/-/issues/667 */
|
|
g_assert_false (class);
|
|
g_assert_false (loaded);
|
|
#endif
|
|
|
|
/* Ref reloads */
|
|
class = g_type_class_ref (DYNAMIC_OBJECT_TYPE);
|
|
g_assert_nonnull (class);
|
|
g_assert_cmpint (class->val, ==, 42);
|
|
g_assert_true (loaded);
|
|
|
|
/* And Unref causes finalize once more*/
|
|
g_type_class_unref (class);
|
|
class = g_type_class_peek (DYNAMIC_OBJECT_TYPE);
|
|
#if 0
|
|
/* Disabled as unloading dynamic types is disabled.
|
|
* See https://gitlab.gnome.org/GNOME/glib/-/issues/667 */
|
|
g_assert_null (class);
|
|
g_assert_false (loaded);
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
test_dynamic_type_query (void)
|
|
{
|
|
DynamicObjectClass *class;
|
|
GTypeQuery query_result;
|
|
|
|
g_test_bug ("https://gitlab.gnome.org/GNOME/glib/-/issues/623");
|
|
|
|
class = g_type_class_ref (DYNAMIC_OBJECT_TYPE);
|
|
g_assert_nonnull (class);
|
|
|
|
g_type_query (DYNAMIC_OBJECT_TYPE, &query_result);
|
|
|
|
g_assert_cmpuint (query_result.type, !=, 0);
|
|
g_assert_cmpstr (query_result.type_name, ==, "DynamicObject");
|
|
g_assert_cmpuint (query_result.class_size, >=, sizeof (DynamicObjectClass));
|
|
g_assert_cmpuint (query_result.instance_size, >=, sizeof (DynamicObject));
|
|
|
|
g_type_class_unref (class);
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
|
|
G_LOG_LEVEL_WARNING |
|
|
G_LOG_LEVEL_CRITICAL);
|
|
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
test_module_new (module_register);
|
|
|
|
g_test_add_func ("/gobject/dynamic-type", test_dynamic_type);
|
|
g_test_add_func ("/gobject/dynamic-type/query", test_dynamic_type_query);
|
|
|
|
return g_test_run ();
|
|
}
|