os/ossrv/glib/gobject/gparam.c
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/ossrv/glib/gobject/gparam.c	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,1517 @@
     1.4 +/* GObject - GLib Type, Object, Parameter and Signal Library
     1.5 + * Copyright (C) 1997-1999, 2000-2001 Tim Janik and Red Hat, Inc.
     1.6 + * Portions copyright (c) 2006-2009 Nokia Corporation.  All rights reserved.
     1.7 + *
     1.8 + * This library is free software; you can redistribute it and/or
     1.9 + * modify it under the terms of the GNU Lesser General Public
    1.10 + * License as published by the Free Software Foundation; either
    1.11 + * version 2 of the License, or (at your option) any later version.
    1.12 + *
    1.13 + * This library is distributed in the hope that it will be useful,
    1.14 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
    1.15 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
    1.16 + * Lesser General Public License for more details.
    1.17 + *
    1.18 + * You should have received a copy of the GNU Lesser General
    1.19 + * Public License along with this library; if not, write to the
    1.20 + * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
    1.21 + * Boston, MA 02111-1307, USA.
    1.22 + */
    1.23 +
    1.24 +/*
    1.25 + * MT safe
    1.26 + */
    1.27 +
    1.28 +#include "config.h"
    1.29 +
    1.30 +#include <string.h>
    1.31 +
    1.32 +#include "gparam.h"
    1.33 +#include "gparamspecs.h"
    1.34 +#include "gvaluecollector.h"
    1.35 +#include "gobjectalias.h"
    1.36 +
    1.37 +#ifdef __SYMBIAN32__
    1.38 +#include <glib_global.h>
    1.39 +#include "gobject_wsd.h"
    1.40 +#include <gobject_global.h>
    1.41 +#endif /* __SYMBIAN32__ */
    1.42 +/**
    1.43 + * SECTION:gparamspec
    1.44 + * @short_description: Metadata for parameter specifications
    1.45 + * @see_also: g_object_class_install_property(), g_object_set(),
    1.46 + *     g_object_get(), g_object_set_property(), g_object_get_property(),
    1.47 + *     g_value_register_transform_func()
    1.48 + * @title: GParamSpec
    1.49 + *
    1.50 + * #GParamSpec is an object structure that encapsulates the metadata
    1.51 + * required to specify parameters, such as e.g. #GObject properties.
    1.52 + *
    1.53 + * <para id="canonical-parameter-name">
    1.54 + * Parameter names need to start with a letter (a-z or A-Z). Subsequent
    1.55 + * characters can be letters, numbers or a '-'.
    1.56 + * All other characters are replaced by a '-' during construction.
    1.57 + * The result of this replacement is called the canonical name of the
    1.58 + * parameter.
    1.59 + * </para>
    1.60 + */
    1.61 +
    1.62 +
    1.63 +/* --- defines --- */
    1.64 +#define PARAM_FLOATING_FLAG                     0x2
    1.65 +#define	G_PARAM_USER_MASK			(~0 << G_PARAM_USER_SHIFT)
    1.66 +#define PSPEC_APPLIES_TO_VALUE(pspec, value)	(G_TYPE_CHECK_VALUE_TYPE ((value), G_PARAM_SPEC_VALUE_TYPE (pspec)))
    1.67 +#define	G_SLOCK(mutex)				g_static_mutex_lock (mutex)
    1.68 +#define	G_SUNLOCK(mutex)			g_static_mutex_unlock (mutex)
    1.69 +
    1.70 +
    1.71 +/* --- prototypes --- */
    1.72 +static void	g_param_spec_class_base_init	 (GParamSpecClass	*class);
    1.73 +static void	g_param_spec_class_base_finalize (GParamSpecClass	*class);
    1.74 +static void	g_param_spec_class_init		 (GParamSpecClass	*class,
    1.75 +						  gpointer               class_data);
    1.76 +static void	g_param_spec_init		 (GParamSpec		*pspec,
    1.77 +						  GParamSpecClass	*class);
    1.78 +static void	g_param_spec_finalize		 (GParamSpec		*pspec);
    1.79 +static void	value_param_init		(GValue		*value);
    1.80 +static void	value_param_free_value		(GValue		*value);
    1.81 +static void	value_param_copy_value		(const GValue	*src_value,
    1.82 +						 GValue		*dest_value);
    1.83 +static void	value_param_transform_value	(const GValue	*src_value,
    1.84 +						 GValue		*dest_value);
    1.85 +static gpointer	value_param_peek_pointer	(const GValue	*value);
    1.86 +static gchar*	value_param_collect_value	(GValue		*value,
    1.87 +						 guint           n_collect_values,
    1.88 +						 GTypeCValue    *collect_values,
    1.89 +						 guint           collect_flags);
    1.90 +static gchar*	value_param_lcopy_value		(const GValue	*value,
    1.91 +						 guint           n_collect_values,
    1.92 +						 GTypeCValue    *collect_values,
    1.93 +						 guint           collect_flags);
    1.94 +
    1.95 +
    1.96 +/* --- functions --- */
    1.97 +void
    1.98 +g_param_type_init (void)
    1.99 +{
   1.100 +  static const GTypeFundamentalInfo finfo = {
   1.101 +    (G_TYPE_FLAG_CLASSED |
   1.102 +     G_TYPE_FLAG_INSTANTIATABLE |
   1.103 +     G_TYPE_FLAG_DERIVABLE |
   1.104 +     G_TYPE_FLAG_DEEP_DERIVABLE),
   1.105 +  };
   1.106 +  static const GTypeValueTable param_value_table = {
   1.107 +    value_param_init,           /* value_init */
   1.108 +    value_param_free_value,     /* value_free */
   1.109 +    value_param_copy_value,     /* value_copy */
   1.110 +    value_param_peek_pointer,   /* value_peek_pointer */
   1.111 +    "p",			/* collect_format */
   1.112 +    value_param_collect_value,  /* collect_value */
   1.113 +    "p",			/* lcopy_format */
   1.114 +    value_param_lcopy_value,    /* lcopy_value */
   1.115 +  };
   1.116 +  static const GTypeInfo param_spec_info = {
   1.117 +    sizeof (GParamSpecClass),
   1.118 +
   1.119 +    (GBaseInitFunc) g_param_spec_class_base_init,
   1.120 +    (GBaseFinalizeFunc) g_param_spec_class_base_finalize,
   1.121 +    (GClassInitFunc) g_param_spec_class_init,
   1.122 +    (GClassFinalizeFunc) NULL,
   1.123 +    NULL,	/* class_data */
   1.124 +
   1.125 +    sizeof (GParamSpec),
   1.126 +    0,		/* n_preallocs */
   1.127 +    (GInstanceInitFunc) g_param_spec_init,
   1.128 +
   1.129 +    &param_value_table,
   1.130 +  };
   1.131 +  GType type;
   1.132 +
   1.133 +  /* This should be registred as GParamSpec instead of GParam, for
   1.134 +   * consistency sake, so that type name can be mapped to struct name,
   1.135 +   * However, some language bindings, most noticable the python ones
   1.136 +   * depends on the "GParam" identifier, see #548689
   1.137 +   */
   1.138 +  type = g_type_register_fundamental (G_TYPE_PARAM, g_intern_static_string ("GParam"), &param_spec_info, &finfo, G_TYPE_FLAG_ABSTRACT);
   1.139 +  g_assert (type == G_TYPE_PARAM);
   1.140 +  g_value_register_transform_func (G_TYPE_PARAM, G_TYPE_PARAM, value_param_transform_value);
   1.141 +}
   1.142 +
   1.143 +static void
   1.144 +g_param_spec_class_base_init (GParamSpecClass *class)
   1.145 +{
   1.146 +}
   1.147 +
   1.148 +static void
   1.149 +g_param_spec_class_base_finalize (GParamSpecClass *class)
   1.150 +{
   1.151 +}
   1.152 +
   1.153 +static void
   1.154 +g_param_spec_class_init (GParamSpecClass *class,
   1.155 +			 gpointer         class_data)
   1.156 +{
   1.157 +  class->value_type = G_TYPE_NONE;
   1.158 +  class->finalize = g_param_spec_finalize;
   1.159 +  class->value_set_default = NULL;
   1.160 +  class->value_validate = NULL;
   1.161 +  class->values_cmp = NULL;
   1.162 +}
   1.163 +
   1.164 +static void
   1.165 +g_param_spec_init (GParamSpec      *pspec,
   1.166 +		   GParamSpecClass *class)
   1.167 +{
   1.168 +  pspec->name = NULL;
   1.169 +  pspec->_nick = NULL;
   1.170 +  pspec->_blurb = NULL;
   1.171 +  pspec->flags = 0;
   1.172 +  pspec->value_type = class->value_type;
   1.173 +  pspec->owner_type = 0;
   1.174 +  pspec->qdata = NULL;
   1.175 +  g_datalist_init (&pspec->qdata);
   1.176 +  g_datalist_set_flags (&pspec->qdata, PARAM_FLOATING_FLAG);
   1.177 +  pspec->ref_count = 1;
   1.178 +  pspec->param_id = 0;
   1.179 +}
   1.180 +
   1.181 +static void
   1.182 +g_param_spec_finalize (GParamSpec *pspec)
   1.183 +{
   1.184 +  g_datalist_clear (&pspec->qdata);
   1.185 +
   1.186 +  if (!(pspec->flags & G_PARAM_STATIC_NAME))
   1.187 +    g_free (pspec->name);
   1.188 +  
   1.189 +  if (!(pspec->flags & G_PARAM_STATIC_NICK))
   1.190 +    g_free (pspec->_nick);
   1.191 +
   1.192 +  if (!(pspec->flags & G_PARAM_STATIC_BLURB))
   1.193 +    g_free (pspec->_blurb);
   1.194 +
   1.195 +  g_type_free_instance ((GTypeInstance*) pspec);
   1.196 +}
   1.197 +
   1.198 +/**
   1.199 + * g_param_spec_ref:
   1.200 + * @pspec: a valid #GParamSpec
   1.201 + *
   1.202 + * Increments the reference count of @pspec.
   1.203 + *
   1.204 + * Returns: the #GParamSpec that was passed into this function
   1.205 + */
   1.206 +EXPORT_C GParamSpec*
   1.207 +g_param_spec_ref (GParamSpec *pspec)
   1.208 +{
   1.209 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.210 +  g_return_val_if_fail (pspec->ref_count > 0, NULL);
   1.211 +
   1.212 +  g_atomic_int_inc ((int *)&pspec->ref_count);
   1.213 +
   1.214 +  return pspec;
   1.215 +}
   1.216 +
   1.217 +/**
   1.218 + * g_param_spec_unref:
   1.219 + * @pspec: a valid #GParamSpec
   1.220 + *
   1.221 + * Decrements the reference count of a @pspec.
   1.222 + */
   1.223 +EXPORT_C void
   1.224 +g_param_spec_unref (GParamSpec *pspec)
   1.225 +{
   1.226 +  gboolean is_zero;
   1.227 +
   1.228 +  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
   1.229 +  g_return_if_fail (pspec->ref_count > 0);
   1.230 +
   1.231 +  is_zero = g_atomic_int_dec_and_test ((int *)&pspec->ref_count);
   1.232 +
   1.233 +  if (G_UNLIKELY (is_zero))
   1.234 +    {
   1.235 +      G_PARAM_SPEC_GET_CLASS (pspec)->finalize (pspec);
   1.236 +    }
   1.237 +}
   1.238 +
   1.239 +/**
   1.240 + * g_param_spec_sink:
   1.241 + * @pspec: a valid #GParamSpec
   1.242 + *
   1.243 + * The initial reference count of a newly created #GParamSpec is 1,
   1.244 + * even though no one has explicitly called g_param_spec_ref() on it
   1.245 + * yet. So the initial reference count is flagged as "floating", until
   1.246 + * someone calls <literal>g_param_spec_ref (pspec); g_param_spec_sink
   1.247 + * (pspec);</literal> in sequence on it, taking over the initial
   1.248 + * reference count (thus ending up with a @pspec that has a reference
   1.249 + * count of 1 still, but is not flagged "floating" anymore).
   1.250 + */
   1.251 +EXPORT_C void
   1.252 +g_param_spec_sink (GParamSpec *pspec)
   1.253 +{
   1.254 +  gpointer oldvalue;
   1.255 +  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
   1.256 +  g_return_if_fail (pspec->ref_count > 0);
   1.257 +
   1.258 +  do
   1.259 +    oldvalue = g_atomic_pointer_get (&pspec->qdata);
   1.260 +  while (!g_atomic_pointer_compare_and_exchange ((void**) &pspec->qdata, oldvalue,
   1.261 +                                                 (gpointer) ((gsize) oldvalue & ~(gsize) PARAM_FLOATING_FLAG)));
   1.262 +  if ((gsize) oldvalue & PARAM_FLOATING_FLAG)
   1.263 +    g_param_spec_unref (pspec);
   1.264 +}
   1.265 +
   1.266 +/**
   1.267 + * g_param_spec_ref_sink:
   1.268 + * @pspec: a valid #GParamSpec
   1.269 + *
   1.270 + * Convenience function to ref and sink a #GParamSpec.
   1.271 + *
   1.272 + * Since: 2.10
   1.273 + * Returns: the #GParamSpec that was passed into this function
   1.274 + */
   1.275 +EXPORT_C GParamSpec*
   1.276 +g_param_spec_ref_sink (GParamSpec *pspec)
   1.277 +{
   1.278 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.279 +  g_return_val_if_fail (pspec->ref_count > 0, NULL);
   1.280 +
   1.281 +  g_param_spec_ref (pspec);
   1.282 +  g_param_spec_sink (pspec);
   1.283 +  return pspec;
   1.284 +}
   1.285 +
   1.286 +/**
   1.287 + * g_param_spec_get_name:
   1.288 + * @pspec: a valid #GParamSpec
   1.289 + *
   1.290 + * Get the name of a #GParamSpec.
   1.291 + *
   1.292 + * Returns: the name of @pspec.
   1.293 + */
   1.294 +EXPORT_C G_CONST_RETURN gchar*
   1.295 +g_param_spec_get_name (GParamSpec *pspec)
   1.296 +{
   1.297 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.298 +
   1.299 +  return pspec->name;
   1.300 +}
   1.301 +
   1.302 +/**
   1.303 + * g_param_spec_get_nick:
   1.304 + * @pspec: a valid #GParamSpec
   1.305 + *
   1.306 + * Get the nickname of a #GParamSpec.
   1.307 + *
   1.308 + * Returns: the nickname of @pspec.
   1.309 + */
   1.310 +EXPORT_C G_CONST_RETURN gchar*
   1.311 +g_param_spec_get_nick (GParamSpec *pspec)
   1.312 +{
   1.313 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.314 +
   1.315 +  if (pspec->_nick)
   1.316 +    return pspec->_nick;
   1.317 +  else
   1.318 +    {
   1.319 +      GParamSpec *redirect_target;
   1.320 +
   1.321 +      redirect_target = g_param_spec_get_redirect_target (pspec);
   1.322 +      if (redirect_target && redirect_target->_nick)
   1.323 +	return redirect_target->_nick;
   1.324 +    }
   1.325 +
   1.326 +  return pspec->name;
   1.327 +}
   1.328 +
   1.329 +/**
   1.330 + * g_param_spec_get_blurb:
   1.331 + * @pspec: a valid #GParamSpec
   1.332 + *
   1.333 + * Get the short description of a #GParamSpec.
   1.334 + *
   1.335 + * Returns: the short description of @pspec.
   1.336 + */
   1.337 +EXPORT_C G_CONST_RETURN gchar*
   1.338 +g_param_spec_get_blurb (GParamSpec *pspec)
   1.339 +{
   1.340 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.341 +
   1.342 +  if (pspec->_blurb)
   1.343 +    return pspec->_blurb;
   1.344 +  else
   1.345 +    {
   1.346 +      GParamSpec *redirect_target;
   1.347 +
   1.348 +      redirect_target = g_param_spec_get_redirect_target (pspec);
   1.349 +      if (redirect_target && redirect_target->_blurb)
   1.350 +	return redirect_target->_blurb;
   1.351 +    }
   1.352 +
   1.353 +  return NULL;
   1.354 +}
   1.355 +
   1.356 +static void
   1.357 +canonicalize_key (gchar *key)
   1.358 +{
   1.359 +  gchar *p;
   1.360 +  
   1.361 +  for (p = key; *p != 0; p++)
   1.362 +    {
   1.363 +      gchar c = *p;
   1.364 +      
   1.365 +      if (c != '-' &&
   1.366 +	  (c < '0' || c > '9') &&
   1.367 +	  (c < 'A' || c > 'Z') &&
   1.368 +	  (c < 'a' || c > 'z'))
   1.369 +	*p = '-';
   1.370 +    }
   1.371 +}
   1.372 +
   1.373 +static gboolean
   1.374 +is_canonical (const gchar *key)
   1.375 +{
   1.376 +  const gchar *p;
   1.377 +
   1.378 +  for (p = key; *p != 0; p++)
   1.379 +    {
   1.380 +      gchar c = *p;
   1.381 +      
   1.382 +      if (c != '-' &&
   1.383 +	  (c < '0' || c > '9') &&
   1.384 +	  (c < 'A' || c > 'Z') &&
   1.385 +	  (c < 'a' || c > 'z'))
   1.386 +	return FALSE;
   1.387 +    }
   1.388 +
   1.389 +  return TRUE;
   1.390 +}
   1.391 +
   1.392 +/**
   1.393 + * g_param_spec_internal:
   1.394 + * @param_type: the #GType for the property; must be derived from #G_TYPE_PARAM
   1.395 + * @name: the canonical name of the property
   1.396 + * @nick: the nickname of the property
   1.397 + * @blurb: a short description of the property
   1.398 + * @flags: a combination of #GParamFlags
   1.399 + *
   1.400 + * Creates a new #GParamSpec instance.
   1.401 + *
   1.402 + * A property name consists of segments consisting of ASCII letters and
   1.403 + * digits, separated by either the '-' or '_' character. The first
   1.404 + * character of a property name must be a letter. Names which violate these
   1.405 + * rules lead to undefined behaviour.
   1.406 + *
   1.407 + * When creating and looking up a #GParamSpec, either separator can be
   1.408 + * used, but they cannot be mixed. Using '-' is considerably more
   1.409 + * efficient and in fact required when using property names as detail
   1.410 + * strings for signals.
   1.411 + *
   1.412 + * Beyond the name, #GParamSpec<!-- -->s have two more descriptive
   1.413 + * strings associated with them, the @nick, which should be suitable
   1.414 + * for use as a label for the property in a property editor, and the
   1.415 + * @blurb, which should be a somewhat longer description, suitable for
   1.416 + * e.g. a tooltip. The @nick and @blurb should ideally be localized.
   1.417 + *
   1.418 + * Returns: a newly allocated #GParamSpec instance
   1.419 + */
   1.420 +EXPORT_C gpointer
   1.421 +g_param_spec_internal (GType        param_type,
   1.422 +		       const gchar *name,
   1.423 +		       const gchar *nick,
   1.424 +		       const gchar *blurb,
   1.425 +		       GParamFlags  flags)
   1.426 +{
   1.427 +  GParamSpec *pspec;
   1.428 +  
   1.429 +  g_return_val_if_fail (G_TYPE_IS_PARAM (param_type) && param_type != G_TYPE_PARAM, NULL);
   1.430 +  g_return_val_if_fail (name != NULL, NULL);
   1.431 +  g_return_val_if_fail ((name[0] >= 'A' && name[0] <= 'Z') || (name[0] >= 'a' && name[0] <= 'z'), NULL);
   1.432 +  g_return_val_if_fail (!(flags & G_PARAM_STATIC_NAME) || is_canonical (name), NULL);
   1.433 +  
   1.434 +  pspec = (gpointer) g_type_create_instance (param_type);
   1.435 +
   1.436 +  if (flags & G_PARAM_STATIC_NAME)
   1.437 +    {
   1.438 +      pspec->name = g_intern_static_string (name);
   1.439 +      if (!is_canonical (pspec->name))
   1.440 +        g_warning ("G_PARAM_STATIC_NAME used with non-canonical pspec name: %s", pspec->name);
   1.441 +    }
   1.442 +  else
   1.443 +    {
   1.444 +      pspec->name = g_strdup (name);
   1.445 +      canonicalize_key (pspec->name);
   1.446 +      g_intern_string (pspec->name);
   1.447 +    }
   1.448 +
   1.449 +  if (flags & G_PARAM_STATIC_NICK)
   1.450 +    pspec->_nick = (gchar*) nick;
   1.451 +  else
   1.452 +    pspec->_nick = g_strdup (nick);
   1.453 +
   1.454 +  if (flags & G_PARAM_STATIC_BLURB)
   1.455 +    pspec->_blurb = (gchar*) blurb;
   1.456 +  else
   1.457 +    pspec->_blurb = g_strdup (blurb);
   1.458 +
   1.459 +  pspec->flags = (flags & G_PARAM_USER_MASK) | (flags & G_PARAM_MASK);
   1.460 +  
   1.461 +  return pspec;
   1.462 +}
   1.463 +
   1.464 +/**
   1.465 + * g_param_spec_get_qdata:
   1.466 + * @pspec: a valid #GParamSpec
   1.467 + * @quark: a #GQuark, naming the user data pointer
   1.468 + *
   1.469 + * Gets back user data pointers stored via g_param_spec_set_qdata().
   1.470 + *
   1.471 + * Returns: the user data pointer set, or %NULL
   1.472 + */
   1.473 +EXPORT_C gpointer
   1.474 +g_param_spec_get_qdata (GParamSpec *pspec,
   1.475 +			GQuark      quark)
   1.476 +{
   1.477 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.478 +  
   1.479 +  return quark ? g_datalist_id_get_data (&pspec->qdata, quark) : NULL;
   1.480 +}
   1.481 +
   1.482 +/**
   1.483 + * g_param_spec_set_qdata:
   1.484 + * @pspec: the #GParamSpec to set store a user data pointer
   1.485 + * @quark: a #GQuark, naming the user data pointer
   1.486 + * @data: an opaque user data pointer
   1.487 + *
   1.488 + * Sets an opaque, named pointer on a #GParamSpec. The name is
   1.489 + * specified through a #GQuark (retrieved e.g. via
   1.490 + * g_quark_from_static_string()), and the pointer can be gotten back
   1.491 + * from the @pspec with g_param_spec_get_qdata().  Setting a
   1.492 + * previously set user data pointer, overrides (frees) the old pointer
   1.493 + * set, using %NULL as pointer essentially removes the data stored.
   1.494 + */
   1.495 +EXPORT_C void
   1.496 +g_param_spec_set_qdata (GParamSpec *pspec,
   1.497 +			GQuark      quark,
   1.498 +			gpointer    data)
   1.499 +{
   1.500 +  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
   1.501 +  g_return_if_fail (quark > 0);
   1.502 +
   1.503 +  g_datalist_id_set_data (&pspec->qdata, quark, data);
   1.504 +}
   1.505 +
   1.506 +/**
   1.507 + * g_param_spec_set_qdata_full:
   1.508 + * @pspec: the #GParamSpec to set store a user data pointer
   1.509 + * @quark: a #GQuark, naming the user data pointer
   1.510 + * @data: an opaque user data pointer
   1.511 + * @destroy: function to invoke with @data as argument, when @data needs to
   1.512 + *  be freed
   1.513 + *
   1.514 + * This function works like g_param_spec_set_qdata(), but in addition,
   1.515 + * a <literal>void (*destroy) (gpointer)</literal> function may be
   1.516 + * specified which is called with @data as argument when the @pspec is
   1.517 + * finalized, or the data is being overwritten by a call to
   1.518 + * g_param_spec_set_qdata() with the same @quark.
   1.519 + */
   1.520 +EXPORT_C void
   1.521 +g_param_spec_set_qdata_full (GParamSpec    *pspec,
   1.522 +			     GQuark         quark,
   1.523 +			     gpointer       data,
   1.524 +			     GDestroyNotify destroy)
   1.525 +{
   1.526 +  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
   1.527 +  g_return_if_fail (quark > 0);
   1.528 +
   1.529 +  g_datalist_id_set_data_full (&pspec->qdata, quark, data, data ? destroy : (GDestroyNotify) NULL);
   1.530 +}
   1.531 +
   1.532 +/**
   1.533 + * g_param_spec_steal_qdata:
   1.534 + * @pspec: the #GParamSpec to get a stored user data pointer from
   1.535 + * @quark: a #GQuark, naming the user data pointer
   1.536 + *
   1.537 + * Gets back user data pointers stored via g_param_spec_set_qdata()
   1.538 + * and removes the @data from @pspec without invoking its destroy()
   1.539 + * function (if any was set).  Usually, calling this function is only
   1.540 + * required to update user data pointers with a destroy notifier.
   1.541 + *
   1.542 + * Returns: the user data pointer set, or %NULL
   1.543 + */
   1.544 +EXPORT_C gpointer
   1.545 +g_param_spec_steal_qdata (GParamSpec *pspec,
   1.546 +			  GQuark      quark)
   1.547 +{
   1.548 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.549 +  g_return_val_if_fail (quark > 0, NULL);
   1.550 +  
   1.551 +  return g_datalist_id_remove_no_notify (&pspec->qdata, quark);
   1.552 +}
   1.553 +
   1.554 +/**
   1.555 + * g_param_spec_get_redirect_target:
   1.556 + * @pspec: a #GParamSpec
   1.557 + *
   1.558 + * If the paramspec redirects operations to another paramspec,
   1.559 + * returns that paramspec. Redirect is used typically for
   1.560 + * providing a new implementation of a property in a derived
   1.561 + * type while preserving all the properties from the parent
   1.562 + * type. Redirection is established by creating a property
   1.563 + * of type #GParamSpecOverride. See g_object_class_override_property()
   1.564 + * for an example of the use of this capability.
   1.565 + *
   1.566 + * Since: 2.4
   1.567 + *
   1.568 + * Returns: paramspec to which requests on this paramspec should
   1.569 + *          be redirected, or %NULL if none.
   1.570 + */
   1.571 +EXPORT_C GParamSpec*
   1.572 +g_param_spec_get_redirect_target (GParamSpec *pspec)
   1.573 +{
   1.574 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), NULL);
   1.575 +
   1.576 +  if (G_IS_PARAM_SPEC_OVERRIDE (pspec))
   1.577 +    {
   1.578 +      GParamSpecOverride *ospec = G_PARAM_SPEC_OVERRIDE (pspec);
   1.579 +
   1.580 +      return ospec->overridden;
   1.581 +    }
   1.582 +  else
   1.583 +    return NULL;
   1.584 +}
   1.585 +
   1.586 +/**
   1.587 + * g_param_value_set_default:
   1.588 + * @pspec: a valid #GParamSpec
   1.589 + * @value: a #GValue of correct type for @pspec
   1.590 + *
   1.591 + * Sets @value to its default value as specified in @pspec.
   1.592 + */
   1.593 +EXPORT_C void
   1.594 +g_param_value_set_default (GParamSpec *pspec,
   1.595 +			   GValue     *value)
   1.596 +{
   1.597 +  g_return_if_fail (G_IS_PARAM_SPEC (pspec));
   1.598 +  g_return_if_fail (G_IS_VALUE (value));
   1.599 +  g_return_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value));
   1.600 +
   1.601 +  g_value_reset (value);
   1.602 +  G_PARAM_SPEC_GET_CLASS (pspec)->value_set_default (pspec, value);
   1.603 +}
   1.604 +
   1.605 +/**
   1.606 + * g_param_value_defaults:
   1.607 + * @pspec: a valid #GParamSpec
   1.608 + * @value: a #GValue of correct type for @pspec
   1.609 + *
   1.610 + * Checks whether @value contains the default value as specified in @pspec.
   1.611 + *
   1.612 + * Returns: whether @value contains the canonical default for this @pspec
   1.613 + */
   1.614 +EXPORT_C gboolean
   1.615 +g_param_value_defaults (GParamSpec *pspec,
   1.616 +			GValue     *value)
   1.617 +{
   1.618 +  GValue dflt_value = { 0, };
   1.619 +  gboolean defaults;
   1.620 +
   1.621 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
   1.622 +  g_return_val_if_fail (G_IS_VALUE (value), FALSE);
   1.623 +  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value), FALSE);
   1.624 +
   1.625 +  g_value_init (&dflt_value, G_PARAM_SPEC_VALUE_TYPE (pspec));
   1.626 +  G_PARAM_SPEC_GET_CLASS (pspec)->value_set_default (pspec, &dflt_value);
   1.627 +  defaults = G_PARAM_SPEC_GET_CLASS (pspec)->values_cmp (pspec, value, &dflt_value) == 0;
   1.628 +  g_value_unset (&dflt_value);
   1.629 +
   1.630 +  return defaults;
   1.631 +}
   1.632 +
   1.633 +/**
   1.634 + * g_param_value_validate:
   1.635 + * @pspec: a valid #GParamSpec
   1.636 + * @value: a #GValue of correct type for @pspec
   1.637 + *
   1.638 + * Ensures that the contents of @value comply with the specifications
   1.639 + * set out by @pspec. For example, a #GParamSpecInt might require
   1.640 + * that integers stored in @value may not be smaller than -42 and not be
   1.641 + * greater than +42. If @value contains an integer outside of this range,
   1.642 + * it is modified accordingly, so the resulting value will fit into the
   1.643 + * range -42 .. +42.
   1.644 + *
   1.645 + * Returns: whether modifying @value was necessary to ensure validity
   1.646 + */
   1.647 +EXPORT_C gboolean
   1.648 +g_param_value_validate (GParamSpec *pspec,
   1.649 +			GValue     *value)
   1.650 +{
   1.651 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
   1.652 +  g_return_val_if_fail (G_IS_VALUE (value), FALSE);
   1.653 +  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value), FALSE);
   1.654 +
   1.655 +  if (G_PARAM_SPEC_GET_CLASS (pspec)->value_validate)
   1.656 +    {
   1.657 +      GValue oval = *value;
   1.658 +
   1.659 +      if (G_PARAM_SPEC_GET_CLASS (pspec)->value_validate (pspec, value) ||
   1.660 +	  memcmp (&oval.data, &value->data, sizeof (oval.data)))
   1.661 +	return TRUE;
   1.662 +    }
   1.663 +
   1.664 +  return FALSE;
   1.665 +}
   1.666 +
   1.667 +/**
   1.668 + * g_param_value_convert:
   1.669 + * @pspec: a valid #GParamSpec
   1.670 + * @src_value: souce #GValue
   1.671 + * @dest_value: destination #GValue of correct type for @pspec
   1.672 + * @strict_validation: %TRUE requires @dest_value to conform to @pspec
   1.673 + * without modifications
   1.674 + *
   1.675 + * Transforms @src_value into @dest_value if possible, and then
   1.676 + * validates @dest_value, in order for it to conform to @pspec.  If
   1.677 + * @strict_validation is %TRUE this function will only succeed if the
   1.678 + * transformed @dest_value complied to @pspec without modifications.
   1.679 + *
   1.680 + * See also g_value_type_transformable(), g_value_transform() and
   1.681 + * g_param_value_validate().
   1.682 + *
   1.683 + * Returns: %TRUE if transformation and validation were successful,
   1.684 + *  %FALSE otherwise and @dest_value is left untouched.
   1.685 + */
   1.686 +EXPORT_C gboolean
   1.687 +g_param_value_convert (GParamSpec   *pspec,
   1.688 +		       const GValue *src_value,
   1.689 +		       GValue       *dest_value,
   1.690 +		       gboolean	     strict_validation)
   1.691 +{
   1.692 +  GValue tmp_value = { 0, };
   1.693 +
   1.694 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
   1.695 +  g_return_val_if_fail (G_IS_VALUE (src_value), FALSE);
   1.696 +  g_return_val_if_fail (G_IS_VALUE (dest_value), FALSE);
   1.697 +  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, dest_value), FALSE);
   1.698 +
   1.699 +  /* better leave dest_value untouched when returning FALSE */
   1.700 +
   1.701 +  g_value_init (&tmp_value, G_VALUE_TYPE (dest_value));
   1.702 +  if (g_value_transform (src_value, &tmp_value) &&
   1.703 +      (!g_param_value_validate (pspec, &tmp_value) || !strict_validation))
   1.704 +    {
   1.705 +      g_value_unset (dest_value);
   1.706 +      
   1.707 +      /* values are relocatable */
   1.708 +      memcpy (dest_value, &tmp_value, sizeof (tmp_value));
   1.709 +      
   1.710 +      return TRUE;
   1.711 +    }
   1.712 +  else
   1.713 +    {
   1.714 +      g_value_unset (&tmp_value);
   1.715 +      
   1.716 +      return FALSE;
   1.717 +    }
   1.718 +}
   1.719 +
   1.720 +/**
   1.721 + * g_param_values_cmp:
   1.722 + * @pspec: a valid #GParamSpec
   1.723 + * @value1: a #GValue of correct type for @pspec
   1.724 + * @value2: a #GValue of correct type for @pspec
   1.725 + *
   1.726 + * Compares @value1 with @value2 according to @pspec, and return -1, 0 or +1,
   1.727 + * if @value1 is found to be less than, equal to or greater than @value2,
   1.728 + * respectively.
   1.729 + *
   1.730 + * Returns: -1, 0 or +1, for a less than, equal to or greater than result
   1.731 + */
   1.732 +EXPORT_C gint
   1.733 +g_param_values_cmp (GParamSpec   *pspec,
   1.734 +		    const GValue *value1,
   1.735 +		    const GValue *value2)
   1.736 +{
   1.737 +  gint cmp;
   1.738 +
   1.739 +  /* param_values_cmp() effectively does: value1 - value2
   1.740 +   * so the return values are:
   1.741 +   * -1)  value1 < value2
   1.742 +   *  0)  value1 == value2
   1.743 +   *  1)  value1 > value2
   1.744 +   */
   1.745 +  g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), 0);
   1.746 +  g_return_val_if_fail (G_IS_VALUE (value1), 0);
   1.747 +  g_return_val_if_fail (G_IS_VALUE (value2), 0);
   1.748 +  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value1), 0);
   1.749 +  g_return_val_if_fail (PSPEC_APPLIES_TO_VALUE (pspec, value2), 0);
   1.750 +
   1.751 +  cmp = G_PARAM_SPEC_GET_CLASS (pspec)->values_cmp (pspec, value1, value2);
   1.752 +
   1.753 +  return CLAMP (cmp, -1, 1);
   1.754 +}
   1.755 +
   1.756 +static void
   1.757 +value_param_init (GValue *value)
   1.758 +{
   1.759 +  value->data[0].v_pointer = NULL;
   1.760 +}
   1.761 +
   1.762 +static void
   1.763 +value_param_free_value (GValue *value)
   1.764 +{
   1.765 +  if (value->data[0].v_pointer)
   1.766 +    g_param_spec_unref (value->data[0].v_pointer);
   1.767 +}
   1.768 +
   1.769 +static void
   1.770 +value_param_copy_value (const GValue *src_value,
   1.771 +			GValue       *dest_value)
   1.772 +{
   1.773 +  if (src_value->data[0].v_pointer)
   1.774 +    dest_value->data[0].v_pointer = g_param_spec_ref (src_value->data[0].v_pointer);
   1.775 +  else
   1.776 +    dest_value->data[0].v_pointer = NULL;
   1.777 +}
   1.778 +
   1.779 +static void
   1.780 +value_param_transform_value (const GValue *src_value,
   1.781 +			     GValue       *dest_value)
   1.782 +{
   1.783 +  if (src_value->data[0].v_pointer &&
   1.784 +      g_type_is_a (G_PARAM_SPEC_TYPE (dest_value->data[0].v_pointer), G_VALUE_TYPE (dest_value)))
   1.785 +    dest_value->data[0].v_pointer = g_param_spec_ref (src_value->data[0].v_pointer);
   1.786 +  else
   1.787 +    dest_value->data[0].v_pointer = NULL;
   1.788 +}
   1.789 +
   1.790 +static gpointer
   1.791 +value_param_peek_pointer (const GValue *value)
   1.792 +{
   1.793 +  return value->data[0].v_pointer;
   1.794 +}
   1.795 +
   1.796 +static gchar*
   1.797 +value_param_collect_value (GValue      *value,
   1.798 +			   guint        n_collect_values,
   1.799 +			   GTypeCValue *collect_values,
   1.800 +			   guint        collect_flags)
   1.801 +{
   1.802 +  if (collect_values[0].v_pointer)
   1.803 +    {
   1.804 +      GParamSpec *param = collect_values[0].v_pointer;
   1.805 +
   1.806 +      if (param->g_type_instance.g_class == NULL)
   1.807 +	return g_strconcat ("invalid unclassed param spec pointer for value type `",
   1.808 +			    G_VALUE_TYPE_NAME (value),
   1.809 +			    "'",
   1.810 +			    NULL);
   1.811 +      else if (!g_value_type_compatible (G_PARAM_SPEC_TYPE (param), G_VALUE_TYPE (value)))
   1.812 +	return g_strconcat ("invalid param spec type `",
   1.813 +			    G_PARAM_SPEC_TYPE_NAME (param),
   1.814 +			    "' for value type `",
   1.815 +			    G_VALUE_TYPE_NAME (value),
   1.816 +			    "'",
   1.817 +			    NULL);
   1.818 +      value->data[0].v_pointer = g_param_spec_ref (param);
   1.819 +    }
   1.820 +  else
   1.821 +    value->data[0].v_pointer = NULL;
   1.822 +
   1.823 +  return NULL;
   1.824 +}
   1.825 +
   1.826 +static gchar*
   1.827 +value_param_lcopy_value (const GValue *value,
   1.828 +			 guint         n_collect_values,
   1.829 +			 GTypeCValue  *collect_values,
   1.830 +			 guint         collect_flags)
   1.831 +{
   1.832 +  GParamSpec **param_p = collect_values[0].v_pointer;
   1.833 +
   1.834 +  if (!param_p)
   1.835 +    return g_strdup_printf ("value location for `%s' passed as NULL", G_VALUE_TYPE_NAME (value));
   1.836 +
   1.837 +  if (!value->data[0].v_pointer)
   1.838 +    *param_p = NULL;
   1.839 +  else if (collect_flags & G_VALUE_NOCOPY_CONTENTS)
   1.840 +    *param_p = value->data[0].v_pointer;
   1.841 +  else
   1.842 +    *param_p = g_param_spec_ref (value->data[0].v_pointer);
   1.843 +
   1.844 +  return NULL;
   1.845 +}
   1.846 +
   1.847 +
   1.848 +/* --- param spec pool --- */
   1.849 +/**
   1.850 + * GParamSpecPool:
   1.851 + *
   1.852 + * A #GParamSpecPool maintains a collection of #GParamSpec<!-- -->s which can be
   1.853 + * quickly accessed by owner and name. The implementation of the #GObject property
   1.854 + * system uses such a pool to store the #GParamSpecs of the properties all object
   1.855 + * types.
   1.856 + */
   1.857 +struct _GParamSpecPool
   1.858 +{
   1.859 +  GStaticMutex smutex;
   1.860 +  gboolean     type_prefixing;
   1.861 +  GHashTable  *hash_table;
   1.862 +};
   1.863 +
   1.864 +static guint
   1.865 +param_spec_pool_hash (gconstpointer key_spec)
   1.866 +{
   1.867 +  const GParamSpec *key = key_spec;
   1.868 +  const gchar *p;
   1.869 +  guint h = key->owner_type;
   1.870 +
   1.871 +  for (p = key->name; *p; p++)
   1.872 +    h = (h << 5) - h + *p;
   1.873 +
   1.874 +  return h;
   1.875 +}
   1.876 +
   1.877 +static gboolean
   1.878 +param_spec_pool_equals (gconstpointer key_spec_1,
   1.879 +			gconstpointer key_spec_2)
   1.880 +{
   1.881 +  const GParamSpec *key1 = key_spec_1;
   1.882 +  const GParamSpec *key2 = key_spec_2;
   1.883 +
   1.884 +  return (key1->owner_type == key2->owner_type &&
   1.885 +	  strcmp (key1->name, key2->name) == 0);
   1.886 +}
   1.887 +
   1.888 +#if EMULATOR
   1.889 +PLS(init_smutex,g_param_spec_pool_new ,GStaticMutex)
   1.890 +#define init_smutex (*FUNCTION_NAME(init_smutex,g_param_spec_pool_new )())
   1.891 +#endif /* EMULATOR */
   1.892 +
   1.893 +/**
   1.894 + * g_param_spec_pool_new:
   1.895 + * @type_prefixing: Whether the pool will support type-prefixed property names.
   1.896 + *
   1.897 + * Creates a new #GParamSpecPool.
   1.898 + *
   1.899 + * If @type_prefixing is %TRUE, lookups in the newly created pool will
   1.900 + * allow to specify the owner as a colon-separated prefix of the
   1.901 + * property name, like "GtkContainer:border-width". This feature is
   1.902 + * deprecated, so you should always set @type_prefixing to %FALSE.
   1.903 + *
   1.904 + * Returns: a newly allocated #GParamSpecPool.
   1.905 + */
   1.906 +EXPORT_C GParamSpecPool*
   1.907 +g_param_spec_pool_new (gboolean type_prefixing)
   1.908 +{
   1.909 +  #if !(EMULATOR)
   1.910 +  static GStaticMutex init_smutex = G_STATIC_MUTEX_INIT;
   1.911 +  #endif /* EMULATOR */
   1.912 +  GParamSpecPool *pool = g_new (GParamSpecPool, 1);
   1.913 +
   1.914 +  memcpy (&pool->smutex, &init_smutex, sizeof (init_smutex));
   1.915 +  pool->type_prefixing = type_prefixing != FALSE;
   1.916 +  pool->hash_table = g_hash_table_new (param_spec_pool_hash, param_spec_pool_equals);
   1.917 +
   1.918 +  return pool;
   1.919 +}
   1.920 +
   1.921 +#if EMULATOR
   1.922 +#undef init_smutex 
   1.923 +#endif /* EMULATOR */
   1.924 +
   1.925 +/**
   1.926 + * g_param_spec_pool_insert:
   1.927 + * @pool: a #GParamSpecPool.
   1.928 + * @pspec: the #GParamSpec to insert
   1.929 + * @owner_type: a #GType identifying the owner of @pspec
   1.930 + *
   1.931 + * Inserts a #GParamSpec in the pool.
   1.932 + */
   1.933 +EXPORT_C void
   1.934 +g_param_spec_pool_insert (GParamSpecPool *pool,
   1.935 +			  GParamSpec     *pspec,
   1.936 +			  GType           owner_type)
   1.937 +{
   1.938 +  gchar *p;
   1.939 +  
   1.940 +  if (pool && pspec && owner_type > 0 && pspec->owner_type == 0)
   1.941 +    {
   1.942 +      G_SLOCK (&pool->smutex);
   1.943 +      for (p = pspec->name; *p; p++)
   1.944 +	{
   1.945 +	  if (!strchr (G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS "-_", *p))
   1.946 +	    {
   1.947 +	      g_warning (G_STRLOC ": pspec name \"%s\" contains invalid characters", pspec->name);
   1.948 +	      G_SUNLOCK (&pool->smutex);
   1.949 +	      return;
   1.950 +	    }
   1.951 +	}
   1.952 +      
   1.953 +      pspec->owner_type = owner_type;
   1.954 +      g_param_spec_ref (pspec);
   1.955 +      g_hash_table_insert (pool->hash_table, pspec, pspec);
   1.956 +      G_SUNLOCK (&pool->smutex);
   1.957 +    }
   1.958 +  else
   1.959 +    {
   1.960 +      g_return_if_fail (pool != NULL);
   1.961 +      g_return_if_fail (pspec);
   1.962 +      g_return_if_fail (owner_type > 0);
   1.963 +      g_return_if_fail (pspec->owner_type == 0);
   1.964 +    }
   1.965 +}
   1.966 +
   1.967 +/**
   1.968 + * g_param_spec_pool_remove:
   1.969 + * @pool: a #GParamSpecPool
   1.970 + * @pspec: the #GParamSpec to remove
   1.971 + *
   1.972 + * Removes a #GParamSpec from the pool.
   1.973 + */
   1.974 +EXPORT_C void
   1.975 +g_param_spec_pool_remove (GParamSpecPool *pool,
   1.976 +			  GParamSpec     *pspec)
   1.977 +{
   1.978 +  if (pool && pspec)
   1.979 +    {
   1.980 +      G_SLOCK (&pool->smutex);
   1.981 +      if (g_hash_table_remove (pool->hash_table, pspec))
   1.982 +	g_param_spec_unref (pspec);
   1.983 +      else
   1.984 +	g_warning (G_STRLOC ": attempt to remove unknown pspec `%s' from pool", pspec->name);
   1.985 +      G_SUNLOCK (&pool->smutex);
   1.986 +    }
   1.987 +  else
   1.988 +    {
   1.989 +      g_return_if_fail (pool != NULL);
   1.990 +      g_return_if_fail (pspec);
   1.991 +    }
   1.992 +}
   1.993 +
   1.994 +static inline GParamSpec*
   1.995 +param_spec_ht_lookup (GHashTable  *hash_table,
   1.996 +		      const gchar *param_name,
   1.997 +		      GType        owner_type,
   1.998 +		      gboolean     walk_ancestors)
   1.999 +{
  1.1000 +  GParamSpec key, *pspec;
  1.1001 +
  1.1002 +  key.owner_type = owner_type;
  1.1003 +  key.name = (gchar*) param_name;
  1.1004 +  if (walk_ancestors)
  1.1005 +    do
  1.1006 +      {
  1.1007 +	pspec = g_hash_table_lookup (hash_table, &key);
  1.1008 +	if (pspec)
  1.1009 +	  return pspec;
  1.1010 +	key.owner_type = g_type_parent (key.owner_type);
  1.1011 +      }
  1.1012 +    while (key.owner_type);
  1.1013 +  else
  1.1014 +    pspec = g_hash_table_lookup (hash_table, &key);
  1.1015 +
  1.1016 +  if (!pspec && !is_canonical (param_name))
  1.1017 +    {
  1.1018 +      /* try canonicalized form */
  1.1019 +      key.name = g_strdup (param_name);
  1.1020 +      key.owner_type = owner_type;
  1.1021 +      
  1.1022 +      canonicalize_key (key.name);
  1.1023 +      if (walk_ancestors)
  1.1024 +	do
  1.1025 +	  {
  1.1026 +	    pspec = g_hash_table_lookup (hash_table, &key);
  1.1027 +	    if (pspec)
  1.1028 +	      {
  1.1029 +		g_free (key.name);
  1.1030 +		return pspec;
  1.1031 +	      }
  1.1032 +	    key.owner_type = g_type_parent (key.owner_type);
  1.1033 +	  }
  1.1034 +	while (key.owner_type);
  1.1035 +      else
  1.1036 +	pspec = g_hash_table_lookup (hash_table, &key);
  1.1037 +      g_free (key.name);
  1.1038 +    }
  1.1039 +
  1.1040 +  return pspec;
  1.1041 +}
  1.1042 +
  1.1043 +/**
  1.1044 + * g_param_spec_pool_lookup:
  1.1045 + * @pool: a #GParamSpecPool
  1.1046 + * @param_name: the name to look for
  1.1047 + * @owner_type: the owner to look for
  1.1048 + * @walk_ancestors: If %TRUE, also try to find a #GParamSpec with @param_name
  1.1049 + *  owned by an ancestor of @owner_type.
  1.1050 + *
  1.1051 + * Looks up a #GParamSpec in the pool.
  1.1052 + *
  1.1053 + * Returns: The found #GParamSpec, or %NULL if no matching #GParamSpec was found.
  1.1054 + */
  1.1055 +EXPORT_C GParamSpec*
  1.1056 +g_param_spec_pool_lookup (GParamSpecPool *pool,
  1.1057 +			  const gchar    *param_name,
  1.1058 +			  GType           owner_type,
  1.1059 +			  gboolean        walk_ancestors)
  1.1060 +{
  1.1061 +  GParamSpec *pspec;
  1.1062 +  gchar *delim;
  1.1063 +
  1.1064 +  if (!pool || !param_name)
  1.1065 +    {
  1.1066 +      g_return_val_if_fail (pool != NULL, NULL);
  1.1067 +      g_return_val_if_fail (param_name != NULL, NULL);
  1.1068 +    }
  1.1069 +
  1.1070 +  G_SLOCK (&pool->smutex);
  1.1071 +
  1.1072 +  delim = pool->type_prefixing ? strchr (param_name, ':') : NULL;
  1.1073 +
  1.1074 +  /* try quick and away, i.e. without prefix */
  1.1075 +  if (!delim)
  1.1076 +    {
  1.1077 +      pspec = param_spec_ht_lookup (pool->hash_table, param_name, owner_type, walk_ancestors);
  1.1078 +      G_SUNLOCK (&pool->smutex);
  1.1079 +
  1.1080 +      return pspec;
  1.1081 +    }
  1.1082 +
  1.1083 +  /* strip type prefix */
  1.1084 +  if (pool->type_prefixing && delim[1] == ':')
  1.1085 +    {
  1.1086 +      guint l = delim - param_name;
  1.1087 +      gchar stack_buffer[32], *buffer = l < 32 ? stack_buffer : g_new (gchar, l + 1);
  1.1088 +      GType type;
  1.1089 +      
  1.1090 +      strncpy (buffer, param_name, delim - param_name);
  1.1091 +      buffer[l] = 0;
  1.1092 +      type = g_type_from_name (buffer);
  1.1093 +      if (l >= 32)
  1.1094 +	g_free (buffer);
  1.1095 +      if (type)		/* type==0 isn't a valid type pefix */
  1.1096 +	{
  1.1097 +	  /* sanity check, these cases don't make a whole lot of sense */
  1.1098 +	  if ((!walk_ancestors && type != owner_type) || !g_type_is_a (owner_type, type))
  1.1099 +	    {
  1.1100 +	      G_SUNLOCK (&pool->smutex);
  1.1101 +
  1.1102 +	      return NULL;
  1.1103 +	    }
  1.1104 +	  owner_type = type;
  1.1105 +	  param_name += l + 2;
  1.1106 +	  pspec = param_spec_ht_lookup (pool->hash_table, param_name, owner_type, walk_ancestors);
  1.1107 +	  G_SUNLOCK (&pool->smutex);
  1.1108 +
  1.1109 +	  return pspec;
  1.1110 +	}
  1.1111 +    }
  1.1112 +  /* malformed param_name */
  1.1113 +
  1.1114 +  G_SUNLOCK (&pool->smutex);
  1.1115 +
  1.1116 +  return NULL;
  1.1117 +}
  1.1118 +
  1.1119 +static void
  1.1120 +pool_list (gpointer key,
  1.1121 +	   gpointer value,
  1.1122 +	   gpointer user_data)
  1.1123 +{
  1.1124 +  GParamSpec *pspec = value;
  1.1125 +  gpointer *data = user_data;
  1.1126 +  GType owner_type = (GType) data[1];
  1.1127 +
  1.1128 +  if (owner_type == pspec->owner_type)
  1.1129 +    data[0] = g_list_prepend (data[0], pspec);
  1.1130 +}
  1.1131 +
  1.1132 +/**
  1.1133 + * g_param_spec_pool_list_owned:
  1.1134 + * @pool: a #GParamSpecPool
  1.1135 + * @owner_type: the owner to look for
  1.1136 + *
  1.1137 + * Gets an #GList of all #GParamSpec<!-- -->s owned by @owner_type in
  1.1138 + * the pool.
  1.1139 + *
  1.1140 + * Returns: a #GList of all #GParamSpec<!-- -->s owned by @owner_type
  1.1141 + *          in the pool#GParamSpec<!-- -->s.
  1.1142 + */
  1.1143 +EXPORT_C GList*
  1.1144 +g_param_spec_pool_list_owned (GParamSpecPool *pool,
  1.1145 +			      GType           owner_type)
  1.1146 +{
  1.1147 +  gpointer data[2];
  1.1148 +
  1.1149 +  g_return_val_if_fail (pool != NULL, NULL);
  1.1150 +  g_return_val_if_fail (owner_type > 0, NULL);
  1.1151 +  
  1.1152 +  G_SLOCK (&pool->smutex);
  1.1153 +  data[0] = NULL;
  1.1154 +  data[1] = (gpointer) owner_type;
  1.1155 +  g_hash_table_foreach (pool->hash_table, pool_list, &data);
  1.1156 +  G_SUNLOCK (&pool->smutex);
  1.1157 +
  1.1158 +  return data[0];
  1.1159 +}
  1.1160 +
  1.1161 +static gint
  1.1162 +pspec_compare_id (gconstpointer a,
  1.1163 +		  gconstpointer b)
  1.1164 +{
  1.1165 +  const GParamSpec *pspec1 = a, *pspec2 = b;
  1.1166 +
  1.1167 +  return pspec1->param_id < pspec2->param_id ? -1 : pspec1->param_id > pspec2->param_id;
  1.1168 +}
  1.1169 +
  1.1170 +static inline GSList*
  1.1171 +pspec_list_remove_overridden_and_redirected (GSList     *plist,
  1.1172 +					     GHashTable *ht,
  1.1173 +					     GType       owner_type,
  1.1174 +					     guint      *n_p)
  1.1175 +{
  1.1176 +  GSList *rlist = NULL;
  1.1177 +
  1.1178 +  while (plist)
  1.1179 +    {
  1.1180 +      GSList *tmp = plist->next;
  1.1181 +      GParamSpec *pspec = plist->data;
  1.1182 +      GParamSpec *found;
  1.1183 +      gboolean remove = FALSE;
  1.1184 +
  1.1185 +      /* Remove paramspecs that are redirected, and also paramspecs
  1.1186 +       * that have are overridden by non-redirected properties.
  1.1187 +       * The idea is to get the single paramspec for each name that
  1.1188 +       * best corresponds to what the application sees.
  1.1189 +       */
  1.1190 +      if (g_param_spec_get_redirect_target (pspec))
  1.1191 +	remove = TRUE;
  1.1192 +      else
  1.1193 +	{
  1.1194 +	  found = param_spec_ht_lookup (ht, pspec->name, owner_type, TRUE);
  1.1195 +	  if (found != pspec)
  1.1196 +	    {
  1.1197 +	      GParamSpec *redirect = g_param_spec_get_redirect_target (found);
  1.1198 +	      if (redirect != pspec)
  1.1199 +		remove = TRUE;
  1.1200 +	    }
  1.1201 +	}
  1.1202 +
  1.1203 +      if (remove)
  1.1204 +	{
  1.1205 +	  g_slist_free_1 (plist);
  1.1206 +	}
  1.1207 +      else
  1.1208 +	{
  1.1209 +	  plist->next = rlist;
  1.1210 +	  rlist = plist;
  1.1211 +	  *n_p += 1;
  1.1212 +	}
  1.1213 +      plist = tmp;
  1.1214 +    }
  1.1215 +  return rlist;
  1.1216 +}
  1.1217 +
  1.1218 +static void
  1.1219 +pool_depth_list (gpointer key,
  1.1220 +		 gpointer value,
  1.1221 +		 gpointer user_data)
  1.1222 +{
  1.1223 +  GParamSpec *pspec = value;
  1.1224 +  gpointer *data = user_data;
  1.1225 +  GSList **slists = data[0];
  1.1226 +  GType owner_type = (GType) data[1];
  1.1227 +
  1.1228 +  if (g_type_is_a (owner_type, pspec->owner_type))
  1.1229 +    {
  1.1230 +      if (G_TYPE_IS_INTERFACE (pspec->owner_type))
  1.1231 +	{
  1.1232 +	  slists[0] = g_slist_prepend (slists[0], pspec);
  1.1233 +	}
  1.1234 +      else
  1.1235 +	{
  1.1236 +	  guint d = g_type_depth (pspec->owner_type);
  1.1237 +
  1.1238 +	  slists[d - 1] = g_slist_prepend (slists[d - 1], pspec);
  1.1239 +	}
  1.1240 +    }
  1.1241 +}
  1.1242 +
  1.1243 +/* We handle interfaces specially since we don't want to
  1.1244 + * count interface prerequisites like normal inheritance;
  1.1245 + * the property comes from the direct inheritance from
  1.1246 + * the prerequisite class, not from the interface that
  1.1247 + * prerequires it.
  1.1248 + * 
  1.1249 + * also 'depth' isn't a meaningful concept for interface
  1.1250 + * prerequites.
  1.1251 + */
  1.1252 +static void
  1.1253 +pool_depth_list_for_interface (gpointer key,
  1.1254 +			       gpointer value,
  1.1255 +			       gpointer user_data)
  1.1256 +{
  1.1257 +  GParamSpec *pspec = value;
  1.1258 +  gpointer *data = user_data;
  1.1259 +  GSList **slists = data[0];
  1.1260 +  GType owner_type = (GType) data[1];
  1.1261 +
  1.1262 +  if (pspec->owner_type == owner_type)
  1.1263 +    slists[0] = g_slist_prepend (slists[0], pspec);
  1.1264 +}
  1.1265 +
  1.1266 +/**
  1.1267 + * g_param_spec_pool_list:
  1.1268 + * @pool: a #GParamSpecPool
  1.1269 + * @owner_type: the owner to look for
  1.1270 + * @n_pspecs_p: return location for the length of the returned array
  1.1271 + *
  1.1272 + * Gets an array of all #GParamSpec<!-- -->s owned by @owner_type in
  1.1273 + * the pool.
  1.1274 + *
  1.1275 + * Returns: a newly allocated array containing pointers to all
  1.1276 + *          #GParamSpec<!-- -->s owned by @owner_type in the pool
  1.1277 + */
  1.1278 +EXPORT_C GParamSpec** /* free result */
  1.1279 +g_param_spec_pool_list (GParamSpecPool *pool,
  1.1280 +			GType           owner_type,
  1.1281 +			guint          *n_pspecs_p)
  1.1282 +{
  1.1283 +  GParamSpec **pspecs, **p;
  1.1284 +  GSList **slists, *node;
  1.1285 +  gpointer data[2];
  1.1286 +  guint d, i;
  1.1287 +
  1.1288 +  g_return_val_if_fail (pool != NULL, NULL);
  1.1289 +  g_return_val_if_fail (owner_type > 0, NULL);
  1.1290 +  g_return_val_if_fail (n_pspecs_p != NULL, NULL);
  1.1291 +  
  1.1292 +  G_SLOCK (&pool->smutex);
  1.1293 +  *n_pspecs_p = 0;
  1.1294 +  d = g_type_depth (owner_type);
  1.1295 +  slists = g_new0 (GSList*, d);
  1.1296 +  data[0] = slists;
  1.1297 +  data[1] = (gpointer) owner_type;
  1.1298 +
  1.1299 +  g_hash_table_foreach (pool->hash_table,
  1.1300 +			G_TYPE_IS_INTERFACE (owner_type) ?
  1.1301 +			   pool_depth_list_for_interface :
  1.1302 +			   pool_depth_list,
  1.1303 +			&data);
  1.1304 +  
  1.1305 +  for (i = 0; i < d; i++)
  1.1306 +    slists[i] = pspec_list_remove_overridden_and_redirected (slists[i], pool->hash_table, owner_type, n_pspecs_p);
  1.1307 +  pspecs = g_new (GParamSpec*, *n_pspecs_p + 1);
  1.1308 +  p = pspecs;
  1.1309 +  for (i = 0; i < d; i++)
  1.1310 +    {
  1.1311 +      slists[i] = g_slist_sort (slists[i], pspec_compare_id);
  1.1312 +      for (node = slists[i]; node; node = node->next)
  1.1313 +	*p++ = node->data;
  1.1314 +      g_slist_free (slists[i]);
  1.1315 +    }
  1.1316 +  *p++ = NULL;
  1.1317 +  g_free (slists);
  1.1318 +  G_SUNLOCK (&pool->smutex);
  1.1319 +
  1.1320 +  return pspecs;
  1.1321 +}
  1.1322 +
  1.1323 +
  1.1324 +/* --- auxillary functions --- */
  1.1325 +typedef struct
  1.1326 +{
  1.1327 +  /* class portion */
  1.1328 +  GType           value_type;
  1.1329 +  void          (*finalize)             (GParamSpec   *pspec);
  1.1330 +  void          (*value_set_default)    (GParamSpec   *pspec,
  1.1331 +					 GValue       *value);
  1.1332 +  gboolean      (*value_validate)       (GParamSpec   *pspec,
  1.1333 +					 GValue       *value);
  1.1334 +  gint          (*values_cmp)           (GParamSpec   *pspec,
  1.1335 +					 const GValue *value1,
  1.1336 +					 const GValue *value2);
  1.1337 +} ParamSpecClassInfo;
  1.1338 +
  1.1339 +static void
  1.1340 +param_spec_generic_class_init (gpointer g_class,
  1.1341 +			       gpointer class_data)
  1.1342 +{
  1.1343 +  GParamSpecClass *class = g_class;
  1.1344 +  ParamSpecClassInfo *info = class_data;
  1.1345 +
  1.1346 +  class->value_type = info->value_type;
  1.1347 +  if (info->finalize)
  1.1348 +    class->finalize = info->finalize;			/* optional */
  1.1349 +  class->value_set_default = info->value_set_default;
  1.1350 +  if (info->value_validate)
  1.1351 +    class->value_validate = info->value_validate;	/* optional */
  1.1352 +  class->values_cmp = info->values_cmp;
  1.1353 +  g_free (class_data);
  1.1354 +}
  1.1355 +
  1.1356 +static void
  1.1357 +default_value_set_default (GParamSpec *pspec,
  1.1358 +			   GValue     *value)
  1.1359 +{
  1.1360 +  /* value is already zero initialized */
  1.1361 +}
  1.1362 +
  1.1363 +static gint
  1.1364 +default_values_cmp (GParamSpec   *pspec,
  1.1365 +		    const GValue *value1,
  1.1366 +		    const GValue *value2)
  1.1367 +{
  1.1368 +  return memcmp (&value1->data, &value2->data, sizeof (value1->data));
  1.1369 +}
  1.1370 +
  1.1371 +/**
  1.1372 + * g_param_type_register_static:
  1.1373 + * @name: 0-terminated string used as the name of the new #GParamSpec type.
  1.1374 + * @pspec_info: The #GParamSpecTypeInfo for this #GParamSpec type.
  1.1375 + *
  1.1376 + * Registers @name as the name of a new static type derived from
  1.1377 + * #G_TYPE_PARAM. The type system uses the information contained in
  1.1378 + * the #GParamSpecTypeInfo structure pointed to by @info to manage the
  1.1379 + * #GParamSpec type and its instances.
  1.1380 + *
  1.1381 + * Returns: The new type identifier.
  1.1382 + */
  1.1383 +EXPORT_C GType
  1.1384 +g_param_type_register_static (const gchar              *name,
  1.1385 +			      const GParamSpecTypeInfo *pspec_info)
  1.1386 +{
  1.1387 +  GTypeInfo info = {
  1.1388 +    sizeof (GParamSpecClass),      /* class_size */
  1.1389 +    NULL,                          /* base_init */
  1.1390 +    NULL,                          /* base_destroy */
  1.1391 +    param_spec_generic_class_init, /* class_init */
  1.1392 +    NULL,                          /* class_destroy */
  1.1393 +    NULL,                          /* class_data */
  1.1394 +    0,                             /* instance_size */
  1.1395 +    16,                            /* n_preallocs */
  1.1396 +    NULL,                          /* instance_init */
  1.1397 +  };
  1.1398 +  ParamSpecClassInfo *cinfo;
  1.1399 +
  1.1400 +  g_return_val_if_fail (name != NULL, 0);
  1.1401 +  g_return_val_if_fail (pspec_info != NULL, 0);
  1.1402 +  g_return_val_if_fail (g_type_from_name (name) == 0, 0);
  1.1403 +  g_return_val_if_fail (pspec_info->instance_size >= sizeof (GParamSpec), 0);
  1.1404 +  g_return_val_if_fail (g_type_name (pspec_info->value_type) != NULL, 0);
  1.1405 +  /* default: g_return_val_if_fail (pspec_info->value_set_default != NULL, 0); */
  1.1406 +  /* optional: g_return_val_if_fail (pspec_info->value_validate != NULL, 0); */
  1.1407 +  /* default: g_return_val_if_fail (pspec_info->values_cmp != NULL, 0); */
  1.1408 +
  1.1409 +  info.instance_size = pspec_info->instance_size;
  1.1410 +  info.n_preallocs = pspec_info->n_preallocs;
  1.1411 +  info.instance_init = (GInstanceInitFunc) pspec_info->instance_init;
  1.1412 +  cinfo = g_new (ParamSpecClassInfo, 1);
  1.1413 +  cinfo->value_type = pspec_info->value_type;
  1.1414 +  cinfo->finalize = pspec_info->finalize;
  1.1415 +  cinfo->value_set_default = pspec_info->value_set_default ? pspec_info->value_set_default : default_value_set_default;
  1.1416 +  cinfo->value_validate = pspec_info->value_validate;
  1.1417 +  cinfo->values_cmp = pspec_info->values_cmp ? pspec_info->values_cmp : default_values_cmp;
  1.1418 +  info.class_data = cinfo;
  1.1419 +
  1.1420 +  return g_type_register_static (G_TYPE_PARAM, name, &info, 0);
  1.1421 +}
  1.1422 +
  1.1423 +/**
  1.1424 + * g_value_set_param:
  1.1425 + * @value: a valid #GValue of type %G_TYPE_PARAM
  1.1426 + * @param: the #GParamSpec to be set
  1.1427 + *
  1.1428 + * Set the contents of a %G_TYPE_PARAM #GValue to @param.
  1.1429 + */
  1.1430 +EXPORT_C void
  1.1431 +g_value_set_param (GValue     *value,
  1.1432 +		   GParamSpec *param)
  1.1433 +{
  1.1434 +  g_return_if_fail (G_VALUE_HOLDS_PARAM (value));
  1.1435 +  if (param)
  1.1436 +    g_return_if_fail (G_IS_PARAM_SPEC (param));
  1.1437 +
  1.1438 +  if (value->data[0].v_pointer)
  1.1439 +    g_param_spec_unref (value->data[0].v_pointer);
  1.1440 +  value->data[0].v_pointer = param;
  1.1441 +  if (value->data[0].v_pointer)
  1.1442 +    g_param_spec_ref (value->data[0].v_pointer);
  1.1443 +}
  1.1444 +
  1.1445 +/**
  1.1446 + * g_value_set_param_take_ownership:
  1.1447 + * @value: a valid #GValue of type %G_TYPE_PARAM
  1.1448 + * @param: the #GParamSpec to be set
  1.1449 + *
  1.1450 + * This is an internal function introduced mainly for C marshallers.
  1.1451 + *
  1.1452 + * Deprecated: 2.4: Use g_value_take_param() instead.
  1.1453 + */
  1.1454 +EXPORT_C void
  1.1455 +g_value_set_param_take_ownership (GValue     *value,
  1.1456 +				  GParamSpec *param)
  1.1457 +{
  1.1458 +  g_value_take_param (value, param);
  1.1459 +}
  1.1460 +
  1.1461 +/**
  1.1462 + * g_value_take_param:
  1.1463 + * @value: a valid #GValue of type %G_TYPE_PARAM
  1.1464 + * @param: the #GParamSpec to be set
  1.1465 + *
  1.1466 + * Sets the contents of a %G_TYPE_PARAM #GValue to @param and takes
  1.1467 + * over the ownership of the callers reference to @param; the caller
  1.1468 + * doesn't have to unref it any more.
  1.1469 + *
  1.1470 + * Since: 2.4
  1.1471 + */
  1.1472 +EXPORT_C void
  1.1473 +g_value_take_param (GValue     *value,
  1.1474 +		    GParamSpec *param)
  1.1475 +{
  1.1476 +  g_return_if_fail (G_VALUE_HOLDS_PARAM (value));
  1.1477 +  if (param)
  1.1478 +    g_return_if_fail (G_IS_PARAM_SPEC (param));
  1.1479 +
  1.1480 +  if (value->data[0].v_pointer)
  1.1481 +    g_param_spec_unref (value->data[0].v_pointer);
  1.1482 +  value->data[0].v_pointer = param; /* we take over the reference count */
  1.1483 +}
  1.1484 +
  1.1485 +/**
  1.1486 + * g_value_get_param:
  1.1487 + * @value: a valid #GValue whose type is derived from %G_TYPE_PARAM
  1.1488 + *
  1.1489 + * Get the contents of a %G_TYPE_PARAM #GValue.
  1.1490 + *
  1.1491 + * Returns: #GParamSpec content of @value
  1.1492 + */
  1.1493 +EXPORT_C GParamSpec*
  1.1494 +g_value_get_param (const GValue *value)
  1.1495 +{
  1.1496 +  g_return_val_if_fail (G_VALUE_HOLDS_PARAM (value), NULL);
  1.1497 +
  1.1498 +  return value->data[0].v_pointer;
  1.1499 +}
  1.1500 +
  1.1501 +/**
  1.1502 + * g_value_dup_param:
  1.1503 + * @value: a valid #GValue whose type is derived from %G_TYPE_PARAM
  1.1504 + *
  1.1505 + * Get the contents of a %G_TYPE_PARAM #GValue, increasing its
  1.1506 + * reference count.
  1.1507 + *
  1.1508 + * Returns: #GParamSpec content of @value, should be unreferenced when
  1.1509 + *          no longer needed.
  1.1510 + */
  1.1511 +EXPORT_C GParamSpec*
  1.1512 +g_value_dup_param (const GValue *value)
  1.1513 +{
  1.1514 +  g_return_val_if_fail (G_VALUE_HOLDS_PARAM (value), NULL);
  1.1515 +
  1.1516 +  return value->data[0].v_pointer ? g_param_spec_ref (value->data[0].v_pointer) : NULL;
  1.1517 +}
  1.1518 +
  1.1519 +#define __G_PARAM_C__
  1.1520 +#include "gobjectaliasdef.c"