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 + ¶m_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"), ¶m_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"