directory.c 18.2 KB
Newer Older
1 2 3
/*****************************************************************************
 * directory.c: expands a directory (directory: access plug-in)
 *****************************************************************************
4
 * Copyright (C) 2002-2007 the VideoLAN team
Carlo Calabrò's avatar
Carlo Calabrò committed
5
 * $Id$
6
 *
7
 * Authors: Derk-Jan Hartman <hartman at videolan dot org>
8
 *          Rémi Denis-Courmont
9 10 11 12 13
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
Clément Stenac's avatar
Clément Stenac committed
14
 *
15 16 17 18 19 20 21
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
Antoine Cellerier's avatar
Antoine Cellerier committed
22
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23 24 25 26 27
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
28

29 30 31 32
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif

33
#include <vlc_common.h>
34
#include <vlc_plugin.h>
35
#include <vlc_playlist.h>
36
#include <vlc_input.h>
Clément Stenac's avatar
Clément Stenac committed
37 38
#include <vlc_access.h>
#include <vlc_demux.h>
39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56

#ifdef HAVE_SYS_TYPES_H
#   include <sys/types.h>
#endif
#ifdef HAVE_SYS_STAT_H
#   include <sys/stat.h>
#endif
#ifdef HAVE_ERRNO_H
#   include <errno.h>
#endif
#ifdef HAVE_FCNTL_H
#   include <fcntl.h>
#endif

#ifdef HAVE_UNISTD_H
#   include <unistd.h>
#elif defined( WIN32 ) && !defined( UNDER_CE )
#   include <io.h>
57 58
#elif defined( UNDER_CE )
#   define strcoll strcmp
59 60
#endif

61
#ifdef HAVE_DIRENT_H
62 63 64
#   include <dirent.h>
#endif

Clément Stenac's avatar
Clément Stenac committed
65
#include <vlc_charset.h>
66

67
/*****************************************************************************
68
 * Module descriptor
69
 *****************************************************************************/
70 71
static int  Open ( vlc_object_t * );
static void Close( vlc_object_t * );
72

73
static int  DemuxOpen ( vlc_object_t * );
74

75
#define RECURSIVE_TEXT N_("Subdirectory behavior")
Clément Stenac's avatar
Clément Stenac committed
76 77
#define RECURSIVE_LONGTEXT N_( \
        "Select whether subdirectories must be expanded.\n" \
Carlo Calabrò's avatar
Carlo Calabrò committed
78 79 80
        "none: subdirectories do not appear in the playlist.\n" \
        "collapse: subdirectories appear but are expanded on first play.\n" \
        "expand: all subdirectories are expanded.\n" )
Clément Stenac's avatar
Clément Stenac committed
81

82 83 84
static const char *const psz_recursive_list[] = { "none", "collapse", "expand" };
static const char *const psz_recursive_list_text[] = {
    N_("none"), N_("collapse"), N_("expand") };
85

Clément Stenac's avatar
Clément Stenac committed
86
#define IGNORE_TEXT N_("Ignored extensions")
87
#define IGNORE_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
88 89 90 91
        "Files with these extensions will not be added to playlist when " \
        "opening a directory.\n" \
        "This is useful if you add directories that contain playlist files " \
        "for instance. Use a comma-separated list of extensions." )
92

93
vlc_module_begin();
Clément Stenac's avatar
Clément Stenac committed
94
    set_category( CAT_INPUT );
95
    set_shortname( N_("Directory" ) );
Clément Stenac's avatar
Clément Stenac committed
96
    set_subcategory( SUBCAT_INPUT_ACCESS );
97
    set_description( N_("Standard filesystem directory input") );
98
    set_capability( "access", 55 );
99 100
    add_shortcut( "directory" );
    add_shortcut( "dir" );
101
    add_shortcut( "file" );
Clément Stenac's avatar
Clément Stenac committed
102
    add_string( "recursive", "expand" , NULL, RECURSIVE_TEXT,
103
                RECURSIVE_LONGTEXT, false );
104
      change_string_list( psz_recursive_list, psz_recursive_list_text, 0 );
105
    add_string( "ignore-filetypes", "m3u,db,nfo,jpg,gif,sfv,txt,sub,idx,srt,cue",
106
                NULL, IGNORE_TEXT, IGNORE_LONGTEXT, false );
107
    set_callbacks( Open, Close );
108 109

    add_submodule();
110
        set_description( "Directory EOF");
111
        set_capability( "demux", 0 );
112
        set_callbacks( DemuxOpen, NULL );
113 114 115
vlc_module_end();


116 117 118 119
/*****************************************************************************
 * Local prototypes, constants, structures
 *****************************************************************************/

120 121 122 123 124 125 126 127
enum
{
    MODE_EXPAND,
    MODE_COLLAPSE,
    MODE_NONE
};

typedef struct stat_list_t stat_list_t;
128

129 130
static ssize_t Read( access_t *, uint8_t *, size_t );
static ssize_t ReadNull( access_t *, uint8_t *, size_t );
131 132 133 134 135 136
static int Control( access_t *, int, va_list );

static int Demux( demux_t *p_demux );
static int DemuxControl( demux_t *p_demux, int i_query, va_list args );


137
static int ReadDir( playlist_t *, const char *psz_name, int i_mode,
138 139 140 141
                    playlist_item_t *, playlist_item_t *, input_item_t *,
                    DIR *handle, stat_list_t *stats );

static DIR *OpenDir (vlc_object_t *obj, const char *psz_name);
142

143 144 145 146 147
/*****************************************************************************
 * Open: open the directory
 *****************************************************************************/
static int Open( vlc_object_t *p_this )
{
148
    access_t *p_access = (access_t*)p_this;
149

150 151 152 153 154 155 156
    if( !p_access->psz_path )
        return VLC_EGENERIC;

    struct stat st;
    if( !stat( p_access->psz_path, &st ) && !S_ISDIR( st.st_mode ) )
        return VLC_EGENERIC;

157 158
    DIR *handle = OpenDir (p_this, p_access->psz_path);
    if (handle == NULL)
Clément Stenac's avatar
Clément Stenac committed
159
        return VLC_EGENERIC;
Gildas Bazin's avatar
 
Gildas Bazin committed
160

161 162
    p_access->p_sys = (access_sys_t *)handle;

163 164 165 166 167
    p_access->pf_read  = Read;
    p_access->pf_block = NULL;
    p_access->pf_seek  = NULL;
    p_access->pf_control= Control;

168
    /* Force a demux */
Rafaël Carré's avatar
Rafaël Carré committed
169
    free( p_access->psz_demux );
170
    p_access->psz_demux = strdup( "directory" );
Gildas Bazin's avatar
 
Gildas Bazin committed
171

172 173 174 175 176 177 178 179
    return VLC_SUCCESS;
}

/*****************************************************************************
 * Close: close the target
 *****************************************************************************/
static void Close( vlc_object_t * p_this )
{
180 181 182
    access_t *p_access = (access_t*)p_this;
    DIR *handle = (DIR *)p_access->p_sys;
    closedir (handle);
183 184 185 186 187
}

/*****************************************************************************
 * ReadNull: read the directory
 *****************************************************************************/
188
static ssize_t ReadNull( access_t *p_access, uint8_t *p_buffer, size_t i_len)
189 190 191 192
{
    /* Return fake data */
    memset( p_buffer, 0, i_len );
    return i_len;
193 194 195
}

/*****************************************************************************
196
 * Read: read the directory
197
 *****************************************************************************/
198
static ssize_t Read( access_t *p_access, uint8_t *p_buffer, size_t i_len)
199
{
200 201
    char               *psz;
    int                 i_mode, i_activity;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
202 203 204 205 206
    char               *psz_name = strdup (p_access->psz_path);

    if( psz_name == NULL )
        return VLC_ENOMEM;

207
    playlist_t         *p_playlist = pl_Yield( p_access );
208 209
    input_thread_t     *p_input = (input_thread_t*)vlc_object_find( p_access, VLC_OBJECT_INPUT, FIND_PARENT );

210
    playlist_item_t    *p_item_in_category;
211 212 213 214 215 216 217 218 219 220 221
    input_item_t       *p_current_input;
    playlist_item_t    *p_current;

    if( !p_input )
    {
        msg_Err( p_access, "unable to find input (internal error)" );
        vlc_object_release( p_playlist );
        return VLC_ENOOBJ;
    }

    p_current_input = input_GetItem( p_input );
222
    p_current = playlist_ItemGetByInput( p_playlist, p_current_input, false );
223

224
    if( !p_current )
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
225
    {
226
        msg_Err( p_access, "unable to find item in playlist" );
227
        vlc_object_release( p_input );
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
228
        vlc_object_release( p_playlist );
229 230 231
        return VLC_ENOOBJ;
    }

232
    /* Remove the ending '/' char */
233
    if( psz_name[0] )
234
    {
235 236
        char *ptr = psz_name + strlen (psz_name);
        switch (*--ptr)
237
        {
238 239 240
            case '/':
            case '\\':
                *ptr = '\0';
241
        }
242
    }
243

244
    /* Handle mode */
245 246
    psz = var_CreateGetString( p_access, "recursive" );
    if( *psz == '\0' || !strncmp( psz, "none" , 4 )  )
247
        i_mode = MODE_NONE;
248
    else if( !strncmp( psz, "collapse", 8 )  )
249 250 251
        i_mode = MODE_COLLAPSE;
    else
        i_mode = MODE_EXPAND;
252
    free( psz );
253

254
    p_current->p_input->i_type = ITEM_TYPE_DIRECTORY;
255
    p_item_in_category = playlist_ItemToNode( p_playlist, p_current,
256
                                              false );
257 258 259 260 261

    i_activity = var_GetInteger( p_playlist, "activity" );
    var_SetInteger( p_playlist, "activity", i_activity +
                    DIRECTORY_ACTIVITY );

262
    ReadDir( p_playlist, psz_name, i_mode, p_current, p_item_in_category,
263
             p_current_input, (DIR *)p_access->p_sys, NULL );
264 265 266 267

    i_activity = var_GetInteger( p_playlist, "activity" );
    var_SetInteger( p_playlist, "activity", i_activity -
                    DIRECTORY_ACTIVITY );
268

269 270
    playlist_Signal( p_playlist );

271
    free( psz_name );
272
    vlc_object_release( p_input );
273
    vlc_object_release( p_playlist );
274 275 276

    /* Return fake data forever */
    p_access->pf_read = ReadNull;
277
    return -1;
278 279 280
}

/*****************************************************************************
281
 * Control:
282 283 284
 *****************************************************************************/
static int Control( access_t *p_access, int i_query, va_list args )
{
285
    bool   *pb_bool;
286 287 288 289 290 291 292 293 294 295
    int          *pi_int;
    int64_t      *pi_64;

    switch( i_query )
    {
        /* */
        case ACCESS_CAN_SEEK:
        case ACCESS_CAN_FASTSEEK:
        case ACCESS_CAN_PAUSE:
        case ACCESS_CAN_CONTROL_PACE:
296 297
            pb_bool = (bool*)va_arg( args, bool* );
            *pb_bool = false;    /* FIXME */
298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315
            break;

        /* */
        case ACCESS_GET_MTU:
            pi_int = (int*)va_arg( args, int * );
            *pi_int = 0;
            break;

        case ACCESS_GET_PTS_DELAY:
            pi_64 = (int64_t*)va_arg( args, int64_t * );
            *pi_64 = DEFAULT_PTS_DELAY * 1000;
            break;

        /* */
        case ACCESS_SET_PAUSE_STATE:
        case ACCESS_GET_TITLE_INFO:
        case ACCESS_SET_TITLE:
        case ACCESS_SET_SEEKPOINT:
316
        case ACCESS_SET_PRIVATE_ID_STATE:
317
        case ACCESS_GET_CONTENT_TYPE:
318 319 320
            return VLC_EGENERIC;

        default:
321
            msg_Warn( p_access, "unimplemented query in control" );
322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339
            return VLC_EGENERIC;
    }
    return VLC_SUCCESS;
}

/*****************************************************************************
 * DemuxOpen:
 *****************************************************************************/
static int DemuxOpen ( vlc_object_t *p_this )
{
    demux_t *p_demux = (demux_t*)p_this;

    if( strcmp( p_demux->psz_demux, "directory" ) )
        return VLC_EGENERIC;

    p_demux->pf_demux   = Demux;
    p_demux->pf_control = DemuxControl;
    return VLC_SUCCESS;
340
}
Clément Stenac's avatar
Clément Stenac committed
341

342 343 344 345 346 347 348
/*****************************************************************************
 * Demux: EOF
 *****************************************************************************/
static int Demux( demux_t *p_demux )
{
    return 0;
}
349

350 351 352 353 354
/*****************************************************************************
 * DemuxControl:
 *****************************************************************************/
static int DemuxControl( demux_t *p_demux, int i_query, va_list args )
{
355
    return demux_vaControlHelper( p_demux->s, 0, 0, 0, 1, i_query, args );
356
}
Clément Stenac's avatar
Clément Stenac committed
357

358 359

static int Sort (const char **a, const char **b)
360
{
361
    return strcoll (*a, *b);
362
}
363

364 365 366 367 368 369 370
struct stat_list_t
{
    stat_list_t *parent;
    struct stat st;
};


Clément Stenac's avatar
Clément Stenac committed
371 372 373
/*****************************************************************************
 * ReadDir: read a directory and add its content to the list
 *****************************************************************************/
374
static int ReadDir( playlist_t *p_playlist, const char *psz_name,
375
                    int i_mode, playlist_item_t *p_parent,
376
                    playlist_item_t *p_parent_category,
377 378
                    input_item_t *p_current_input,
                    DIR *handle, stat_list_t *stparent )
Clément Stenac's avatar
Clément Stenac committed
379
{
380
    char **pp_dir_content = NULL;
381
    int             i_dir_content, i, i_return = VLC_SUCCESS;
382
    playlist_item_t *p_node;
383

384
    char **ppsz_extensions = NULL;
385
    int i_extensions = 0;
386 387
    char *psz_ignore;

388
    struct stat_list_t stself;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
389
#ifndef WIN32
390 391 392 393
    int fd = dirfd (handle);

    if ((fd == -1) || fstat (fd, &stself.st))
    {
394
        msg_Err (p_playlist, "cannot stat `%s': %m", psz_name);
395 396 397 398 399 400 401 402 403 404 405 406 407
        return VLC_EGENERIC;
    }

    for (stat_list_t *stats = stparent; stats != NULL; stats = stats->parent)
    {
        if ((stself.st.st_ino == stats->st.st_ino)
         && (stself.st.st_dev == stats->st.st_dev))
        {
            msg_Warn (p_playlist,
                      "ignoring infinitely recursive directory `%s'",
                      psz_name);
            return VLC_SUCCESS;
        }
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
408
    }
409 410
#else
        /* Windows has st_dev (driver letter - 'A'), but it zeroes st_ino,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
411 412
         * so that the test above will always incorrectly succeed.
         * Besides, Windows does not have dirfd(). */
413 414 415 416
#endif

    stself.parent = stparent;

417
    /* Get the first directory entry */
418
    i_dir_content = utf8_loaddir (handle, &pp_dir_content, NULL, Sort);
419 420
    if( i_dir_content == -1 )
    {
421
        msg_Err (p_playlist, "cannot read `%s': %m", psz_name);
422 423 424 425 426
        return VLC_EGENERIC;
    }
    else if( i_dir_content <= 0 )
    {
        /* directory is empty */
427 428
        msg_Dbg( p_playlist, "%s directory is empty", psz_name );
        free( pp_dir_content );
429 430 431 432 433
        return VLC_SUCCESS;
    }

    /* Build array with ignores */
    psz_ignore = var_CreateGetString( p_playlist, "ignore-filetypes" );
434 435
    if( psz_ignore && *psz_ignore )
    {
436 437
        char *psz_parser = psz_ignore;
        int a;
438

439 440 441 442
        for( a = 0; psz_parser[a] != '\0'; a++ )
        {
            if( psz_parser[a] == ',' ) i_extensions++;
        }
443

444
        ppsz_extensions = (char **)calloc (i_extensions, sizeof (char *));
445

446 447
        for( a = 0; a < i_extensions; a++ )
        {
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
448
            char *tmp, *ptr;
449

450
            while( psz_parser[0] != '\0' && psz_parser[0] == ' ' ) psz_parser++;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
451 452 453 454 455
            ptr = strchr( psz_parser, ',');
            tmp = ( ptr == NULL )
                 ? strdup( psz_parser )
                 : strndup( psz_parser, ptr - psz_parser );

456
            ppsz_extensions[a] = tmp;
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
457
            psz_parser = ptr + 1;
458
        }
459
    }
460
    free( psz_ignore );
461

462 463
    /* While we still have entries in the directory */
    for( i = 0; i < i_dir_content; i++ )
Clément Stenac's avatar
Clément Stenac committed
464
    {
465
        const char *entry = pp_dir_content[i];
Clément Stenac's avatar
Clément Stenac committed
466
        int i_size_entry = strlen( psz_name ) +
467
                           strlen( entry ) + 2 + 7 /* strlen("file://") */;
468
        char psz_uri[i_size_entry];
Clément Stenac's avatar
Clément Stenac committed
469

470
        sprintf( psz_uri, "%s/%s", psz_name, entry);
Clément Stenac's avatar
Clément Stenac committed
471

472
        /* if it starts with '.' then forget it */
473
        if (entry[0] != '.')
Clément Stenac's avatar
Clément Stenac committed
474
        {
475 476
            DIR *subdir = (i_mode != MODE_COLLAPSE)
                    ? OpenDir (VLC_OBJECT (p_playlist), psz_uri) : NULL;
477

478
            if (subdir != NULL) /* Recurse into subdirectory */
Clément Stenac's avatar
Clément Stenac committed
479 480 481
            {
                if( i_mode == MODE_NONE )
                {
482 483 484
                    msg_Dbg( p_playlist, "skipping subdirectory `%s'",
                             psz_uri );
                    closedir (subdir);
Clément Stenac's avatar
Clément Stenac committed
485 486
                    continue;
                }
487 488 489 490 491

                msg_Dbg (p_playlist, "creating subdirectory %s", psz_uri);

                p_node = playlist_NodeCreate( p_playlist, entry,
                                              p_parent_category,
492
                                              PLAYLIST_NO_REBUILD, NULL );
493 494 495 496 497 498 499 500 501

                /* If we had the parent in category, the it is now node.
                 * Else, we still don't have  */
                i_return = ReadDir( p_playlist, psz_uri , MODE_EXPAND,
                                    p_node, p_parent_category ? p_node : NULL,
                                    p_current_input, subdir, &stself );
                closedir (subdir);
                if (i_return)
                    break; // error :-(
Clément Stenac's avatar
Clément Stenac committed
502 503 504
            }
            else
            {
505
                input_item_t *p_input;
506

507 508
                if( i_extensions > 0 )
                {
509
                    const char *psz_dot = strrchr (entry, '.' );
510
                    if( psz_dot++ && *psz_dot )
511 512 513 514 515 516 517 518 519
                    {
                        int a;
                        for( a = 0; a < i_extensions; a++ )
                        {
                            if( !strcmp( psz_dot, ppsz_extensions[a] ) )
                                break;
                        }
                        if( a < i_extensions )
                        {
520
                            msg_Dbg( p_playlist, "ignoring file %s", psz_uri );
521 522 523 524
                            continue;
                        }
                    }
                }
525

526 527
                memmove (psz_uri + 7, psz_uri, sizeof (psz_uri) - 7);
                memcpy (psz_uri, "file://", 7);
528
                p_input = input_ItemNewWithType( VLC_OBJECT(p_playlist),
529
                                                 psz_uri, entry, 0, NULL,
530
                                                 -1, ITEM_TYPE_FILE );
531
                if (p_input != NULL)
532 533 534
                {
                    if( p_current_input )
                        input_ItemCopyOptions( p_current_input, p_input );
535
                    int i_ret = playlist_BothAddInput( p_playlist, p_input,
536
                                           p_parent_category,
537 538
                                           PLAYLIST_APPEND|PLAYLIST_PREPARSE|
                                           PLAYLIST_NO_REBUILD,
539
                                           PLAYLIST_END, NULL, NULL,
540
                                           false );
541
                    vlc_gc_decref( p_input );
542 543
                    if( i_ret != VLC_SUCCESS )
                        return VLC_EGENERIC;
544
                }
Clément Stenac's avatar
Clément Stenac committed
545 546 547
            }
        }
    }
548

549
    for( i = 0; i < i_extensions; i++ )
550 551
        free( ppsz_extensions[i] );
    free( ppsz_extensions );
552

553
    for( i = 0; i < i_dir_content; i++ )
554 555
        free( pp_dir_content[i] );
    free( pp_dir_content );
556 557

    return i_return;
Clément Stenac's avatar
Clément Stenac committed
558
}
559 560 561 562 563 564 565 566 567 568


static DIR *OpenDir (vlc_object_t *obj, const char *path)
{
    msg_Dbg (obj, "opening directory `%s'", path);
    DIR *handle = utf8_opendir (path);
    if (handle == NULL)
    {
        int err = errno;
        if (err != ENOTDIR)
569
            msg_Err (obj, "%s: %m", path);
570 571 572 573 574 575 576 577
        else
            msg_Dbg (obj, "skipping non-directory `%s'", path);
        errno = err;

        return NULL;
    }
    return handle;
}