]> git.sesse.net Git - vlc/blob - src/misc/events.c
misc/events.c: Fix previous commit.
[vlc] / src / misc / events.c
1 /*****************************************************************************
2  * events.c: events interface
3  * This library provides an interface to the send and receive events.
4  * It is more lightweight than variable based callback.
5  * Methode
6  *****************************************************************************
7  * Copyright (C) 1998-2005 the VideoLAN team
8  * $Id$
9  *
10  * Authors: Pierre d'Herbemont <pdherbemont # videolan.org >
11  *
12  * This program is free software; you can redistribute it and/or modify
13  * it under the terms of the GNU General Public License as published by
14  * the Free Software Foundation; either version 2 of the License, or
15  * (at your option) any later version.
16  *
17  * This program is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20  * GNU General Public License for more details.
21  *
22  * You should have received a copy of the GNU General Public License
23  * along with this program; if not, write to the Free Software
24  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
25  *****************************************************************************/
26
27 /*****************************************************************************
28  * Preamble
29  *****************************************************************************/
30
31 #include <vlc/vlc.h>
32
33 #include <assert.h>
34
35 #include <vlc_events.h>
36 #include <vlc_arrays.h>
37
38 /*****************************************************************************
39  * Documentation : Read vlc_events.h
40  *****************************************************************************/
41
42 //#define DEBUG_EVENT
43
44 /*****************************************************************************
45  *  Private types.
46  *****************************************************************************/
47
48 typedef struct vlc_event_listener_t
49 {
50     void *               p_user_data;
51     vlc_event_callback_t pf_callback;
52 #ifdef DEBUG_EVENT
53     char *               psz_debug_name;
54 #endif
55 } vlc_event_listener_t;
56
57 typedef struct vlc_event_listeners_group_t
58 {
59     vlc_event_type_t    event_type;
60     DECL_ARRAY(struct vlc_event_listener_t *) listeners;
61 } vlc_event_listeners_group_t;
62
63 #ifdef DEBUG_EVENT
64 static const char * ppsz_event_type_to_name[] = 
65 {
66     [vlc_InputItemMetaChanged]          = "vlc_InputItemMetaChanged",
67     [vlc_InputItemSubItemAdded]         = "vlc_InputItemSubItemAdded",
68     [vlc_ServicesDiscoveryItemAdded]    = "vlc_ServicesDiscoveryItemAdded",
69     [vlc_ServicesDiscoveryItemRemoved]  = "vlc_ServicesDiscoveryItemRemoved"
70 };
71 #endif
72
73 /*****************************************************************************
74  * 
75  *****************************************************************************/
76
77 /**
78  * Initialize event manager object
79  * p_obj is the object that contains the event manager. But not
80  * necessarily a vlc_object_t (an input_item_t is not a vlc_object_t
81  * for instance).
82  * p_parent_obj gives a libvlc instance
83  */
84 int vlc_event_manager_init( vlc_event_manager_t * p_em, void * p_obj,
85                             vlc_object_t * p_parent_obj )
86 {
87     p_em->p_obj = p_obj;
88     p_em->p_parent_object = p_parent_obj;
89     vlc_mutex_init( p_parent_obj, &p_em->object_lock );
90     ARRAY_INIT( p_em->listeners_groups );
91     return VLC_SUCCESS;
92 }
93
94 /**
95  * Destroy the event manager
96  */
97 void vlc_event_manager_fini( vlc_event_manager_t * p_em )
98 {
99     struct vlc_event_listeners_group_t * listeners_group;
100     struct vlc_event_listener_t * listener;
101
102     vlc_mutex_destroy( &p_em->object_lock );
103
104     FOREACH_ARRAY( listeners_group, p_em->listeners_groups )
105         FOREACH_ARRAY( listener, listeners_group->listeners )
106             free( listener );
107         FOREACH_END()
108         free( listeners_group );
109     FOREACH_END()
110 }
111
112 /**
113  * Destroy the event manager
114  */
115 int vlc_event_manager_register_event_type(
116         vlc_event_manager_t * p_em,
117         vlc_event_type_t event_type )
118 {
119     vlc_event_listeners_group_t * listeners_group;
120     listeners_group = malloc(sizeof(vlc_event_listeners_group_t));
121
122     if( !listeners_group )
123         return VLC_ENOMEM;
124
125     listeners_group->event_type = event_type;
126     ARRAY_INIT( listeners_group->listeners );
127     
128     vlc_mutex_lock( &p_em->object_lock );
129     ARRAY_APPEND( p_em->listeners_groups, listeners_group );
130     vlc_mutex_unlock( &p_em->object_lock );
131
132     return VLC_SUCCESS;
133 }
134
135 /**
136  * Send an event to the listener attached to this p_em.
137  */
138 void vlc_event_send( vlc_event_manager_t * p_em,
139                      vlc_event_t * p_event )
140 {
141     vlc_event_listeners_group_t * listeners_group;
142     vlc_event_listener_t * listener;
143     vlc_event_listener_t * array_of_cached_listeners = NULL;
144     vlc_event_listener_t * cached_listener;
145     int i, i_cached_listeners = 0;
146
147     /* Fill event with the sending object now */
148     p_event->p_obj = p_em->p_obj;
149
150     vlc_mutex_lock( &p_em->object_lock );
151     FOREACH_ARRAY( listeners_group, p_em->listeners_groups )
152         if( listeners_group->event_type == p_event->type )
153         {
154             if( listeners_group->listeners.i_size <= 0 )
155                 break;
156
157             /* Save the function to call */
158             i_cached_listeners = listeners_group->listeners.i_size;
159             array_of_cached_listeners = malloc(
160                     sizeof(vlc_event_listener_t)*i_cached_listeners );
161             if( !array_of_cached_listeners )
162             {
163                 msg_Err( p_em->p_parent_object, "Not enough memory in vlc_event_send" );
164                 return;
165             }
166
167             cached_listener = array_of_cached_listeners;
168             FOREACH_ARRAY( listener, listeners_group->listeners )
169                 memcpy( cached_listener, listener, sizeof(vlc_event_listener_t));
170 #ifdef DEBUG_EVENT
171                 cached_listener->psz_debug_name = strdup(cached_listener->psz_debug_name);
172 #endif
173                 cached_listener += sizeof(vlc_event_listener_t);
174             FOREACH_END()
175
176             break;
177         }
178     FOREACH_END()
179     vlc_mutex_unlock( &p_em->object_lock );
180     
181     /* Call the function attached */
182     cached_listener = array_of_cached_listeners;
183     for( i = 0; i < i_cached_listeners; i++ )
184     {
185 #ifdef DEBUG_EVENT
186         msg_Dbg( p_em->p_parent_object,
187                     "Calling '%s' with a '%s' event (data %p)",
188                     cached_listener->psz_debug_name,
189                     ppsz_event_type_to_name[p_event->type],
190                     cached_listener->p_user_data );
191         free(cached_listener->psz_debug_name);
192 #endif
193
194         cached_listener->pf_callback( p_event, cached_listener->p_user_data );
195         cached_listener += sizeof(vlc_event_listener_t) ;
196     }
197     free( array_of_cached_listeners );
198
199 }
200
201 /**
202  * Add a callback for an event.
203  */
204 int __vlc_event_attach( vlc_event_manager_t * p_em,
205                         vlc_event_type_t event_type,
206                         vlc_event_callback_t pf_callback,
207                         void *p_user_data,
208                         const char * psz_debug_name )
209 {
210     vlc_event_listeners_group_t * listeners_group;
211     vlc_event_listener_t * listener;
212     listener = malloc(sizeof(vlc_event_listener_t));
213     if( !listener )
214         return VLC_ENOMEM;
215     
216     listener->p_user_data = p_user_data;
217     listener->pf_callback = pf_callback;
218 #ifdef DEBUG_EVENT
219     listener->psz_debug_name = strdup( psz_debug_name );
220 #else
221     (void)psz_debug_name;
222 #endif
223
224     vlc_mutex_lock( &p_em->object_lock );
225     FOREACH_ARRAY( listeners_group, p_em->listeners_groups )
226         if( listeners_group->event_type == event_type )
227         {
228             ARRAY_APPEND( listeners_group->listeners, listener );
229 #ifdef DEBUG_EVENT
230                 msg_Dbg( p_em->p_parent_object,
231                     "Listening to '%s' event with '%s' (data %p)",
232                     ppsz_event_type_to_name[event_type],
233                     listener->psz_debug_name,
234                     listener->p_user_data );
235 #endif
236             vlc_mutex_unlock( &p_em->object_lock );
237             return VLC_SUCCESS;
238         }
239     FOREACH_END()
240     vlc_mutex_unlock( &p_em->object_lock );
241
242     msg_Err( p_em->p_parent_object, "Can't attach to an object event manager event" );
243     free(listener);
244     return VLC_EGENERIC;
245 }
246
247 /**
248  * Remove a callback for an event.
249  */
250 int vlc_event_detach( vlc_event_manager_t *p_em,
251                       vlc_event_type_t event_type,
252                       vlc_event_callback_t pf_callback,
253                       void *p_user_data )
254 {
255     vlc_event_listeners_group_t * listeners_group;
256     struct vlc_event_listener_t * listener;
257
258     vlc_mutex_lock( &p_em->object_lock );
259     FOREACH_ARRAY( listeners_group, p_em->listeners_groups )
260         if( listeners_group->event_type == event_type )
261         {
262             FOREACH_ARRAY( listener, listeners_group->listeners )
263                 if( listener->pf_callback == pf_callback &&
264                     listener->p_user_data == p_user_data )
265                 {
266                     /* that's our listener */
267                     ARRAY_REMOVE( listeners_group->listeners,
268                         fe_idx /* This comes from the macro (and that's why
269                                   I hate macro) */ );
270 #ifdef DEBUG_EVENT
271                     msg_Dbg( p_em->p_parent_object,
272                         "Detaching '%s' from '%s' event (data %p)",
273                         listener->psz_debug_name,
274                         ppsz_event_type_to_name[event_type],
275                         listener->p_user_data );
276
277                     free( listener->psz_debug_name );
278 #endif
279                     free( listener );
280                     vlc_mutex_unlock( &p_em->object_lock );
281                     return VLC_SUCCESS;
282                 }
283             FOREACH_END()
284         }
285     FOREACH_END()
286     vlc_mutex_unlock( &p_em->object_lock );
287
288     msg_Warn( p_em->p_parent_object, "Can't detach to an object event manager event" );
289
290     return VLC_EGENERIC;
291 }