variant of class_peek() which works for static types only.

Thu Nov 27 17:53:52 2003  Tim Janik  <timj@gtk.org>

        * gtype.[hc]:
        (g_type_class_peek_static): variant of class_peek() which works for
        static types only.

        * gobject.c:
        (g_object_do_class_init): make ::notify a run-action signal.
        (g_object_newv): use g_type_class_peek_static() by default to
        speed up common code path (trades two write-locks for one read-lock).
        (g_object_disconnect):
        (g_object_connect): allow signal specification words to be
        seperated by '-'.
        (g_object_set_valist):
        (g_object_new_valist): don't leak values.
        (g_object_get_property): check property for readability.
        (g_object_set_property): check property for writability and to
        not be construct-only.
        (g_object_set_valist): check property to not be construct-only.
This commit is contained in:
Tim Janik 2003-11-27 17:08:06 +00:00 committed by Tim Janik
parent 6e3b71aec3
commit 0642df0ab3
4 changed files with 112 additions and 47 deletions

View File

@ -1,3 +1,23 @@
Thu Nov 27 17:53:52 2003 Tim Janik <timj@gtk.org>
* gtype.[hc]:
(g_type_class_peek_static): variant of class_peek() which works for
static types only.
* gobject.c:
(g_object_do_class_init): make ::notify a run-action signal.
(g_object_newv): use g_type_class_peek_static() by default to
speed up common code path (trades two write-locks for one read-lock).
(g_object_disconnect):
(g_object_connect): allow signal specification words to be
seperated by '-'.
(g_object_set_valist):
(g_object_new_valist): don't leak values.
(g_object_get_property): check property for readability.
(g_object_set_property): check property for writability and to
not be construct-only.
(g_object_set_valist): check property to not be construct-only.
Sat Oct 25 01:09:17 2003 Matthias Clasen <maclas@gmx.de> Sat Oct 25 01:09:17 2003 Matthias Clasen <maclas@gmx.de>
* gparam.h (struct _GParamSpecClass): Add /*< private >*/ * gparam.h (struct _GParamSpecClass): Add /*< private >*/

View File

@ -253,7 +253,7 @@ g_object_do_class_init (GObjectClass *class)
gobject_signals[NOTIFY] = gobject_signals[NOTIFY] =
g_signal_new ("notify", g_signal_new ("notify",
G_TYPE_FROM_CLASS (class), G_TYPE_FROM_CLASS (class),
G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE | G_SIGNAL_DETAILED | G_SIGNAL_NO_HOOKS, G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE | G_SIGNAL_DETAILED | G_SIGNAL_NO_HOOKS | G_SIGNAL_ACTION,
G_STRUCT_OFFSET (GObjectClass, notify), G_STRUCT_OFFSET (GObjectClass, notify),
NULL, NULL, NULL, NULL,
g_cclosure_marshal_VOID__PARAM, g_cclosure_marshal_VOID__PARAM,
@ -414,7 +414,7 @@ g_object_class_override_property (GObjectClass *oclass,
if (!overridden) if (!overridden)
{ {
g_warning ("%s: Can't find property to override for '%s::%s'", g_warning ("%s: Can't find property to override for '%s::%s'",
G_STRLOC, G_OBJECT_CLASS_NAME (oclass), name); G_STRFUNC, G_OBJECT_CLASS_NAME (oclass), name);
return; return;
} }
@ -631,7 +631,7 @@ g_object_notify (GObject *object,
if (!pspec) if (!pspec)
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
G_OBJECT_TYPE_NAME (object), G_OBJECT_TYPE_NAME (object),
property_name); property_name);
else else
@ -656,8 +656,8 @@ g_object_thaw_notify (GObject *object)
g_object_ref (object); g_object_ref (object);
nqueue = g_object_notify_queue_from_object (object, &property_notify_context); nqueue = g_object_notify_queue_from_object (object, &property_notify_context);
if (!nqueue || !nqueue->freeze_count) if (!nqueue || !nqueue->freeze_count)
g_warning (G_STRLOC ": property-changed notification for %s(%p) is not frozen", g_warning ("%s: property-changed notification for %s(%p) is not frozen",
G_OBJECT_TYPE_NAME (object), object); G_STRFUNC, G_OBJECT_TYPE_NAME (object), object);
else else
g_object_notify_queue_thaw (object, nqueue); g_object_notify_queue_thaw (object, nqueue);
g_object_unref (object); g_object_unref (object);
@ -826,7 +826,7 @@ g_object_newv (GType object_type,
GObjectConstructParam *cparams, *oparams; GObjectConstructParam *cparams, *oparams;
GObjectNotifyQueue *nqueue; GObjectNotifyQueue *nqueue;
GObject *object; GObject *object;
GObjectClass *class; GObjectClass *class, *unref_class = NULL;
GSList *slist; GSList *slist;
guint n_total_cparams = 0, n_cparams = 0, n_oparams = 0, n_cvalues; guint n_total_cparams = 0, n_cparams = 0, n_oparams = 0, n_cvalues;
GValue *cvalues; GValue *cvalues;
@ -835,7 +835,9 @@ g_object_newv (GType object_type,
g_return_val_if_fail (G_TYPE_IS_OBJECT (object_type), NULL); g_return_val_if_fail (G_TYPE_IS_OBJECT (object_type), NULL);
class = g_type_class_ref (object_type); class = g_type_class_peek_static (object_type);
if (!class)
class = unref_class = g_type_class_ref (object_type);
for (slist = class->construct_properties; slist; slist = slist->next) for (slist = class->construct_properties; slist; slist = slist->next)
{ {
clist = g_list_prepend (clist, slist->data); clist = g_list_prepend (clist, slist->data);
@ -855,7 +857,7 @@ g_object_newv (GType object_type,
if (!pspec) if (!pspec)
{ {
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
g_type_name (object_type), g_type_name (object_type),
parameters[i].name); parameters[i].name);
continue; continue;
@ -863,7 +865,7 @@ g_object_newv (GType object_type,
if (!(pspec->flags & G_PARAM_WRITABLE)) if (!(pspec->flags & G_PARAM_WRITABLE))
{ {
g_warning ("%s: property `%s' of object class `%s' is not writable", g_warning ("%s: property `%s' of object class `%s' is not writable",
G_STRLOC, G_STRFUNC,
pspec->name, pspec->name,
g_type_name (object_type)); g_type_name (object_type));
continue; continue;
@ -874,8 +876,8 @@ g_object_newv (GType object_type,
if (!list) if (!list)
{ {
g_warning (G_STRLOC ": construct property \"%s\" for object `%s' can't be set twice", g_warning ("%s: construct property \"%s\" for object `%s' can't be set twice",
pspec->name, g_type_name (object_type)); G_STRFUNC, pspec->name, g_type_name (object_type));
continue; continue;
} }
cparams[n_cparams].pspec = pspec; cparams[n_cparams].pspec = pspec;
@ -936,7 +938,8 @@ g_object_newv (GType object_type,
object_set_property (object, oparams[i].pspec, oparams[i].value, nqueue); object_set_property (object, oparams[i].pspec, oparams[i].value, nqueue);
g_free (oparams); g_free (oparams);
g_type_class_unref (class); if (unref_class)
g_type_class_unref (unref_class);
/* release our own freeze count and handle notifications */ /* release our own freeze count and handle notifications */
g_object_notify_queue_thaw (object, nqueue); g_object_notify_queue_thaw (object, nqueue);
@ -974,7 +977,7 @@ g_object_new_valist (GType object_type,
if (!pspec) if (!pspec)
{ {
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
g_type_name (object_type), g_type_name (object_type),
name); name);
break; break;
@ -990,12 +993,9 @@ g_object_new_valist (GType object_type,
G_VALUE_COLLECT (&params[n_params].value, var_args, 0, &error); G_VALUE_COLLECT (&params[n_params].value, var_args, 0, &error);
if (error) if (error)
{ {
g_warning ("%s: %s", G_STRLOC, error); g_warning ("%s: %s", G_STRFUNC, error);
g_free (error); g_free (error);
g_value_unset (&params[n_params].value);
/* we purposely leak the value here, it might not be
* in a sane state if an error condition occoured
*/
break; break;
} }
n_params++; n_params++;
@ -1074,7 +1074,7 @@ g_object_set_valist (GObject *object,
if (!pspec) if (!pspec)
{ {
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
G_OBJECT_TYPE_NAME (object), G_OBJECT_TYPE_NAME (object),
name); name);
break; break;
@ -1082,23 +1082,26 @@ g_object_set_valist (GObject *object,
if (!(pspec->flags & G_PARAM_WRITABLE)) if (!(pspec->flags & G_PARAM_WRITABLE))
{ {
g_warning ("%s: property `%s' of object class `%s' is not writable", g_warning ("%s: property `%s' of object class `%s' is not writable",
G_STRLOC, G_STRFUNC,
pspec->name, pspec->name,
G_OBJECT_TYPE_NAME (object)); G_OBJECT_TYPE_NAME (object));
break; break;
} }
if (pspec->flags & G_PARAM_CONSTRUCT_ONLY)
{
g_warning ("%s: construct property \"%s\" for object `%s' can't be set after construction",
G_STRFUNC, pspec->name, G_OBJECT_TYPE_NAME (object));
break;
}
g_value_init (&value, G_PARAM_SPEC_VALUE_TYPE (pspec)); g_value_init (&value, G_PARAM_SPEC_VALUE_TYPE (pspec));
G_VALUE_COLLECT (&value, var_args, 0, &error); G_VALUE_COLLECT (&value, var_args, 0, &error);
if (error) if (error)
{ {
g_warning ("%s: %s", G_STRLOC, error); g_warning ("%s: %s", G_STRFUNC, error);
g_free (error); g_free (error);
g_value_unset (&value);
/* we purposely leak the value here, it might not be
* in a sane state if an error condition occoured
*/
break; break;
} }
@ -1138,7 +1141,7 @@ g_object_get_valist (GObject *object,
if (!pspec) if (!pspec)
{ {
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
G_OBJECT_TYPE_NAME (object), G_OBJECT_TYPE_NAME (object),
name); name);
break; break;
@ -1146,7 +1149,7 @@ g_object_get_valist (GObject *object,
if (!(pspec->flags & G_PARAM_READABLE)) if (!(pspec->flags & G_PARAM_READABLE))
{ {
g_warning ("%s: property `%s' of object class `%s' is not readable", g_warning ("%s: property `%s' of object class `%s' is not readable",
G_STRLOC, G_STRFUNC,
pspec->name, pspec->name,
G_OBJECT_TYPE_NAME (object)); G_OBJECT_TYPE_NAME (object));
break; break;
@ -1159,7 +1162,7 @@ g_object_get_valist (GObject *object,
G_VALUE_LCOPY (&value, var_args, 0, &error); G_VALUE_LCOPY (&value, var_args, 0, &error);
if (error) if (error)
{ {
g_warning ("%s: %s", G_STRLOC, error); g_warning ("%s: %s", G_STRFUNC, error);
g_free (error); g_free (error);
g_value_unset (&value); g_value_unset (&value);
break; break;
@ -1224,9 +1227,17 @@ g_object_set_property (GObject *object,
TRUE); TRUE);
if (!pspec) if (!pspec)
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
G_OBJECT_TYPE_NAME (object), G_OBJECT_TYPE_NAME (object),
property_name); property_name);
else if (!(pspec->flags & G_PARAM_WRITABLE))
g_warning ("%s: property `%s' of object class `%s' is not writable",
G_STRFUNC,
pspec->name,
G_OBJECT_TYPE_NAME (object));
else if (pspec->flags & G_PARAM_CONSTRUCT_ONLY)
g_warning ("%s: construct property \"%s\" for object `%s' can't be set after construction",
G_STRFUNC, pspec->name, G_OBJECT_TYPE_NAME (object));
else else
object_set_property (object, pspec, value, nqueue); object_set_property (object, pspec, value, nqueue);
@ -1253,9 +1264,14 @@ g_object_get_property (GObject *object,
TRUE); TRUE);
if (!pspec) if (!pspec)
g_warning ("%s: object class `%s' has no property named `%s'", g_warning ("%s: object class `%s' has no property named `%s'",
G_STRLOC, G_STRFUNC,
G_OBJECT_TYPE_NAME (object), G_OBJECT_TYPE_NAME (object),
property_name); property_name);
else if (!(pspec->flags & G_PARAM_READABLE))
g_warning ("%s: property `%s' of object class `%s' is not readable",
G_STRFUNC,
pspec->name,
G_OBJECT_TYPE_NAME (object));
else else
{ {
GValue *prop_value, tmp_value = { 0, }; GValue *prop_value, tmp_value = { 0, };
@ -1269,8 +1285,8 @@ g_object_get_property (GObject *object,
} }
else if (!g_value_type_transformable (G_PARAM_SPEC_VALUE_TYPE (pspec), G_VALUE_TYPE (value))) else if (!g_value_type_transformable (G_PARAM_SPEC_VALUE_TYPE (pspec), G_VALUE_TYPE (value)))
{ {
g_warning ("can't retrieve property `%s' of type `%s' as value of type `%s'", g_warning ("%s: can't retrieve property `%s' of type `%s' as value of type `%s'",
pspec->name, G_STRFUNC, pspec->name,
g_type_name (G_PARAM_SPEC_VALUE_TYPE (pspec)), g_type_name (G_PARAM_SPEC_VALUE_TYPE (pspec)),
G_VALUE_TYPE_NAME (value)); G_VALUE_TYPE_NAME (value));
g_object_unref (object); g_object_unref (object);
@ -1314,37 +1330,44 @@ g_object_connect (gpointer _object,
sid = g_signal_connect_data (object, signal_spec + 8, sid = g_signal_connect_data (object, signal_spec + 8,
callback, data, NULL, callback, data, NULL,
0); 0);
else if (strncmp (signal_spec, "object_signal::", 15) == 0) else if (strncmp (signal_spec, "object_signal::", 15) == 0 ||
strncmp (signal_spec, "object-signal::", 15) == 0)
sid = g_signal_connect_object (object, signal_spec + 15, sid = g_signal_connect_object (object, signal_spec + 15,
callback, data, callback, data,
0); 0);
else if (strncmp (signal_spec, "swapped_signal::", 16) == 0) else if (strncmp (signal_spec, "swapped_signal::", 16) == 0 ||
strncmp (signal_spec, "swapped-signal::", 16) == 0)
sid = g_signal_connect_data (object, signal_spec + 16, sid = g_signal_connect_data (object, signal_spec + 16,
callback, data, NULL, callback, data, NULL,
G_CONNECT_SWAPPED); G_CONNECT_SWAPPED);
else if (strncmp (signal_spec, "swapped_object_signal::", 23) == 0) else if (strncmp (signal_spec, "swapped_object_signal::", 23) == 0 ||
strncmp (signal_spec, "swapped-object-signal::", 23) == 0)
sid = g_signal_connect_object (object, signal_spec + 23, sid = g_signal_connect_object (object, signal_spec + 23,
callback, data, callback, data,
G_CONNECT_SWAPPED); G_CONNECT_SWAPPED);
else if (strncmp (signal_spec, "signal_after::", 14) == 0) else if (strncmp (signal_spec, "signal_after::", 14) == 0 ||
strncmp (signal_spec, "signal-after::", 14) == 0)
sid = g_signal_connect_data (object, signal_spec + 14, sid = g_signal_connect_data (object, signal_spec + 14,
callback, data, NULL, callback, data, NULL,
G_CONNECT_AFTER); G_CONNECT_AFTER);
else if (strncmp (signal_spec, "object_signal_after::", 21) == 0) else if (strncmp (signal_spec, "object_signal_after::", 21) == 0 ||
strncmp (signal_spec, "object-signal-after::", 21) == 0)
sid = g_signal_connect_object (object, signal_spec + 21, sid = g_signal_connect_object (object, signal_spec + 21,
callback, data, callback, data,
G_CONNECT_AFTER); G_CONNECT_AFTER);
else if (strncmp (signal_spec, "swapped_signal_after::", 22) == 0) else if (strncmp (signal_spec, "swapped_signal_after::", 22) == 0 ||
strncmp (signal_spec, "swapped-signal-after::", 22) == 0)
sid = g_signal_connect_data (object, signal_spec + 22, sid = g_signal_connect_data (object, signal_spec + 22,
callback, data, NULL, callback, data, NULL,
G_CONNECT_SWAPPED | G_CONNECT_AFTER); G_CONNECT_SWAPPED | G_CONNECT_AFTER);
else if (strncmp (signal_spec, "swapped_object_signal_after::", 29) == 0) else if (strncmp (signal_spec, "swapped_object_signal_after::", 29) == 0 ||
strncmp (signal_spec, "swapped-object-signal-after::", 29) == 0)
sid = g_signal_connect_object (object, signal_spec + 29, sid = g_signal_connect_object (object, signal_spec + 29,
callback, data, callback, data,
G_CONNECT_SWAPPED | G_CONNECT_AFTER); G_CONNECT_SWAPPED | G_CONNECT_AFTER);
else else
{ {
g_warning ("%s: invalid signal spec \"%s\"", G_STRLOC, signal_spec); g_warning ("%s: invalid signal spec \"%s\"", G_STRFUNC, signal_spec);
break; break;
} }
signal_spec = va_arg (var_args, gchar*); signal_spec = va_arg (var_args, gchar*);
@ -1372,29 +1395,31 @@ g_object_disconnect (gpointer _object,
gpointer data = va_arg (var_args, gpointer); gpointer data = va_arg (var_args, gpointer);
guint sid = 0, detail = 0, mask = 0; guint sid = 0, detail = 0, mask = 0;
if (strncmp (signal_spec, "any_signal::", 12) == 0) if (strncmp (signal_spec, "any_signal::", 12) == 0 ||
strncmp (signal_spec, "any-signal::", 12) == 0)
{ {
signal_spec += 12; signal_spec += 12;
mask = G_SIGNAL_MATCH_ID | G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA; mask = G_SIGNAL_MATCH_ID | G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA;
} }
else if (strcmp (signal_spec, "any_signal") == 0) else if (strcmp (signal_spec, "any_signal") == 0 ||
strcmp (signal_spec, "any-signal") == 0)
{ {
signal_spec += 10; signal_spec += 10;
mask = G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA; mask = G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA;
} }
else else
{ {
g_warning ("%s: invalid signal spec \"%s\"", G_STRLOC, signal_spec); g_warning ("%s: invalid signal spec \"%s\"", G_STRFUNC, signal_spec);
break; break;
} }
if ((mask & G_SIGNAL_MATCH_ID) && if ((mask & G_SIGNAL_MATCH_ID) &&
!g_signal_parse_name (signal_spec, G_OBJECT_TYPE (object), &sid, &detail, FALSE)) !g_signal_parse_name (signal_spec, G_OBJECT_TYPE (object), &sid, &detail, FALSE))
g_warning ("%s: invalid signal name \"%s\"", G_STRLOC, signal_spec); g_warning ("%s: invalid signal name \"%s\"", G_STRFUNC, signal_spec);
else if (!g_signal_handlers_disconnect_matched (object, mask | (detail ? G_SIGNAL_MATCH_DETAIL : 0), else if (!g_signal_handlers_disconnect_matched (object, mask | (detail ? G_SIGNAL_MATCH_DETAIL : 0),
sid, detail, sid, detail,
NULL, (gpointer)callback, data)) NULL, (gpointer)callback, data))
g_warning (G_STRLOC ": signal handler %p(%p) is not connected", callback, data); g_warning ("%s: signal handler %p(%p) is not connected", G_STRFUNC, callback, data);
signal_spec = va_arg (var_args, gchar*); signal_spec = va_arg (var_args, gchar*);
} }
va_end (var_args); va_end (var_args);
@ -1481,7 +1506,7 @@ g_object_weak_unref (GObject *object,
} }
} }
if (!found_one) if (!found_one)
g_warning (G_STRLOC ": couldn't find weak ref %p(%p)", notify, data); g_warning ("%s: couldn't find weak ref %p(%p)", G_STRFUNC, notify, data);
} }
void void

View File

@ -2462,6 +2462,25 @@ g_type_class_peek (GType type)
return class; return class;
} }
gpointer
g_type_class_peek_static (GType type)
{
TypeNode *node;
gpointer class;
node = lookup_type_node_I (type);
G_READ_LOCK (&type_rw_lock);
if (node && node->is_classed && node->data &&
/* peek only static types: */ node->plugin == NULL &&
node->data->class.class) /* common.ref_count _may_ be 0 */
class = node->data->class.class;
else
class = NULL;
G_READ_UNLOCK (&type_rw_lock);
return class;
}
gpointer gpointer
g_type_class_peek_parent (gpointer g_class) g_type_class_peek_parent (gpointer g_class)
{ {

View File

@ -177,6 +177,7 @@ gboolean g_type_is_a (GType type,
GType is_a_type); GType is_a_type);
gpointer g_type_class_ref (GType type); gpointer g_type_class_ref (GType type);
gpointer g_type_class_peek (GType type); gpointer g_type_class_peek (GType type);
gpointer g_type_class_peek_static (GType type);
void g_type_class_unref (gpointer g_class); void g_type_class_unref (gpointer g_class);
gpointer g_type_class_peek_parent (gpointer g_class); gpointer g_type_class_peek_parent (gpointer g_class);
gpointer g_type_interface_peek (gpointer instance_class, gpointer g_type_interface_peek (gpointer instance_class,