2 * @author Hallyson Melo <hallyson.melo@indt.org.br>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU Lesser General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
27 #include <libgnomevfs/gnome-vfs-module.h>
28 #include <libgnomevfs/gnome-vfs-utils.h>
30 #include <gmyth/gmyth_file_transfer.h>
31 #include <gmyth/gmyth_livetv.h>
32 #include <gmyth/gmyth_uri.h>
33 #include <gmyth/gmyth_recorder.h>
34 #include <gmyth/gmyth_backendinfo.h>
35 #include <gmyth/gmyth_util.h>
37 #define GST_MYTHTV_ID_NUM 1
38 #define MYTHTV_VERSION_DEFAULT 30
39 #define MYTHTV_TRANSFER_MAX_WAITS 100
41 #define MYTHTV_BUFFER_SIZE 1024*64
43 static GnomeVFSResult do_read (GnomeVFSMethod *method,
44 GnomeVFSMethodHandle *method_handle,
46 GnomeVFSFileSize num_bytes,
47 GnomeVFSFileSize *bytes_read,
48 GnomeVFSContext *context);
51 GMythFileTransfer *file_transfer;
64 do_open (GnomeVFSMethod *method,
65 GnomeVFSMethodHandle **method_handle,
67 GnomeVFSOpenMode mode,
68 GnomeVFSContext *context)
70 MythtvHandle *myth_handle;
71 GMythBackendInfo *backend_info;
72 GMythURI *gmyth_uri = NULL;
74 gboolean is_livetv = FALSE;
76 _GNOME_VFS_METHOD_PARAM_CHECK (method_handle != NULL);
77 _GNOME_VFS_METHOD_PARAM_CHECK (uri != NULL);
79 myth_handle = g_new0 (MythtvHandle, 1);
81 if (mode & GNOME_VFS_OPEN_WRITE) {
82 return GNOME_VFS_ERROR_NOT_PERMITTED;
85 if (gnome_vfs_uri_get_host_name (uri) == NULL) {
86 return GNOME_VFS_ERROR_INVALID_HOST_NAME;
89 /* Initialize mythtv handler*/
90 myth_handle->file_transfer = NULL;
91 myth_handle->livetv = NULL;
92 myth_handle->mythtv_version = MYTHTV_VERSION_DEFAULT;
93 myth_handle->bytes_read = 0;
94 myth_handle->content_size = 0;
96 /* Creates and fills out the backend info structure */
97 backend_info = gmyth_backend_info_new_with_uri (
98 gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
100 /* creates an instance of */
101 gmyth_uri = gmyth_uri_new_with_value(
102 gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
104 is_livetv = gmyth_uri_is_livetv( gmyth_uri );
106 /* Connect to the backend */
107 if ( gmyth_uri != NULL && is_livetv == TRUE ) {
108 myth_handle->livetv = gmyth_livetv_new ();
110 myth_handle->channel_num = gmyth_uri_get_channel_num( gmyth_uri );
112 if ( myth_handle->channel_num != -1 ) {
113 if (gmyth_livetv_channel_setup (myth_handle->livetv, myth_handle->channel_num,
114 backend_info) == FALSE) {
115 g_object_unref( gmyth_uri );
119 if ( gmyth_livetv_setup (myth_handle->livetv, backend_info) == FALSE ) {
120 g_object_unref( gmyth_uri );
125 myth_handle->file_transfer = gmyth_livetv_create_file_transfer (myth_handle->livetv);
127 if (NULL == myth_handle->file_transfer) {
133 myth_handle->file_transfer = gmyth_file_transfer_new (backend_info);
135 /* Verifies if the file exists */
136 if (!gmyth_util_file_exists (backend_info, gmyth_uri_get_path (gmyth_uri))) {
137 g_object_unref (backend_info);
138 return GNOME_VFS_ERROR_NOT_FOUND;
141 /* sets the Playback monitor connection */
142 ret = gmyth_file_transfer_open ( myth_handle->file_transfer, gmyth_uri_get_path (gmyth_uri) );
144 } /* if - LiveTV or not? */
147 g_warning ("MythTV FileTransfer open error\n");
148 return GNOME_VFS_ERROR_NOT_OPEN;
151 g_object_unref (backend_info);
153 //if ( gmyth_uri != NULL )
154 // g_object_unref( gmyth_uri );
156 g_return_val_if_fail (myth_handle->file_transfer != NULL, GNOME_VFS_ERROR_NOT_OPEN);
158 myth_handle->content_size = myth_handle->file_transfer->filesize;
160 myth_handle->buffer = g_byte_array_sized_new (MYTHTV_BUFFER_SIZE);
161 myth_handle->buffer_remain = 0;
163 *method_handle = (GnomeVFSMethodHandle *) myth_handle;
168 static GnomeVFSResult
169 do_read (GnomeVFSMethod *method,
170 GnomeVFSMethodHandle *method_handle,
172 GnomeVFSFileSize num_bytes,
173 GnomeVFSFileSize *bytes_read,
174 GnomeVFSContext *context)
176 MythtvHandle *myth_handle = (MythtvHandle *) method_handle;
177 GnomeVFSFileSize bytes_to_read;
181 if (myth_handle->bytes_read >= myth_handle->content_size)
182 return GNOME_VFS_ERROR_EOF;
184 // fixme: change this to min math function
185 if (num_bytes > myth_handle->content_size - myth_handle->bytes_read)
186 bytes_to_read = myth_handle->content_size - myth_handle->bytes_read;
188 bytes_to_read = num_bytes;
190 /* Loop sending the Myth File Transfer request:
191 * Retry whilst authentication fails and we supply it. */
192 //if (myth_handle->buffer_remain < MYTHTV_BUFFER_SIZE) {
193 if ( bytes_to_read > myth_handle->buffer_remain ) {
194 GByteArray *tmp_buffer = g_byte_array_new();
196 printf ("XXXXXXXXXXXXXX Pedindo %d %d\n", MYTHTV_BUFFER_SIZE, myth_handle->buffer_remain);
198 gint len = gmyth_file_transfer_read (myth_handle->file_transfer,
199 tmp_buffer, MYTHTV_BUFFER_SIZE - myth_handle->buffer_remain, TRUE);
202 g_byte_array_free (tmp_buffer, TRUE);
203 g_warning ("Fail to read bytes");
204 return GNOME_VFS_ERROR_IO;
205 } /*else if (len == 0) {
206 g_byte_array_free (tmp_buffer, TRUE);
207 g_warning ("End of file probably achieved");
208 return GNOME_VFS_ERROR_EOF;
211 myth_handle->buffer = g_byte_array_append (myth_handle->buffer,
212 tmp_buffer->data, len);
214 myth_handle->buffer_remain += len;
216 g_byte_array_free (tmp_buffer, TRUE);
220 bytes_to_read = (bytes_to_read > myth_handle->buffer_remain) ? myth_handle->buffer_remain : bytes_to_read;
221 /* gets the first buffer_size bytes from the byte array buffer variable */
223 g_memmove (buffer, myth_handle->buffer->data, bytes_to_read);
225 myth_handle->bytes_read += bytes_to_read;
226 myth_handle->buffer_remain -= bytes_to_read;
228 /* flushs the newly buffer got from byte array */
229 myth_handle->buffer = g_byte_array_remove_range (myth_handle->buffer, 0, bytes_to_read);
230 *bytes_read = bytes_to_read;
235 static GnomeVFSResult
236 do_close (GnomeVFSMethod *method,
237 GnomeVFSMethodHandle *method_handle,
238 GnomeVFSContext *context)
241 MythtvHandle *myth_handle = (MythtvHandle *) method_handle;
243 if (myth_handle->file_transfer) {
244 gmyth_file_transfer_close (myth_handle->file_transfer);
245 g_object_unref (myth_handle->file_transfer);
246 myth_handle->file_transfer = NULL;
249 if (myth_handle->livetv) {
250 g_object_unref (myth_handle->livetv);
251 myth_handle->livetv = NULL;
254 if (myth_handle->buffer) {
255 g_byte_array_free (myth_handle->buffer, TRUE);
256 myth_handle->buffer = NULL;
259 g_free (myth_handle);
264 static GnomeVFSResult
265 do_get_file_info (GnomeVFSMethod *method,
267 GnomeVFSFileInfo *file_info,
268 GnomeVFSFileInfoOptions options,
269 GnomeVFSContext *context)
271 GMythFileTransfer *file_transfer = NULL;
272 GMythLiveTV *livetv = NULL;
273 GMythBackendInfo *backend_info = NULL;
274 GMythURI *gmyth_uri = NULL;
275 gboolean is_livetv = FALSE;
278 /* Creates and fills out the backend info structure */
279 backend_info = gmyth_backend_info_new_with_uri (
280 gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
282 /* creates an instance of */
283 gmyth_uri = gmyth_uri_new_with_value(
284 gnome_vfs_unescape_string( gnome_vfs_uri_to_string( uri, GNOME_VFS_URI_HIDE_NONE ), "" ) );
286 is_livetv = gmyth_uri_is_livetv( gmyth_uri );
288 file_info->name = g_strdup (gmyth_uri_get_path (gmyth_uri)+1);
289 file_info->valid_fields = file_info->valid_fields
290 | GNOME_VFS_FILE_INFO_FIELDS_TYPE
291 | GNOME_VFS_FILE_INFO_FIELDS_MIME_TYPE
292 | GNOME_VFS_FILE_INFO_FIELDS_PERMISSIONS;
293 file_info->type = GNOME_VFS_FILE_TYPE_REGULAR;
294 /* fixme: get from file extension? */
295 file_info->mime_type = g_strdup ("video/x-nuv");
296 file_info->permissions =
297 GNOME_VFS_PERM_USER_READ |
298 GNOME_VFS_PERM_OTHER_READ |
299 GNOME_VFS_PERM_GROUP_READ;
301 /* Connect to the backend */
302 if ( gmyth_uri != NULL && is_livetv == TRUE ) {
303 livetv = gmyth_livetv_new ();
305 gint channel_num = gmyth_uri_get_channel_num( gmyth_uri );
307 if ( channel_num != -1 ) {
308 if (gmyth_livetv_channel_setup (livetv, channel_num,
309 backend_info) == FALSE) {
310 g_object_unref( gmyth_uri );
314 if ( gmyth_livetv_setup (livetv, backend_info) == FALSE ) {
315 g_object_unref( gmyth_uri );
320 file_transfer = gmyth_livetv_create_file_transfer (livetv);
322 if (NULL == file_transfer) {
328 file_transfer = gmyth_file_transfer_new (backend_info);
330 /* Verifies if the file exists */
331 if (!gmyth_util_file_exists (backend_info, gmyth_uri_get_path (gmyth_uri))) {
332 g_object_unref (backend_info);
333 return GNOME_VFS_ERROR_NOT_FOUND;
336 /* sets the Playback monitor connection */
337 ret = gmyth_file_transfer_open ( file_transfer, gmyth_uri_get_path (gmyth_uri) );
339 } /* if - LiveTV or not? */
343 g_warning ("MythTV FileTransfer open error\n");
344 return GNOME_VFS_ERROR_NOT_OPEN;
348 file_info->size = gmyth_file_transfer_get_filesize (file_transfer);
349 file_info->block_count = GNOME_VFS_FILE_INFO_FIELDS_BLOCK_COUNT;
350 file_info->io_block_size = GNOME_VFS_FILE_INFO_FIELDS_IO_BLOCK_SIZE;
352 g_object_unref (file_transfer);
353 g_object_unref (livetv);
354 g_object_unref (backend_info);
360 do_is_local (GnomeVFSMethod *method,
361 const GnomeVFSURI *uri)
366 static GnomeVFSMethod method = {
367 sizeof (GnomeVFSMethod),
398 vfs_module_init (const char *method_name, const char *args)
404 vfs_module_shutdown (GnomeVFSMethod *method)