]> git.sesse.net Git - vlc/blob - modules/access/eyetv.m
Use var_Inherit* when applicable.
[vlc] / modules / access / eyetv.m
1 /*****************************************************************************
2  * eyetv.c : Access module to connect to our plugin running within EyeTV
3  *****************************************************************************
4  * Copyright (C) 2006-2007 the VideoLAN team
5  * $Id$
6  *
7  * Author: Felix Kühne <fkuehne at videolan dot org>
8  *
9  * This program is free software; you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation; either version 2 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27
28 #ifdef HAVE_CONFIG_H
29 # include "config.h"
30 #endif
31
32 #include <vlc_common.h>
33 #include <vlc_plugin.h>
34 #include <vlc_access.h>
35
36 #include <vlc_network.h>
37
38 #include <sys/types.h>
39 #include <sys/socket.h>
40 #include <sys/un.h>
41 #include <unistd.h>
42
43 #import <Foundation/Foundation.h>
44
45 #define MTU 65535
46
47 /* TODO:
48  * watch for PluginQuit or DeviceRemoved to stop output to VLC's core then */
49
50 /*****************************************************************************
51  * Module descriptior
52  *****************************************************************************/
53 static int  Open ( vlc_object_t * );
54 static void Close( vlc_object_t * );
55
56 #define CHANNEL_TEXT N_("Channel number")
57 #define CHANNEL_LONGTEXT N_( \
58     "EyeTV program number, or use 0 for last channel, " \
59     "-1 for S-Video input, -2 for Composite input" )
60
61 #define CACHING_TEXT N_("Caching value in ms")
62 #define CACHING_LONGTEXT N_( \
63     "Caching value for EyeTV captures. This " \
64     "value should be set in milliseconds." )
65
66 vlc_module_begin ()
67     set_shortname( "EyeTV" )
68     set_description( N_("EyeTV input") )
69     set_category( CAT_INPUT )
70     set_subcategory( SUBCAT_INPUT_ACCESS )
71
72     add_integer( "eyetv-channel", 0, NULL,
73                  CHANNEL_TEXT, CHANNEL_LONGTEXT, false )
74
75     set_capability( "access", 0 )
76     add_shortcut( "eyetv" )
77     set_callbacks( Open, Close )
78     add_integer( "eyetv-caching", DEFAULT_PTS_DELAY / 1000, NULL,
79                  CACHING_TEXT, CACHING_LONGTEXT, true);
80 vlc_module_end ()
81
82 /*****************************************************************************
83  * Access: local prototypes
84  *****************************************************************************/
85 struct access_sys_t
86 {
87     int eyetvSock;
88     int i_pts_delay;
89 };
90
91 static block_t *BlockRead( access_t *);
92 static int Control( access_t *, int, va_list );
93
94 static void selectChannel( vlc_object_t *p_this, int theChannelNum )
95 {
96     NSAppleScript *script;
97     NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
98     switch( theChannelNum )
99     {
100         case -2: // Composite
101             script = [[NSAppleScript alloc] initWithSource:
102                         @"tell application \"EyeTV\"\n"
103                          "  input_change input source composite video input\n"
104                          "  volume_change level 0\n"
105                          "  show player_window\n"
106                          "  tell application \"System Events\" to set visible of process \"EyeTV\" to false\n"
107                          "end tell"];
108             break;
109         case -1: // S-Video
110             script = [[NSAppleScript alloc] initWithSource:
111                         @"tell application \"EyeTV\"\n"
112                          "  input_change input source S video input\n"
113                          "  volume_change level 0\n"
114                          "  show player_window\n"
115                          "  tell application \"System Events\" to set visible of process \"EyeTV\" to false\n"
116                          "end tell"];
117             break;
118         case 0: // Last
119             script = [[NSAppleScript alloc] initWithSource:
120                         @"tell application \"EyeTV\"\n"
121                          "  volume_change level 0\n"
122                          "  show player_window\n"
123                          "  tell application \"System Events\" to set visible of process \"EyeTV\" to false\n"
124                          "end tell"];
125             break;
126         default:
127             if( theChannelNum > 0 )
128             {
129                 NSString *channel_change = [NSString stringWithFormat:
130                     @"tell application \"EyeTV\"\n"
131                      "  channel_change channel number %d\n"
132                      "  volume_change level 0\n"
133                      "  show player_window\n"
134                      "  tell application \"System Events\" to set visible of process \"EyeTV\" to false\n"
135                      "end tell", theChannelNum];
136                 script = [[NSAppleScript alloc] initWithSource:channel_change];
137             }
138             else
139                 return;
140     }
141     NSDictionary *errorDict;
142     NSAppleEventDescriptor *descriptor = [script executeAndReturnError:&errorDict];
143     if( nil == descriptor ) 
144     {
145         NSString *errorString = [errorDict objectForKey:NSAppleScriptErrorMessage];
146         msg_Err( p_this, "EyeTV source change failed with error status '%s'", [errorString UTF8String] );
147     }
148     [script release];
149     [pool release];
150 }
151
152 /*****************************************************************************
153  * Open: sets up the module and its threads
154  *****************************************************************************/
155 static int Open( vlc_object_t *p_this )
156 {
157     access_t        *p_access = (access_t *)p_this;
158     access_sys_t    *p_sys;
159
160     struct sockaddr_un publicAddr, peerAddr;
161     int publicSock;
162  
163     /* Init p_access */
164     access_InitFields( p_access );
165     ACCESS_SET_CALLBACKS( NULL, BlockRead, Control, NULL );
166     p_sys = p_access->p_sys = calloc( 1, sizeof( access_sys_t ) );
167     if( !p_sys )
168         return VLC_ENOMEM;
169
170     p_sys->i_pts_delay = var_InheritInteger( p_access, "eyetv-caching" );
171
172     msg_Dbg( p_access, "coming up" );
173     selectChannel( p_this, var_InheritInteger( p_access, "eyetv-channel" ) );
174
175     /* socket */
176     memset(&publicAddr, 0, sizeof(publicAddr));
177     publicAddr.sun_family = AF_UNIX;
178     strncpy(publicAddr.sun_path, "/tmp/.vlc-eyetv-bridge", sizeof(publicAddr.sun_path)-1);
179     /* remove previous public path if it wasn't cleanly removed */
180     if( (0 != unlink(publicAddr.sun_path)) && (ENOENT != errno) )
181     {
182         msg_Err( p_access, "local socket path is not usable (errno=%d)", errno );
183         free( p_sys );
184         return VLC_EGENERIC;
185     }
186
187     publicSock = socket(PF_UNIX, SOCK_STREAM, 0);
188     if( publicSock == -1 )
189     {
190         msg_Err( p_access, "create local socket failed (errno=%d)", errno );
191         free( p_sys );
192         return VLC_EGENERIC;
193     }
194
195     if( bind(publicSock, (struct sockaddr *)&publicAddr, sizeof(struct sockaddr_un)) == -1 )
196     {
197         msg_Err( p_access, "bind local socket failed (errno=%d)", errno );
198         close( publicSock );
199         free( p_sys );
200         return VLC_EGENERIC;
201     }
202
203     /* we are not expecting more than one connection */
204     if( listen(publicSock, 1) == -1 )
205     {
206         msg_Err( p_access, "cannot accept connection (errno=%d)", errno );
207         close( publicSock );
208         free( p_sys );
209         return VLC_EGENERIC;
210     }
211     else
212     {
213         socklen_t peerSockLen = sizeof(struct sockaddr_un);
214         int peerSock;
215
216         /* tell the EyeTV plugin to open start sending */
217         CFNotificationCenterPostNotification( CFNotificationCenterGetDistributedCenter (),
218                                               CFSTR("VLCAccessStartDataSending"),
219                                               CFSTR("VLCEyeTVSupport"),
220                                               /*userInfo*/ NULL,
221                                               TRUE );
222
223         msg_Dbg( p_access, "plugin notified" );
224
225         peerSock = accept(publicSock, (struct sockaddr *)&peerAddr, &peerSockLen);
226         if( peerSock == -1 )
227         {
228             msg_Err( p_access, "cannot wait for connection (errno=%d)", errno );
229             close( publicSock );
230             free( p_sys );
231             return VLC_EGENERIC;
232         }
233
234         msg_Dbg( p_access, "plugin connected" );
235
236         p_sys->eyetvSock = peerSock;
237
238         /* remove public access */
239         close(publicSock);
240         unlink(publicAddr.sun_path);
241     }
242     return VLC_SUCCESS;
243 }
244
245 /*****************************************************************************
246  * Close: closes msg-port, free resources
247  *****************************************************************************/
248 static void Close( vlc_object_t *p_this )
249 {
250     access_t     *p_access = (access_t *)p_this;
251     access_sys_t *p_sys = p_access->p_sys;
252  
253     msg_Dbg( p_access, "closing" );
254  
255     /* tell the EyeTV plugin to close its msg port and stop sending */
256     CFNotificationCenterPostNotification( CFNotificationCenterGetDistributedCenter (),
257                                           CFSTR("VLCAccessStopDataSending"),
258                                           CFSTR("VLCEyeTVSupport"),
259                                           /*userInfo*/ NULL,
260                                           TRUE );
261  
262     msg_Dbg( p_access, "plugin notified" );
263
264         close(p_sys->eyetvSock);
265  
266     msg_Dbg( p_access, "msg port closed and freed" );
267  
268     free( p_sys );
269 }
270
271 /*****************************************************************************
272 * BlockRead: forwarding data from EyeTV plugin which was received above
273 *****************************************************************************/
274 static block_t *BlockRead( access_t *p_access )
275 {
276     access_sys_t *p_sys = p_access->p_sys;
277     block_t      *p_block;
278     ssize_t len;
279
280     if( p_access->info.b_eof )
281         return NULL;
282
283     /* Read data */
284     p_block = block_New( p_access, MTU );
285     len = net_Read( p_access, p_sys->eyetvSock, NULL,
286                     p_block->p_buffer, MTU, false );
287
288     if( len < 0 )
289     {
290         block_Release( p_block );
291         return NULL;
292     }
293
294     return block_Realloc( p_block, 0, p_block->i_buffer = len );
295 }
296
297 /*****************************************************************************
298  * Control:
299  *****************************************************************************/
300 static int Control( access_t *p_access, int i_query, va_list args )
301 {
302     bool   *pb_bool;
303     int          *pi_int;
304     int64_t      *pi_64;
305     access_sys_t  *p_sys = (access_sys_t *) p_access->p_sys;
306  
307     switch( i_query )
308     {
309         case ACCESS_CAN_SEEK:
310         case ACCESS_CAN_FASTSEEK:
311             pb_bool = (bool*)va_arg( args, bool* );
312             *pb_bool = false;
313             break;
314         case ACCESS_CAN_PAUSE:
315             pb_bool = (bool*)va_arg( args, bool* );
316             *pb_bool = false;
317             break;
318         case ACCESS_CAN_CONTROL_PACE:
319             pb_bool = (bool*)va_arg( args, bool* );
320             *pb_bool = false;
321             break;
322
323         /* */
324         case ACCESS_GET_PTS_DELAY:
325             pi_64 = (int64_t*)va_arg( args, int64_t * );
326             *pi_64 = (int64_t) p_sys->i_pts_delay * 1000;
327             break;
328         
329         case ACCESS_SET_PAUSE_STATE:
330         case ACCESS_GET_TITLE_INFO:
331         case ACCESS_SET_TITLE:
332         case ACCESS_SET_SEEKPOINT:
333         case ACCESS_SET_PRIVATE_ID_STATE:
334         case ACCESS_GET_CONTENT_TYPE:
335             return VLC_EGENERIC;
336  
337         default:
338             msg_Warn( p_access, "unimplemented query in control" );
339             return VLC_EGENERIC;
340  
341     }
342     return VLC_SUCCESS;
343 }