os/ossrv/glib/gobject/gboxed.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* GObject - GLib Type, Object, Parameter and Signal Library
sl@0
     2
 * Copyright (C) 2000-2001 Red Hat, Inc.
sl@0
     3
 * Portions copyright (c) 2006-2009 Nokia Corporation.  All rights reserved.
sl@0
     4
 *
sl@0
     5
 * This library is free software; you can redistribute it and/or
sl@0
     6
 * modify it under the terms of the GNU Lesser General Public
sl@0
     7
 * License as published by the Free Software Foundation; either
sl@0
     8
 * version 2 of the License, or (at your option) any later version.
sl@0
     9
 *
sl@0
    10
 * This library is distributed in the hope that it will be useful,
sl@0
    11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
sl@0
    13
 * Lesser General Public License for more details.
sl@0
    14
 *
sl@0
    15
 * You should have received a copy of the GNU Lesser General
sl@0
    16
 * Public License along with this library; if not, write to the
sl@0
    17
 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
sl@0
    18
 * Boston, MA 02111-1307, USA.
sl@0
    19
 */
sl@0
    20
sl@0
    21
#include "config.h"
sl@0
    22
sl@0
    23
#include <string.h>
sl@0
    24
sl@0
    25
#include "gboxed.h"
sl@0
    26
#include "gbsearcharray.h"
sl@0
    27
#include "gvalue.h"
sl@0
    28
#include "gvaluearray.h"
sl@0
    29
#include "gclosure.h"
sl@0
    30
#include "gvaluecollector.h"
sl@0
    31
#include "gobjectalias.h"
sl@0
    32
sl@0
    33
#ifdef __SYMBIAN32__
sl@0
    34
#include "gobject_wsd.h"
sl@0
    35
#endif /* __SYMBIAN32__ */
sl@0
    36
sl@0
    37
/**
sl@0
    38
 * SECTION:gboxed
sl@0
    39
 * @short_description: A mechanism to wrap opaque C structures registered
sl@0
    40
 *     by the type system
sl@0
    41
 * @see_also: #GParamSpecBoxed, g_param_spec_boxed()
sl@0
    42
 * @title: Boxed Types
sl@0
    43
 *
sl@0
    44
 * GBoxed is a generic wrapper mechanism for arbitrary C structures. The only
sl@0
    45
 * thing the type system needs to know about the structures is how to copy and
sl@0
    46
 * free them, beyond that they are treated as opaque chunks of memory.
sl@0
    47
 *
sl@0
    48
 * Boxed types are useful for simple value-holder structures like rectangles or
sl@0
    49
 * points. They can also be used for wrapping structures defined in non-GObject
sl@0
    50
 * based libraries.
sl@0
    51
 */
sl@0
    52
sl@0
    53
/* --- typedefs & structures --- */
sl@0
    54
typedef struct
sl@0
    55
{
sl@0
    56
  GType		 type;
sl@0
    57
  GBoxedCopyFunc copy;
sl@0
    58
  GBoxedFreeFunc free;
sl@0
    59
} BoxedNode;
sl@0
    60
sl@0
    61
sl@0
    62
/* --- prototypes --- */
sl@0
    63
static gint	boxed_nodes_cmp		(gconstpointer	p1,
sl@0
    64
					 gconstpointer	p2);
sl@0
    65
sl@0
    66
sl@0
    67
/* --- variables --- */
sl@0
    68
#if EMULATOR	
sl@0
    69
PLS(boxed_bsa,gboxed,GBSearchArray *)
sl@0
    70
#define boxed_bsa (*FUNCTION_NAME(boxed_bsa,gboxed)())
sl@0
    71
#else
sl@0
    72
static GBSearchArray *boxed_bsa = NULL;
sl@0
    73
#endif /*EMULATOR */
sl@0
    74
static const GBSearchConfig boxed_bconfig = {
sl@0
    75
  sizeof (BoxedNode),
sl@0
    76
  boxed_nodes_cmp,
sl@0
    77
  0,
sl@0
    78
};
sl@0
    79
sl@0
    80
sl@0
    81
/* --- functions --- */
sl@0
    82
static gint
sl@0
    83
boxed_nodes_cmp	(gconstpointer p1,
sl@0
    84
		 gconstpointer p2)
sl@0
    85
{
sl@0
    86
  const BoxedNode *node1 = p1, *node2 = p2;
sl@0
    87
sl@0
    88
  return G_BSEARCH_ARRAY_CMP (node1->type, node2->type);
sl@0
    89
}
sl@0
    90
sl@0
    91
static inline void              /* keep this function in sync with gvalue.c */
sl@0
    92
value_meminit (GValue *value,
sl@0
    93
	       GType   value_type)
sl@0
    94
{
sl@0
    95
  value->g_type = value_type;
sl@0
    96
  memset (value->data, 0, sizeof (value->data));
sl@0
    97
}
sl@0
    98
sl@0
    99
static gpointer
sl@0
   100
value_copy (gpointer boxed)
sl@0
   101
{
sl@0
   102
  const GValue *src_value = boxed;
sl@0
   103
  GValue *dest_value = g_new0 (GValue, 1);
sl@0
   104
sl@0
   105
  if (G_VALUE_TYPE (src_value))
sl@0
   106
    {
sl@0
   107
      g_value_init (dest_value, G_VALUE_TYPE (src_value));
sl@0
   108
      g_value_copy (src_value, dest_value);
sl@0
   109
    }
sl@0
   110
  return dest_value;
sl@0
   111
}
sl@0
   112
sl@0
   113
static void
sl@0
   114
value_free (gpointer boxed)
sl@0
   115
{
sl@0
   116
  GValue *value = boxed;
sl@0
   117
sl@0
   118
  if (G_VALUE_TYPE (value))
sl@0
   119
    g_value_unset (value);
sl@0
   120
  g_free (value);
sl@0
   121
}
sl@0
   122
sl@0
   123
void
sl@0
   124
g_boxed_type_init (void)
sl@0
   125
{
sl@0
   126
  static const GTypeInfo info = {
sl@0
   127
    0,                          /* class_size */
sl@0
   128
    NULL,                       /* base_init */
sl@0
   129
    NULL,                       /* base_destroy */
sl@0
   130
    NULL,                       /* class_init */
sl@0
   131
    NULL,                       /* class_destroy */
sl@0
   132
    NULL,                       /* class_data */
sl@0
   133
    0,                          /* instance_size */
sl@0
   134
    0,                          /* n_preallocs */
sl@0
   135
    NULL,                       /* instance_init */
sl@0
   136
    NULL,                       /* value_table */
sl@0
   137
  };
sl@0
   138
  const GTypeFundamentalInfo finfo = { G_TYPE_FLAG_DERIVABLE, };
sl@0
   139
  GType type;
sl@0
   140
sl@0
   141
  boxed_bsa = g_bsearch_array_create (&boxed_bconfig);
sl@0
   142
sl@0
   143
  /* G_TYPE_BOXED
sl@0
   144
   */
sl@0
   145
  type = g_type_register_fundamental (G_TYPE_BOXED, g_intern_static_string ("GBoxed"), &info, &finfo,
sl@0
   146
				      G_TYPE_FLAG_ABSTRACT | G_TYPE_FLAG_VALUE_ABSTRACT);
sl@0
   147
  g_assert (type == G_TYPE_BOXED);
sl@0
   148
}
sl@0
   149
sl@0
   150
#if EMULATOR
sl@0
   151
PLS(type_id,g_closure_get_type ,GType)
sl@0
   152
#define type_id (*FUNCTION_NAME(type_id,g_closure_get_type )())
sl@0
   153
#endif /* EMULATOR */
sl@0
   154
sl@0
   155
EXPORT_C GType
sl@0
   156
g_closure_get_type (void)
sl@0
   157
{
sl@0
   158
  #if !(EMULATOR)
sl@0
   159
  static GType type_id = 0;
sl@0
   160
  #endif /* EMULATOR */
sl@0
   161
sl@0
   162
  if (!type_id)
sl@0
   163
    type_id = g_boxed_type_register_static (g_intern_static_string ("GClosure"),
sl@0
   164
					    (GBoxedCopyFunc) g_closure_ref,
sl@0
   165
					    (GBoxedFreeFunc) g_closure_unref);
sl@0
   166
  return type_id;
sl@0
   167
}
sl@0
   168
sl@0
   169
#if EMULATOR
sl@0
   170
#undef type_id
sl@0
   171
PLS(type_id,g_value_get_type ,GType)
sl@0
   172
#define type_id (*FUNCTION_NAME(type_id,g_value_get_type )())
sl@0
   173
#endif /* EMULATOR */
sl@0
   174
sl@0
   175
EXPORT_C GType
sl@0
   176
g_value_get_type (void)
sl@0
   177
{
sl@0
   178
  #if !(EMULATOR)
sl@0
   179
  static GType type_id = 0;
sl@0
   180
  #endif /* EMULATOR */
sl@0
   181
sl@0
   182
  if (!type_id)
sl@0
   183
    type_id = g_boxed_type_register_static (g_intern_static_string ("GValue"),
sl@0
   184
					    value_copy,
sl@0
   185
					    value_free);
sl@0
   186
  return type_id;
sl@0
   187
}
sl@0
   188
sl@0
   189
#if EMULATOR
sl@0
   190
#undef type_id
sl@0
   191
PLS(type_id,g_value_array_get_type ,GType)
sl@0
   192
#define type_id (*FUNCTION_NAME(type_id,g_value_array_get_type )())
sl@0
   193
#endif /* EMULATOR */
sl@0
   194
sl@0
   195
EXPORT_C GType
sl@0
   196
g_value_array_get_type (void)
sl@0
   197
{
sl@0
   198
  #if !(EMULATOR)
sl@0
   199
  static GType type_id = 0;
sl@0
   200
  #endif /* EMULATOR */
sl@0
   201
sl@0
   202
  if (!type_id)
sl@0
   203
    type_id = g_boxed_type_register_static (g_intern_static_string ("GValueArray"),
sl@0
   204
					    (GBoxedCopyFunc) g_value_array_copy,
sl@0
   205
					    (GBoxedFreeFunc) g_value_array_free);
sl@0
   206
  return type_id;
sl@0
   207
}
sl@0
   208
sl@0
   209
static gpointer
sl@0
   210
gdate_copy (gpointer boxed)
sl@0
   211
{
sl@0
   212
  const GDate *date = (const GDate*) boxed;
sl@0
   213
sl@0
   214
  return g_date_new_julian (g_date_get_julian (date));
sl@0
   215
}
sl@0
   216
sl@0
   217
#if EMULATOR
sl@0
   218
#undef type_id
sl@0
   219
PLS(type_id,g_date_get_type ,GType)
sl@0
   220
#define type_id (*FUNCTION_NAME(type_id,g_date_get_type )())
sl@0
   221
#endif /* EMULATOR */
sl@0
   222
sl@0
   223
EXPORT_C GType
sl@0
   224
g_date_get_type (void)
sl@0
   225
{
sl@0
   226
  #if !(EMULATOR)
sl@0
   227
  static GType type_id = 0;
sl@0
   228
  #endif /* EMULATOR */
sl@0
   229
sl@0
   230
  if (!type_id)
sl@0
   231
    type_id = g_boxed_type_register_static (g_intern_static_string ("GDate"),
sl@0
   232
					    (GBoxedCopyFunc) gdate_copy,
sl@0
   233
					    (GBoxedFreeFunc) g_date_free);
sl@0
   234
  return type_id;
sl@0
   235
}
sl@0
   236
sl@0
   237
#if EMULATOR
sl@0
   238
#undef type_id
sl@0
   239
PLS(type_id,g_strv_get_type ,GType)
sl@0
   240
#define type_id (*FUNCTION_NAME(type_id,g_strv_get_type )())
sl@0
   241
#endif /* EMULATOR */
sl@0
   242
sl@0
   243
EXPORT_C GType
sl@0
   244
g_strv_get_type (void)
sl@0
   245
{
sl@0
   246
  #if !(EMULATOR)
sl@0
   247
  static GType type_id = 0;
sl@0
   248
  #endif /* EMULATOR */
sl@0
   249
sl@0
   250
  if (!type_id)
sl@0
   251
    type_id = g_boxed_type_register_static (g_intern_static_string ("GStrv"),
sl@0
   252
					    (GBoxedCopyFunc) g_strdupv,
sl@0
   253
					    (GBoxedFreeFunc) g_strfreev);
sl@0
   254
  return type_id;
sl@0
   255
}
sl@0
   256
sl@0
   257
static gpointer
sl@0
   258
gstring_copy (gpointer boxed)
sl@0
   259
{
sl@0
   260
  const GString *src_gstring = boxed;
sl@0
   261
sl@0
   262
  return g_string_new_len (src_gstring->str, src_gstring->len);
sl@0
   263
}
sl@0
   264
sl@0
   265
static void
sl@0
   266
gstring_free (gpointer boxed)
sl@0
   267
{
sl@0
   268
  GString *gstring = boxed;
sl@0
   269
sl@0
   270
  g_string_free (gstring, TRUE);
sl@0
   271
}
sl@0
   272
sl@0
   273
#if EMULATOR
sl@0
   274
#undef type_id
sl@0
   275
PLS(type_id,g_gstring_get_type ,GType)
sl@0
   276
#define type_id (*FUNCTION_NAME(type_id,g_gstring_get_type )())
sl@0
   277
#endif /* EMULATOR */
sl@0
   278
sl@0
   279
EXPORT_C GType
sl@0
   280
g_gstring_get_type (void)
sl@0
   281
{
sl@0
   282
  #if !(EMULATOR)
sl@0
   283
  static GType type_id = 0;
sl@0
   284
  #endif /* !(EMULATOR) */
sl@0
   285
sl@0
   286
  if (!type_id)
sl@0
   287
    type_id = g_boxed_type_register_static (g_intern_static_string ("GString"),
sl@0
   288
                                            /* the naming is a bit odd, but GString is obviously not G_TYPE_STRING */
sl@0
   289
					    gstring_copy,
sl@0
   290
					    gstring_free);
sl@0
   291
  return type_id;
sl@0
   292
}
sl@0
   293
sl@0
   294
static gpointer
sl@0
   295
hash_table_copy (gpointer boxed)
sl@0
   296
{
sl@0
   297
  GHashTable *hash_table = boxed;
sl@0
   298
  return g_hash_table_ref (hash_table);
sl@0
   299
}
sl@0
   300
sl@0
   301
static void
sl@0
   302
hash_table_free (gpointer boxed)
sl@0
   303
{
sl@0
   304
  GHashTable *hash_table = boxed;
sl@0
   305
  g_hash_table_unref (hash_table);
sl@0
   306
}
sl@0
   307
sl@0
   308
#if EMULATOR
sl@0
   309
#undef type_id
sl@0
   310
PLS(type_id,g_hash_table_get_type ,GType)
sl@0
   311
#define type_id (*FUNCTION_NAME(type_id,g_hash_table_get_type )())
sl@0
   312
#endif /* EMULATOR */
sl@0
   313
sl@0
   314
EXPORT_C GType
sl@0
   315
g_hash_table_get_type (void)
sl@0
   316
{
sl@0
   317
  #if !(EMULATOR)
sl@0
   318
  static GType type_id = 0;
sl@0
   319
  #endif */ !(EMULATOR) */
sl@0
   320
  if (!type_id)
sl@0
   321
    type_id = g_boxed_type_register_static (g_intern_static_string ("GHashTable"),
sl@0
   322
					    hash_table_copy, hash_table_free);
sl@0
   323
  return type_id;
sl@0
   324
}
sl@0
   325
sl@0
   326
#if EMULATOR
sl@0
   327
#undef type_id
sl@0
   328
PLS(type_id,g_regex_get_type ,GType)
sl@0
   329
#define type_id (*FUNCTION_NAME(type_id,g_regex_get_type )())
sl@0
   330
#endif /* EMULATOR */
sl@0
   331
sl@0
   332
EXPORT_C GType
sl@0
   333
g_regex_get_type (void)
sl@0
   334
{
sl@0
   335
 #if !(EMULATOR)
sl@0
   336
  static GType type_id = 0;
sl@0
   337
  #endif */ !(EMULATOR) */
sl@0
   338
sl@0
   339
#ifdef ENABLE_REGEX
sl@0
   340
  if (!type_id)
sl@0
   341
    type_id = g_boxed_type_register_static (g_intern_static_string ("GRegex"),
sl@0
   342
					    (GBoxedCopyFunc) g_regex_ref,
sl@0
   343
					    (GBoxedFreeFunc) g_regex_unref);
sl@0
   344
#endif
sl@0
   345
sl@0
   346
  return type_id;
sl@0
   347
}
sl@0
   348
sl@0
   349
#if EMULATOR
sl@0
   350
#undef type_id
sl@0
   351
#endif /* EMULATOR */
sl@0
   352
sl@0
   353
static void
sl@0
   354
boxed_proxy_value_init (GValue *value)
sl@0
   355
{
sl@0
   356
  value->data[0].v_pointer = NULL;
sl@0
   357
}
sl@0
   358
sl@0
   359
static void
sl@0
   360
boxed_proxy_value_free (GValue *value)
sl@0
   361
{
sl@0
   362
  if (value->data[0].v_pointer && !(value->data[1].v_uint & G_VALUE_NOCOPY_CONTENTS))
sl@0
   363
    {
sl@0
   364
      BoxedNode key, *node;
sl@0
   365
sl@0
   366
      key.type = G_VALUE_TYPE (value);
sl@0
   367
      node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   368
      node->free (value->data[0].v_pointer);
sl@0
   369
    }
sl@0
   370
}
sl@0
   371
sl@0
   372
static void
sl@0
   373
boxed_proxy_value_copy (const GValue *src_value,
sl@0
   374
			GValue       *dest_value)
sl@0
   375
{
sl@0
   376
  if (src_value->data[0].v_pointer)
sl@0
   377
    {
sl@0
   378
      BoxedNode key, *node;
sl@0
   379
sl@0
   380
      key.type = G_VALUE_TYPE (src_value);
sl@0
   381
      node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   382
      dest_value->data[0].v_pointer = node->copy (src_value->data[0].v_pointer);
sl@0
   383
    }
sl@0
   384
  else
sl@0
   385
    dest_value->data[0].v_pointer = src_value->data[0].v_pointer;
sl@0
   386
}
sl@0
   387
sl@0
   388
static gpointer
sl@0
   389
boxed_proxy_value_peek_pointer (const GValue *value)
sl@0
   390
{
sl@0
   391
  return value->data[0].v_pointer;
sl@0
   392
}
sl@0
   393
sl@0
   394
static gchar*
sl@0
   395
boxed_proxy_collect_value (GValue      *value,
sl@0
   396
			   guint        n_collect_values,
sl@0
   397
			   GTypeCValue *collect_values,
sl@0
   398
			   guint        collect_flags)
sl@0
   399
{
sl@0
   400
  BoxedNode key, *node;
sl@0
   401
sl@0
   402
  key.type = G_VALUE_TYPE (value);
sl@0
   403
  node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   404
sl@0
   405
  if (!collect_values[0].v_pointer)
sl@0
   406
    value->data[0].v_pointer = NULL;
sl@0
   407
  else
sl@0
   408
    {
sl@0
   409
      if (collect_flags & G_VALUE_NOCOPY_CONTENTS)
sl@0
   410
	{
sl@0
   411
	  value->data[0].v_pointer = collect_values[0].v_pointer;
sl@0
   412
	  value->data[1].v_uint = G_VALUE_NOCOPY_CONTENTS;
sl@0
   413
	}
sl@0
   414
      else
sl@0
   415
	value->data[0].v_pointer = node->copy (collect_values[0].v_pointer);
sl@0
   416
    }
sl@0
   417
sl@0
   418
  return NULL;
sl@0
   419
}
sl@0
   420
sl@0
   421
static gchar*
sl@0
   422
boxed_proxy_lcopy_value (const GValue *value,
sl@0
   423
			 guint         n_collect_values,
sl@0
   424
			 GTypeCValue  *collect_values,
sl@0
   425
			 guint         collect_flags)
sl@0
   426
{
sl@0
   427
  gpointer *boxed_p = collect_values[0].v_pointer;
sl@0
   428
sl@0
   429
  if (!boxed_p)
sl@0
   430
    return g_strdup_printf ("value location for `%s' passed as NULL", G_VALUE_TYPE_NAME (value));
sl@0
   431
sl@0
   432
  if (!value->data[0].v_pointer)
sl@0
   433
    *boxed_p = NULL;
sl@0
   434
  else if (collect_flags & G_VALUE_NOCOPY_CONTENTS)
sl@0
   435
    *boxed_p = value->data[0].v_pointer;
sl@0
   436
  else
sl@0
   437
    {
sl@0
   438
      BoxedNode key, *node;
sl@0
   439
sl@0
   440
      key.type = G_VALUE_TYPE (value);
sl@0
   441
      node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   442
      *boxed_p = node->copy (value->data[0].v_pointer);
sl@0
   443
    }
sl@0
   444
sl@0
   445
  return NULL;
sl@0
   446
}
sl@0
   447
sl@0
   448
/**
sl@0
   449
 * g_boxed_type_register_static:
sl@0
   450
 * @name: Name of the new boxed type.
sl@0
   451
 * @boxed_copy: Boxed structure copy function.
sl@0
   452
 * @boxed_free: Boxed structure free function.
sl@0
   453
 *
sl@0
   454
 * This function creates a new %G_TYPE_BOXED derived type id for a new
sl@0
   455
 * boxed type with name @name. Boxed type handling functions have to be
sl@0
   456
 * provided to copy and free opaque boxed structures of this type.
sl@0
   457
 *
sl@0
   458
 * Returns: New %G_TYPE_BOXED derived type id for @name.
sl@0
   459
 */
sl@0
   460
EXPORT_C GType
sl@0
   461
g_boxed_type_register_static (const gchar   *name,
sl@0
   462
			      GBoxedCopyFunc boxed_copy,
sl@0
   463
			      GBoxedFreeFunc boxed_free)
sl@0
   464
{
sl@0
   465
  static const GTypeValueTable vtable = {
sl@0
   466
    boxed_proxy_value_init,
sl@0
   467
    boxed_proxy_value_free,
sl@0
   468
    boxed_proxy_value_copy,
sl@0
   469
    boxed_proxy_value_peek_pointer,
sl@0
   470
    "p",
sl@0
   471
    boxed_proxy_collect_value,
sl@0
   472
    "p",
sl@0
   473
    boxed_proxy_lcopy_value,
sl@0
   474
  };
sl@0
   475
  static const GTypeInfo type_info = {
sl@0
   476
    0,			/* class_size */
sl@0
   477
    NULL,		/* base_init */
sl@0
   478
    NULL,		/* base_finalize */
sl@0
   479
    NULL,		/* class_init */
sl@0
   480
    NULL,		/* class_finalize */
sl@0
   481
    NULL,		/* class_data */
sl@0
   482
    0,			/* instance_size */
sl@0
   483
    0,			/* n_preallocs */
sl@0
   484
    NULL,		/* instance_init */
sl@0
   485
    &vtable,		/* value_table */
sl@0
   486
  };
sl@0
   487
  GType type;
sl@0
   488
sl@0
   489
  g_return_val_if_fail (name != NULL, 0);
sl@0
   490
  g_return_val_if_fail (boxed_copy != NULL, 0);
sl@0
   491
  g_return_val_if_fail (boxed_free != NULL, 0);
sl@0
   492
  g_return_val_if_fail (g_type_from_name (name) == 0, 0);
sl@0
   493
sl@0
   494
  type = g_type_register_static (G_TYPE_BOXED, name, &type_info, 0);
sl@0
   495
sl@0
   496
  /* install proxy functions upon successfull registration */
sl@0
   497
  if (type)
sl@0
   498
    {
sl@0
   499
      BoxedNode key;
sl@0
   500
sl@0
   501
      key.type = type;
sl@0
   502
      key.copy = boxed_copy;
sl@0
   503
      key.free = boxed_free;
sl@0
   504
      boxed_bsa = g_bsearch_array_insert (boxed_bsa, &boxed_bconfig, &key);
sl@0
   505
    }
sl@0
   506
sl@0
   507
  return type;
sl@0
   508
}
sl@0
   509
sl@0
   510
/**
sl@0
   511
 * g_boxed_copy:
sl@0
   512
 * @boxed_type: The type of @src_boxed.
sl@0
   513
 * @src_boxed: The boxed structure to be copied.
sl@0
   514
 * 
sl@0
   515
 * Provide a copy of a boxed structure @src_boxed which is of type @boxed_type.
sl@0
   516
 * 
sl@0
   517
 * Returns: The newly created copy of the boxed structure.
sl@0
   518
 */
sl@0
   519
EXPORT_C gpointer
sl@0
   520
g_boxed_copy (GType         boxed_type,
sl@0
   521
	      gconstpointer src_boxed)
sl@0
   522
{
sl@0
   523
  GTypeValueTable *value_table;
sl@0
   524
  gpointer dest_boxed;
sl@0
   525
sl@0
   526
  g_return_val_if_fail (G_TYPE_IS_BOXED (boxed_type), NULL);
sl@0
   527
  g_return_val_if_fail (G_TYPE_IS_ABSTRACT (boxed_type) == FALSE, NULL);
sl@0
   528
  g_return_val_if_fail (src_boxed != NULL, NULL);
sl@0
   529
sl@0
   530
  value_table = g_type_value_table_peek (boxed_type);
sl@0
   531
  if (!value_table)
sl@0
   532
    g_return_val_if_fail (G_TYPE_IS_VALUE_TYPE (boxed_type), NULL);
sl@0
   533
sl@0
   534
  /* check if our proxying implementation is used, we can short-cut here */
sl@0
   535
  if (value_table->value_copy == boxed_proxy_value_copy)
sl@0
   536
    {
sl@0
   537
      BoxedNode key, *node;
sl@0
   538
sl@0
   539
      key.type = boxed_type;
sl@0
   540
      node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   541
      dest_boxed = node->copy ((gpointer) src_boxed);
sl@0
   542
    }
sl@0
   543
  else
sl@0
   544
    {
sl@0
   545
      GValue src_value, dest_value;
sl@0
   546
sl@0
   547
      /* we heavily rely on third-party boxed type value vtable
sl@0
   548
       * implementations to follow normal boxed value storage
sl@0
   549
       * (data[0].v_pointer is the boxed struct, and
sl@0
   550
       * data[1].v_uint holds the G_VALUE_NOCOPY_CONTENTS flag,
sl@0
   551
       * rest zero).
sl@0
   552
       * but then, we can expect that since we layed out the
sl@0
   553
       * g_boxed_*() API.
sl@0
   554
       * data[1].v_uint&G_VALUE_NOCOPY_CONTENTS shouldn't be set
sl@0
   555
       * after a copy.
sl@0
   556
       */
sl@0
   557
      /* equiv. to g_value_set_static_boxed() */
sl@0
   558
      value_meminit (&src_value, boxed_type);
sl@0
   559
      src_value.data[0].v_pointer = (gpointer) src_boxed;
sl@0
   560
      src_value.data[1].v_uint = G_VALUE_NOCOPY_CONTENTS;
sl@0
   561
sl@0
   562
      /* call third-party code copy function, fingers-crossed */
sl@0
   563
      value_meminit (&dest_value, boxed_type);
sl@0
   564
      value_table->value_copy (&src_value, &dest_value);
sl@0
   565
sl@0
   566
      /* double check and grouse if things went wrong */
sl@0
   567
      if (dest_value.data[1].v_ulong)
sl@0
   568
	g_warning ("the copy_value() implementation of type `%s' seems to make use of reserved GValue fields",
sl@0
   569
		   g_type_name (boxed_type));
sl@0
   570
sl@0
   571
      dest_boxed = dest_value.data[0].v_pointer;
sl@0
   572
    }
sl@0
   573
sl@0
   574
  return dest_boxed;
sl@0
   575
}
sl@0
   576
sl@0
   577
/**
sl@0
   578
 * g_boxed_free:
sl@0
   579
 * @boxed_type: The type of @boxed.
sl@0
   580
 * @boxed: The boxed structure to be freed.
sl@0
   581
 *
sl@0
   582
 * Free the boxed structure @boxed which is of type @boxed_type.
sl@0
   583
 */
sl@0
   584
EXPORT_C void
sl@0
   585
g_boxed_free (GType    boxed_type,
sl@0
   586
	      gpointer boxed)
sl@0
   587
{
sl@0
   588
  GTypeValueTable *value_table;
sl@0
   589
sl@0
   590
  g_return_if_fail (G_TYPE_IS_BOXED (boxed_type));
sl@0
   591
  g_return_if_fail (G_TYPE_IS_ABSTRACT (boxed_type) == FALSE);
sl@0
   592
  g_return_if_fail (boxed != NULL);
sl@0
   593
sl@0
   594
  value_table = g_type_value_table_peek (boxed_type);
sl@0
   595
  if (!value_table)
sl@0
   596
    g_return_if_fail (G_TYPE_IS_VALUE_TYPE (boxed_type));
sl@0
   597
sl@0
   598
  /* check if our proxying implementation is used, we can short-cut here */
sl@0
   599
  if (value_table->value_free == boxed_proxy_value_free)
sl@0
   600
    {
sl@0
   601
      BoxedNode key, *node;
sl@0
   602
sl@0
   603
      key.type = boxed_type;
sl@0
   604
      node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   605
      node->free (boxed);
sl@0
   606
    }
sl@0
   607
  else
sl@0
   608
    {
sl@0
   609
      GValue value;
sl@0
   610
sl@0
   611
      /* see g_boxed_copy() on why we think we can do this */
sl@0
   612
      value_meminit (&value, boxed_type);
sl@0
   613
      value.data[0].v_pointer = boxed;
sl@0
   614
      value_table->value_free (&value);
sl@0
   615
    }
sl@0
   616
}
sl@0
   617
sl@0
   618
/**
sl@0
   619
 * g_value_get_boxed:
sl@0
   620
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   621
 *
sl@0
   622
 * Get the contents of a %G_TYPE_BOXED derived #GValue.
sl@0
   623
 *
sl@0
   624
 * Returns: boxed contents of @value
sl@0
   625
 */
sl@0
   626
EXPORT_C gpointer
sl@0
   627
g_value_get_boxed (const GValue *value)
sl@0
   628
{
sl@0
   629
  g_return_val_if_fail (G_VALUE_HOLDS_BOXED (value), NULL);
sl@0
   630
  g_return_val_if_fail (G_TYPE_IS_VALUE (G_VALUE_TYPE (value)), NULL);
sl@0
   631
sl@0
   632
  return value->data[0].v_pointer;
sl@0
   633
}
sl@0
   634
sl@0
   635
/**
sl@0
   636
 * g_value_dup_boxed:
sl@0
   637
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   638
 *
sl@0
   639
 * Get the contents of a %G_TYPE_BOXED derived #GValue.  Upon getting,
sl@0
   640
 * the boxed value is duplicated and needs to be later freed with
sl@0
   641
 * g_boxed_free(), e.g. like: g_boxed_free (G_VALUE_TYPE (@value),
sl@0
   642
 * return_value);
sl@0
   643
 *
sl@0
   644
 * Returns: boxed contents of @value
sl@0
   645
 */
sl@0
   646
EXPORT_C gpointer
sl@0
   647
g_value_dup_boxed (const GValue *value)
sl@0
   648
{
sl@0
   649
  g_return_val_if_fail (G_VALUE_HOLDS_BOXED (value), NULL);
sl@0
   650
  g_return_val_if_fail (G_TYPE_IS_VALUE (G_VALUE_TYPE (value)), NULL);
sl@0
   651
sl@0
   652
  return value->data[0].v_pointer ? g_boxed_copy (G_VALUE_TYPE (value), value->data[0].v_pointer) : NULL;
sl@0
   653
}
sl@0
   654
sl@0
   655
static inline void
sl@0
   656
value_set_boxed_internal (GValue       *value,
sl@0
   657
			  gconstpointer const_boxed,
sl@0
   658
			  gboolean      need_copy,
sl@0
   659
			  gboolean      need_free)
sl@0
   660
{
sl@0
   661
  BoxedNode key, *node;
sl@0
   662
  gpointer boxed = (gpointer) const_boxed;
sl@0
   663
sl@0
   664
  if (!boxed)
sl@0
   665
    {
sl@0
   666
      /* just resetting to NULL might not be desired, need to
sl@0
   667
       * have value reinitialized also (for values defaulting
sl@0
   668
       * to other default value states than a NULL data pointer),
sl@0
   669
       * g_value_reset() will handle this
sl@0
   670
       */
sl@0
   671
      g_value_reset (value);
sl@0
   672
      return;
sl@0
   673
    }
sl@0
   674
sl@0
   675
  key.type = G_VALUE_TYPE (value);
sl@0
   676
  node = g_bsearch_array_lookup (boxed_bsa, &boxed_bconfig, &key);
sl@0
   677
sl@0
   678
  if (node)
sl@0
   679
    {
sl@0
   680
      /* we proxy this type, free contents and copy right away */
sl@0
   681
      if (value->data[0].v_pointer && !(value->data[1].v_uint & G_VALUE_NOCOPY_CONTENTS))
sl@0
   682
	node->free (value->data[0].v_pointer);
sl@0
   683
      value->data[1].v_uint = need_free ? 0 : G_VALUE_NOCOPY_CONTENTS;
sl@0
   684
      value->data[0].v_pointer = need_copy ? node->copy (boxed) : boxed;
sl@0
   685
    }
sl@0
   686
  else
sl@0
   687
    {
sl@0
   688
      /* we don't handle this type, free contents and let g_boxed_copy()
sl@0
   689
       * figure what's required
sl@0
   690
       */
sl@0
   691
      if (value->data[0].v_pointer && !(value->data[1].v_uint & G_VALUE_NOCOPY_CONTENTS))
sl@0
   692
	g_boxed_free (G_VALUE_TYPE (value), value->data[0].v_pointer);
sl@0
   693
      value->data[1].v_uint = need_free ? 0 : G_VALUE_NOCOPY_CONTENTS;
sl@0
   694
      value->data[0].v_pointer = need_copy ? g_boxed_copy (G_VALUE_TYPE (value), boxed) : boxed;
sl@0
   695
    }
sl@0
   696
}
sl@0
   697
sl@0
   698
/**
sl@0
   699
 * g_value_set_boxed:
sl@0
   700
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   701
 * @v_boxed: boxed value to be set
sl@0
   702
 *
sl@0
   703
 * Set the contents of a %G_TYPE_BOXED derived #GValue to @v_boxed.
sl@0
   704
 */
sl@0
   705
EXPORT_C void
sl@0
   706
g_value_set_boxed (GValue       *value,
sl@0
   707
		   gconstpointer boxed)
sl@0
   708
{
sl@0
   709
  g_return_if_fail (G_VALUE_HOLDS_BOXED (value));
sl@0
   710
  g_return_if_fail (G_TYPE_IS_VALUE (G_VALUE_TYPE (value)));
sl@0
   711
sl@0
   712
  value_set_boxed_internal (value, boxed, TRUE, TRUE);
sl@0
   713
}
sl@0
   714
sl@0
   715
/**
sl@0
   716
 * g_value_set_static_boxed:
sl@0
   717
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   718
 * @v_boxed: static boxed value to be set
sl@0
   719
 *
sl@0
   720
 * Set the contents of a %G_TYPE_BOXED derived #GValue to @v_boxed.
sl@0
   721
 * The boxed value is assumed to be static, and is thus not duplicated
sl@0
   722
 * when setting the #GValue.
sl@0
   723
 */
sl@0
   724
EXPORT_C void
sl@0
   725
g_value_set_static_boxed (GValue       *value,
sl@0
   726
			  gconstpointer boxed)
sl@0
   727
{
sl@0
   728
  g_return_if_fail (G_VALUE_HOLDS_BOXED (value));
sl@0
   729
  g_return_if_fail (G_TYPE_IS_VALUE (G_VALUE_TYPE (value)));
sl@0
   730
sl@0
   731
  value_set_boxed_internal (value, boxed, FALSE, FALSE);
sl@0
   732
}
sl@0
   733
sl@0
   734
/**
sl@0
   735
 * g_value_set_boxed_take_ownership:
sl@0
   736
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   737
 * @v_boxed: duplicated unowned boxed value to be set
sl@0
   738
 *
sl@0
   739
 * This is an internal function introduced mainly for C marshallers.
sl@0
   740
 *
sl@0
   741
 * Deprecated: 2.4: Use g_value_take_boxed() instead.
sl@0
   742
 */
sl@0
   743
EXPORT_C void
sl@0
   744
g_value_set_boxed_take_ownership (GValue       *value,
sl@0
   745
				  gconstpointer boxed)
sl@0
   746
{
sl@0
   747
  g_value_take_boxed (value, boxed);
sl@0
   748
}
sl@0
   749
sl@0
   750
/**
sl@0
   751
 * g_value_take_boxed:
sl@0
   752
 * @value: a valid #GValue of %G_TYPE_BOXED derived type
sl@0
   753
 * @v_boxed: duplicated unowned boxed value to be set
sl@0
   754
 *
sl@0
   755
 * Sets the contents of a %G_TYPE_BOXED derived #GValue to @v_boxed
sl@0
   756
 * and takes over the ownership of the callers reference to @v_boxed;
sl@0
   757
 * the caller doesn't have to unref it any more.
sl@0
   758
 *
sl@0
   759
 * Since: 2.4
sl@0
   760
 */
sl@0
   761
EXPORT_C void
sl@0
   762
g_value_take_boxed (GValue       *value,
sl@0
   763
		    gconstpointer boxed)
sl@0
   764
{
sl@0
   765
  g_return_if_fail (G_VALUE_HOLDS_BOXED (value));
sl@0
   766
  g_return_if_fail (G_TYPE_IS_VALUE (G_VALUE_TYPE (value)));
sl@0
   767
sl@0
   768
  value_set_boxed_internal (value, boxed, FALSE, TRUE);
sl@0
   769
}
sl@0
   770
sl@0
   771
#define __G_BOXED_C__
sl@0
   772
#include "gobjectaliasdef.c"