cdda.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:22k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * cdda.c : CD digital audio input module for vlc
  3.  *****************************************************************************
  4.  * Copyright (C) 2000, 2003 the VideoLAN team
  5.  * $Id: bddbfbfd900e0f8e28079886f967471ed8e59b55 $
  6.  *
  7.  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
  8.  *          Gildas Bazin <gbazin@netcourrier.com>
  9.  *
  10.  * This program is free software; you can redistribute it and/or modify
  11.  * it under the terms of the GNU General Public License as published by
  12.  * the Free Software Foundation; either version 2 of the License, or
  13.  * (at your option) any later version.
  14.  *
  15.  * This program is distributed in the hope that it will be useful,
  16.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  18.  * GNU General Public License for more details.
  19.  *
  20.  * You should have received a copy of the GNU General Public License
  21.  * along with this program; if not, write to the Free Software
  22.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  23.  *****************************************************************************/
  24. /**
  25.  * Todo:
  26.  *   - Improve CDDB support (non-blocking, cache, ...)
  27.  *   - Fix tracknumber in MRL
  28.  */
  29. /*****************************************************************************
  30.  * Preamble
  31.  *****************************************************************************/
  32. #ifdef HAVE_CONFIG_H
  33. # include "config.h"
  34. #endif
  35. #include <vlc_common.h>
  36. #include <vlc_plugin.h>
  37. #include <vlc_input.h>
  38. #include <vlc_access.h>
  39. #include <vlc_meta.h>
  40. #include <vlc_charset.h>
  41. #include <vlc_codecs.h> /* For WAVEHEADER */
  42. #include "vcd/cdrom.h"
  43. #ifdef HAVE_LIBCDDB
  44. #include <cddb/cddb.h>
  45. #endif
  46. #include <errno.h>
  47. /*****************************************************************************
  48.  * Module descriptior
  49.  *****************************************************************************/
  50. static int  Open ( vlc_object_t * );
  51. static void Close( vlc_object_t * );
  52. #define CACHING_TEXT N_("Caching value in ms")
  53. #define CACHING_LONGTEXT N_( 
  54.     "Default caching value for Audio CDs. This " 
  55.     "value should be set in milliseconds." )
  56. vlc_module_begin ()
  57.     set_shortname( N_("Audio CD"))
  58.     set_description( N_("Audio CD input") )
  59.     set_capability( "access", 10 )
  60.     set_category( CAT_INPUT )
  61.     set_subcategory( SUBCAT_INPUT_ACCESS )
  62.     set_callbacks( Open, Close )
  63.     add_usage_hint( N_("[cdda:][device][@[track]]") )
  64.     add_integer( "cdda-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT,
  65.                  CACHING_LONGTEXT, true )
  66.         change_safe()
  67.     add_integer( "cdda-track", 0 , NULL, NULL, NULL, true )
  68.         change_internal ()
  69.     add_integer( "cdda-first-sector", -1, NULL, NULL, NULL, true )
  70.         change_internal ()
  71.     add_integer( "cdda-last-sector", -1, NULL, NULL, NULL, true )
  72.         change_internal ()
  73.     add_string( "cddb-server", "freedb.freedb.org", NULL,
  74.                 N_( "CDDB Server" ), N_( "Address of the CDDB server to use." ),
  75.                 true )
  76.     add_integer( "cddb-port", 8880, NULL,
  77.                 N_( "CDDB port" ), N_( "CDDB Server port to use." ),
  78.                 true )
  79.     add_shortcut( "cdda" )
  80.     add_shortcut( "cddasimple" )
  81. vlc_module_end ()
  82. /* how many blocks VCDRead will read in each loop */
  83. #define CDDA_BLOCKS_ONCE 20
  84. #define CDDA_DATA_ONCE   (CDDA_BLOCKS_ONCE * CDDA_DATA_SIZE)
  85. /*****************************************************************************
  86.  * Access: local prototypes
  87.  *****************************************************************************/
  88. struct access_sys_t
  89. {
  90.     vcddev_t    *vcddev;                            /* vcd device descriptor */
  91.     /* Current position */
  92.     int         i_sector;                                  /* Current Sector */
  93.     int *       p_sectors;                                  /* Track sectors */
  94.     /* Wave header for the output data */
  95.     WAVEHEADER  waveheader;
  96.     bool  b_header;
  97.     int         i_track;
  98.     int         i_first_sector;
  99.     int         i_last_sector;
  100. };
  101. static block_t *Block( access_t * );
  102. static int      Seek( access_t *, int64_t );
  103. static int      Control( access_t *, int, va_list );
  104. static int GetTracks( access_t *p_access, input_item_t *p_current );
  105. #ifdef HAVE_LIBCDDB
  106. static cddb_disc_t *GetCDDBInfo( access_t *p_access, int i_titles, int *p_sectors );
  107. #endif
  108. /*****************************************************************************
  109.  * Open: open cdda
  110.  *****************************************************************************/
  111. static int Open( vlc_object_t *p_this )
  112. {
  113.     access_t     *p_access = (access_t*)p_this;
  114.     access_sys_t *p_sys;
  115.     vcddev_t *vcddev;
  116.     char *psz_name;
  117.     int i_ret;
  118.     if( !p_access->psz_path || !*p_access->psz_path )
  119.     {
  120.         /* Only when selected */
  121.         if( !p_this->b_force ) return VLC_EGENERIC;
  122.         psz_name = var_CreateGetString( p_this, "cd-audio" );
  123.         if( !psz_name || !*psz_name )
  124.         {
  125.             free( psz_name );
  126.             return VLC_EGENERIC;
  127.         }
  128.     }
  129.     else psz_name = ToLocaleDup( p_access->psz_path );
  130. #ifdef WIN32
  131.     if( psz_name[0] && psz_name[1] == ':' &&
  132.         psz_name[2] == '\' && psz_name[3] == '' ) psz_name[2] = '';
  133. #endif
  134.     /* Open CDDA */
  135.     if( (vcddev = ioctl_Open( VLC_OBJECT(p_access), psz_name )) == NULL )
  136.     {
  137.         msg_Warn( p_access, "could not open %s", psz_name );
  138.         free( psz_name );
  139.         return VLC_EGENERIC;
  140.     }
  141.     free( psz_name );
  142.     /* Set up p_access */
  143.     STANDARD_BLOCK_ACCESS_INIT
  144.     p_sys->vcddev = vcddev;
  145.    /* Do we play a single track ? */
  146.    p_sys->i_track = var_CreateGetInteger( p_access, "cdda-track" ) - 1;
  147.    if( p_sys->i_track < 0 )
  148.    {
  149.         /* We only do separate items if the whole disc is requested */
  150.         input_thread_t *p_input = (input_thread_t*)vlc_object_find( p_access, VLC_OBJECT_INPUT, FIND_PARENT );
  151.         i_ret = -1;
  152.         if( p_input )
  153.         {
  154.             input_item_t *p_current = input_GetItem( p_input );
  155.             if( p_current )
  156.                 i_ret = GetTracks( p_access, p_current );
  157.             vlc_object_release( p_input );
  158.         }
  159.         if( i_ret < 0 )
  160.             goto error;
  161.     }
  162.     else
  163.     {
  164.         /* Build a WAV header for the output data */
  165.         memset( &p_sys->waveheader, 0, sizeof(WAVEHEADER) );
  166.         SetWLE( &p_sys->waveheader.Format, 1 ); /*WAVE_FORMAT_PCM*/
  167.         SetWLE( &p_sys->waveheader.BitsPerSample, 16);
  168.         p_sys->waveheader.MainChunkID = VLC_FOURCC('R', 'I', 'F', 'F');
  169.         p_sys->waveheader.Length = 0;               /* we just don't know */
  170.         p_sys->waveheader.ChunkTypeID = VLC_FOURCC('W', 'A', 'V', 'E');
  171.         p_sys->waveheader.SubChunkID = VLC_FOURCC('f', 'm', 't', ' ');
  172.         SetDWLE( &p_sys->waveheader.SubChunkLength, 16);
  173.         SetWLE( &p_sys->waveheader.Modus, 2);
  174.         SetDWLE( &p_sys->waveheader.SampleFreq, 44100);
  175.         SetWLE( &p_sys->waveheader.BytesPerSample,
  176.                     2 /*Modus*/ * 16 /*BitsPerSample*/ / 8 );
  177.         SetDWLE( &p_sys->waveheader.BytesPerSec,
  178.                     2*16/8 /*BytesPerSample*/ * 44100 /*SampleFreq*/ );
  179.         p_sys->waveheader.DataChunkID = VLC_FOURCC('d', 'a', 't', 'a');
  180.         p_sys->waveheader.DataLength = 0;           /* we just don't know */
  181.         p_sys->i_first_sector = var_CreateGetInteger( p_access,
  182.                                                       "cdda-first-sector" );
  183.         p_sys->i_last_sector  = var_CreateGetInteger( p_access,
  184.                                                       "cdda-last-sector" );
  185.         /* Tracknumber in MRL */
  186.         if( p_sys->i_first_sector < 0 || p_sys->i_last_sector < 0 )
  187.         {
  188.             const int i_titles = ioctl_GetTracksMap( VLC_OBJECT(p_access),
  189.                                                      p_sys->vcddev, &p_sys->p_sectors );
  190.             if( p_sys->i_track >= i_titles )
  191.             {
  192.                 msg_Err( p_access, "invalid track number" );
  193.                 goto error;
  194.             }
  195.             p_sys->i_first_sector = p_sys->p_sectors[p_sys->i_track];
  196.             p_sys->i_last_sector = p_sys->p_sectors[p_sys->i_track+1];
  197.         }
  198.         p_sys->i_sector = p_sys->i_first_sector;
  199.         p_access->info.i_size = (p_sys->i_last_sector - p_sys->i_first_sector)
  200.                                      * (int64_t)CDDA_DATA_SIZE;
  201.     }
  202.     /* PTS delay */
  203.     var_Create( p_access, "cdda-caching", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
  204.     return VLC_SUCCESS;
  205. error:
  206.     free( p_sys->p_sectors );
  207.     ioctl_Close( VLC_OBJECT(p_access), p_sys->vcddev );
  208.     free( p_sys );
  209.     return VLC_EGENERIC;
  210. }
  211. /*****************************************************************************
  212.  * Close: closes cdda
  213.  *****************************************************************************/
  214. static void Close( vlc_object_t *p_this )
  215. {
  216.     access_t     *p_access = (access_t *)p_this;
  217.     access_sys_t *p_sys = p_access->p_sys;
  218.     free( p_sys->p_sectors );
  219.     ioctl_Close( p_this, p_sys->vcddev );
  220.     free( p_sys );
  221. }
  222. /*****************************************************************************
  223.  * Block: read data (CDDA_DATA_ONCE)
  224.  *****************************************************************************/
  225. static block_t *Block( access_t *p_access )
  226. {
  227.     access_sys_t *p_sys = p_access->p_sys;
  228.     int i_blocks = CDDA_BLOCKS_ONCE;
  229.     block_t *p_block;
  230.     if( p_sys->i_track < 0 ) p_access->info.b_eof = true;
  231.     /* Check end of file */
  232.     if( p_access->info.b_eof ) return NULL;
  233.     if( !p_sys->b_header )
  234.     {
  235.         /* Return only the header */
  236.         p_block = block_New( p_access, sizeof( WAVEHEADER ) );
  237.         memcpy( p_block->p_buffer, &p_sys->waveheader, sizeof(WAVEHEADER) );
  238.         p_sys->b_header = true;
  239.         return p_block;
  240.     }
  241.     if( p_sys->i_sector >= p_sys->i_last_sector )
  242.     {
  243.         p_access->info.b_eof = true;
  244.         return NULL;
  245.     }
  246.     /* Don't read too far */
  247.     if( p_sys->i_sector + i_blocks >= p_sys->i_last_sector )
  248.         i_blocks = p_sys->i_last_sector - p_sys->i_sector;
  249.     /* Do the actual reading */
  250.     if( !( p_block = block_New( p_access, i_blocks * CDDA_DATA_SIZE ) ) )
  251.     {
  252.         msg_Err( p_access, "cannot get a new block of size: %i",
  253.                  i_blocks * CDDA_DATA_SIZE );
  254.         return NULL;
  255.     }
  256.     if( ioctl_ReadSectors( VLC_OBJECT(p_access), p_sys->vcddev,
  257.             p_sys->i_sector, p_block->p_buffer, i_blocks, CDDA_TYPE ) < 0 )
  258.     {
  259.         msg_Err( p_access, "cannot read sector %i", p_sys->i_sector );
  260.         block_Release( p_block );
  261.         /* Try to skip one sector (in case of bad sectors) */
  262.         p_sys->i_sector++;
  263.         p_access->info.i_pos += CDDA_DATA_SIZE;
  264.         return NULL;
  265.     }
  266.     /* Update a few values */
  267.     p_sys->i_sector += i_blocks;
  268.     p_access->info.i_pos += p_block->i_buffer;
  269.     return p_block;
  270. }
  271. /****************************************************************************
  272.  * Seek
  273.  ****************************************************************************/
  274. static int Seek( access_t *p_access, int64_t i_pos )
  275. {
  276.     access_sys_t *p_sys = p_access->p_sys;
  277.     /* Next sector to read */
  278.     p_sys->i_sector = p_sys->i_first_sector + i_pos / CDDA_DATA_SIZE;
  279.     p_access->info.i_pos = i_pos;
  280.     return VLC_SUCCESS;
  281. }
  282. /*****************************************************************************
  283.  * Control:
  284.  *****************************************************************************/
  285. static int Control( access_t *p_access, int i_query, va_list args )
  286. {
  287.     bool    *pb_bool;
  288.     int64_t *pi_64;
  289.     switch( i_query )
  290.     {
  291.         case ACCESS_CAN_SEEK:
  292.         case ACCESS_CAN_FASTSEEK:
  293.         case ACCESS_CAN_PAUSE:
  294.         case ACCESS_CAN_CONTROL_PACE:
  295.             pb_bool = (bool*)va_arg( args, bool* );
  296.             *pb_bool = true;
  297.             break;
  298.         case ACCESS_GET_PTS_DELAY:
  299.             pi_64 = (int64_t*)va_arg( args, int64_t * );
  300.             *pi_64 = var_GetInteger( p_access, "cdda-caching" ) * INT64_C(1000);
  301.             break;
  302.         case ACCESS_SET_PAUSE_STATE:
  303.             break;
  304.         case ACCESS_GET_TITLE_INFO:
  305.         case ACCESS_SET_TITLE:
  306.         case ACCESS_GET_META:
  307.         case ACCESS_SET_SEEKPOINT:
  308.         case ACCESS_SET_PRIVATE_ID_STATE:
  309.         case ACCESS_GET_CONTENT_TYPE:
  310.             return VLC_EGENERIC;
  311.         default:
  312.             msg_Warn( p_access, "unimplemented query in control" );
  313.             return VLC_EGENERIC;
  314.     }
  315.     return VLC_SUCCESS;
  316. }
  317. static int GetTracks( access_t *p_access, input_item_t *p_current )
  318. {
  319.     access_sys_t *p_sys = p_access->p_sys;
  320.     const int i_titles = ioctl_GetTracksMap( VLC_OBJECT(p_access),
  321.                                              p_sys->vcddev, &p_sys->p_sectors );
  322.     if( i_titles <= 0 )
  323.     {
  324.         if( i_titles < 0 )
  325.             msg_Err( p_access, "unable to count tracks" );
  326.         else if( i_titles <= 0 )
  327.             msg_Err( p_access, "no audio tracks found" );
  328.         return VLC_EGENERIC;;
  329.     }
  330.     /* */
  331.     input_item_SetName( p_current, "Audio CD" );
  332.     const char *psz_album = NULL;
  333.     const char *psz_year = NULL;
  334.     const char *psz_genre = NULL;
  335.     const char *psz_artist = NULL;
  336.     const char *psz_description = NULL;
  337. /* Return true if the given string is not NULL and not empty */
  338. #define NONEMPTY( psz ) ( (psz) && *(psz) )
  339. /* If the given string is NULL or empty, fill it by the return value of 'code' */
  340. #define ON_EMPTY( psz, code ) do { if( !NONEMPTY( psz) ) { (psz) = code; } } while(0)
  341.     /* Retreive CDDB informations */
  342. #ifdef HAVE_LIBCDDB
  343.     char psz_year_buffer[4+1];
  344.     cddb_disc_t *p_disc = GetCDDBInfo( p_access, i_titles, p_sys->p_sectors );
  345.     if( p_disc )
  346.     {
  347.         psz_album = cddb_disc_get_title( p_disc );
  348.         psz_genre = cddb_disc_get_genre( p_disc );
  349.         /* */
  350.         const unsigned i_year = cddb_disc_get_year( p_disc );
  351.         if( i_year > 0 )
  352.         {
  353.             psz_year = psz_year_buffer;
  354.             snprintf( psz_year_buffer, sizeof(psz_year_buffer), "%u", i_year );
  355.         }
  356.         /* Set artist only if unique */
  357.         for( int i = 0; i < i_titles; i++ )
  358.         {
  359.             cddb_track_t *t = cddb_disc_get_track( p_disc, i );
  360.             if( !t )
  361.                 continue;
  362.             const char *psz_track_artist = cddb_track_get_artist( t );
  363.             if( psz_artist && psz_track_artist &&
  364.                 strcmp( psz_artist, psz_track_artist ) )
  365.             {
  366.                 psz_artist = NULL;
  367.                 break;
  368.             }
  369.             psz_artist = psz_track_artist;
  370.         }
  371.     }
  372. #endif
  373.     /* */
  374.     vlc_meta_t **pp_cd_text;
  375.     int        i_cd_text;
  376.     if( ioctl_GetCdText( VLC_OBJECT(p_access), p_sys->vcddev, &pp_cd_text, &i_cd_text ) )
  377.     {
  378.         msg_Dbg( p_access, "CD-TEXT information missing" );
  379.         i_cd_text = 0;
  380.         pp_cd_text = NULL;
  381.     }
  382.     /* Retreive CD-TEXT informations but prefer CDDB */
  383.     if( i_cd_text > 0 && pp_cd_text[0] )
  384.     {
  385.         const vlc_meta_t *p_disc = pp_cd_text[0];
  386.         ON_EMPTY( psz_album,       vlc_meta_Get( p_disc, vlc_meta_Album ) );
  387.         ON_EMPTY( psz_genre,       vlc_meta_Get( p_disc, vlc_meta_Genre ) );
  388.         ON_EMPTY( psz_artist,      vlc_meta_Get( p_disc, vlc_meta_Artist ) );
  389.         ON_EMPTY( psz_description, vlc_meta_Get( p_disc, vlc_meta_Description ) );
  390.     }
  391.     if( NONEMPTY( psz_album ) )
  392.     {
  393.         input_item_SetName( p_current, psz_album );
  394.         input_item_SetAlbum( p_current, psz_album );
  395.     }
  396.     if( NONEMPTY( psz_genre ) )
  397.         input_item_SetGenre( p_current, psz_genre );
  398.     if( NONEMPTY( psz_artist ) )
  399.         input_item_SetArtist( p_current, psz_artist );
  400.     if( NONEMPTY( psz_year ) )
  401.         input_item_SetDate( p_current, psz_year );
  402.     if( NONEMPTY( psz_description ) )
  403.         input_item_SetDescription( p_current, psz_description );
  404.     const mtime_t i_duration = (int64_t)( p_sys->p_sectors[i_titles] - p_sys->p_sectors[0] ) *
  405.                                CDDA_DATA_SIZE * 1000000 / 44100 / 2 / 2;
  406.     input_item_SetDuration( p_current, i_duration );
  407.     /* Build title table */
  408.     for( int i = 0; i < i_titles; i++ )
  409.     {
  410.         input_item_t *p_input_item;
  411.         char *psz_uri, *psz_opt, *psz_first, *psz_last;
  412.         char *psz_name;
  413.         msg_Dbg( p_access, "track[%d] start=%d", i, p_sys->p_sectors[i] );
  414.         /* */
  415.         if( asprintf( &psz_uri, "cdda://%s", p_access->psz_path ) == -1 )
  416.             psz_uri = NULL;
  417.         if( asprintf( &psz_opt, "cdda-track=%i", i+1 ) == -1 )
  418.             psz_opt = NULL;
  419.         if( asprintf( &psz_first, "cdda-first-sector=%i",p_sys->p_sectors[i] ) == -1 )
  420.             psz_first = NULL;
  421.         if( asprintf( &psz_last, "cdda-last-sector=%i", p_sys->p_sectors[i+1] ) == -1 )
  422.             psz_last = NULL;
  423.         /* Define a "default name" */
  424.         if( asprintf( &psz_name, _("Audio CD - Track %02i"), (i+1) ) == -1 )
  425.             psz_name = NULL;
  426.         /* Create playlist items */
  427.         const mtime_t i_duration = (int64_t)( p_sys->p_sectors[i+1] - p_sys->p_sectors[i] ) *
  428.                                    CDDA_DATA_SIZE * 1000000 / 44100 / 2 / 2;
  429.         p_input_item = input_item_NewWithType( VLC_OBJECT( p_access ),
  430.                                               psz_uri, psz_name, 0, NULL, 0, i_duration,
  431.                                               ITEM_TYPE_DISC );
  432.         input_item_CopyOptions( p_current, p_input_item );
  433.         input_item_AddOption( p_input_item, psz_first, VLC_INPUT_OPTION_TRUSTED );
  434.         input_item_AddOption( p_input_item, psz_last, VLC_INPUT_OPTION_TRUSTED );
  435.         input_item_AddOption( p_input_item, psz_opt, VLC_INPUT_OPTION_TRUSTED );
  436.         const char *psz_track_title = NULL;
  437.         const char *psz_track_artist = NULL;
  438.         const char *psz_track_genre = NULL;
  439.         const char *psz_track_description = NULL;
  440. #ifdef HAVE_LIBCDDB
  441.         /* Retreive CDDB informations */
  442.         if( p_disc )
  443.         {
  444.             cddb_track_t *t = cddb_disc_get_track( p_disc, i );
  445.             if( t != NULL )
  446.             {
  447.                 psz_track_title = cddb_track_get_title( t );
  448.                 psz_track_artist = cddb_track_get_artist( t );
  449.             }
  450.         }
  451. #endif
  452.         /* Retreive CD-TEXT informations but prefer CDDB */
  453.         if( i+1 < i_cd_text && pp_cd_text[i+1] )
  454.         {
  455.             const vlc_meta_t *t = pp_cd_text[i+1];
  456.             ON_EMPTY( psz_track_title,       vlc_meta_Get( t, vlc_meta_Title ) );
  457.             ON_EMPTY( psz_track_artist,      vlc_meta_Get( t, vlc_meta_Artist ) );
  458.             ON_EMPTY( psz_track_genre,       vlc_meta_Get( t, vlc_meta_Genre ) );
  459.             ON_EMPTY( psz_track_description, vlc_meta_Get( t, vlc_meta_Description ) );
  460.         }
  461.         /* */
  462.         ON_EMPTY( psz_track_artist,       psz_artist );
  463.         ON_EMPTY( psz_track_genre,        psz_genre );
  464.         ON_EMPTY( psz_track_description,  psz_description );
  465.         /* */
  466.         if( NONEMPTY( psz_track_title ) )
  467.         {
  468.             input_item_SetName( p_input_item, psz_track_title );
  469.             input_item_SetTitle( p_input_item, psz_track_title );
  470.         }
  471.         if( NONEMPTY( psz_track_artist ) )
  472.             input_item_SetArtist( p_input_item, psz_track_artist );
  473.         if( NONEMPTY( psz_track_genre ) )
  474.             input_item_SetGenre( p_input_item, psz_track_genre );
  475.         if( NONEMPTY( psz_track_description ) )
  476.             input_item_SetDescription( p_input_item, psz_track_description );
  477.         if( NONEMPTY( psz_album ) )
  478.             input_item_SetAlbum( p_input_item, psz_album );
  479.         if( NONEMPTY( psz_year ) )
  480.             input_item_SetDate( p_input_item, psz_year );
  481.         char psz_num[3+1];
  482.         snprintf( psz_num, sizeof(psz_num), "%d", 1+i );
  483.         input_item_SetTrackNum( p_input_item, psz_num );
  484.         input_item_AddSubItem( p_current, p_input_item );
  485.         vlc_gc_decref( p_input_item );
  486.         free( psz_uri ); free( psz_opt ); free( psz_name );
  487.         free( psz_first ); free( psz_last );
  488.     }
  489. #undef ON_EMPTY
  490. #undef NONEMPTY
  491.     /* */
  492.     for( int i = 0; i < i_cd_text; i++ )
  493.     {
  494.         vlc_meta_t *p_meta = pp_cd_text[i];
  495.         if( !p_meta )
  496.             continue;
  497.         vlc_meta_Delete( p_meta );
  498.     }
  499.     free( pp_cd_text );
  500. #ifdef HAVE_LIBCDDB
  501.     if( p_disc )
  502.         cddb_disc_destroy( p_disc );
  503. #endif
  504.     return VLC_SUCCESS;
  505. }
  506. #ifdef HAVE_LIBCDDB
  507. static cddb_disc_t *GetCDDBInfo( access_t *p_access, int i_titles, int *p_sectors )
  508. {
  509.     if( var_CreateGetInteger( p_access, "album-art" ) == ALBUM_ART_WHEN_ASKED )
  510.         return NULL;
  511.     /* */
  512.     cddb_conn_t *p_cddb = cddb_new();
  513.     if( !p_cddb )
  514.     {
  515.         msg_Warn( p_access, "unable to use CDDB" );
  516.         return NULL;
  517.     }
  518.     /* */
  519.     char *psz_tmp = config_GetPsz( p_access, "cddb-server" );
  520.     cddb_set_server_name( p_cddb, psz_tmp );
  521.     free( psz_tmp );
  522.     cddb_set_server_port( p_cddb, config_GetInt( p_access, "cddb-port" ) );
  523.     cddb_set_email_address( p_cddb, "vlc@videolan.org" );
  524.     /// todo
  525.     cddb_cache_disable( p_cddb );
  526. //    cddb_cache_set_dir( p_cddb,
  527. //                     config_GetPsz( p_access,
  528. //                                    MODULE_STRING "-cddb-cachedir") );
  529.     cddb_set_timeout( p_cddb, 10 );
  530.     /// todo
  531.     cddb_http_disable( p_cddb);
  532.     /* */
  533.     cddb_disc_t *p_disc = cddb_disc_new();
  534.     if( !p_disc )
  535.     {
  536.         msg_Err( p_access, "unable to create CDDB disc structure." );
  537.         goto error;
  538.     }
  539.     int64_t i_length = 0;
  540.     for( int i = 0; i < i_titles; i++ )
  541.     {
  542.         cddb_track_t *t = cddb_track_new();
  543.         cddb_track_set_frame_offset( t, p_sectors[i] );
  544.         cddb_disc_add_track( p_disc, t );
  545.         const int64_t i_size = ( p_sectors[i+1] - p_sectors[i] ) *
  546.                                (int64_t)CDDA_DATA_SIZE;
  547.         i_length += INT64_C(1000000) * i_size / 44100 / 4  ;
  548.     }
  549.     cddb_disc_set_length( p_disc, (int)(i_length/1000000) );
  550.     if( !cddb_disc_calc_discid( p_disc ) )
  551.     {
  552.         msg_Err( p_access, "CDDB disc ID calculation failed" );
  553.         goto error;
  554.     }
  555.     const int i_matches = cddb_query( p_cddb, p_disc );
  556.     if( i_matches <= 0 )
  557.     {
  558.         msg_Warn( p_access, "CDDB error: %s", cddb_error_str(errno));
  559.         goto error;
  560.     }
  561.     if( i_matches > 1 )
  562.         msg_Warn( p_access, "found %d matches in CDDB. Using first one.", i_matches );
  563.     cddb_read( p_cddb, p_disc );
  564.     cddb_destroy( p_cddb);
  565.     return p_disc;
  566. error:
  567.     if( p_disc )
  568.         cddb_disc_destroy( p_disc );
  569.     cddb_destroy( p_cddb );
  570.     return NULL;
  571. }
  572. #endif /*HAVE_LIBCDDB*/