os/ossrv/glib/gobject/gobjectnotifyqueue.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* GObject - GLib Type, Object, Parameter and Signal Library
sl@0
     2
 * Copyright (C) 1998-1999, 2000-2001 Tim Janik and Red Hat, Inc.
sl@0
     3
 *
sl@0
     4
 * This library is free software; you can redistribute it and/or
sl@0
     5
 * modify it under the terms of the GNU Lesser General Public
sl@0
     6
 * License as published by the Free Software Foundation; either
sl@0
     7
 * version 2 of the License, or (at your option) any later version.
sl@0
     8
 *
sl@0
     9
 * This library is distributed in the hope that it will be useful,
sl@0
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
sl@0
    12
 * Lesser General Public License for more details.
sl@0
    13
 *
sl@0
    14
 * You should have received a copy of the GNU Lesser General
sl@0
    15
 * Public License along with this library; if not, write to the
sl@0
    16
 * Free Software Foundation, Inc., 59 Temple Place, Suite 330,
sl@0
    17
 * Boston, MA 02111-1307, USA.
sl@0
    18
 */
sl@0
    19
sl@0
    20
#ifndef __G_OBJECT_NOTIFY_QUEUE_H__
sl@0
    21
#define __G_OBJECT_NOTIFY_QUEUE_H__
sl@0
    22
sl@0
    23
#include <string.h> /* memset */
sl@0
    24
sl@0
    25
#include <glib-object.h>
sl@0
    26
sl@0
    27
G_BEGIN_DECLS
sl@0
    28
sl@0
    29
sl@0
    30
/* --- typedefs --- */
sl@0
    31
#if (!EMULATOR)
sl@0
    32
typedef struct _GObjectNotifyContext          GObjectNotifyContext;
sl@0
    33
#endif
sl@0
    34
typedef struct _GObjectNotifyQueue            GObjectNotifyQueue;
sl@0
    35
#if (!EMULATOR)
sl@0
    36
typedef void (*GObjectNotifyQueueDispatcher) (GObject     *object,
sl@0
    37
					      guint        n_pspecs,
sl@0
    38
					      GParamSpec **pspecs);
sl@0
    39
sl@0
    40
sl@0
    41
/* --- structures --- */
sl@0
    42
struct _GObjectNotifyContext
sl@0
    43
{
sl@0
    44
  GQuark                       quark_notify_queue;
sl@0
    45
  GObjectNotifyQueueDispatcher dispatcher;
sl@0
    46
  GTrashStack                 *_nqueue_trash; /* unused */
sl@0
    47
};
sl@0
    48
#endif
sl@0
    49
struct _GObjectNotifyQueue
sl@0
    50
{
sl@0
    51
  GObjectNotifyContext *context;
sl@0
    52
  GSList               *pspecs;
sl@0
    53
  guint16               n_pspecs;
sl@0
    54
  guint16               freeze_count;
sl@0
    55
  /* currently, this structure abuses the GList allocation chain and thus
sl@0
    56
   * must be <= sizeof (GList)
sl@0
    57
   */
sl@0
    58
};
sl@0
    59
sl@0
    60
sl@0
    61
/* --- functions --- */
sl@0
    62
static void
sl@0
    63
g_object_notify_queue_free (gpointer data)
sl@0
    64
{
sl@0
    65
  GObjectNotifyQueue *nqueue = data;
sl@0
    66
sl@0
    67
  g_slist_free (nqueue->pspecs);
sl@0
    68
  g_list_free_1 ((void*) nqueue);
sl@0
    69
}
sl@0
    70
sl@0
    71
static inline GObjectNotifyQueue*
sl@0
    72
g_object_notify_queue_freeze (GObject		   *object,
sl@0
    73
			      GObjectNotifyContext *context)
sl@0
    74
{
sl@0
    75
  GObjectNotifyQueue *nqueue;
sl@0
    76
sl@0
    77
  nqueue = g_datalist_id_get_data (&object->qdata, context->quark_notify_queue);
sl@0
    78
  if (!nqueue)
sl@0
    79
    {
sl@0
    80
      nqueue = (void*) g_list_alloc ();
sl@0
    81
      memset (nqueue, 0, sizeof (*nqueue));
sl@0
    82
      nqueue->context = context;
sl@0
    83
      g_datalist_id_set_data_full (&object->qdata, context->quark_notify_queue,
sl@0
    84
				   nqueue, g_object_notify_queue_free);
sl@0
    85
    }
sl@0
    86
sl@0
    87
  g_return_val_if_fail (nqueue->freeze_count < 65535, nqueue);
sl@0
    88
  nqueue->freeze_count++;
sl@0
    89
sl@0
    90
  return nqueue;
sl@0
    91
}
sl@0
    92
sl@0
    93
static inline void
sl@0
    94
g_object_notify_queue_thaw (GObject            *object,
sl@0
    95
			    GObjectNotifyQueue *nqueue)
sl@0
    96
{
sl@0
    97
  GObjectNotifyContext *context = nqueue->context;
sl@0
    98
  GParamSpec *pspecs_mem[16], **pspecs, **free_me = NULL;
sl@0
    99
  GSList *slist;
sl@0
   100
  guint n_pspecs = 0;
sl@0
   101
sl@0
   102
  g_return_if_fail (nqueue->freeze_count > 0);
sl@0
   103
sl@0
   104
  nqueue->freeze_count--;
sl@0
   105
  if (nqueue->freeze_count)
sl@0
   106
    return;
sl@0
   107
  g_return_if_fail (object->ref_count > 0);
sl@0
   108
sl@0
   109
  pspecs = nqueue->n_pspecs > 16 ? free_me = g_new (GParamSpec*, nqueue->n_pspecs) : pspecs_mem;
sl@0
   110
  /* set first entry to NULL since it's checked unconditionally */
sl@0
   111
  pspecs[0] = NULL;
sl@0
   112
  for (slist = nqueue->pspecs; slist; slist = slist->next)
sl@0
   113
    {
sl@0
   114
      GParamSpec *pspec = slist->data;
sl@0
   115
      guint i = 0;
sl@0
   116
sl@0
   117
      /* dedup, make pspecs in the list unique */
sl@0
   118
    redo_dedup_check:
sl@0
   119
      if (pspecs[i] == pspec)
sl@0
   120
	continue;
sl@0
   121
      if (++i < n_pspecs)
sl@0
   122
	goto redo_dedup_check;
sl@0
   123
sl@0
   124
      pspecs[n_pspecs++] = pspec;
sl@0
   125
    }
sl@0
   126
  g_datalist_id_set_data (&object->qdata, context->quark_notify_queue, NULL);
sl@0
   127
sl@0
   128
  if (n_pspecs)
sl@0
   129
    context->dispatcher (object, n_pspecs, pspecs);
sl@0
   130
  g_free (free_me);
sl@0
   131
}
sl@0
   132
sl@0
   133
static inline void
sl@0
   134
g_object_notify_queue_clear (GObject            *object,
sl@0
   135
			     GObjectNotifyQueue *nqueue)
sl@0
   136
{
sl@0
   137
  g_return_if_fail (nqueue->freeze_count > 0);
sl@0
   138
sl@0
   139
  g_slist_free (nqueue->pspecs);
sl@0
   140
  nqueue->pspecs = NULL;
sl@0
   141
  nqueue->n_pspecs = 0;
sl@0
   142
}
sl@0
   143
sl@0
   144
static inline void
sl@0
   145
g_object_notify_queue_add (GObject            *object,
sl@0
   146
			   GObjectNotifyQueue *nqueue,
sl@0
   147
			   GParamSpec	      *pspec)
sl@0
   148
{
sl@0
   149
  if (pspec->flags & G_PARAM_READABLE)
sl@0
   150
    {
sl@0
   151
      GParamSpec *redirect;
sl@0
   152
sl@0
   153
      g_return_if_fail (nqueue->n_pspecs < 65535);
sl@0
   154
sl@0
   155
      redirect = g_param_spec_get_redirect_target (pspec);
sl@0
   156
      if (redirect)
sl@0
   157
	pspec = redirect;
sl@0
   158
	    
sl@0
   159
      /* we do the deduping in _thaw */
sl@0
   160
      nqueue->pspecs = g_slist_prepend (nqueue->pspecs, pspec);
sl@0
   161
      nqueue->n_pspecs++;
sl@0
   162
    }
sl@0
   163
}
sl@0
   164
sl@0
   165
static inline GObjectNotifyQueue*
sl@0
   166
g_object_notify_queue_from_object (GObject              *object,
sl@0
   167
				   GObjectNotifyContext *context)
sl@0
   168
{
sl@0
   169
  return g_datalist_id_get_data (&object->qdata, context->quark_notify_queue);
sl@0
   170
}
sl@0
   171
sl@0
   172
sl@0
   173
G_END_DECLS
sl@0
   174
sl@0
   175
#endif /* __G_OBJECT_NOTIFY_QUEUE_H__ */