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/vlc.h>
34
#include <vlc_playlist.h>
35
#include <vlc_input.h>
Clément Stenac's avatar
Clément Stenac committed
36 37
#include <vlc_access.h>
#include <vlc_demux.h>
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55

#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>
56 57
#elif defined( UNDER_CE )
#   define strcoll strcmp
58 59
#endif

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

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

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

72
static int  DemuxOpen ( vlc_object_t * );
73

74
#define RECURSIVE_TEXT N_("Subdirectory behavior")
Clément Stenac's avatar
Clément Stenac committed
75 76
#define RECURSIVE_LONGTEXT N_( \
        "Select whether subdirectories must be expanded.\n" \
Carlo Calabrò's avatar
Carlo Calabrò committed
77 78 79
        "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
80

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

Clément Stenac's avatar
Clément Stenac committed
85
#define IGNORE_TEXT N_("Ignored extensions")
86
#define IGNORE_LONGTEXT N_( \
Clément Stenac's avatar
Clément Stenac committed
87 88 89 90
        "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." )
91

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

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


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

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

typedef struct stat_list_t stat_list_t;
127

128 129
static ssize_t Read( access_t *, uint8_t *, size_t );
static ssize_t ReadNull( access_t *, uint8_t *, size_t );
130 131 132 133 134 135
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 );


136
static int ReadDir( playlist_t *, const char *psz_name, int i_mode,
137 138 139 140
                    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);
141

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

149 150 151 152 153 154 155
    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;

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

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

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

167
    /* Force a demux */
168
    p_access->psz_demux = strdup( "directory" );
Gildas Bazin's avatar
 
Gildas Bazin committed
169

170 171 172 173 174 175 176 177
    return VLC_SUCCESS;
}

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

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

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

    if( psz_name == NULL )
        return VLC_ENOMEM;

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

208
    playlist_item_t    *p_item_in_category;
209 210 211 212 213 214 215 216 217 218 219
    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 );
220
    p_current = playlist_ItemGetByInput( p_playlist, p_current_input, false );
221

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

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

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

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

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

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

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

267 268
    playlist_Signal( p_playlist );

269
    free( psz_name );
270
    vlc_object_release( p_input );
271
    vlc_object_release( p_playlist );
272 273 274

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

/*****************************************************************************
279
 * Control:
280 281 282
 *****************************************************************************/
static int Control( access_t *p_access, int i_query, va_list args )
{
283
    bool   *pb_bool;
284 285 286 287 288 289 290 291 292 293
    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:
294 295
            pb_bool = (bool*)va_arg( args, bool* );
            *pb_bool = false;    /* FIXME */
296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313
            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:
314
        case ACCESS_SET_PRIVATE_ID_STATE:
315
        case ACCESS_GET_CONTENT_TYPE:
316 317 318
            return VLC_EGENERIC;

        default:
319
            msg_Warn( p_access, "unimplemented query in control" );
320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337
            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;
338
}
Clément Stenac's avatar
Clément Stenac committed
339

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

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

356 357

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

362 363 364 365 366 367 368
struct stat_list_t
{
    stat_list_t *parent;
    struct stat st;
};


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

382
    char **ppsz_extensions = NULL;
383
    int i_extensions = 0;
384 385
    char *psz_ignore;

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

    if ((fd == -1) || fstat (fd, &stself.st))
    {
392
        msg_Err (p_playlist, "cannot stat `%s': %m", psz_name);
393 394 395 396 397 398 399 400 401 402 403 404 405
        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
406
    }
407 408
#else
        /* Windows has st_dev (driver letter - 'A'), but it zeroes st_ino,
Rémi Denis-Courmont's avatar
Rémi Denis-Courmont committed
409 410
         * so that the test above will always incorrectly succeed.
         * Besides, Windows does not have dirfd(). */
411 412 413 414
#endif

    stself.parent = stparent;

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

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

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

442
        ppsz_extensions = (char **)calloc (i_extensions, sizeof (char *));
443

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

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

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

460 461
    /* 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
462
    {
463
        const char *entry = pp_dir_content[i];
Clément Stenac's avatar
Clément Stenac committed
464
        int i_size_entry = strlen( psz_name ) +
465
                           strlen( entry ) + 2 + 7 /* strlen("file://") */;
466
        char psz_uri[i_size_entry];
Clément Stenac's avatar
Clément Stenac committed
467

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

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

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

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

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

                /* 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
500 501 502
            }
            else
            {
503
                input_item_t *p_input;
504

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

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

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

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

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


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)
567
            msg_Err (obj, "%s: %m", path);
568 569 570 571 572 573 574 575
        else
            msg_Dbg (obj, "skipping non-directory `%s'", path);
        errno = err;

        return NULL;
    }
    return handle;
}