Fix g_autoptr_cleanup_gstring_free( ) argument to avoid of confliting arument name
[platform/upstream/glib.git] / gobject / tests / accumulator.c
1 /* GObject - GLib Type, Object, Parameter and Signal Library
2  * Copyright (C) 2001, 2003 Red Hat, Inc.
3  *
4  * SPDX-License-Identifier: LGPL-2.1-or-later
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General
17  * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <glib-object.h>
21
22 #include "marshalers.h"
23 #include "testcommon.h"
24
25 /* What this test tests is the behavior of signal accumulators
26  * Two accumulators are tested:
27  *
28  * 1: A custom accumulator that appends the returned strings
29  * 2: The standard g_signal_accumulator_true_handled that stops
30  *    emission on TRUE returns.
31  */
32
33 /* TestObject, a parent class for TestObject */
34 #define TEST_TYPE_OBJECT          (test_object_get_type ())
35 typedef struct _TestObject        TestObject;
36 typedef struct _TestObjectClass   TestObjectClass;
37
38 struct _TestObject
39 {
40   GObject parent_instance;
41 };
42 struct _TestObjectClass
43 {
44   GObjectClass parent_class;
45
46   gchar*   (*test_signal1) (TestObject *tobject,
47                             gint        param);
48   gboolean (*test_signal2) (TestObject *tobject,
49                             gint        param);
50   GVariant* (*test_signal3) (TestObject *tobject,
51                              gboolean *weak_ptr);
52 };
53
54 static GType test_object_get_type (void);
55
56 static gboolean
57 test_signal1_accumulator (GSignalInvocationHint *ihint,
58                           GValue                *return_accu,
59                           const GValue          *handler_return,
60                           gpointer               data)
61 {
62   const gchar *accu_string = g_value_get_string (return_accu);
63   const gchar *new_string = g_value_get_string (handler_return);
64   gchar *result_string;
65
66   if (accu_string)
67     result_string = g_strconcat (accu_string, new_string, NULL);
68   else if (new_string)
69     result_string = g_strdup (new_string);
70   else
71     result_string = NULL;
72
73   g_value_take_string (return_accu, result_string);
74
75   return TRUE;
76 }
77
78 static gchar *
79 test_object_signal1_callback_before (TestObject *tobject,
80                                      gint        param,
81                                      gpointer    data)
82 {
83   return g_strdup ("<before>");
84 }
85
86 static gchar *
87 test_object_real_signal1 (TestObject *tobject,
88                           gint        param)
89 {
90   return g_strdup ("<default>");
91 }
92
93 static gchar *
94 test_object_signal1_callback_after (TestObject *tobject,
95                                     gint        param,
96                                     gpointer    data)
97 {
98   return g_strdup ("<after>");
99 }
100
101 static gboolean
102 test_object_signal2_callback_before (TestObject *tobject,
103                                      gint        param)
104 {
105   switch (param)
106     {
107     case 1: return TRUE;
108     case 2: return FALSE;
109     case 3: return FALSE;
110     case 4: return FALSE;
111     }
112
113   g_assert_not_reached ();
114   return FALSE;
115 }
116
117 static gboolean
118 test_object_real_signal2 (TestObject *tobject,
119                           gint        param)
120 {
121   switch (param)
122     {
123     case 1: g_assert_not_reached (); return FALSE;
124     case 2: return TRUE;
125     case 3: return FALSE;
126     case 4: return FALSE;
127     }
128
129   g_assert_not_reached ();
130   return FALSE;
131 }
132
133 static gboolean
134 test_object_signal2_callback_after (TestObject *tobject,
135                                     gint        param)
136 {
137   switch (param)
138     {
139     case 1: g_assert_not_reached (); return FALSE;
140     case 2: g_assert_not_reached (); return FALSE;
141     case 3: return TRUE;
142     case 4: return FALSE;
143     }
144
145   g_assert_not_reached ();
146   return FALSE;
147 }
148
149 static gboolean
150 test_signal3_accumulator (GSignalInvocationHint *ihint,
151                           GValue                *return_accu,
152                           const GValue          *handler_return,
153                           gpointer               data)
154 {
155   GVariant *variant;
156
157   variant = g_value_get_variant (handler_return);
158   g_assert_false (g_variant_is_floating (variant));
159
160   g_value_set_variant (return_accu, variant);
161
162   return variant == NULL;
163 }
164
165 /* To be notified when the variant is finalised, we construct
166  * it from data with a custom GDestroyNotify.
167  */
168
169 typedef struct {
170   char *mem;
171   gsize n;
172   gboolean *weak_ptr;
173 } VariantData;
174
175 static void
176 free_data (VariantData *data)
177 {
178   *(data->weak_ptr) = TRUE;
179   g_free (data->mem);
180   g_slice_free (VariantData, data);
181 }
182
183 static GVariant *
184 test_object_real_signal3 (TestObject *tobject,
185                           gboolean *weak_ptr)
186 {
187   GVariant *variant;
188   VariantData *data;
189
190   variant = g_variant_ref_sink (g_variant_new_uint32 (42));
191   data = g_slice_new (VariantData);
192   data->weak_ptr = weak_ptr;
193   data->n = g_variant_get_size (variant);
194   data->mem = g_malloc (data->n);
195   g_variant_store (variant, data->mem);
196   g_variant_unref (variant);
197
198   variant = g_variant_new_from_data (G_VARIANT_TYPE ("u"),
199                                      data->mem,
200                                      data->n,
201                                      TRUE,
202                                      (GDestroyNotify) free_data,
203                                      data);
204   return g_variant_ref_sink (variant);
205 }
206
207 static void
208 test_object_class_init (TestObjectClass *class)
209 {
210   class->test_signal1 = test_object_real_signal1;
211   class->test_signal2 = test_object_real_signal2;
212   class->test_signal3 = test_object_real_signal3;
213
214   g_signal_new ("test-signal1",
215                 G_OBJECT_CLASS_TYPE (class),
216                 G_SIGNAL_RUN_LAST,
217                 G_STRUCT_OFFSET (TestObjectClass, test_signal1),
218                 test_signal1_accumulator, NULL,
219                 test_STRING__INT,
220                 G_TYPE_STRING, 1, G_TYPE_INT);
221   g_signal_new ("test-signal2",
222                 G_OBJECT_CLASS_TYPE (class),
223                 G_SIGNAL_RUN_LAST,
224                 G_STRUCT_OFFSET (TestObjectClass, test_signal2),
225                 g_signal_accumulator_true_handled, NULL,
226                 test_BOOLEAN__INT,
227                 G_TYPE_BOOLEAN, 1, G_TYPE_INT);
228   g_signal_new ("test-signal3",
229                 G_OBJECT_CLASS_TYPE (class),
230                 G_SIGNAL_RUN_LAST,
231                 G_STRUCT_OFFSET (TestObjectClass, test_signal3),
232                 test_signal3_accumulator, NULL,
233                 test_VARIANT__POINTER,
234                 G_TYPE_VARIANT, 1, G_TYPE_POINTER);
235 }
236
237 static DEFINE_TYPE(TestObject, test_object,
238                    test_object_class_init, NULL, NULL,
239                    G_TYPE_OBJECT);
240
241 static void
242 test_accumulator (void)
243 {
244   TestObject *object;
245   gchar *string_result;
246   gboolean bool_result;
247   gboolean variant_finalised;
248   GVariant *variant_result;
249
250   object = g_object_new (TEST_TYPE_OBJECT, NULL);
251
252   g_signal_connect (object, "test-signal1",
253                     G_CALLBACK (test_object_signal1_callback_before), NULL);
254   g_signal_connect_after (object, "test-signal1",
255                           G_CALLBACK (test_object_signal1_callback_after), NULL);
256
257   g_signal_emit_by_name (object, "test-signal1", 0, &string_result);
258   g_assert_cmpstr (string_result, ==, "<before><default><after>");
259   g_free (string_result);
260
261   g_signal_connect (object, "test-signal2",
262                     G_CALLBACK (test_object_signal2_callback_before), NULL);
263   g_signal_connect_after (object, "test-signal2",
264                           G_CALLBACK (test_object_signal2_callback_after), NULL);
265
266   bool_result = FALSE;
267   g_signal_emit_by_name (object, "test-signal2", 1, &bool_result);
268   g_assert_true (bool_result);
269   bool_result = FALSE;
270   g_signal_emit_by_name (object, "test-signal2", 2, &bool_result);
271   g_assert_true (bool_result);
272   bool_result = FALSE;
273   g_signal_emit_by_name (object, "test-signal2", 3, &bool_result);
274   g_assert_true (bool_result);
275   bool_result = TRUE;
276   g_signal_emit_by_name (object, "test-signal2", 4, &bool_result);
277   g_assert_false (bool_result);
278
279   variant_finalised = FALSE;
280   variant_result = NULL;
281   g_signal_emit_by_name (object, "test-signal3", &variant_finalised, &variant_result);
282   g_assert_nonnull (variant_result);
283   g_assert_false (g_variant_is_floating (variant_result));
284
285   /* Test that variant_result had refcount 1 */
286   g_assert_false (variant_finalised);
287   g_variant_unref (variant_result);
288   g_assert_true (variant_finalised);
289
290   g_object_unref (object);
291 }
292
293 int
294 main (int  argc,
295       char *argv[])
296 {
297   g_log_set_always_fatal (g_log_set_always_fatal (G_LOG_FATAL_MASK) |
298                           G_LOG_LEVEL_WARNING |
299                           G_LOG_LEVEL_CRITICAL);
300
301   g_test_init (&argc, &argv, NULL);
302
303   g_test_add_func ("/gobject/accumulator", test_accumulator);
304
305   return g_test_run ();
306 }