os/ossrv/glib/gobject/gvalue.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) 1997-1999, 2000-2001 Tim Janik and 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
/*
sl@0
    22
 * FIXME: MT-safety
sl@0
    23
 */
sl@0
    24
sl@0
    25
#include "config.h"
sl@0
    26
sl@0
    27
#include <string.h>
sl@0
    28
sl@0
    29
#include "gvalue.h"
sl@0
    30
#include "gvaluecollector.h"
sl@0
    31
#include "gbsearcharray.h"
sl@0
    32
#include "gobjectalias.h"
sl@0
    33
sl@0
    34
#ifdef __SYMBIAN32__
sl@0
    35
#include "gobject_wsd.h"
sl@0
    36
#endif /* __SYMBIAN32__ */
sl@0
    37
sl@0
    38
/**
sl@0
    39
 * SECTION:generic_values
sl@0
    40
 * @short_description: A polymorphic type that can hold values of any
sl@0
    41
 *     other type
sl@0
    42
 * @see_also: The fundamental types which all support #GValue
sl@0
    43
 *     operations and thus can be used as a type initializer for
sl@0
    44
 *     g_value_init() are defined by a separate interface.  See the <link
sl@0
    45
 *     linkend="gobject-Standard-Parameter-and-Value-Types">Standard
sl@0
    46
 *     Values API</link> for details.
sl@0
    47
 * @title: Generic values
sl@0
    48
 *
sl@0
    49
 * The #GValue structure is basically a variable container that consists
sl@0
    50
 * of a type identifier and a specific value of that type.
sl@0
    51
 * The type identifier within a #GValue structure always determines the
sl@0
    52
 * type of the associated value.
sl@0
    53
 * To create a undefined #GValue structure, simply create a zero-filled
sl@0
    54
 * #GValue structure. To initialize the #GValue, use the g_value_init()
sl@0
    55
 * function. A #GValue cannot be used until it is initialized.
sl@0
    56
 * The basic type operations (such as freeing and copying) are determined
sl@0
    57
 * by the #GTypeValueTable associated with the type ID stored in the #GValue.
sl@0
    58
 * Other #GValue operations (such as converting values between types) are
sl@0
    59
 * provided by this interface.
sl@0
    60
 *
sl@0
    61
 * The code in the example program below demonstrates #GValue's
sl@0
    62
 * features.
sl@0
    63
 *
sl@0
    64
 * |[
sl@0
    65
 * #include &lt;glib-object.h&gt;
sl@0
    66
 *
sl@0
    67
 * static void
sl@0
    68
 * int2string (const GValue *src_value,
sl@0
    69
 *             GValue       *dest_value)
sl@0
    70
 * {
sl@0
    71
 *   if (g_value_get_int (src_value) == 42)
sl@0
    72
 *     g_value_set_static_string (dest_value, "An important number");
sl@0
    73
 *   else
sl@0
    74
 *     g_value_set_static_string (dest_value, "What's that?");
sl@0
    75
 * }
sl@0
    76
 *
sl@0
    77
 * int
sl@0
    78
 * main (int   argc,
sl@0
    79
 *       char *argv[])
sl@0
    80
 * {
sl@0
    81
 *   /&ast; GValues must start zero-filled &ast;/
sl@0
    82
 *   GValue a = {0};
sl@0
    83
 *   GValue b = {0};
sl@0
    84
 *   const gchar *message;
sl@0
    85
 *
sl@0
    86
 *   g_type_init ();
sl@0
    87
 *
sl@0
    88
 *   /&ast; The GValue starts empty &ast;/
sl@0
    89
 *   g_assert (!G_VALUE_HOLDS_STRING (&amp;a));
sl@0
    90
 *
sl@0
    91
 *   /&ast; Put a string in it &ast;/
sl@0
    92
 *   g_value_init (&amp;a, G_TYPE_STRING);
sl@0
    93
 *   g_assert (G_VALUE_HOLDS_STRING (&amp;a));
sl@0
    94
 *   g_value_set_static_string (&amp;a, "Hello, world!");
sl@0
    95
 *   g_printf ("%s\n", g_value_get_string (&amp;a));
sl@0
    96
 *
sl@0
    97
 *   /&ast; Reset it to its pristine state &ast;/
sl@0
    98
 *   g_value_unset (&amp;a);
sl@0
    99
 *
sl@0
   100
 *   /&ast; It can then be reused for another type &ast;/
sl@0
   101
 *   g_value_init (&amp;a, G_TYPE_INT);
sl@0
   102
 *   g_value_set_int (&amp;a, 42);
sl@0
   103
 *
sl@0
   104
 *   /&ast; Attempt to transform it into a GValue of type STRING &ast;/
sl@0
   105
 *   g_value_init (&amp;b, G_TYPE_STRING);
sl@0
   106
 *
sl@0
   107
 *   /&ast; An INT is transformable to a STRING &ast;/
sl@0
   108
 *   g_assert (g_value_type_transformable (G_TYPE_INT, G_TYPE_STRING));
sl@0
   109
 *
sl@0
   110
 *   g_value_transform (&amp;a, &amp;b);
sl@0
   111
 *   g_printf ("%s\n", g_value_get_string (&amp;b));
sl@0
   112
 *
sl@0
   113
 *   /&ast; Attempt to transform it again using a custom transform function &ast;/
sl@0
   114
 *   g_value_register_transform_func (G_TYPE_INT, G_TYPE_STRING, int2string);
sl@0
   115
 *   g_value_transform (&amp;a, &amp;b);
sl@0
   116
 *   g_printf ("%s\n", g_value_get_string (&amp;b));
sl@0
   117
 *   return 0;
sl@0
   118
 * }
sl@0
   119
 * ]|
sl@0
   120
 */
sl@0
   121
sl@0
   122
sl@0
   123
/* --- typedefs & structures --- */
sl@0
   124
typedef struct {
sl@0
   125
  GType src_type;
sl@0
   126
  GType dest_type;
sl@0
   127
  GValueTransform func;
sl@0
   128
} TransformEntry;
sl@0
   129
sl@0
   130
sl@0
   131
/* --- prototypes --- */
sl@0
   132
static gint	transform_entries_cmp	(gconstpointer bsearch_node1,
sl@0
   133
					 gconstpointer bsearch_node2);
sl@0
   134
sl@0
   135
sl@0
   136
/* --- variables --- */
sl@0
   137
#if EMULATOR
sl@0
   138
sl@0
   139
PLS(transform_array,gvalue,GBSearchArray *) 
sl@0
   140
PLS(transform_bconfig,gvalue,GBSearchConfig )
sl@0
   141
sl@0
   142
#define transform_array (*FUNCTION_NAME(transform_array,gvalue)())
sl@0
   143
#define transform_bconfig  (*FUNCTION_NAME(transform_bconfig ,gvalue)())
sl@0
   144
sl@0
   145
const GBSearchConfig temp_transform_bconfig = {
sl@0
   146
  sizeof (TransformEntry),
sl@0
   147
  transform_entries_cmp,
sl@0
   148
  0,
sl@0
   149
};
sl@0
   150
sl@0
   151
sl@0
   152
#else
sl@0
   153
sl@0
   154
static GBSearchArray *transform_array = NULL;
sl@0
   155
static GBSearchConfig transform_bconfig = {
sl@0
   156
  sizeof (TransformEntry),
sl@0
   157
  transform_entries_cmp,
sl@0
   158
  0,
sl@0
   159
};
sl@0
   160
sl@0
   161
#endif /* EMULATOR */
sl@0
   162
sl@0
   163
sl@0
   164
/* --- functions --- */
sl@0
   165
void
sl@0
   166
g_value_c_init (void)
sl@0
   167
{
sl@0
   168
  transform_array = g_bsearch_array_create (&transform_bconfig);
sl@0
   169
}
sl@0
   170
sl@0
   171
static inline void		/* keep this function in sync with gvaluecollector.h and gboxed.c */
sl@0
   172
value_meminit (GValue *value,
sl@0
   173
	       GType   value_type)
sl@0
   174
{
sl@0
   175
  value->g_type = value_type;
sl@0
   176
  memset (value->data, 0, sizeof (value->data));
sl@0
   177
}
sl@0
   178
sl@0
   179
/**
sl@0
   180
 * g_value_init:
sl@0
   181
 * @value: A zero-filled (uninitialized) #GValue structure.
sl@0
   182
 * @g_type: Type the #GValue should hold values of.
sl@0
   183
 *
sl@0
   184
 * Initializes @value with the default value of @type.
sl@0
   185
 *
sl@0
   186
 * Returns: the #GValue structure that has been passed in
sl@0
   187
 */
sl@0
   188
EXPORT_C GValue*
sl@0
   189
g_value_init (GValue *value,
sl@0
   190
	      GType   g_type)
sl@0
   191
{
sl@0
   192
  /* g_return_val_if_fail (G_TYPE_IS_VALUE (g_type), NULL);	be more elaborate below */
sl@0
   193
  g_return_val_if_fail (value != NULL, NULL);
sl@0
   194
  /* g_return_val_if_fail (G_VALUE_TYPE (value) == 0, NULL);	be more elaborate below */
sl@0
   195
sl@0
   196
  if (G_TYPE_IS_VALUE (g_type) && G_VALUE_TYPE (value) == 0)
sl@0
   197
    {
sl@0
   198
      GTypeValueTable *value_table = g_type_value_table_peek (g_type);
sl@0
   199
sl@0
   200
      /* setup and init */
sl@0
   201
      value_meminit (value, g_type);
sl@0
   202
      value_table->value_init (value);
sl@0
   203
    }
sl@0
   204
  else if (G_VALUE_TYPE (value))
sl@0
   205
    g_warning ("%s: cannot initialize GValue with type `%s', the value has already been initialized as `%s'",
sl@0
   206
	       G_STRLOC,
sl@0
   207
	       g_type_name (g_type),
sl@0
   208
	       g_type_name (G_VALUE_TYPE (value)));
sl@0
   209
  else /* !G_TYPE_IS_VALUE (g_type) */
sl@0
   210
    g_warning ("%s: cannot initialize GValue with type `%s', %s",
sl@0
   211
	       G_STRLOC,
sl@0
   212
	       g_type_name (g_type),
sl@0
   213
	       g_type_value_table_peek (g_type) ?
sl@0
   214
	       "this type is abstract with regards to GValue use, use a more specific (derived) type" :
sl@0
   215
	       "this type has no GTypeValueTable implementation");
sl@0
   216
  return value;
sl@0
   217
}
sl@0
   218
sl@0
   219
/**
sl@0
   220
 * g_value_copy:
sl@0
   221
 * @src_value: An initialized #GValue structure.
sl@0
   222
 * @dest_value: An initialized #GValue structure of the same type as @src_value.
sl@0
   223
 *
sl@0
   224
 * Copies the value of @src_value into @dest_value.
sl@0
   225
 */
sl@0
   226
EXPORT_C void
sl@0
   227
g_value_copy (const GValue *src_value,
sl@0
   228
	      GValue       *dest_value)
sl@0
   229
{
sl@0
   230
  g_return_if_fail (G_IS_VALUE (src_value));
sl@0
   231
  g_return_if_fail (G_IS_VALUE (dest_value));
sl@0
   232
  g_return_if_fail (g_value_type_compatible (G_VALUE_TYPE (src_value), G_VALUE_TYPE (dest_value)));
sl@0
   233
  
sl@0
   234
  if (src_value != dest_value)
sl@0
   235
    {
sl@0
   236
      GType dest_type = G_VALUE_TYPE (dest_value);
sl@0
   237
      GTypeValueTable *value_table = g_type_value_table_peek (dest_type);
sl@0
   238
sl@0
   239
      /* make sure dest_value's value is free()d */
sl@0
   240
      if (value_table->value_free)
sl@0
   241
	value_table->value_free (dest_value);
sl@0
   242
sl@0
   243
      /* setup and copy */
sl@0
   244
      value_meminit (dest_value, dest_type);
sl@0
   245
      value_table->value_copy (src_value, dest_value);
sl@0
   246
    }
sl@0
   247
}
sl@0
   248
sl@0
   249
/**
sl@0
   250
 * g_value_reset:
sl@0
   251
 * @value: An initialized #GValue structure.
sl@0
   252
 *
sl@0
   253
 * Clears the current value in @value and resets it to the default value
sl@0
   254
 * (as if the value had just been initialized).
sl@0
   255
 *
sl@0
   256
 * Returns: the #GValue structure that has been passed in
sl@0
   257
 */
sl@0
   258
EXPORT_C GValue*
sl@0
   259
g_value_reset (GValue *value)
sl@0
   260
{
sl@0
   261
  GTypeValueTable *value_table;
sl@0
   262
  GType g_type;
sl@0
   263
  
sl@0
   264
  g_return_val_if_fail (G_IS_VALUE (value), NULL);
sl@0
   265
  
sl@0
   266
  g_type = G_VALUE_TYPE (value);
sl@0
   267
  value_table = g_type_value_table_peek (g_type);
sl@0
   268
sl@0
   269
  /* make sure value's value is free()d */
sl@0
   270
  if (value_table->value_free)
sl@0
   271
    value_table->value_free (value);
sl@0
   272
sl@0
   273
  /* setup and init */
sl@0
   274
  value_meminit (value, g_type);
sl@0
   275
  value_table->value_init (value);
sl@0
   276
sl@0
   277
  return value;
sl@0
   278
}
sl@0
   279
sl@0
   280
/**
sl@0
   281
 * g_value_unset:
sl@0
   282
 * @value: An initialized #GValue structure.
sl@0
   283
 *
sl@0
   284
 * Clears the current value in @value and "unsets" the type,
sl@0
   285
 * this releases all resources associated with this GValue.
sl@0
   286
 * An unset value is the same as an uninitialized (zero-filled)
sl@0
   287
 * #GValue structure.
sl@0
   288
 */
sl@0
   289
EXPORT_C void
sl@0
   290
g_value_unset (GValue *value)
sl@0
   291
{
sl@0
   292
  GTypeValueTable *value_table;
sl@0
   293
  
sl@0
   294
  g_return_if_fail (G_IS_VALUE (value));
sl@0
   295
sl@0
   296
  value_table = g_type_value_table_peek (G_VALUE_TYPE (value));
sl@0
   297
sl@0
   298
  if (value_table->value_free)
sl@0
   299
    value_table->value_free (value);
sl@0
   300
  memset (value, 0, sizeof (*value));
sl@0
   301
}
sl@0
   302
sl@0
   303
/**
sl@0
   304
 * g_value_fits_pointer:
sl@0
   305
 * @value: An initialized #GValue structure.
sl@0
   306
 *
sl@0
   307
 * Determines if @value will fit inside the size of a pointer value.
sl@0
   308
 * This is an internal function introduced mainly for C marshallers.
sl@0
   309
 *
sl@0
   310
 * Returns: %TRUE if @value will fit inside a pointer value.
sl@0
   311
 */
sl@0
   312
EXPORT_C gboolean
sl@0
   313
g_value_fits_pointer (const GValue *value)
sl@0
   314
{
sl@0
   315
  GTypeValueTable *value_table;
sl@0
   316
sl@0
   317
  g_return_val_if_fail (G_IS_VALUE (value), FALSE);
sl@0
   318
sl@0
   319
  value_table = g_type_value_table_peek (G_VALUE_TYPE (value));
sl@0
   320
sl@0
   321
  return value_table->value_peek_pointer != NULL;
sl@0
   322
}
sl@0
   323
sl@0
   324
/**
sl@0
   325
 * g_value_peek_pointer:
sl@0
   326
 * @value: An initialized #GValue structure.
sl@0
   327
 *
sl@0
   328
 * Return the value contents as pointer. This function asserts that
sl@0
   329
 * g_value_fits_pointer() returned %TRUE for the passed in value.
sl@0
   330
 * This is an internal function introduced mainly for C marshallers.
sl@0
   331
 *
sl@0
   332
 * Returns: %TRUE if @value will fit inside a pointer value.
sl@0
   333
 */
sl@0
   334
EXPORT_C gpointer
sl@0
   335
g_value_peek_pointer (const GValue *value)
sl@0
   336
{
sl@0
   337
  GTypeValueTable *value_table;
sl@0
   338
sl@0
   339
  g_return_val_if_fail (G_IS_VALUE (value), NULL);
sl@0
   340
sl@0
   341
  value_table = g_type_value_table_peek (G_VALUE_TYPE (value));
sl@0
   342
  if (!value_table->value_peek_pointer)
sl@0
   343
    {
sl@0
   344
      g_return_val_if_fail (g_value_fits_pointer (value) == TRUE, NULL);
sl@0
   345
      return NULL;
sl@0
   346
    }
sl@0
   347
sl@0
   348
  return value_table->value_peek_pointer (value);
sl@0
   349
}
sl@0
   350
sl@0
   351
/**
sl@0
   352
 * g_value_set_instance:
sl@0
   353
 * @value: An initialized #GValue structure.
sl@0
   354
 * @instance: the instance
sl@0
   355
 *
sl@0
   356
 * Sets @value from an instantiatable type via the
sl@0
   357
 * value_table's collect_value() function.
sl@0
   358
 */
sl@0
   359
EXPORT_C void
sl@0
   360
g_value_set_instance (GValue  *value,
sl@0
   361
		      gpointer instance)
sl@0
   362
{
sl@0
   363
  GType g_type;
sl@0
   364
  GTypeValueTable *value_table;
sl@0
   365
  GTypeCValue cvalue;
sl@0
   366
  gchar *error_msg;
sl@0
   367
  
sl@0
   368
  g_return_if_fail (G_IS_VALUE (value));
sl@0
   369
  if (instance)
sl@0
   370
    {
sl@0
   371
      g_return_if_fail (G_TYPE_CHECK_INSTANCE (instance));
sl@0
   372
      g_return_if_fail (g_value_type_compatible (G_TYPE_FROM_INSTANCE (instance), G_VALUE_TYPE (value)));
sl@0
   373
    }
sl@0
   374
  
sl@0
   375
  g_type = G_VALUE_TYPE (value);
sl@0
   376
  value_table = g_type_value_table_peek (g_type);
sl@0
   377
  
sl@0
   378
  g_return_if_fail (strcmp (value_table->collect_format, "p") == 0);
sl@0
   379
  
sl@0
   380
  memset (&cvalue, 0, sizeof (cvalue));
sl@0
   381
  cvalue.v_pointer = instance;
sl@0
   382
  
sl@0
   383
  /* make sure value's value is free()d */
sl@0
   384
  if (value_table->value_free)
sl@0
   385
    value_table->value_free (value);
sl@0
   386
sl@0
   387
  /* setup and collect */
sl@0
   388
  value_meminit (value, g_type);
sl@0
   389
  error_msg = value_table->collect_value (value, 1, &cvalue, 0);
sl@0
   390
  if (error_msg)
sl@0
   391
    {
sl@0
   392
      g_warning ("%s: %s", G_STRLOC, error_msg);
sl@0
   393
      g_free (error_msg);
sl@0
   394
      
sl@0
   395
      /* we purposely leak the value here, it might not be
sl@0
   396
       * in a sane state if an error condition occoured
sl@0
   397
       */
sl@0
   398
      value_meminit (value, g_type);
sl@0
   399
      value_table->value_init (value);
sl@0
   400
    }
sl@0
   401
}
sl@0
   402
sl@0
   403
static GValueTransform
sl@0
   404
transform_func_lookup (GType src_type,
sl@0
   405
		       GType dest_type)
sl@0
   406
{
sl@0
   407
  TransformEntry entry;
sl@0
   408
sl@0
   409
  entry.src_type = src_type;
sl@0
   410
  do
sl@0
   411
    {
sl@0
   412
      entry.dest_type = dest_type;
sl@0
   413
      do
sl@0
   414
	{
sl@0
   415
	  TransformEntry *e;
sl@0
   416
	  
sl@0
   417
	  e = g_bsearch_array_lookup (transform_array, &transform_bconfig, &entry);
sl@0
   418
	  if (e)
sl@0
   419
	    {
sl@0
   420
	      /* need to check that there hasn't been a change in value handling */
sl@0
   421
	      if (g_type_value_table_peek (entry.dest_type) == g_type_value_table_peek (dest_type) &&
sl@0
   422
		  g_type_value_table_peek (entry.src_type) == g_type_value_table_peek (src_type))
sl@0
   423
		return e->func;
sl@0
   424
	    }
sl@0
   425
	  entry.dest_type = g_type_parent (entry.dest_type);
sl@0
   426
	}
sl@0
   427
      while (entry.dest_type);
sl@0
   428
      
sl@0
   429
      entry.src_type = g_type_parent (entry.src_type);
sl@0
   430
    }
sl@0
   431
  while (entry.src_type);
sl@0
   432
sl@0
   433
  return NULL;
sl@0
   434
}
sl@0
   435
sl@0
   436
static gint
sl@0
   437
transform_entries_cmp (gconstpointer bsearch_node1,
sl@0
   438
		       gconstpointer bsearch_node2)
sl@0
   439
{
sl@0
   440
  const TransformEntry *e1 = bsearch_node1;
sl@0
   441
  const TransformEntry *e2 = bsearch_node2;
sl@0
   442
  gint cmp = G_BSEARCH_ARRAY_CMP (e1->src_type, e2->src_type);
sl@0
   443
sl@0
   444
  if (cmp)
sl@0
   445
    return cmp;
sl@0
   446
  else
sl@0
   447
    return G_BSEARCH_ARRAY_CMP (e1->dest_type, e2->dest_type);
sl@0
   448
}
sl@0
   449
sl@0
   450
/**
sl@0
   451
 * g_value_register_transform_func:
sl@0
   452
 * @src_type: Source type.
sl@0
   453
 * @dest_type: Target type.
sl@0
   454
 * @transform_func: a function which transforms values of type @src_type
sl@0
   455
 *  into value of type @dest_type
sl@0
   456
 *
sl@0
   457
 * Registers a value transformation function for use in g_value_transform().
sl@0
   458
 * A previously registered transformation function for @src_type and @dest_type
sl@0
   459
 * will be replaced.
sl@0
   460
 */
sl@0
   461
EXPORT_C void
sl@0
   462
g_value_register_transform_func (GType           src_type,
sl@0
   463
				 GType           dest_type,
sl@0
   464
				 GValueTransform transform_func)
sl@0
   465
{
sl@0
   466
  TransformEntry entry;
sl@0
   467
sl@0
   468
  /* these checks won't pass for dynamic types.
sl@0
   469
   * g_return_if_fail (G_TYPE_HAS_VALUE_TABLE (src_type));
sl@0
   470
   * g_return_if_fail (G_TYPE_HAS_VALUE_TABLE (dest_type));
sl@0
   471
   */
sl@0
   472
  g_return_if_fail (transform_func != NULL);
sl@0
   473
sl@0
   474
  entry.src_type = src_type;
sl@0
   475
  entry.dest_type = dest_type;
sl@0
   476
sl@0
   477
#if 0 /* let transform function replacement be a valid operation */
sl@0
   478
  if (g_bsearch_array_lookup (transform_array, &transform_bconfig, &entry))
sl@0
   479
    g_warning ("reregistering value transformation function (%p) for `%s' to `%s'",
sl@0
   480
	       transform_func,
sl@0
   481
	       g_type_name (src_type),
sl@0
   482
	       g_type_name (dest_type));
sl@0
   483
#endif
sl@0
   484
sl@0
   485
  entry.func = transform_func;
sl@0
   486
  transform_array = g_bsearch_array_replace (transform_array, &transform_bconfig, &entry);
sl@0
   487
}
sl@0
   488
sl@0
   489
/**
sl@0
   490
 * g_value_type_transformable:
sl@0
   491
 * @src_type: Source type.
sl@0
   492
 * @dest_type: Target type.
sl@0
   493
 *
sl@0
   494
 * Check whether g_value_transform() is able to transform values
sl@0
   495
 * of type @src_type into values of type @dest_type.
sl@0
   496
 *
sl@0
   497
 * Returns: %TRUE if the transformation is possible, %FALSE otherwise.
sl@0
   498
 */
sl@0
   499
EXPORT_C gboolean
sl@0
   500
g_value_type_transformable (GType src_type,
sl@0
   501
			    GType dest_type)
sl@0
   502
{
sl@0
   503
  g_return_val_if_fail (G_TYPE_IS_VALUE (src_type), FALSE);
sl@0
   504
  g_return_val_if_fail (G_TYPE_IS_VALUE (dest_type), FALSE);
sl@0
   505
sl@0
   506
  return (g_value_type_compatible (src_type, dest_type) ||
sl@0
   507
	  transform_func_lookup (src_type, dest_type) != NULL);
sl@0
   508
}
sl@0
   509
sl@0
   510
/**
sl@0
   511
 * g_value_type_compatible:
sl@0
   512
 * @src_type: source type to be copied.
sl@0
   513
 * @dest_type: destination type for copying.
sl@0
   514
 *
sl@0
   515
 * Returns whether a #GValue of type @src_type can be copied into
sl@0
   516
 * a #GValue of type @dest_type.
sl@0
   517
 *
sl@0
   518
 * Returns: %TRUE if g_value_copy() is possible with @src_type and @dest_type.
sl@0
   519
 */
sl@0
   520
EXPORT_C gboolean
sl@0
   521
g_value_type_compatible (GType src_type,
sl@0
   522
			 GType dest_type)
sl@0
   523
{
sl@0
   524
  g_return_val_if_fail (G_TYPE_IS_VALUE (src_type), FALSE);
sl@0
   525
  g_return_val_if_fail (G_TYPE_IS_VALUE (dest_type), FALSE);
sl@0
   526
sl@0
   527
  return (g_type_is_a (src_type, dest_type) &&
sl@0
   528
	  g_type_value_table_peek (dest_type) == g_type_value_table_peek (src_type));
sl@0
   529
}
sl@0
   530
sl@0
   531
/**
sl@0
   532
 * g_value_transform:
sl@0
   533
 * @src_value: Source value.
sl@0
   534
 * @dest_value: Target value.
sl@0
   535
 *
sl@0
   536
 * Tries to cast the contents of @src_value into a type appropriate
sl@0
   537
 * to store in @dest_value, e.g. to transform a %G_TYPE_INT value
sl@0
   538
 * into a %G_TYPE_FLOAT value. Performing transformations between
sl@0
   539
 * value types might incur precision lossage. Especially
sl@0
   540
 * transformations into strings might reveal seemingly arbitrary
sl@0
   541
 * results and shouldn't be relied upon for production code (such
sl@0
   542
 * as rcfile value or object property serialization).
sl@0
   543
 *
sl@0
   544
 * Returns: Whether a transformation rule was found and could be applied.
sl@0
   545
 *  Upon failing transformations, @dest_value is left untouched.
sl@0
   546
 */
sl@0
   547
EXPORT_C gboolean
sl@0
   548
g_value_transform (const GValue *src_value,
sl@0
   549
		   GValue       *dest_value)
sl@0
   550
{
sl@0
   551
  GType dest_type;
sl@0
   552
sl@0
   553
  g_return_val_if_fail (G_IS_VALUE (src_value), FALSE);
sl@0
   554
  g_return_val_if_fail (G_IS_VALUE (dest_value), FALSE);
sl@0
   555
sl@0
   556
  dest_type = G_VALUE_TYPE (dest_value);
sl@0
   557
  if (g_value_type_compatible (G_VALUE_TYPE (src_value), dest_type))
sl@0
   558
    {
sl@0
   559
      g_value_copy (src_value, dest_value);
sl@0
   560
      
sl@0
   561
      return TRUE;
sl@0
   562
    }
sl@0
   563
  else
sl@0
   564
    {
sl@0
   565
      GValueTransform transform = transform_func_lookup (G_VALUE_TYPE (src_value), dest_type);
sl@0
   566
sl@0
   567
      if (transform)
sl@0
   568
	{
sl@0
   569
	  g_value_unset (dest_value);
sl@0
   570
	  
sl@0
   571
	  /* setup and transform */
sl@0
   572
	  value_meminit (dest_value, dest_type);
sl@0
   573
	  transform (src_value, dest_value);
sl@0
   574
	  
sl@0
   575
	  return TRUE;
sl@0
   576
	}
sl@0
   577
    }
sl@0
   578
  return FALSE;
sl@0
   579
}
sl@0
   580
sl@0
   581
#define __G_VALUE_C__
sl@0
   582
#include "gobjectaliasdef.c"