|
Packit |
ae235b |
/* GObject - GLib Type, Object, Parameter and Signal Library
|
|
Packit |
ae235b |
* Copyright (C) 2001, 2003 Red Hat, Inc.
|
|
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
|
|
Packit |
ae235b |
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
Packit |
ae235b |
*/
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
#undef G_LOG_DOMAIN
|
|
Packit |
ae235b |
#define G_LOG_DOMAIN "TestAccumulator"
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
#undef G_DISABLE_ASSERT
|
|
Packit |
ae235b |
#undef G_DISABLE_CHECKS
|
|
Packit |
ae235b |
#undef G_DISABLE_CAST_CHECKS
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
#include <string.h>
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
#include <glib-object.h>
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
#include "testmarshal.h"
|
|
Packit |
ae235b |
#include "testcommon.h"
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
/* What this test tests is the behavior of signal accumulators
|
|
Packit |
ae235b |
* Two accumulators are tested:
|
|
Packit |
ae235b |
*
|
|
Packit |
ae235b |
* 1: A custom accumulator that appends the returned strings
|
|
Packit |
ae235b |
* 2: The standard g_signal_accumulator_true_handled that stops
|
|
Packit |
ae235b |
* emission on TRUE returns.
|
|
Packit |
ae235b |
*/
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
/*
|
|
Packit |
ae235b |
* TestObject, a parent class for TestObject
|
|
Packit |
ae235b |
*/
|
|
Packit |
ae235b |
#define TEST_TYPE_OBJECT (test_object_get_type ())
|
|
Packit |
ae235b |
typedef struct _TestObject TestObject;
|
|
Packit |
ae235b |
typedef struct _TestObjectClass TestObjectClass;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
struct _TestObject
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
GObject parent_instance;
|
|
Packit |
ae235b |
};
|
|
Packit |
ae235b |
struct _TestObjectClass
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
GObjectClass parent_class;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
gchar* (*test_signal1) (TestObject *tobject,
|
|
Packit |
ae235b |
gint param);
|
|
Packit |
ae235b |
gboolean (*test_signal2) (TestObject *tobject,
|
|
Packit |
ae235b |
gint param);
|
|
Packit |
ae235b |
GVariant* (*test_signal3) (TestObject *tobject,
|
|
Packit |
ae235b |
gboolean *weak_ptr);
|
|
Packit |
ae235b |
};
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static GType test_object_get_type (void);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gboolean
|
|
Packit |
ae235b |
test_signal1_accumulator (GSignalInvocationHint *ihint,
|
|
Packit |
ae235b |
GValue *return_accu,
|
|
Packit |
ae235b |
const GValue *handler_return,
|
|
Packit |
ae235b |
gpointer data)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
const gchar *accu_string = g_value_get_string (return_accu);
|
|
Packit |
ae235b |
const gchar *new_string = g_value_get_string (handler_return);
|
|
Packit |
ae235b |
gchar *result_string;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
if (accu_string)
|
|
Packit |
ae235b |
result_string = g_strconcat (accu_string, new_string, NULL);
|
|
Packit |
ae235b |
else if (new_string)
|
|
Packit |
ae235b |
result_string = g_strdup (new_string);
|
|
Packit |
ae235b |
else
|
|
Packit |
ae235b |
result_string = NULL;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_value_set_string_take_ownership (return_accu, result_string);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
return TRUE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gchar *
|
|
Packit |
ae235b |
test_object_signal1_callback_before (TestObject *tobject,
|
|
Packit |
ae235b |
gint param,
|
|
Packit |
ae235b |
gpointer data)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
return g_strdup ("<before>");
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gchar *
|
|
Packit |
ae235b |
test_object_real_signal1 (TestObject *tobject,
|
|
Packit |
ae235b |
gint param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
return g_strdup ("<default>");
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gchar *
|
|
Packit |
ae235b |
test_object_signal1_callback_after (TestObject *tobject,
|
|
Packit |
ae235b |
gint param,
|
|
Packit |
ae235b |
gpointer data)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
return g_strdup ("<after>");
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gboolean
|
|
Packit |
ae235b |
test_object_signal2_callback_before (TestObject *tobject,
|
|
Packit |
ae235b |
gint param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
switch (param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
case 1: return TRUE;
|
|
Packit |
ae235b |
case 2: return FALSE;
|
|
Packit |
ae235b |
case 3: return FALSE;
|
|
Packit |
ae235b |
case 4: return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_assert_not_reached ();
|
|
Packit |
ae235b |
return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gboolean
|
|
Packit |
ae235b |
test_object_real_signal2 (TestObject *tobject,
|
|
Packit |
ae235b |
gint param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
switch (param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
case 1: g_assert_not_reached (); return FALSE;
|
|
Packit |
ae235b |
case 2: return TRUE;
|
|
Packit |
ae235b |
case 3: return FALSE;
|
|
Packit |
ae235b |
case 4: return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_assert_not_reached ();
|
|
Packit |
ae235b |
return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gboolean
|
|
Packit |
ae235b |
test_object_signal2_callback_after (TestObject *tobject,
|
|
Packit |
ae235b |
gint param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
switch (param)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
case 1: g_assert_not_reached (); return FALSE;
|
|
Packit |
ae235b |
case 2: g_assert_not_reached (); return FALSE;
|
|
Packit |
ae235b |
case 3: return TRUE;
|
|
Packit |
ae235b |
case 4: return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_assert_not_reached ();
|
|
Packit |
ae235b |
return FALSE;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static gboolean
|
|
Packit |
ae235b |
test_signal3_accumulator (GSignalInvocationHint *ihint,
|
|
Packit |
ae235b |
GValue *return_accu,
|
|
Packit |
ae235b |
const GValue *handler_return,
|
|
Packit |
ae235b |
gpointer data)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
GVariant *variant;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
variant = g_value_get_variant (handler_return);
|
|
Packit |
ae235b |
g_assert (!g_variant_is_floating (variant));
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_value_set_variant (return_accu, variant);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
return variant == NULL;
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
/* To be notified when the variant is finalised, we construct
|
|
Packit |
ae235b |
* it from data with a custom GDestroyNotify.
|
|
Packit |
ae235b |
*/
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
typedef struct {
|
|
Packit |
ae235b |
char *mem;
|
|
Packit |
ae235b |
gsize n;
|
|
Packit |
ae235b |
gboolean *weak_ptr;
|
|
Packit |
ae235b |
} VariantData;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static void
|
|
Packit |
ae235b |
free_data (VariantData *data)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
*(data->weak_ptr) = TRUE;
|
|
Packit |
ae235b |
g_free (data->mem);
|
|
Packit |
ae235b |
g_slice_free (VariantData, data);
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static GVariant *
|
|
Packit |
ae235b |
test_object_real_signal3 (TestObject *tobject,
|
|
Packit |
ae235b |
gboolean *weak_ptr)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
GVariant *variant;
|
|
Packit |
ae235b |
VariantData *data;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
variant = g_variant_ref_sink (g_variant_new_uint32 (42));
|
|
Packit |
ae235b |
data = g_slice_new (VariantData);
|
|
Packit |
ae235b |
data->weak_ptr = weak_ptr;
|
|
Packit |
ae235b |
data->n = g_variant_get_size (variant);
|
|
Packit |
ae235b |
data->mem = g_malloc (data->n);
|
|
Packit |
ae235b |
g_variant_store (variant, data->mem);
|
|
Packit |
ae235b |
g_variant_unref (variant);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
variant = g_variant_new_from_data (G_VARIANT_TYPE ("u"),
|
|
Packit |
ae235b |
data->mem,
|
|
Packit |
ae235b |
data->n,
|
|
Packit |
ae235b |
TRUE,
|
|
Packit |
ae235b |
(GDestroyNotify) free_data,
|
|
Packit |
ae235b |
data);
|
|
Packit |
ae235b |
return g_variant_ref_sink (variant);
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static void
|
|
Packit |
ae235b |
test_object_class_init (TestObjectClass *class)
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
class->test_signal1 = test_object_real_signal1;
|
|
Packit |
ae235b |
class->test_signal2 = test_object_real_signal2;
|
|
Packit |
ae235b |
class->test_signal3 = test_object_real_signal3;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_signal_new ("test-signal1",
|
|
Packit |
ae235b |
G_OBJECT_CLASS_TYPE (class),
|
|
Packit |
ae235b |
G_SIGNAL_RUN_LAST,
|
|
Packit |
ae235b |
G_STRUCT_OFFSET (TestObjectClass, test_signal1),
|
|
Packit |
ae235b |
test_signal1_accumulator, NULL,
|
|
Packit |
ae235b |
test_marshal_STRING__INT,
|
|
Packit |
ae235b |
G_TYPE_STRING, 1, G_TYPE_INT);
|
|
Packit |
ae235b |
g_signal_new ("test-signal2",
|
|
Packit |
ae235b |
G_OBJECT_CLASS_TYPE (class),
|
|
Packit |
ae235b |
G_SIGNAL_RUN_LAST,
|
|
Packit |
ae235b |
G_STRUCT_OFFSET (TestObjectClass, test_signal2),
|
|
Packit |
ae235b |
g_signal_accumulator_true_handled, NULL,
|
|
Packit |
ae235b |
test_marshal_BOOLEAN__INT,
|
|
Packit |
ae235b |
G_TYPE_BOOLEAN, 1, G_TYPE_INT);
|
|
Packit |
ae235b |
g_signal_new ("test-signal3",
|
|
Packit |
ae235b |
G_OBJECT_CLASS_TYPE (class),
|
|
Packit |
ae235b |
G_SIGNAL_RUN_LAST,
|
|
Packit |
ae235b |
G_STRUCT_OFFSET (TestObjectClass, test_signal3),
|
|
Packit |
ae235b |
test_signal3_accumulator, NULL,
|
|
Packit |
ae235b |
test_marshal_VARIANT__POINTER,
|
|
Packit |
ae235b |
G_TYPE_VARIANT, 1, G_TYPE_POINTER);
|
|
Packit |
ae235b |
}
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
static DEFINE_TYPE(TestObject, test_object,
|
|
Packit |
ae235b |
test_object_class_init, NULL, NULL,
|
|
Packit |
ae235b |
G_TYPE_OBJECT)
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
int
|
|
Packit |
ae235b |
main (int argc,
|
|
Packit |
ae235b |
char *argv[])
|
|
Packit |
ae235b |
{
|
|
Packit |
ae235b |
TestObject *object;
|
|
Packit |
ae235b |
gchar *string_result;
|
|
Packit |
ae235b |
gboolean bool_result;
|
|
Packit |
ae235b |
gboolean variant_finalised;
|
|
Packit |
ae235b |
GVariant *variant_result;
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
|
|
Packit |
ae235b |
G_LOG_LEVEL_WARNING |
|
|
Packit |
ae235b |
G_LOG_LEVEL_CRITICAL);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
object = g_object_new (TEST_TYPE_OBJECT, NULL);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_signal_connect (object, "test-signal1",
|
|
Packit |
ae235b |
G_CALLBACK (test_object_signal1_callback_before), NULL);
|
|
Packit |
ae235b |
g_signal_connect_after (object, "test-signal1",
|
|
Packit |
ae235b |
G_CALLBACK (test_object_signal1_callback_after), NULL);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal1", 0, &string_result);
|
|
Packit |
ae235b |
g_assert (strcmp (string_result, "<before><default><after>") == 0);
|
|
Packit |
ae235b |
g_free (string_result);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_signal_connect (object, "test-signal2",
|
|
Packit |
ae235b |
G_CALLBACK (test_object_signal2_callback_before), NULL);
|
|
Packit |
ae235b |
g_signal_connect_after (object, "test-signal2",
|
|
Packit |
ae235b |
G_CALLBACK (test_object_signal2_callback_after), NULL);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
bool_result = FALSE;
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal2", 1, &bool_result);
|
|
Packit |
ae235b |
g_assert (bool_result == TRUE);
|
|
Packit |
ae235b |
bool_result = FALSE;
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal2", 2, &bool_result);
|
|
Packit |
ae235b |
g_assert (bool_result == TRUE);
|
|
Packit |
ae235b |
bool_result = FALSE;
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal2", 3, &bool_result);
|
|
Packit |
ae235b |
g_assert (bool_result == TRUE);
|
|
Packit |
ae235b |
bool_result = TRUE;
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal2", 4, &bool_result);
|
|
Packit |
ae235b |
g_assert (bool_result == FALSE);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
variant_finalised = FALSE;
|
|
Packit |
ae235b |
variant_result = NULL;
|
|
Packit |
ae235b |
g_signal_emit_by_name (object, "test-signal3", &variant_finalised, &variant_result);
|
|
Packit |
ae235b |
g_assert (variant_result != NULL);
|
|
Packit |
ae235b |
g_assert (!g_variant_is_floating (variant_result));
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
/* Test that variant_result had refcount 1 */
|
|
Packit |
ae235b |
g_assert (!variant_finalised);
|
|
Packit |
ae235b |
g_variant_unref (variant_result);
|
|
Packit |
ae235b |
g_assert (variant_finalised);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
g_object_unref (object);
|
|
Packit |
ae235b |
|
|
Packit |
ae235b |
return 0;
|
|
Packit |
ae235b |
}
|