aac.c
上传用户:riyaled888
上传日期:2009-03-27
资源大小:7338k
文件大小:8k
源码类别:

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * aac.c : Raw aac Stream input module for vlc
  3.  *****************************************************************************
  4.  * Copyright (C) 2001-2003 VideoLAN
  5.  * $Id: aac.c 7665 2004-05-15 10:52:56Z fenrir $
  6.  *
  7.  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
  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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #include <stdlib.h>                                      /* malloc(), free() */
  27. #include <vlc/vlc.h>
  28. #include <vlc/input.h>
  29. /*****************************************************************************
  30.  * Module descriptor
  31.  *****************************************************************************/
  32. static int  Open    ( vlc_object_t * );
  33. static void Close  ( vlc_object_t * );
  34. vlc_module_begin();
  35.     set_description( _("AAC demuxer" ) );
  36.     set_capability( "demux2", 100 );
  37.     set_callbacks( Open, Close );
  38.     add_shortcut( "aac" );
  39. vlc_module_end();
  40. /* TODO:
  41.  * - adif support ?
  42.  *
  43.  */
  44. /*****************************************************************************
  45.  * Local prototypes
  46.  *****************************************************************************/
  47. struct demux_sys_t
  48. {
  49.     mtime_t         i_time;
  50.     es_out_id_t     *p_es;
  51. };
  52. static int Demux  ( demux_t * );
  53. static int Control( demux_t *, int, va_list );
  54. static int i_aac_samplerate[16] =
  55. {
  56.     96000, 88200, 64000, 48000, 44100, 32000,
  57.     24000, 22050, 16000, 12000, 11025, 8000,
  58.     7350,  0,     0,     0
  59. };
  60. #define AAC_ID( p )          ( ((p)[1]>>3)&0x01 )
  61. #define AAC_SAMPLE_RATE( p ) i_aac_samplerate[((p)[2]>>2)&0x0f]
  62. #define AAC_CHANNELS( p )    ( (((p)[2]&0x01)<<2) | (((p)[3]>>6)&0x03) )
  63. #define AAC_FRAME_SIZE( p )  ( (((p)[3]&0x03) << 11)|( (p)[4] << 3 )|( (((p)[5]) >>5)&0x7 ) )
  64. /* FIXME it's plain wrong */
  65. #define AAC_FRAME_SAMPLES( p )  1024
  66. static inline int HeaderCheck( uint8_t *p )
  67. {
  68.     if( p[0] != 0xff ||
  69.         ( p[1]&0xf6 ) != 0xf0 ||
  70.         AAC_SAMPLE_RATE( p ) == 0 ||
  71.         AAC_CHANNELS( p ) == 0 ||
  72.         AAC_FRAME_SIZE( p ) == 0 )
  73.     {
  74.         return VLC_FALSE;
  75.     }
  76.     return VLC_TRUE;
  77. }
  78. /*****************************************************************************
  79.  * Open: initializes AAC demux structures
  80.  *****************************************************************************/
  81. static int Open( vlc_object_t * p_this )
  82. {
  83.     demux_t     *p_demux = (demux_t*)p_this;
  84.     demux_sys_t *p_sys;
  85.     int         b_forced = VLC_FALSE;
  86.     uint8_t     *p_peek;
  87.     module_t    *p_id3;
  88.     es_format_t fmt;
  89.     if( !strncmp( p_demux->psz_demux, "aac", 3 ) )
  90.     {
  91.         b_forced = VLC_TRUE;
  92.     }
  93.     if( p_demux->psz_path )
  94.     {
  95.         int  i_len = strlen( p_demux->psz_path );
  96.         if( i_len > 4 && !strcasecmp( &p_demux->psz_path[i_len - 4], ".aac" ) )
  97.         {
  98.             b_forced = VLC_TRUE;
  99.         }
  100.     }
  101.     if( !b_forced )
  102.     {
  103.         /* I haven't find any sure working aac detection so only forced or
  104.          * extention check
  105.          */
  106.         msg_Warn( p_demux, "AAC module discarded" );
  107.         return VLC_EGENERIC;
  108.     }
  109.     /* skip possible id3 header */
  110.     if( ( p_id3 = module_Need( p_demux, "id3", NULL, 0 ) ) )
  111.     {
  112.         module_Unneed( p_demux, p_id3 );
  113.     }
  114.     p_demux->pf_demux   = Demux;
  115.     p_demux->pf_control = Control;
  116.     p_demux->p_sys = p_sys = malloc( sizeof( demux_sys_t ) );
  117.     p_sys->i_time = 1;
  118.     /* peek the begining (10 is for adts header) */
  119.     if( stream_Peek( p_demux->s, &p_peek, 10 ) < 10 )
  120.     {
  121.         msg_Err( p_demux, "cannot peek" );
  122.         goto error;
  123.     }
  124.     if( !strncmp( p_peek, "ADIF", 4 ) )
  125.     {
  126.         msg_Err( p_demux, "ADIF file. Not yet supported. (Please report)" );
  127.         goto error;
  128.     }
  129.     es_format_Init( &fmt, AUDIO_ES, VLC_FOURCC( 'm', 'p', '4', 'a' ) );
  130.     if( HeaderCheck( p_peek ) )
  131.     {
  132.         fmt.audio.i_channels = AAC_CHANNELS( p_peek );
  133.         fmt.audio.i_rate     = AAC_SAMPLE_RATE( p_peek );
  134.         msg_Dbg( p_demux,
  135.                  "adts header: id=%d channels=%d sample_rate=%d",
  136.                  AAC_ID( p_peek ),
  137.                  AAC_CHANNELS( p_peek ),
  138.                  AAC_SAMPLE_RATE( p_peek ) );
  139.     }
  140.     p_sys->p_es = es_out_Add( p_demux->out, &fmt );
  141.     return VLC_SUCCESS;
  142. error:
  143.     free( p_sys );
  144.     return VLC_EGENERIC;
  145. }
  146. /*****************************************************************************
  147.  * Demux: reads and demuxes data packets
  148.  *****************************************************************************
  149.  * Returns -1 in case of error, 0 in case of EOF, 1 otherwise
  150.  *****************************************************************************/
  151. static int Demux( demux_t *p_demux )
  152. {
  153.     demux_sys_t *p_sys = p_demux->p_sys;
  154.     block_t     *p_block;
  155.     uint8_t     h[8];
  156.     uint8_t     *p_peek;
  157.     if( stream_Peek( p_demux->s, &p_peek, 8 ) < 8 )
  158.     {
  159.         msg_Warn( p_demux, "cannot peek" );
  160.         return 0;
  161.     }
  162.     if( !HeaderCheck( p_peek ) )
  163.     {
  164.         /* we need to resynch */
  165.         vlc_bool_t  b_ok = VLC_FALSE;
  166.         int         i_skip = 0;
  167.         int         i_peek;
  168.         i_peek = stream_Peek( p_demux->s, &p_peek, 8096 );
  169.         if( i_peek < 8 )
  170.         {
  171.             msg_Warn( p_demux, "cannot peek" );
  172.             return 0;
  173.         }
  174.         while( i_peek >= 8 )
  175.         {
  176.             if( HeaderCheck( p_peek ) )
  177.             {
  178.                 b_ok = VLC_TRUE;
  179.                 break;
  180.             }
  181.             p_peek++;
  182.             i_peek--;
  183.             i_skip++;
  184.         }
  185.         msg_Warn( p_demux, "garbage=%d bytes", i_skip );
  186.         stream_Read( p_demux->s, NULL, i_skip );
  187.         return 1;
  188.     }
  189.     memcpy( h, p_peek, 8 );    /* can't use p_peek after stream_*  */
  190.     /* set PCR */
  191.     es_out_Control( p_demux->out, ES_OUT_SET_PCR, p_sys->i_time );
  192.     if( ( p_block = stream_Block( p_demux->s, AAC_FRAME_SIZE( h ) ) ) == NULL )
  193.     {
  194.         msg_Warn( p_demux, "cannot read data" );
  195.         return 0;
  196.     }
  197.     p_block->i_dts = p_block->i_pts = p_sys->i_time;
  198.     es_out_Send( p_demux->out, p_sys->p_es, p_block );
  199.     p_sys->i_time += (mtime_t)1000000 *
  200.                      (mtime_t)AAC_FRAME_SAMPLES( h ) /
  201.                      (mtime_t)AAC_SAMPLE_RATE( h );
  202.     return( 1 );
  203. }
  204. /*****************************************************************************
  205.  * Close: frees unused data
  206.  *****************************************************************************/
  207. static void Close( vlc_object_t * p_this )
  208. {
  209.     demux_t     *p_demux = (demux_t*)p_this;
  210.     demux_sys_t *p_sys = p_demux->p_sys;
  211.     free( p_sys );
  212. }
  213. /*****************************************************************************
  214.  * Control:
  215.  *****************************************************************************/
  216. static int Control( demux_t *p_demux, int i_query, va_list args )
  217. {
  218.     /* demux_sys_t *p_sys  = p_demux->p_sys; */
  219.     /* FIXME calculate the bitrate */
  220.     if( i_query == DEMUX_SET_TIME )
  221.         return VLC_EGENERIC;
  222.     else
  223.         return demux2_vaControlHelper( p_demux->s,
  224.                                        0, -1,
  225.                                        8*0, 1, i_query, args );
  226. }