os/ossrv/glib/gobject/gparam.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
 * MT safe
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 "gparam.h"
sl@0
    30
#include "gparamspecs.h"
sl@0
    31
#include "gvaluecollector.h"
sl@0
    32
#include "gobjectalias.h"
sl@0
    33
sl@0
    34
#ifdef __SYMBIAN32__
sl@0
    35
#include <glib_global.h>
sl@0
    36
#include "gobject_wsd.h"
sl@0
    37
#include <gobject_global.h>
sl@0
    38
#endif /* __SYMBIAN32__ */
sl@0
    39
/**
sl@0
    40
 * SECTION:gparamspec
sl@0
    41
 * @short_description: Metadata for parameter specifications
sl@0
    42
 * @see_also: g_object_class_install_property(), g_object_set(),
sl@0
    43
 *     g_object_get(), g_object_set_property(), g_object_get_property(),
sl@0
    44
 *     g_value_register_transform_func()
sl@0
    45
 * @title: GParamSpec
sl@0
    46
 *
sl@0
    47
 * #GParamSpec is an object structure that encapsulates the metadata
sl@0
    48
 * required to specify parameters, such as e.g. #GObject properties.
sl@0
    49
 *
sl@0
    50
 * <para id="canonical-parameter-name">
sl@0
    51
 * Parameter names need to start with a letter (a-z or A-Z). Subsequent
sl@0
    52
 * characters can be letters, numbers or a '-'.
sl@0
    53
 * All other characters are replaced by a '-' during construction.
sl@0
    54
 * The result of this replacement is called the canonical name of the
sl@0
    55
 * parameter.
sl@0
    56
 * </para>
sl@0
    57
 */
sl@0
    58
sl@0
    59
sl@0
    60
/* --- defines --- */
sl@0
    61
#define PARAM_FLOATING_FLAG                     0x2
sl@0
    62
#define	G_PARAM_USER_MASK			(~0 << G_PARAM_USER_SHIFT)
sl@0
    63
#define PSPEC_APPLIES_TO_VALUE(pspec, value)	(G_TYPE_CHECK_VALUE_TYPE ((value), G_PARAM_SPEC_VALUE_TYPE (pspec)))
sl@0
    64
#define	G_SLOCK(mutex)				g_static_mutex_lock (mutex)
sl@0
    65
#define	G_SUNLOCK(mutex)			g_static_mutex_unlock (mutex)
sl@0
    66
sl@0
    67
sl@0
    68
/* --- prototypes --- */
sl@0
    69
static void	g_param_spec_class_base_init	 (GParamSpecClass	*class);
sl@0
    70
static void	g_param_spec_class_base_finalize (GParamSpecClass	*class);
sl@0
    71
static void	g_param_spec_class_init		 (GParamSpecClass	*class,
sl@0
    72
						  gpointer               class_data);
sl@0
    73
static void	g_param_spec_init		 (GParamSpec		*pspec,
sl@0
    74
						  GParamSpecClass	*class);
sl@0
    75
static void	g_param_spec_finalize		 (GParamSpec		*pspec);
sl@0
    76
static void	value_param_init		(GValue		*value);
sl@0
    77
static void	value_param_free_value		(GValue		*value);
sl@0
    78
static void	value_param_copy_value		(const GValue	*src_value,
sl@0
    79
						 GValue		*dest_value);
sl@0
    80
static void	value_param_transform_value	(const GValue	*src_value,
sl@0
    81
						 GValue		*dest_value);
sl@0
    82
static gpointer	value_param_peek_pointer	(const GValue	*value);
sl@0
    83
static gchar*	value_param_collect_value	(GValue		*value,
sl@0
    84
						 guint           n_collect_values,
sl@0
    85
						 GTypeCValue    *collect_values,
sl@0
    86
						 guint           collect_flags);
sl@0
    87
static gchar*	value_param_lcopy_value		(const GValue	*value,
sl@0
    88
						 guint           n_collect_values,
sl@0
    89
						 GTypeCValue    *collect_values,
sl@0
    90
						 guint           collect_flags);
sl@0
    91
sl@0
    92
sl@0
    93
/* --- functions --- */
sl@0
    94
void
sl@0
    95
g_param_type_init (void)
sl@0
    96
{
sl@0
    97
  static const GTypeFundamentalInfo finfo = {
sl@0
    98
    (G_TYPE_FLAG_CLASSED |
sl@0
    99
     G_TYPE_FLAG_INSTANTIATABLE |
sl@0
   100
     G_TYPE_FLAG_DERIVABLE |
sl@0
   101
     G_TYPE_FLAG_DEEP_DERIVABLE),
sl@0
   102
  };
sl@0
   103
  static const GTypeValueTable param_value_table = {
sl@0
   104
    value_param_init,           /* value_init */
sl@0
   105
    value_param_free_value,     /* value_free */
sl@0
   106
    value_param_copy_value,     /* value_copy */
sl@0
   107
    value_param_peek_pointer,   /* value_peek_pointer */
sl@0
   108
    "p",			/* collect_format */
sl@0
   109
    value_param_collect_value,  /* collect_value */
sl@0
   110
    "p",			/* lcopy_format */
sl@0
   111
    value_param_lcopy_value,    /* lcopy_value */
sl@0
   112
  };
sl@0
   113
  static const GTypeInfo param_spec_info = {
sl@0
   114
    sizeof (GParamSpecClass),
sl@0
   115
sl@0
   116
    (GBaseInitFunc) g_param_spec_class_base_init,
sl@0
   117
    (GBaseFinalizeFunc) g_param_spec_class_base_finalize,
sl@0
   118
    (GClassInitFunc) g_param_spec_class_init,
sl@0
   119
    (GClassFinalizeFunc) NULL,
sl@0
   120
    NULL,	/* class_data */
sl@0
   121
sl@0
   122
    sizeof (GParamSpec),
sl@0
   123
    0,		/* n_preallocs */
sl@0
   124
    (GInstanceInitFunc) g_param_spec_init,
sl@0
   125
sl@0
   126
    &param_value_table,
sl@0
   127
  };
sl@0
   128
  GType type;
sl@0
   129
sl@0
   130
  /* This should be registred as GParamSpec instead of GParam, for
sl@0
   131
   * consistency sake, so that type name can be mapped to struct name,
sl@0
   132
   * However, some language bindings, most noticable the python ones
sl@0
   133
   * depends on the "GParam" identifier, see #548689
sl@0
   134
   */
sl@0
   135
  type = g_type_register_fundamental (G_TYPE_PARAM, g_intern_static_string ("GParam"), &param_spec_info, &finfo, G_TYPE_FLAG_ABSTRACT);
sl@0
   136
  g_assert (type == G_TYPE_PARAM);
sl@0
   137
  g_value_register_transform_func (G_TYPE_PARAM, G_TYPE_PARAM, value_param_transform_value);
sl@0
   138
}
sl@0
   139
sl@0
   140
static void
sl@0
   141
g_param_spec_class_base_init (GParamSpecClass *class)
sl@0
   142
{
sl@0
   143
}
sl@0
   144
sl@0
   145
static void
sl@0
   146
g_param_spec_class_base_finalize (GParamSpecClass *class)
sl@0
   147
{
sl@0
   148
}
sl@0
   149
sl@0
   150
static void
sl@0
   151
g_param_spec_class_init (GParamSpecClass *class,
sl@0
   152
			 gpointer         class_data)
sl@0
   153
{
sl@0
   154
  class->value_type = G_TYPE_NONE;
sl@0
   155
  class->finalize = g_param_spec_finalize;
sl@0
   156
  class->value_set_default = NULL;
sl@0
   157
  class->value_validate = NULL;
sl@0
   158
  class->values_cmp = NULL;
sl@0
   159
}
sl@0
   160
sl@0
   161
static void
sl@0
   162
g_param_spec_init (GParamSpec      *pspec,
sl@0
   163
		   GParamSpecClass *class)
sl@0
   164
{
sl@0
   165
  pspec->name = NULL;
sl@0
   166
  pspec->_nick = NULL;
sl@0
   167
  pspec->_blurb = NULL;
sl@0
   168
  pspec->flags = 0;
sl@0
   169
  pspec->value_type = class->value_type;
sl@0
   170
  pspec->owner_type = 0;
sl@0
   171
  pspec->qdata = NULL;
sl@0
   172
  g_datalist_init (&pspec->qdata);
sl@0
   173
  g_datalist_set_flags (&pspec->qdata, PARAM_FLOATING_FLAG);
sl@0
   174
  pspec->ref_count = 1;
sl@0
   175
  pspec->param_id = 0;
sl@0
   176
}
sl@0
   177
sl@0
   178
static void
sl@0
   179
g_param_spec_finalize (GParamSpec *pspec)
sl@0
   180
{
sl@0
   181
  g_datalist_clear (&pspec->qdata);
sl@0
   182
sl@0
   183
  if (!(pspec->flags & G_PARAM_STATIC_NAME))
sl@0
   184
    g_free (pspec->name);
sl@0
   185
  
sl@0
   186
  if (!(pspec->flags & G_PARAM_STATIC_NICK))
sl@0
   187
    g_free (pspec->_nick);
sl@0
   188
sl@0
   189
  if (!(pspec->flags & G_PARAM_STATIC_BLURB))
sl@0
   190
    g_free (pspec->_blurb);
sl@0
   191
sl@0
   192
  g_type_free_instance ((GTypeInstance*) pspec);
sl@0
   193
}
sl@0
   194
sl@0
   195
/**
sl@0
   196
 * g_param_spec_ref:
sl@0
   197
 * @pspec: a valid #GParamSpec
sl@0
   198
 *
sl@0
   199
 * Increments the reference count of @pspec.
sl@0
   200
 *
sl@0
   201
 * Returns: the #GParamSpec that was passed into this function
sl@0
   202
 */
sl@0
   203
EXPORT_C GParamSpec*
sl@0
   204
g_param_spec_ref (GParamSpec *pspec)
sl@0
   205
{
sl@0
   206
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   207
  g_return_val_if_fail (pspec->ref_count > 0, NULL);
sl@0
   208
sl@0
   209
  g_atomic_int_inc ((int *)&pspec->ref_count);
sl@0
   210
sl@0
   211
  return pspec;
sl@0
   212
}
sl@0
   213
sl@0
   214
/**
sl@0
   215
 * g_param_spec_unref:
sl@0
   216
 * @pspec: a valid #GParamSpec
sl@0
   217
 *
sl@0
   218
 * Decrements the reference count of a @pspec.
sl@0
   219
 */
sl@0
   220
EXPORT_C void
sl@0
   221
g_param_spec_unref (GParamSpec *pspec)
sl@0
   222
{
sl@0
   223
  gboolean is_zero;
sl@0
   224
sl@0
   225
  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
sl@0
   226
  g_return_if_fail (pspec->ref_count > 0);
sl@0
   227
sl@0
   228
  is_zero = g_atomic_int_dec_and_test ((int *)&pspec->ref_count);
sl@0
   229
sl@0
   230
  if (G_UNLIKELY (is_zero))
sl@0
   231
    {
sl@0
   232
      G_PARAM_SPEC_GET_CLASS (pspec)->finalize (pspec);
sl@0
   233
    }
sl@0
   234
}
sl@0
   235
sl@0
   236
/**
sl@0
   237
 * g_param_spec_sink:
sl@0
   238
 * @pspec: a valid #GParamSpec
sl@0
   239
 *
sl@0
   240
 * The initial reference count of a newly created #GParamSpec is 1,
sl@0
   241
 * even though no one has explicitly called g_param_spec_ref() on it
sl@0
   242
 * yet. So the initial reference count is flagged as "floating", until
sl@0
   243
 * someone calls <literal>g_param_spec_ref (pspec); g_param_spec_sink
sl@0
   244
 * (pspec);</literal> in sequence on it, taking over the initial
sl@0
   245
 * reference count (thus ending up with a @pspec that has a reference
sl@0
   246
 * count of 1 still, but is not flagged "floating" anymore).
sl@0
   247
 */
sl@0
   248
EXPORT_C void
sl@0
   249
g_param_spec_sink (GParamSpec *pspec)
sl@0
   250
{
sl@0
   251
  gpointer oldvalue;
sl@0
   252
  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
sl@0
   253
  g_return_if_fail (pspec->ref_count > 0);
sl@0
   254
sl@0
   255
  do
sl@0
   256
    oldvalue = g_atomic_pointer_get (&pspec->qdata);
sl@0
   257
  while (!g_atomic_pointer_compare_and_exchange ((void**) &pspec->qdata, oldvalue,
sl@0
   258
                                                 (gpointer) ((gsize) oldvalue & ~(gsize) PARAM_FLOATING_FLAG)));
sl@0
   259
  if ((gsize) oldvalue & PARAM_FLOATING_FLAG)
sl@0
   260
    g_param_spec_unref (pspec);
sl@0
   261
}
sl@0
   262
sl@0
   263
/**
sl@0
   264
 * g_param_spec_ref_sink:
sl@0
   265
 * @pspec: a valid #GParamSpec
sl@0
   266
 *
sl@0
   267
 * Convenience function to ref and sink a #GParamSpec.
sl@0
   268
 *
sl@0
   269
 * Since: 2.10
sl@0
   270
 * Returns: the #GParamSpec that was passed into this function
sl@0
   271
 */
sl@0
   272
EXPORT_C GParamSpec*
sl@0
   273
g_param_spec_ref_sink (GParamSpec *pspec)
sl@0
   274
{
sl@0
   275
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   276
  g_return_val_if_fail (pspec->ref_count > 0, NULL);
sl@0
   277
sl@0
   278
  g_param_spec_ref (pspec);
sl@0
   279
  g_param_spec_sink (pspec);
sl@0
   280
  return pspec;
sl@0
   281
}
sl@0
   282
sl@0
   283
/**
sl@0
   284
 * g_param_spec_get_name:
sl@0
   285
 * @pspec: a valid #GParamSpec
sl@0
   286
 *
sl@0
   287
 * Get the name of a #GParamSpec.
sl@0
   288
 *
sl@0
   289
 * Returns: the name of @pspec.
sl@0
   290
 */
sl@0
   291
EXPORT_C G_CONST_RETURN gchar*
sl@0
   292
g_param_spec_get_name (GParamSpec *pspec)
sl@0
   293
{
sl@0
   294
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   295
sl@0
   296
  return pspec->name;
sl@0
   297
}
sl@0
   298
sl@0
   299
/**
sl@0
   300
 * g_param_spec_get_nick:
sl@0
   301
 * @pspec: a valid #GParamSpec
sl@0
   302
 *
sl@0
   303
 * Get the nickname of a #GParamSpec.
sl@0
   304
 *
sl@0
   305
 * Returns: the nickname of @pspec.
sl@0
   306
 */
sl@0
   307
EXPORT_C G_CONST_RETURN gchar*
sl@0
   308
g_param_spec_get_nick (GParamSpec *pspec)
sl@0
   309
{
sl@0
   310
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   311
sl@0
   312
  if (pspec->_nick)
sl@0
   313
    return pspec->_nick;
sl@0
   314
  else
sl@0
   315
    {
sl@0
   316
      GParamSpec *redirect_target;
sl@0
   317
sl@0
   318
      redirect_target = g_param_spec_get_redirect_target (pspec);
sl@0
   319
      if (redirect_target && redirect_target->_nick)
sl@0
   320
	return redirect_target->_nick;
sl@0
   321
    }
sl@0
   322
sl@0
   323
  return pspec->name;
sl@0
   324
}
sl@0
   325
sl@0
   326
/**
sl@0
   327
 * g_param_spec_get_blurb:
sl@0
   328
 * @pspec: a valid #GParamSpec
sl@0
   329
 *
sl@0
   330
 * Get the short description of a #GParamSpec.
sl@0
   331
 *
sl@0
   332
 * Returns: the short description of @pspec.
sl@0
   333
 */
sl@0
   334
EXPORT_C G_CONST_RETURN gchar*
sl@0
   335
g_param_spec_get_blurb (GParamSpec *pspec)
sl@0
   336
{
sl@0
   337
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   338
sl@0
   339
  if (pspec->_blurb)
sl@0
   340
    return pspec->_blurb;
sl@0
   341
  else
sl@0
   342
    {
sl@0
   343
      GParamSpec *redirect_target;
sl@0
   344
sl@0
   345
      redirect_target = g_param_spec_get_redirect_target (pspec);
sl@0
   346
      if (redirect_target && redirect_target->_blurb)
sl@0
   347
	return redirect_target->_blurb;
sl@0
   348
    }
sl@0
   349
sl@0
   350
  return NULL;
sl@0
   351
}
sl@0
   352
sl@0
   353
static void
sl@0
   354
canonicalize_key (gchar *key)
sl@0
   355
{
sl@0
   356
  gchar *p;
sl@0
   357
  
sl@0
   358
  for (p = key; *p != 0; p++)
sl@0
   359
    {
sl@0
   360
      gchar c = *p;
sl@0
   361
      
sl@0
   362
      if (c != '-' &&
sl@0
   363
	  (c < '0' || c > '9') &&
sl@0
   364
	  (c < 'A' || c > 'Z') &&
sl@0
   365
	  (c < 'a' || c > 'z'))
sl@0
   366
	*p = '-';
sl@0
   367
    }
sl@0
   368
}
sl@0
   369
sl@0
   370
static gboolean
sl@0
   371
is_canonical (const gchar *key)
sl@0
   372
{
sl@0
   373
  const gchar *p;
sl@0
   374
sl@0
   375
  for (p = key; *p != 0; p++)
sl@0
   376
    {
sl@0
   377
      gchar c = *p;
sl@0
   378
      
sl@0
   379
      if (c != '-' &&
sl@0
   380
	  (c < '0' || c > '9') &&
sl@0
   381
	  (c < 'A' || c > 'Z') &&
sl@0
   382
	  (c < 'a' || c > 'z'))
sl@0
   383
	return FALSE;
sl@0
   384
    }
sl@0
   385
sl@0
   386
  return TRUE;
sl@0
   387
}
sl@0
   388
sl@0
   389
/**
sl@0
   390
 * g_param_spec_internal:
sl@0
   391
 * @param_type: the #GType for the property; must be derived from #G_TYPE_PARAM
sl@0
   392
 * @name: the canonical name of the property
sl@0
   393
 * @nick: the nickname of the property
sl@0
   394
 * @blurb: a short description of the property
sl@0
   395
 * @flags: a combination of #GParamFlags
sl@0
   396
 *
sl@0
   397
 * Creates a new #GParamSpec instance.
sl@0
   398
 *
sl@0
   399
 * A property name consists of segments consisting of ASCII letters and
sl@0
   400
 * digits, separated by either the '-' or '_' character. The first
sl@0
   401
 * character of a property name must be a letter. Names which violate these
sl@0
   402
 * rules lead to undefined behaviour.
sl@0
   403
 *
sl@0
   404
 * When creating and looking up a #GParamSpec, either separator can be
sl@0
   405
 * used, but they cannot be mixed. Using '-' is considerably more
sl@0
   406
 * efficient and in fact required when using property names as detail
sl@0
   407
 * strings for signals.
sl@0
   408
 *
sl@0
   409
 * Beyond the name, #GParamSpec<!-- -->s have two more descriptive
sl@0
   410
 * strings associated with them, the @nick, which should be suitable
sl@0
   411
 * for use as a label for the property in a property editor, and the
sl@0
   412
 * @blurb, which should be a somewhat longer description, suitable for
sl@0
   413
 * e.g. a tooltip. The @nick and @blurb should ideally be localized.
sl@0
   414
 *
sl@0
   415
 * Returns: a newly allocated #GParamSpec instance
sl@0
   416
 */
sl@0
   417
EXPORT_C gpointer
sl@0
   418
g_param_spec_internal (GType        param_type,
sl@0
   419
		       const gchar *name,
sl@0
   420
		       const gchar *nick,
sl@0
   421
		       const gchar *blurb,
sl@0
   422
		       GParamFlags  flags)
sl@0
   423
{
sl@0
   424
  GParamSpec *pspec;
sl@0
   425
  
sl@0
   426
  g_return_val_if_fail (G_TYPE_IS_PARAM (param_type) && param_type != G_TYPE_PARAM, NULL);
sl@0
   427
  g_return_val_if_fail (name != NULL, NULL);
sl@0
   428
  g_return_val_if_fail ((name[0] >= 'A' && name[0] <= 'Z') || (name[0] >= 'a' && name[0] <= 'z'), NULL);
sl@0
   429
  g_return_val_if_fail (!(flags & G_PARAM_STATIC_NAME) || is_canonical (name), NULL);
sl@0
   430
  
sl@0
   431
  pspec = (gpointer) g_type_create_instance (param_type);
sl@0
   432
sl@0
   433
  if (flags & G_PARAM_STATIC_NAME)
sl@0
   434
    {
sl@0
   435
      pspec->name = g_intern_static_string (name);
sl@0
   436
      if (!is_canonical (pspec->name))
sl@0
   437
        g_warning ("G_PARAM_STATIC_NAME used with non-canonical pspec name: %s", pspec->name);
sl@0
   438
    }
sl@0
   439
  else
sl@0
   440
    {
sl@0
   441
      pspec->name = g_strdup (name);
sl@0
   442
      canonicalize_key (pspec->name);
sl@0
   443
      g_intern_string (pspec->name);
sl@0
   444
    }
sl@0
   445
sl@0
   446
  if (flags & G_PARAM_STATIC_NICK)
sl@0
   447
    pspec->_nick = (gchar*) nick;
sl@0
   448
  else
sl@0
   449
    pspec->_nick = g_strdup (nick);
sl@0
   450
sl@0
   451
  if (flags & G_PARAM_STATIC_BLURB)
sl@0
   452
    pspec->_blurb = (gchar*) blurb;
sl@0
   453
  else
sl@0
   454
    pspec->_blurb = g_strdup (blurb);
sl@0
   455
sl@0
   456
  pspec->flags = (flags & G_PARAM_USER_MASK) | (flags & G_PARAM_MASK);
sl@0
   457
  
sl@0
   458
  return pspec;
sl@0
   459
}
sl@0
   460
sl@0
   461
/**
sl@0
   462
 * g_param_spec_get_qdata:
sl@0
   463
 * @pspec: a valid #GParamSpec
sl@0
   464
 * @quark: a #GQuark, naming the user data pointer
sl@0
   465
 *
sl@0
   466
 * Gets back user data pointers stored via g_param_spec_set_qdata().
sl@0
   467
 *
sl@0
   468
 * Returns: the user data pointer set, or %NULL
sl@0
   469
 */
sl@0
   470
EXPORT_C gpointer
sl@0
   471
g_param_spec_get_qdata (GParamSpec *pspec,
sl@0
   472
			GQuark      quark)
sl@0
   473
{
sl@0
   474
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   475
  
sl@0
   476
  return quark ? g_datalist_id_get_data (&pspec->qdata, quark) : NULL;
sl@0
   477
}
sl@0
   478
sl@0
   479
/**
sl@0
   480
 * g_param_spec_set_qdata:
sl@0
   481
 * @pspec: the #GParamSpec to set store a user data pointer
sl@0
   482
 * @quark: a #GQuark, naming the user data pointer
sl@0
   483
 * @data: an opaque user data pointer
sl@0
   484
 *
sl@0
   485
 * Sets an opaque, named pointer on a #GParamSpec. The name is
sl@0
   486
 * specified through a #GQuark (retrieved e.g. via
sl@0
   487
 * g_quark_from_static_string()), and the pointer can be gotten back
sl@0
   488
 * from the @pspec with g_param_spec_get_qdata().  Setting a
sl@0
   489
 * previously set user data pointer, overrides (frees) the old pointer
sl@0
   490
 * set, using %NULL as pointer essentially removes the data stored.
sl@0
   491
 */
sl@0
   492
EXPORT_C void
sl@0
   493
g_param_spec_set_qdata (GParamSpec *pspec,
sl@0
   494
			GQuark      quark,
sl@0
   495
			gpointer    data)
sl@0
   496
{
sl@0
   497
  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
sl@0
   498
  g_return_if_fail (quark > 0);
sl@0
   499
sl@0
   500
  g_datalist_id_set_data (&pspec->qdata, quark, data);
sl@0
   501
}
sl@0
   502
sl@0
   503
/**
sl@0
   504
 * g_param_spec_set_qdata_full:
sl@0
   505
 * @pspec: the #GParamSpec to set store a user data pointer
sl@0
   506
 * @quark: a #GQuark, naming the user data pointer
sl@0
   507
 * @data: an opaque user data pointer
sl@0
   508
 * @destroy: function to invoke with @data as argument, when @data needs to
sl@0
   509
 *  be freed
sl@0
   510
 *
sl@0
   511
 * This function works like g_param_spec_set_qdata(), but in addition,
sl@0
   512
 * a <literal>void (*destroy) (gpointer)</literal> function may be
sl@0
   513
 * specified which is called with @data as argument when the @pspec is
sl@0
   514
 * finalized, or the data is being overwritten by a call to
sl@0
   515
 * g_param_spec_set_qdata() with the same @quark.
sl@0
   516
 */
sl@0
   517
EXPORT_C void
sl@0
   518
g_param_spec_set_qdata_full (GParamSpec    *pspec,
sl@0
   519
			     GQuark         quark,
sl@0
   520
			     gpointer       data,
sl@0
   521
			     GDestroyNotify destroy)
sl@0
   522
{
sl@0
   523
  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
sl@0
   524
  g_return_if_fail (quark > 0);
sl@0
   525
sl@0
   526
  g_datalist_id_set_data_full (&pspec->qdata, quark, data, data ? destroy : (GDestroyNotify) NULL);
sl@0
   527
}
sl@0
   528
sl@0
   529
/**
sl@0
   530
 * g_param_spec_steal_qdata:
sl@0
   531
 * @pspec: the #GParamSpec to get a stored user data pointer from
sl@0
   532
 * @quark: a #GQuark, naming the user data pointer
sl@0
   533
 *
sl@0
   534
 * Gets back user data pointers stored via g_param_spec_set_qdata()
sl@0
   535
 * and removes the @data from @pspec without invoking its destroy()
sl@0
   536
 * function (if any was set).  Usually, calling this function is only
sl@0
   537
 * required to update user data pointers with a destroy notifier.
sl@0
   538
 *
sl@0
   539
 * Returns: the user data pointer set, or %NULL
sl@0
   540
 */
sl@0
   541
EXPORT_C gpointer
sl@0
   542
g_param_spec_steal_qdata (GParamSpec *pspec,
sl@0
   543
			  GQuark      quark)
sl@0
   544
{
sl@0
   545
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   546
  g_return_val_if_fail (quark > 0, NULL);
sl@0
   547
  
sl@0
   548
  return g_datalist_id_remove_no_notify (&pspec->qdata, quark);
sl@0
   549
}
sl@0
   550
sl@0
   551
/**
sl@0
   552
 * g_param_spec_get_redirect_target:
sl@0
   553
 * @pspec: a #GParamSpec
sl@0
   554
 *
sl@0
   555
 * If the paramspec redirects operations to another paramspec,
sl@0
   556
 * returns that paramspec. Redirect is used typically for
sl@0
   557
 * providing a new implementation of a property in a derived
sl@0
   558
 * type while preserving all the properties from the parent
sl@0
   559
 * type. Redirection is established by creating a property
sl@0
   560
 * of type #GParamSpecOverride. See g_object_class_override_property()
sl@0
   561
 * for an example of the use of this capability.
sl@0
   562
 *
sl@0
   563
 * Since: 2.4
sl@0
   564
 *
sl@0
   565
 * Returns: paramspec to which requests on this paramspec should
sl@0
   566
 *          be redirected, or %NULL if none.
sl@0
   567
 */
sl@0
   568
EXPORT_C GParamSpec*
sl@0
   569
g_param_spec_get_redirect_target (GParamSpec *pspec)
sl@0
   570
{
sl@0
   571
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
sl@0
   572
sl@0
   573
  if (G_IS_PARAM_SPEC_OVERRIDE (pspec))
sl@0
   574
    {
sl@0
   575
      GParamSpecOverride *ospec = G_PARAM_SPEC_OVERRIDE (pspec);
sl@0
   576
sl@0
   577
      return ospec->overridden;
sl@0
   578
    }
sl@0
   579
  else
sl@0
   580
    return NULL;
sl@0
   581
}
sl@0
   582
sl@0
   583
/**
sl@0
   584
 * g_param_value_set_default:
sl@0
   585
 * @pspec: a valid #GParamSpec
sl@0
   586
 * @value: a #GValue of correct type for @pspec
sl@0
   587
 *
sl@0
   588
 * Sets @value to its default value as specified in @pspec.
sl@0
   589
 */
sl@0
   590
EXPORT_C void
sl@0
   591
g_param_value_set_default (GParamSpec *pspec,
sl@0
   592
			   GValue     *value)
sl@0
   593
{
sl@0
   594
  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
sl@0
   595
  g_return_if_fail (G_IS_VALUE (value));
sl@0
   596
  g_return_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value));
sl@0
   597
sl@0
   598
  g_value_reset (value);
sl@0
   599
  G_PARAM_SPEC_GET_CLASS (pspec)->value_set_default (pspec, value);
sl@0
   600
}
sl@0
   601
sl@0
   602
/**
sl@0
   603
 * g_param_value_defaults:
sl@0
   604
 * @pspec: a valid #GParamSpec
sl@0
   605
 * @value: a #GValue of correct type for @pspec
sl@0
   606
 *
sl@0
   607
 * Checks whether @value contains the default value as specified in @pspec.
sl@0
   608
 *
sl@0
   609
 * Returns: whether @value contains the canonical default for this @pspec
sl@0
   610
 */
sl@0
   611
EXPORT_C gboolean
sl@0
   612
g_param_value_defaults (GParamSpec *pspec,
sl@0
   613
			GValue     *value)
sl@0
   614
{
sl@0
   615
  GValue dflt_value = { 0, };
sl@0
   616
  gboolean defaults;
sl@0
   617
sl@0
   618
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
sl@0
   619
  g_return_val_if_fail (G_IS_VALUE (value), FALSE);
sl@0
   620
  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value), FALSE);
sl@0
   621
sl@0
   622
  g_value_init (&dflt_value, G_PARAM_SPEC_VALUE_TYPE (pspec));
sl@0
   623
  G_PARAM_SPEC_GET_CLASS (pspec)->value_set_default (pspec, &dflt_value);
sl@0
   624
  defaults = G_PARAM_SPEC_GET_CLASS (pspec)->values_cmp (pspec, value, &dflt_value) == 0;
sl@0
   625
  g_value_unset (&dflt_value);
sl@0
   626
sl@0
   627
  return defaults;
sl@0
   628
}
sl@0
   629
sl@0
   630
/**
sl@0
   631
 * g_param_value_validate:
sl@0
   632
 * @pspec: a valid #GParamSpec
sl@0
   633
 * @value: a #GValue of correct type for @pspec
sl@0
   634
 *
sl@0
   635
 * Ensures that the contents of @value comply with the specifications
sl@0
   636
 * set out by @pspec. For example, a #GParamSpecInt might require
sl@0
   637
 * that integers stored in @value may not be smaller than -42 and not be
sl@0
   638
 * greater than +42. If @value contains an integer outside of this range,
sl@0
   639
 * it is modified accordingly, so the resulting value will fit into the
sl@0
   640
 * range -42 .. +42.
sl@0
   641
 *
sl@0
   642
 * Returns: whether modifying @value was necessary to ensure validity
sl@0
   643
 */
sl@0
   644
EXPORT_C gboolean
sl@0
   645
g_param_value_validate (GParamSpec *pspec,
sl@0
   646
			GValue     *value)
sl@0
   647
{
sl@0
   648
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
sl@0
   649
  g_return_val_if_fail (G_IS_VALUE (value), FALSE);
sl@0
   650
  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value), FALSE);
sl@0
   651
sl@0
   652
  if (G_PARAM_SPEC_GET_CLASS (pspec)->value_validate)
sl@0
   653
    {
sl@0
   654
      GValue oval = *value;
sl@0
   655
sl@0
   656
      if (G_PARAM_SPEC_GET_CLASS (pspec)->value_validate (pspec, value) ||
sl@0
   657
	  memcmp (&oval.data, &value->data, sizeof (oval.data)))
sl@0
   658
	return TRUE;
sl@0
   659
    }
sl@0
   660
sl@0
   661
  return FALSE;
sl@0
   662
}
sl@0
   663
sl@0
   664
/**
sl@0
   665
 * g_param_value_convert:
sl@0
   666
 * @pspec: a valid #GParamSpec
sl@0
   667
 * @src_value: souce #GValue
sl@0
   668
 * @dest_value: destination #GValue of correct type for @pspec
sl@0
   669
 * @strict_validation: %TRUE requires @dest_value to conform to @pspec
sl@0
   670
 * without modifications
sl@0
   671
 *
sl@0
   672
 * Transforms @src_value into @dest_value if possible, and then
sl@0
   673
 * validates @dest_value, in order for it to conform to @pspec.  If
sl@0
   674
 * @strict_validation is %TRUE this function will only succeed if the
sl@0
   675
 * transformed @dest_value complied to @pspec without modifications.
sl@0
   676
 *
sl@0
   677
 * See also g_value_type_transformable(), g_value_transform() and
sl@0
   678
 * g_param_value_validate().
sl@0
   679
 *
sl@0
   680
 * Returns: %TRUE if transformation and validation were successful,
sl@0
   681
 *  %FALSE otherwise and @dest_value is left untouched.
sl@0
   682
 */
sl@0
   683
EXPORT_C gboolean
sl@0
   684
g_param_value_convert (GParamSpec   *pspec,
sl@0
   685
		       const GValue *src_value,
sl@0
   686
		       GValue       *dest_value,
sl@0
   687
		       gboolean	     strict_validation)
sl@0
   688
{
sl@0
   689
  GValue tmp_value = { 0, };
sl@0
   690
sl@0
   691
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
sl@0
   692
  g_return_val_if_fail (G_IS_VALUE (src_value), FALSE);
sl@0
   693
  g_return_val_if_fail (G_IS_VALUE (dest_value), FALSE);
sl@0
   694
  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, dest_value), FALSE);
sl@0
   695
sl@0
   696
  /* better leave dest_value untouched when returning FALSE */
sl@0
   697
sl@0
   698
  g_value_init (&tmp_value, G_VALUE_TYPE (dest_value));
sl@0
   699
  if (g_value_transform (src_value, &tmp_value) &&
sl@0
   700
      (!g_param_value_validate (pspec, &tmp_value) || !strict_validation))
sl@0
   701
    {
sl@0
   702
      g_value_unset (dest_value);
sl@0
   703
      
sl@0
   704
      /* values are relocatable */
sl@0
   705
      memcpy (dest_value, &tmp_value, sizeof (tmp_value));
sl@0
   706
      
sl@0
   707
      return TRUE;
sl@0
   708
    }
sl@0
   709
  else
sl@0
   710
    {
sl@0
   711
      g_value_unset (&tmp_value);
sl@0
   712
      
sl@0
   713
      return FALSE;
sl@0
   714
    }
sl@0
   715
}
sl@0
   716
sl@0
   717
/**
sl@0
   718
 * g_param_values_cmp:
sl@0
   719
 * @pspec: a valid #GParamSpec
sl@0
   720
 * @value1: a #GValue of correct type for @pspec
sl@0
   721
 * @value2: a #GValue of correct type for @pspec
sl@0
   722
 *
sl@0
   723
 * Compares @value1 with @value2 according to @pspec, and return -1, 0 or +1,
sl@0
   724
 * if @value1 is found to be less than, equal to or greater than @value2,
sl@0
   725
 * respectively.
sl@0
   726
 *
sl@0
   727
 * Returns: -1, 0 or +1, for a less than, equal to or greater than result
sl@0
   728
 */
sl@0
   729
EXPORT_C gint
sl@0
   730
g_param_values_cmp (GParamSpec   *pspec,
sl@0
   731
		    const GValue *value1,
sl@0
   732
		    const GValue *value2)
sl@0
   733
{
sl@0
   734
  gint cmp;
sl@0
   735
sl@0
   736
  /* param_values_cmp() effectively does: value1 - value2
sl@0
   737
   * so the return values are:
sl@0
   738
   * -1)  value1 < value2
sl@0
   739
   *  0)  value1 == value2
sl@0
   740
   *  1)  value1 > value2
sl@0
   741
   */
sl@0
   742
  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), 0);
sl@0
   743
  g_return_val_if_fail (G_IS_VALUE (value1), 0);
sl@0
   744
  g_return_val_if_fail (G_IS_VALUE (value2), 0);
sl@0
   745
  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value1), 0);
sl@0
   746
  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value2), 0);
sl@0
   747
sl@0
   748
  cmp = G_PARAM_SPEC_GET_CLASS (pspec)->values_cmp (pspec, value1, value2);
sl@0
   749
sl@0
   750
  return CLAMP (cmp, -1, 1);
sl@0
   751
}
sl@0
   752
sl@0
   753
static void
sl@0
   754
value_param_init (GValue *value)
sl@0
   755
{
sl@0
   756
  value->data[0].v_pointer = NULL;
sl@0
   757
}
sl@0
   758
sl@0
   759
static void
sl@0
   760
value_param_free_value (GValue *value)
sl@0
   761
{
sl@0
   762
  if (value->data[0].v_pointer)
sl@0
   763
    g_param_spec_unref (value->data[0].v_pointer);
sl@0
   764
}
sl@0
   765
sl@0
   766
static void
sl@0
   767
value_param_copy_value (const GValue *src_value,
sl@0
   768
			GValue       *dest_value)
sl@0
   769
{
sl@0
   770
  if (src_value->data[0].v_pointer)
sl@0
   771
    dest_value->data[0].v_pointer = g_param_spec_ref (src_value->data[0].v_pointer);
sl@0
   772
  else
sl@0
   773
    dest_value->data[0].v_pointer = NULL;
sl@0
   774
}
sl@0
   775
sl@0
   776
static void
sl@0
   777
value_param_transform_value (const GValue *src_value,
sl@0
   778
			     GValue       *dest_value)
sl@0
   779
{
sl@0
   780
  if (src_value->data[0].v_pointer &&
sl@0
   781
      g_type_is_a (G_PARAM_SPEC_TYPE (dest_value->data[0].v_pointer), G_VALUE_TYPE (dest_value)))
sl@0
   782
    dest_value->data[0].v_pointer = g_param_spec_ref (src_value->data[0].v_pointer);
sl@0
   783
  else
sl@0
   784
    dest_value->data[0].v_pointer = NULL;
sl@0
   785
}
sl@0
   786
sl@0
   787
static gpointer
sl@0
   788
value_param_peek_pointer (const GValue *value)
sl@0
   789
{
sl@0
   790
  return value->data[0].v_pointer;
sl@0
   791
}
sl@0
   792
sl@0
   793
static gchar*
sl@0
   794
value_param_collect_value (GValue      *value,
sl@0
   795
			   guint        n_collect_values,
sl@0
   796
			   GTypeCValue *collect_values,
sl@0
   797
			   guint        collect_flags)
sl@0
   798
{
sl@0
   799
  if (collect_values[0].v_pointer)
sl@0
   800
    {
sl@0
   801
      GParamSpec *param = collect_values[0].v_pointer;
sl@0
   802
sl@0
   803
      if (param->g_type_instance.g_class == NULL)
sl@0
   804
	return g_strconcat ("invalid unclassed param spec pointer for value type `",
sl@0
   805
			    G_VALUE_TYPE_NAME (value),
sl@0
   806
			    "'",
sl@0
   807
			    NULL);
sl@0
   808
      else if (!g_value_type_compatible (G_PARAM_SPEC_TYPE (param), G_VALUE_TYPE (value)))
sl@0
   809
	return g_strconcat ("invalid param spec type `",
sl@0
   810
			    G_PARAM_SPEC_TYPE_NAME (param),
sl@0
   811
			    "' for value type `",
sl@0
   812
			    G_VALUE_TYPE_NAME (value),
sl@0
   813
			    "'",
sl@0
   814
			    NULL);
sl@0
   815
      value->data[0].v_pointer = g_param_spec_ref (param);
sl@0
   816
    }
sl@0
   817
  else
sl@0
   818
    value->data[0].v_pointer = NULL;
sl@0
   819
sl@0
   820
  return NULL;
sl@0
   821
}
sl@0
   822
sl@0
   823
static gchar*
sl@0
   824
value_param_lcopy_value (const GValue *value,
sl@0
   825
			 guint         n_collect_values,
sl@0
   826
			 GTypeCValue  *collect_values,
sl@0
   827
			 guint         collect_flags)
sl@0
   828
{
sl@0
   829
  GParamSpec **param_p = collect_values[0].v_pointer;
sl@0
   830
sl@0
   831
  if (!param_p)
sl@0
   832
    return g_strdup_printf ("value location for `%s' passed as NULL", G_VALUE_TYPE_NAME (value));
sl@0
   833
sl@0
   834
  if (!value->data[0].v_pointer)
sl@0
   835
    *param_p = NULL;
sl@0
   836
  else if (collect_flags & G_VALUE_NOCOPY_CONTENTS)
sl@0
   837
    *param_p = value->data[0].v_pointer;
sl@0
   838
  else
sl@0
   839
    *param_p = g_param_spec_ref (value->data[0].v_pointer);
sl@0
   840
sl@0
   841
  return NULL;
sl@0
   842
}
sl@0
   843
sl@0
   844
sl@0
   845
/* --- param spec pool --- */
sl@0
   846
/**
sl@0
   847
 * GParamSpecPool:
sl@0
   848
 *
sl@0
   849
 * A #GParamSpecPool maintains a collection of #GParamSpec<!-- -->s which can be
sl@0
   850
 * quickly accessed by owner and name. The implementation of the #GObject property
sl@0
   851
 * system uses such a pool to store the #GParamSpecs of the properties all object
sl@0
   852
 * types.
sl@0
   853
 */
sl@0
   854
struct _GParamSpecPool
sl@0
   855
{
sl@0
   856
  GStaticMutex smutex;
sl@0
   857
  gboolean     type_prefixing;
sl@0
   858
  GHashTable  *hash_table;
sl@0
   859
};
sl@0
   860
sl@0
   861
static guint
sl@0
   862
param_spec_pool_hash (gconstpointer key_spec)
sl@0
   863
{
sl@0
   864
  const GParamSpec *key = key_spec;
sl@0
   865
  const gchar *p;
sl@0
   866
  guint h = key->owner_type;
sl@0
   867
sl@0
   868
  for (p = key->name; *p; p++)
sl@0
   869
    h = (h << 5) - h + *p;
sl@0
   870
sl@0
   871
  return h;
sl@0
   872
}
sl@0
   873
sl@0
   874
static gboolean
sl@0
   875
param_spec_pool_equals (gconstpointer key_spec_1,
sl@0
   876
			gconstpointer key_spec_2)
sl@0
   877
{
sl@0
   878
  const GParamSpec *key1 = key_spec_1;
sl@0
   879
  const GParamSpec *key2 = key_spec_2;
sl@0
   880
sl@0
   881
  return (key1->owner_type == key2->owner_type &&
sl@0
   882
	  strcmp (key1->name, key2->name) == 0);
sl@0
   883
}
sl@0
   884
sl@0
   885
#if EMULATOR
sl@0
   886
PLS(init_smutex,g_param_spec_pool_new ,GStaticMutex)
sl@0
   887
#define init_smutex (*FUNCTION_NAME(init_smutex,g_param_spec_pool_new )())
sl@0
   888
#endif /* EMULATOR */
sl@0
   889
sl@0
   890
/**
sl@0
   891
 * g_param_spec_pool_new:
sl@0
   892
 * @type_prefixing: Whether the pool will support type-prefixed property names.
sl@0
   893
 *
sl@0
   894
 * Creates a new #GParamSpecPool.
sl@0
   895
 *
sl@0
   896
 * If @type_prefixing is %TRUE, lookups in the newly created pool will
sl@0
   897
 * allow to specify the owner as a colon-separated prefix of the
sl@0
   898
 * property name, like "GtkContainer:border-width". This feature is
sl@0
   899
 * deprecated, so you should always set @type_prefixing to %FALSE.
sl@0
   900
 *
sl@0
   901
 * Returns: a newly allocated #GParamSpecPool.
sl@0
   902
 */
sl@0
   903
EXPORT_C GParamSpecPool*
sl@0
   904
g_param_spec_pool_new (gboolean type_prefixing)
sl@0
   905
{
sl@0
   906
  #if !(EMULATOR)
sl@0
   907
  static GStaticMutex init_smutex = G_STATIC_MUTEX_INIT;
sl@0
   908
  #endif /* EMULATOR */
sl@0
   909
  GParamSpecPool *pool = g_new (GParamSpecPool, 1);
sl@0
   910
sl@0
   911
  memcpy (&pool->smutex, &init_smutex, sizeof (init_smutex));
sl@0
   912
  pool->type_prefixing = type_prefixing != FALSE;
sl@0
   913
  pool->hash_table = g_hash_table_new (param_spec_pool_hash, param_spec_pool_equals);
sl@0
   914
sl@0
   915
  return pool;
sl@0
   916
}
sl@0
   917
sl@0
   918
#if EMULATOR
sl@0
   919
#undef init_smutex 
sl@0
   920
#endif /* EMULATOR */
sl@0
   921
sl@0
   922
/**
sl@0
   923
 * g_param_spec_pool_insert:
sl@0
   924
 * @pool: a #GParamSpecPool.
sl@0
   925
 * @pspec: the #GParamSpec to insert
sl@0
   926
 * @owner_type: a #GType identifying the owner of @pspec
sl@0
   927
 *
sl@0
   928
 * Inserts a #GParamSpec in the pool.
sl@0
   929
 */
sl@0
   930
EXPORT_C void
sl@0
   931
g_param_spec_pool_insert (GParamSpecPool *pool,
sl@0
   932
			  GParamSpec     *pspec,
sl@0
   933
			  GType           owner_type)
sl@0
   934
{
sl@0
   935
  gchar *p;
sl@0
   936
  
sl@0
   937
  if (pool && pspec && owner_type > 0 && pspec->owner_type == 0)
sl@0
   938
    {
sl@0
   939
      G_SLOCK (&pool->smutex);
sl@0
   940
      for (p = pspec->name; *p; p++)
sl@0
   941
	{
sl@0
   942
	  if (!strchr (G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS "-_", *p))
sl@0
   943
	    {
sl@0
   944
	      g_warning (G_STRLOC ": pspec name \"%s\" contains invalid characters", pspec->name);
sl@0
   945
	      G_SUNLOCK (&pool->smutex);
sl@0
   946
	      return;
sl@0
   947
	    }
sl@0
   948
	}
sl@0
   949
      
sl@0
   950
      pspec->owner_type = owner_type;
sl@0
   951
      g_param_spec_ref (pspec);
sl@0
   952
      g_hash_table_insert (pool->hash_table, pspec, pspec);
sl@0
   953
      G_SUNLOCK (&pool->smutex);
sl@0
   954
    }
sl@0
   955
  else
sl@0
   956
    {
sl@0
   957
      g_return_if_fail (pool != NULL);
sl@0
   958
      g_return_if_fail (pspec);
sl@0
   959
      g_return_if_fail (owner_type > 0);
sl@0
   960
      g_return_if_fail (pspec->owner_type == 0);
sl@0
   961
    }
sl@0
   962
}
sl@0
   963
sl@0
   964
/**
sl@0
   965
 * g_param_spec_pool_remove:
sl@0
   966
 * @pool: a #GParamSpecPool
sl@0
   967
 * @pspec: the #GParamSpec to remove
sl@0
   968
 *
sl@0
   969
 * Removes a #GParamSpec from the pool.
sl@0
   970
 */
sl@0
   971
EXPORT_C void
sl@0
   972
g_param_spec_pool_remove (GParamSpecPool *pool,
sl@0
   973
			  GParamSpec     *pspec)
sl@0
   974
{
sl@0
   975
  if (pool && pspec)
sl@0
   976
    {
sl@0
   977
      G_SLOCK (&pool->smutex);
sl@0
   978
      if (g_hash_table_remove (pool->hash_table, pspec))
sl@0
   979
	g_param_spec_unref (pspec);
sl@0
   980
      else
sl@0
   981
	g_warning (G_STRLOC ": attempt to remove unknown pspec `%s' from pool", pspec->name);
sl@0
   982
      G_SUNLOCK (&pool->smutex);
sl@0
   983
    }
sl@0
   984
  else
sl@0
   985
    {
sl@0
   986
      g_return_if_fail (pool != NULL);
sl@0
   987
      g_return_if_fail (pspec);
sl@0
   988
    }
sl@0
   989
}
sl@0
   990
sl@0
   991
static inline GParamSpec*
sl@0
   992
param_spec_ht_lookup (GHashTable  *hash_table,
sl@0
   993
		      const gchar *param_name,
sl@0
   994
		      GType        owner_type,
sl@0
   995
		      gboolean     walk_ancestors)
sl@0
   996
{
sl@0
   997
  GParamSpec key, *pspec;
sl@0
   998
sl@0
   999
  key.owner_type = owner_type;
sl@0
  1000
  key.name = (gchar*) param_name;
sl@0
  1001
  if (walk_ancestors)
sl@0
  1002
    do
sl@0
  1003
      {
sl@0
  1004
	pspec = g_hash_table_lookup (hash_table, &key);
sl@0
  1005
	if (pspec)
sl@0
  1006
	  return pspec;
sl@0
  1007
	key.owner_type = g_type_parent (key.owner_type);
sl@0
  1008
      }
sl@0
  1009
    while (key.owner_type);
sl@0
  1010
  else
sl@0
  1011
    pspec = g_hash_table_lookup (hash_table, &key);
sl@0
  1012
sl@0
  1013
  if (!pspec && !is_canonical (param_name))
sl@0
  1014
    {
sl@0
  1015
      /* try canonicalized form */
sl@0
  1016
      key.name = g_strdup (param_name);
sl@0
  1017
      key.owner_type = owner_type;
sl@0
  1018
      
sl@0
  1019
      canonicalize_key (key.name);
sl@0
  1020
      if (walk_ancestors)
sl@0
  1021
	do
sl@0
  1022
	  {
sl@0
  1023
	    pspec = g_hash_table_lookup (hash_table, &key);
sl@0
  1024
	    if (pspec)
sl@0
  1025
	      {
sl@0
  1026
		g_free (key.name);
sl@0
  1027
		return pspec;
sl@0
  1028
	      }
sl@0
  1029
	    key.owner_type = g_type_parent (key.owner_type);
sl@0
  1030
	  }
sl@0
  1031
	while (key.owner_type);
sl@0
  1032
      else
sl@0
  1033
	pspec = g_hash_table_lookup (hash_table, &key);
sl@0
  1034
      g_free (key.name);
sl@0
  1035
    }
sl@0
  1036
sl@0
  1037
  return pspec;
sl@0
  1038
}
sl@0
  1039
sl@0
  1040
/**
sl@0
  1041
 * g_param_spec_pool_lookup:
sl@0
  1042
 * @pool: a #GParamSpecPool
sl@0
  1043
 * @param_name: the name to look for
sl@0
  1044
 * @owner_type: the owner to look for
sl@0
  1045
 * @walk_ancestors: If %TRUE, also try to find a #GParamSpec with @param_name
sl@0
  1046
 *  owned by an ancestor of @owner_type.
sl@0
  1047
 *
sl@0
  1048
 * Looks up a #GParamSpec in the pool.
sl@0
  1049
 *
sl@0
  1050
 * Returns: The found #GParamSpec, or %NULL if no matching #GParamSpec was found.
sl@0
  1051
 */
sl@0
  1052
EXPORT_C GParamSpec*
sl@0
  1053
g_param_spec_pool_lookup (GParamSpecPool *pool,
sl@0
  1054
			  const gchar    *param_name,
sl@0
  1055
			  GType           owner_type,
sl@0
  1056
			  gboolean        walk_ancestors)
sl@0
  1057
{
sl@0
  1058
  GParamSpec *pspec;
sl@0
  1059
  gchar *delim;
sl@0
  1060
sl@0
  1061
  if (!pool || !param_name)
sl@0
  1062
    {
sl@0
  1063
      g_return_val_if_fail (pool != NULL, NULL);
sl@0
  1064
      g_return_val_if_fail (param_name != NULL, NULL);
sl@0
  1065
    }
sl@0
  1066
sl@0
  1067
  G_SLOCK (&pool->smutex);
sl@0
  1068
sl@0
  1069
  delim = pool->type_prefixing ? strchr (param_name, ':') : NULL;
sl@0
  1070
sl@0
  1071
  /* try quick and away, i.e. without prefix */
sl@0
  1072
  if (!delim)
sl@0
  1073
    {
sl@0
  1074
      pspec = param_spec_ht_lookup (pool->hash_table, param_name, owner_type, walk_ancestors);
sl@0
  1075
      G_SUNLOCK (&pool->smutex);
sl@0
  1076
sl@0
  1077
      return pspec;
sl@0
  1078
    }
sl@0
  1079
sl@0
  1080
  /* strip type prefix */
sl@0
  1081
  if (pool->type_prefixing && delim[1] == ':')
sl@0
  1082
    {
sl@0
  1083
      guint l = delim - param_name;
sl@0
  1084
      gchar stack_buffer[32], *buffer = l < 32 ? stack_buffer : g_new (gchar, l + 1);
sl@0
  1085
      GType type;
sl@0
  1086
      
sl@0
  1087
      strncpy (buffer, param_name, delim - param_name);
sl@0
  1088
      buffer[l] = 0;
sl@0
  1089
      type = g_type_from_name (buffer);
sl@0
  1090
      if (l >= 32)
sl@0
  1091
	g_free (buffer);
sl@0
  1092
      if (type)		/* type==0 isn't a valid type pefix */
sl@0
  1093
	{
sl@0
  1094
	  /* sanity check, these cases don't make a whole lot of sense */
sl@0
  1095
	  if ((!walk_ancestors && type != owner_type) || !g_type_is_a (owner_type, type))
sl@0
  1096
	    {
sl@0
  1097
	      G_SUNLOCK (&pool->smutex);
sl@0
  1098
sl@0
  1099
	      return NULL;
sl@0
  1100
	    }
sl@0
  1101
	  owner_type = type;
sl@0
  1102
	  param_name += l + 2;
sl@0
  1103
	  pspec = param_spec_ht_lookup (pool->hash_table, param_name, owner_type, walk_ancestors);
sl@0
  1104
	  G_SUNLOCK (&pool->smutex);
sl@0
  1105
sl@0
  1106
	  return pspec;
sl@0
  1107
	}
sl@0
  1108
    }
sl@0
  1109
  /* malformed param_name */
sl@0
  1110
sl@0
  1111
  G_SUNLOCK (&pool->smutex);
sl@0
  1112
sl@0
  1113
  return NULL;
sl@0
  1114
}
sl@0
  1115
sl@0
  1116
static void
sl@0
  1117
pool_list (gpointer key,
sl@0
  1118
	   gpointer value,
sl@0
  1119
	   gpointer user_data)
sl@0
  1120
{
sl@0
  1121
  GParamSpec *pspec = value;
sl@0
  1122
  gpointer *data = user_data;
sl@0
  1123
  GType owner_type = (GType) data[1];
sl@0
  1124
sl@0
  1125
  if (owner_type == pspec->owner_type)
sl@0
  1126
    data[0] = g_list_prepend (data[0], pspec);
sl@0
  1127
}
sl@0
  1128
sl@0
  1129
/**
sl@0
  1130
 * g_param_spec_pool_list_owned:
sl@0
  1131
 * @pool: a #GParamSpecPool
sl@0
  1132
 * @owner_type: the owner to look for
sl@0
  1133
 *
sl@0
  1134
 * Gets an #GList of all #GParamSpec<!-- -->s owned by @owner_type in
sl@0
  1135
 * the pool.
sl@0
  1136
 *
sl@0
  1137
 * Returns: a #GList of all #GParamSpec<!-- -->s owned by @owner_type
sl@0
  1138
 *          in the pool#GParamSpec<!-- -->s.
sl@0
  1139
 */
sl@0
  1140
EXPORT_C GList*
sl@0
  1141
g_param_spec_pool_list_owned (GParamSpecPool *pool,
sl@0
  1142
			      GType           owner_type)
sl@0
  1143
{
sl@0
  1144
  gpointer data[2];
sl@0
  1145
sl@0
  1146
  g_return_val_if_fail (pool != NULL, NULL);
sl@0
  1147
  g_return_val_if_fail (owner_type > 0, NULL);
sl@0
  1148
  
sl@0
  1149
  G_SLOCK (&pool->smutex);
sl@0
  1150
  data[0] = NULL;
sl@0
  1151
  data[1] = (gpointer) owner_type;
sl@0
  1152
  g_hash_table_foreach (pool->hash_table, pool_list, &data);
sl@0
  1153
  G_SUNLOCK (&pool->smutex);
sl@0
  1154
sl@0
  1155
  return data[0];
sl@0
  1156
}
sl@0
  1157
sl@0
  1158
static gint
sl@0
  1159
pspec_compare_id (gconstpointer a,
sl@0
  1160
		  gconstpointer b)
sl@0
  1161
{
sl@0
  1162
  const GParamSpec *pspec1 = a, *pspec2 = b;
sl@0
  1163
sl@0
  1164
  return pspec1->param_id < pspec2->param_id ? -1 : pspec1->param_id > pspec2->param_id;
sl@0
  1165
}
sl@0
  1166
sl@0
  1167
static inline GSList*
sl@0
  1168
pspec_list_remove_overridden_and_redirected (GSList     *plist,
sl@0
  1169
					     GHashTable *ht,
sl@0
  1170
					     GType       owner_type,
sl@0
  1171
					     guint      *n_p)
sl@0
  1172
{
sl@0
  1173
  GSList *rlist = NULL;
sl@0
  1174
sl@0
  1175
  while (plist)
sl@0
  1176
    {
sl@0
  1177
      GSList *tmp = plist->next;
sl@0
  1178
      GParamSpec *pspec = plist->data;
sl@0
  1179
      GParamSpec *found;
sl@0
  1180
      gboolean remove = FALSE;
sl@0
  1181
sl@0
  1182
      /* Remove paramspecs that are redirected, and also paramspecs
sl@0
  1183
       * that have are overridden by non-redirected properties.
sl@0
  1184
       * The idea is to get the single paramspec for each name that
sl@0
  1185
       * best corresponds to what the application sees.
sl@0
  1186
       */
sl@0
  1187
      if (g_param_spec_get_redirect_target (pspec))
sl@0
  1188
	remove = TRUE;
sl@0
  1189
      else
sl@0
  1190
	{
sl@0
  1191
	  found = param_spec_ht_lookup (ht, pspec->name, owner_type, TRUE);
sl@0
  1192
	  if (found != pspec)
sl@0
  1193
	    {
sl@0
  1194
	      GParamSpec *redirect = g_param_spec_get_redirect_target (found);
sl@0
  1195
	      if (redirect != pspec)
sl@0
  1196
		remove = TRUE;
sl@0
  1197
	    }
sl@0
  1198
	}
sl@0
  1199
sl@0
  1200
      if (remove)
sl@0
  1201
	{
sl@0
  1202
	  g_slist_free_1 (plist);
sl@0
  1203
	}
sl@0
  1204
      else
sl@0
  1205
	{
sl@0
  1206
	  plist->next = rlist;
sl@0
  1207
	  rlist = plist;
sl@0
  1208
	  *n_p += 1;
sl@0
  1209
	}
sl@0
  1210
      plist = tmp;
sl@0
  1211
    }
sl@0
  1212
  return rlist;
sl@0
  1213
}
sl@0
  1214
sl@0
  1215
static void
sl@0
  1216
pool_depth_list (gpointer key,
sl@0
  1217
		 gpointer value,
sl@0
  1218
		 gpointer user_data)
sl@0
  1219
{
sl@0
  1220
  GParamSpec *pspec = value;
sl@0
  1221
  gpointer *data = user_data;
sl@0
  1222
  GSList **slists = data[0];
sl@0
  1223
  GType owner_type = (GType) data[1];
sl@0
  1224
sl@0
  1225
  if (g_type_is_a (owner_type, pspec->owner_type))
sl@0
  1226
    {
sl@0
  1227
      if (G_TYPE_IS_INTERFACE (pspec->owner_type))
sl@0
  1228
	{
sl@0
  1229
	  slists[0] = g_slist_prepend (slists[0], pspec);
sl@0
  1230
	}
sl@0
  1231
      else
sl@0
  1232
	{
sl@0
  1233
	  guint d = g_type_depth (pspec->owner_type);
sl@0
  1234
sl@0
  1235
	  slists[d - 1] = g_slist_prepend (slists[d - 1], pspec);
sl@0
  1236
	}
sl@0
  1237
    }
sl@0
  1238
}
sl@0
  1239
sl@0
  1240
/* We handle interfaces specially since we don't want to
sl@0
  1241
 * count interface prerequisites like normal inheritance;
sl@0
  1242
 * the property comes from the direct inheritance from
sl@0
  1243
 * the prerequisite class, not from the interface that
sl@0
  1244
 * prerequires it.
sl@0
  1245
 * 
sl@0
  1246
 * also 'depth' isn't a meaningful concept for interface
sl@0
  1247
 * prerequites.
sl@0
  1248
 */
sl@0
  1249
static void
sl@0
  1250
pool_depth_list_for_interface (gpointer key,
sl@0
  1251
			       gpointer value,
sl@0
  1252
			       gpointer user_data)
sl@0
  1253
{
sl@0
  1254
  GParamSpec *pspec = value;
sl@0
  1255
  gpointer *data = user_data;
sl@0
  1256
  GSList **slists = data[0];
sl@0
  1257
  GType owner_type = (GType) data[1];
sl@0
  1258
sl@0
  1259
  if (pspec->owner_type == owner_type)
sl@0
  1260
    slists[0] = g_slist_prepend (slists[0], pspec);
sl@0
  1261
}
sl@0
  1262
sl@0
  1263
/**
sl@0
  1264
 * g_param_spec_pool_list:
sl@0
  1265
 * @pool: a #GParamSpecPool
sl@0
  1266
 * @owner_type: the owner to look for
sl@0
  1267
 * @n_pspecs_p: return location for the length of the returned array
sl@0
  1268
 *
sl@0
  1269
 * Gets an array of all #GParamSpec<!-- -->s owned by @owner_type in
sl@0
  1270
 * the pool.
sl@0
  1271
 *
sl@0
  1272
 * Returns: a newly allocated array containing pointers to all
sl@0
  1273
 *          #GParamSpec<!-- -->s owned by @owner_type in the pool
sl@0
  1274
 */
sl@0
  1275
EXPORT_C GParamSpec** /* free result */
sl@0
  1276
g_param_spec_pool_list (GParamSpecPool *pool,
sl@0
  1277
			GType           owner_type,
sl@0
  1278
			guint          *n_pspecs_p)
sl@0
  1279
{
sl@0
  1280
  GParamSpec **pspecs, **p;
sl@0
  1281
  GSList **slists, *node;
sl@0
  1282
  gpointer data[2];
sl@0
  1283
  guint d, i;
sl@0
  1284
sl@0
  1285
  g_return_val_if_fail (pool != NULL, NULL);
sl@0
  1286
  g_return_val_if_fail (owner_type > 0, NULL);
sl@0
  1287
  g_return_val_if_fail (n_pspecs_p != NULL, NULL);
sl@0
  1288
  
sl@0
  1289
  G_SLOCK (&pool->smutex);
sl@0
  1290
  *n_pspecs_p = 0;
sl@0
  1291
  d = g_type_depth (owner_type);
sl@0
  1292
  slists = g_new0 (GSList*, d);
sl@0
  1293
  data[0] = slists;
sl@0
  1294
  data[1] = (gpointer) owner_type;
sl@0
  1295
sl@0
  1296
  g_hash_table_foreach (pool->hash_table,
sl@0
  1297
			G_TYPE_IS_INTERFACE (owner_type) ?
sl@0
  1298
			   pool_depth_list_for_interface :
sl@0
  1299
			   pool_depth_list,
sl@0
  1300
			&data);
sl@0
  1301
  
sl@0
  1302
  for (i = 0; i < d; i++)
sl@0
  1303
    slists[i] = pspec_list_remove_overridden_and_redirected (slists[i], pool->hash_table, owner_type, n_pspecs_p);
sl@0
  1304
  pspecs = g_new (GParamSpec*, *n_pspecs_p + 1);
sl@0
  1305
  p = pspecs;
sl@0
  1306
  for (i = 0; i < d; i++)
sl@0
  1307
    {
sl@0
  1308
      slists[i] = g_slist_sort (slists[i], pspec_compare_id);
sl@0
  1309
      for (node = slists[i]; node; node = node->next)
sl@0
  1310
	*p++ = node->data;
sl@0
  1311
      g_slist_free (slists[i]);
sl@0
  1312
    }
sl@0
  1313
  *p++ = NULL;
sl@0
  1314
  g_free (slists);
sl@0
  1315
  G_SUNLOCK (&pool->smutex);
sl@0
  1316
sl@0
  1317
  return pspecs;
sl@0
  1318
}
sl@0
  1319
sl@0
  1320
sl@0
  1321
/* --- auxillary functions --- */
sl@0
  1322
typedef struct
sl@0
  1323
{
sl@0
  1324
  /* class portion */
sl@0
  1325
  GType           value_type;
sl@0
  1326
  void          (*finalize)             (GParamSpec   *pspec);
sl@0
  1327
  void          (*value_set_default)    (GParamSpec   *pspec,
sl@0
  1328
					 GValue       *value);
sl@0
  1329
  gboolean      (*value_validate)       (GParamSpec   *pspec,
sl@0
  1330
					 GValue       *value);
sl@0
  1331
  gint          (*values_cmp)           (GParamSpec   *pspec,
sl@0
  1332
					 const GValue *value1,
sl@0
  1333
					 const GValue *value2);
sl@0
  1334
} ParamSpecClassInfo;
sl@0
  1335
sl@0
  1336
static void
sl@0
  1337
param_spec_generic_class_init (gpointer g_class,
sl@0
  1338
			       gpointer class_data)
sl@0
  1339
{
sl@0
  1340
  GParamSpecClass *class = g_class;
sl@0
  1341
  ParamSpecClassInfo *info = class_data;
sl@0
  1342
sl@0
  1343
  class->value_type = info->value_type;
sl@0
  1344
  if (info->finalize)
sl@0
  1345
    class->finalize = info->finalize;			/* optional */
sl@0
  1346
  class->value_set_default = info->value_set_default;
sl@0
  1347
  if (info->value_validate)
sl@0
  1348
    class->value_validate = info->value_validate;	/* optional */
sl@0
  1349
  class->values_cmp = info->values_cmp;
sl@0
  1350
  g_free (class_data);
sl@0
  1351
}
sl@0
  1352
sl@0
  1353
static void
sl@0
  1354
default_value_set_default (GParamSpec *pspec,
sl@0
  1355
			   GValue     *value)
sl@0
  1356
{
sl@0
  1357
  /* value is already zero initialized */
sl@0
  1358
}
sl@0
  1359
sl@0
  1360
static gint
sl@0
  1361
default_values_cmp (GParamSpec   *pspec,
sl@0
  1362
		    const GValue *value1,
sl@0
  1363
		    const GValue *value2)
sl@0
  1364
{
sl@0
  1365
  return memcmp (&value1->data, &value2->data, sizeof (value1->data));
sl@0
  1366
}
sl@0
  1367
sl@0
  1368
/**
sl@0
  1369
 * g_param_type_register_static:
sl@0
  1370
 * @name: 0-terminated string used as the name of the new #GParamSpec type.
sl@0
  1371
 * @pspec_info: The #GParamSpecTypeInfo for this #GParamSpec type.
sl@0
  1372
 *
sl@0
  1373
 * Registers @name as the name of a new static type derived from
sl@0
  1374
 * #G_TYPE_PARAM. The type system uses the information contained in
sl@0
  1375
 * the #GParamSpecTypeInfo structure pointed to by @info to manage the
sl@0
  1376
 * #GParamSpec type and its instances.
sl@0
  1377
 *
sl@0
  1378
 * Returns: The new type identifier.
sl@0
  1379
 */
sl@0
  1380
EXPORT_C GType
sl@0
  1381
g_param_type_register_static (const gchar              *name,
sl@0
  1382
			      const GParamSpecTypeInfo *pspec_info)
sl@0
  1383
{
sl@0
  1384
  GTypeInfo info = {
sl@0
  1385
    sizeof (GParamSpecClass),      /* class_size */
sl@0
  1386
    NULL,                          /* base_init */
sl@0
  1387
    NULL,                          /* base_destroy */
sl@0
  1388
    param_spec_generic_class_init, /* class_init */
sl@0
  1389
    NULL,                          /* class_destroy */
sl@0
  1390
    NULL,                          /* class_data */
sl@0
  1391
    0,                             /* instance_size */
sl@0
  1392
    16,                            /* n_preallocs */
sl@0
  1393
    NULL,                          /* instance_init */
sl@0
  1394
  };
sl@0
  1395
  ParamSpecClassInfo *cinfo;
sl@0
  1396
sl@0
  1397
  g_return_val_if_fail (name != NULL, 0);
sl@0
  1398
  g_return_val_if_fail (pspec_info != NULL, 0);
sl@0
  1399
  g_return_val_if_fail (g_type_from_name (name) == 0, 0);
sl@0
  1400
  g_return_val_if_fail (pspec_info->instance_size >= sizeof (GParamSpec), 0);
sl@0
  1401
  g_return_val_if_fail (g_type_name (pspec_info->value_type) != NULL, 0);
sl@0
  1402
  /* default: g_return_val_if_fail (pspec_info->value_set_default != NULL, 0); */
sl@0
  1403
  /* optional: g_return_val_if_fail (pspec_info->value_validate != NULL, 0); */
sl@0
  1404
  /* default: g_return_val_if_fail (pspec_info->values_cmp != NULL, 0); */
sl@0
  1405
sl@0
  1406
  info.instance_size = pspec_info->instance_size;
sl@0
  1407
  info.n_preallocs = pspec_info->n_preallocs;
sl@0
  1408
  info.instance_init = (GInstanceInitFunc) pspec_info->instance_init;
sl@0
  1409
  cinfo = g_new (ParamSpecClassInfo, 1);
sl@0
  1410
  cinfo->value_type = pspec_info->value_type;
sl@0
  1411
  cinfo->finalize = pspec_info->finalize;
sl@0
  1412
  cinfo->value_set_default = pspec_info->value_set_default ? pspec_info->value_set_default : default_value_set_default;
sl@0
  1413
  cinfo->value_validate = pspec_info->value_validate;
sl@0
  1414
  cinfo->values_cmp = pspec_info->values_cmp ? pspec_info->values_cmp : default_values_cmp;
sl@0
  1415
  info.class_data = cinfo;
sl@0
  1416
sl@0
  1417
  return g_type_register_static (G_TYPE_PARAM, name, &info, 0);
sl@0
  1418
}
sl@0
  1419
sl@0
  1420
/**
sl@0
  1421
 * g_value_set_param:
sl@0
  1422
 * @value: a valid #GValue of type %G_TYPE_PARAM
sl@0
  1423
 * @param: the #GParamSpec to be set
sl@0
  1424
 *
sl@0
  1425
 * Set the contents of a %G_TYPE_PARAM #GValue to @param.
sl@0
  1426
 */
sl@0
  1427
EXPORT_C void
sl@0
  1428
g_value_set_param (GValue     *value,
sl@0
  1429
		   GParamSpec *param)
sl@0
  1430
{
sl@0
  1431
  g_return_if_fail (G_VALUE_HOLDS_PARAM (value));
sl@0
  1432
  if (param)
sl@0
  1433
    g_return_if_fail (G_IS_PARAM_SPEC (param));
sl@0
  1434
sl@0
  1435
  if (value->data[0].v_pointer)
sl@0
  1436
    g_param_spec_unref (value->data[0].v_pointer);
sl@0
  1437
  value->data[0].v_pointer = param;
sl@0
  1438
  if (value->data[0].v_pointer)
sl@0
  1439
    g_param_spec_ref (value->data[0].v_pointer);
sl@0
  1440
}
sl@0
  1441
sl@0
  1442
/**
sl@0
  1443
 * g_value_set_param_take_ownership:
sl@0
  1444
 * @value: a valid #GValue of type %G_TYPE_PARAM
sl@0
  1445
 * @param: the #GParamSpec to be set
sl@0
  1446
 *
sl@0
  1447
 * This is an internal function introduced mainly for C marshallers.
sl@0
  1448
 *
sl@0
  1449
 * Deprecated: 2.4: Use g_value_take_param() instead.
sl@0
  1450
 */
sl@0
  1451
EXPORT_C void
sl@0
  1452
g_value_set_param_take_ownership (GValue     *value,
sl@0
  1453
				  GParamSpec *param)
sl@0
  1454
{
sl@0
  1455
  g_value_take_param (value, param);
sl@0
  1456
}
sl@0
  1457
sl@0
  1458
/**
sl@0
  1459
 * g_value_take_param:
sl@0
  1460
 * @value: a valid #GValue of type %G_TYPE_PARAM
sl@0
  1461
 * @param: the #GParamSpec to be set
sl@0
  1462
 *
sl@0
  1463
 * Sets the contents of a %G_TYPE_PARAM #GValue to @param and takes
sl@0
  1464
 * over the ownership of the callers reference to @param; the caller
sl@0
  1465
 * doesn't have to unref it any more.
sl@0
  1466
 *
sl@0
  1467
 * Since: 2.4
sl@0
  1468
 */
sl@0
  1469
EXPORT_C void
sl@0
  1470
g_value_take_param (GValue     *value,
sl@0
  1471
		    GParamSpec *param)
sl@0
  1472
{
sl@0
  1473
  g_return_if_fail (G_VALUE_HOLDS_PARAM (value));
sl@0
  1474
  if (param)
sl@0
  1475
    g_return_if_fail (G_IS_PARAM_SPEC (param));
sl@0
  1476
sl@0
  1477
  if (value->data[0].v_pointer)
sl@0
  1478
    g_param_spec_unref (value->data[0].v_pointer);
sl@0
  1479
  value->data[0].v_pointer = param; /* we take over the reference count */
sl@0
  1480
}
sl@0
  1481
sl@0
  1482
/**
sl@0
  1483
 * g_value_get_param:
sl@0
  1484
 * @value: a valid #GValue whose type is derived from %G_TYPE_PARAM
sl@0
  1485
 *
sl@0
  1486
 * Get the contents of a %G_TYPE_PARAM #GValue.
sl@0
  1487
 *
sl@0
  1488
 * Returns: #GParamSpec content of @value
sl@0
  1489
 */
sl@0
  1490
EXPORT_C GParamSpec*
sl@0
  1491
g_value_get_param (const GValue *value)
sl@0
  1492
{
sl@0
  1493
  g_return_val_if_fail (G_VALUE_HOLDS_PARAM (value), NULL);
sl@0
  1494
sl@0
  1495
  return value->data[0].v_pointer;
sl@0
  1496
}
sl@0
  1497
sl@0
  1498
/**
sl@0
  1499
 * g_value_dup_param:
sl@0
  1500
 * @value: a valid #GValue whose type is derived from %G_TYPE_PARAM
sl@0
  1501
 *
sl@0
  1502
 * Get the contents of a %G_TYPE_PARAM #GValue, increasing its
sl@0
  1503
 * reference count.
sl@0
  1504
 *
sl@0
  1505
 * Returns: #GParamSpec content of @value, should be unreferenced when
sl@0
  1506
 *          no longer needed.
sl@0
  1507
 */
sl@0
  1508
EXPORT_C GParamSpec*
sl@0
  1509
g_value_dup_param (const GValue *value)
sl@0
  1510
{
sl@0
  1511
  g_return_val_if_fail (G_VALUE_HOLDS_PARAM (value), NULL);
sl@0
  1512
sl@0
  1513
  return value->data[0].v_pointer ? g_param_spec_ref (value->data[0].v_pointer) : NULL;
sl@0
  1514
}
sl@0
  1515
sl@0
  1516
#define __G_PARAM_C__
sl@0
  1517
#include "gobjectaliasdef.c"