mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-01 10:26:13 +01:00
180 lines
4.3 KiB
C
180 lines
4.3 KiB
C
|
/* flags.c
|
||
|
* Copyright (C) 2018 Arthur Demchenkov
|
||
|
*
|
||
|
* 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>
|
||
|
|
||
|
/* Check that validation of flags works on architectures where
|
||
|
* #gint and #glong are different sizes, as the flags are cast
|
||
|
* between types a few times.
|
||
|
*
|
||
|
* See: https://gitlab.gnome.org/GNOME/glib/issues/1572
|
||
|
*/
|
||
|
|
||
|
enum {
|
||
|
PROP_FLAGS = 1
|
||
|
};
|
||
|
|
||
|
typedef struct _GTest GTest;
|
||
|
typedef struct _GTestClass GTestClass;
|
||
|
|
||
|
typedef enum {
|
||
|
NO_FLAG = 0,
|
||
|
LOWEST_FLAG = 1,
|
||
|
HIGHEST_FLAG = 1 << 31
|
||
|
} MyFlagsEnum;
|
||
|
|
||
|
struct _GTest {
|
||
|
GObject object;
|
||
|
MyFlagsEnum flags;
|
||
|
};
|
||
|
|
||
|
struct _GTestClass {
|
||
|
GObjectClass parent_class;
|
||
|
};
|
||
|
|
||
|
static GType my_test_get_type (void);
|
||
|
static GType my_test_flags_get_type (void);
|
||
|
|
||
|
#define G_TYPE_TEST (my_test_get_type())
|
||
|
#define MY_TEST(test) (G_TYPE_CHECK_INSTANCE_CAST ((test), G_TYPE_TEST, GTest))
|
||
|
G_DEFINE_TYPE (GTest, my_test, G_TYPE_OBJECT)
|
||
|
|
||
|
static void my_test_class_init (GTestClass * klass);
|
||
|
static void my_test_init (GTest * test);
|
||
|
static void my_test_get_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
GValue *value,
|
||
|
GParamSpec *pspec);
|
||
|
static void my_test_set_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
const GValue *value,
|
||
|
GParamSpec *pspec);
|
||
|
|
||
|
static GType
|
||
|
my_test_flags_get_type (void)
|
||
|
{
|
||
|
static GType flags_type = 0;
|
||
|
|
||
|
if (G_UNLIKELY(flags_type == 0))
|
||
|
{
|
||
|
static const GFlagsValue values[] = {
|
||
|
{ LOWEST_FLAG, "LOWEST_FLAG", "lowest" },
|
||
|
{ HIGHEST_FLAG, "HIGHEST_FLAG", "highest" },
|
||
|
{ 0, NULL, NULL }
|
||
|
};
|
||
|
|
||
|
flags_type = g_flags_register_static (g_intern_static_string ("GTestFlags"), values);
|
||
|
}
|
||
|
return flags_type;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
my_test_class_init (GTestClass *klass)
|
||
|
{
|
||
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||
|
|
||
|
gobject_class->get_property = my_test_get_property;
|
||
|
gobject_class->set_property = my_test_set_property;
|
||
|
|
||
|
g_object_class_install_property (gobject_class, 1,
|
||
|
g_param_spec_flags ("flags",
|
||
|
"Flags",
|
||
|
"Flags test property",
|
||
|
my_test_flags_get_type(), 0,
|
||
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
|
||
|
}
|
||
|
|
||
|
static void my_test_init (GTest *test)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
my_test_get_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
GValue *value,
|
||
|
GParamSpec *pspec)
|
||
|
{
|
||
|
GTest *test = MY_TEST (object);
|
||
|
|
||
|
switch (prop_id)
|
||
|
{
|
||
|
case PROP_FLAGS:
|
||
|
g_value_set_flags (value, test->flags);
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
my_test_set_property (GObject *object,
|
||
|
guint prop_id,
|
||
|
const GValue *value,
|
||
|
GParamSpec *pspec)
|
||
|
{
|
||
|
GTest *test = MY_TEST (object);
|
||
|
|
||
|
switch (prop_id)
|
||
|
{
|
||
|
case PROP_FLAGS:
|
||
|
test->flags = g_value_get_flags (value);
|
||
|
break;
|
||
|
default:
|
||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
check_flags_validation (void)
|
||
|
{
|
||
|
guint test_flags[] = {
|
||
|
NO_FLAG,
|
||
|
LOWEST_FLAG,
|
||
|
HIGHEST_FLAG,
|
||
|
LOWEST_FLAG | HIGHEST_FLAG
|
||
|
};
|
||
|
guint flag_read;
|
||
|
gsize i;
|
||
|
|
||
|
for (i = 0; i < G_N_ELEMENTS (test_flags); i++)
|
||
|
{
|
||
|
guint flag_set = test_flags[i];
|
||
|
GObject *test = g_object_new (G_TYPE_TEST,
|
||
|
"flags", flag_set,
|
||
|
NULL);
|
||
|
|
||
|
g_object_get (test, "flags", &flag_read, NULL);
|
||
|
|
||
|
/* This check will fail in case of gint -> glong conversion
|
||
|
* in value_flags_enum_collect_value() */
|
||
|
g_assert_cmpint (flag_read, ==, flag_set);
|
||
|
|
||
|
g_object_unref (test);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int
|
||
|
main (int argc, char **argv)
|
||
|
{
|
||
|
g_test_init (&argc, &argv, NULL);
|
||
|
g_test_add_func ("/gobject/flags/validate", check_flags_validation);
|
||
|
return g_test_run ();
|
||
|
}
|