Blame tests/gobject/accumulator.c

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
}