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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * au.c : au file input module for vlc
  3.  *****************************************************************************
  4.  * Copyright (C) 2001-2007 the VideoLAN team
  5.  * $Id: 08f7e986417f0d84d2dd2f418d98a09742dcc6b2 $
  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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <vlc_common.h>
  30. #include <vlc_plugin.h>
  31. #include <vlc_demux.h>
  32. /* TODO:
  33.  *  - all adpcm things (I _NEED_ samples)
  34.  *  - ...
  35.  */
  36. /*****************************************************************************
  37.  * Module descriptor
  38.  *****************************************************************************/
  39. static int  Open ( vlc_object_t * );
  40. static void Close( vlc_object_t * );
  41. vlc_module_begin ()
  42.     set_category( CAT_INPUT )
  43.     set_subcategory( SUBCAT_INPUT_DEMUX )
  44.     set_description( N_("AU demuxer") )
  45.     set_capability( "demux", 10 )
  46.     set_callbacks( Open, Close )
  47.     add_shortcut( "au" )
  48. vlc_module_end ()
  49. /*****************************************************************************
  50.  * Local prototypes
  51.  *****************************************************************************/
  52. enum AuType_e
  53. {
  54.     AU_UNKNOWN      =  0,
  55.     AU_MULAW_8      =  1,  /* 8-bit ISDN u-law */
  56.     AU_LINEAR_8     =  2,  /* 8-bit linear PCM */
  57.     AU_LINEAR_16    =  3,  /* 16-bit linear PCM */
  58.     AU_LINEAR_24    =  4,  /* 24-bit linear PCM */
  59.     AU_LINEAR_32    =  5,  /* 32-bit linear PCM */
  60.     AU_FLOAT        =  6,  /* 32-bit IEEE floating point */
  61.     AU_DOUBLE       =  7,  /* 64-bit IEEE floating point */
  62.     AU_ADPCM_G721   =  23, /* 4-bit CCITT g.721 ADPCM */
  63.     AU_ADPCM_G722   =  24, /* CCITT g.722 ADPCM */
  64.     AU_ADPCM_G723_3 =  25, /* CCITT g.723 3-bit ADPCM */
  65.     AU_ADPCM_G723_5 =  26, /* CCITT g.723 5-bit ADPCM */
  66.     AU_ALAW_8       =  27  /* 8-bit ISDN A-law */
  67. };
  68. enum AuCat_e
  69. {
  70.     AU_CAT_UNKNOWN  = 0,
  71.     AU_CAT_PCM      = 1,
  72.     AU_CAT_ADPCM    = 2
  73. };
  74. struct demux_sys_t
  75. {
  76.     es_format_t     fmt;
  77.     es_out_id_t     *es;
  78.     mtime_t         i_time;
  79.     int             i_frame_size;
  80.     mtime_t         i_frame_length;
  81.     int             i_header_size;
  82. };
  83. static int Demux( demux_t * );
  84. static int Control ( demux_t *, int i_query, va_list args );
  85. /*****************************************************************************
  86.  * Open: check file and initializes structures
  87.  *****************************************************************************/
  88. static int Open( vlc_object_t *p_this )
  89. {
  90.     demux_t     *p_demux = (demux_t*)p_this;
  91.     demux_sys_t *p_sys;
  92.     uint8_t      hdr[20];
  93.     const uint8_t *p_peek;
  94.     int          i_cat;
  95.     int          i_samples, i_modulo;
  96.     if( stream_Peek( p_demux->s , &p_peek, 4 ) < 4 )
  97.         return VLC_EGENERIC;
  98.     if( memcmp( p_peek, ".snd", 4 ) )
  99.         return VLC_EGENERIC;
  100.     /* skip signature */
  101.     stream_Read( p_demux->s, NULL, 4 );   /* cannot fail */
  102.     /* read header */
  103.     if( stream_Read( p_demux->s, hdr, 20 ) < 20 )
  104.     {
  105.         msg_Err( p_demux, "cannot read" );
  106.         return VLC_EGENERIC;
  107.     }
  108.     if( GetDWBE( &hdr[0]  ) < 24 )
  109.     {
  110.         msg_Err( p_demux, "invalid file" );
  111.         return VLC_EGENERIC;
  112.     }
  113.     DEMUX_INIT_COMMON(); p_sys = p_demux->p_sys;
  114.     p_sys->i_time = 1;
  115.     p_sys->i_header_size = GetDWBE( &hdr[0] );
  116.     /* skip extra header data */
  117.     if( p_sys->i_header_size > 24 )
  118.     {
  119.         stream_Read( p_demux->s, NULL, p_sys->i_header_size - 24 );
  120.     }
  121.     /* init fmt */
  122.     es_format_Init( &p_sys->fmt, AUDIO_ES, 0 );
  123.     p_sys->fmt.audio.i_rate     = GetDWBE( &hdr[12] );
  124.     p_sys->fmt.audio.i_channels = GetDWBE( &hdr[16] );
  125. #if 0
  126.     p_sys->au.i_header_size   = GetDWBE( &p_sys->au.i_header_size );
  127.     p_sys->au.i_data_size     = GetDWBE( &p_sys->au.i_data_size );
  128.     p_sys->au.i_encoding      = GetDWBE( &p_sys->au.i_encoding );
  129.     p_sys->au.i_sample_rate   = GetDWBE( &p_sys->au.i_sample_rate );
  130.     p_sys->au.i_channels      = GetDWBE( &p_sys->au.i_channels );
  131. #endif
  132.     switch( GetDWBE( &hdr[8] ) )
  133.     {
  134.         case AU_ALAW_8:        /* 8-bit ISDN A-law */
  135.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a','l','a','w' );
  136.             p_sys->fmt.audio.i_bitspersample = 8;
  137.             p_sys->fmt.audio.i_blockalign    = 1 * p_sys->fmt.audio.i_channels;
  138.             i_cat                    = AU_CAT_PCM;
  139.             break;
  140.         case AU_MULAW_8:       /* 8-bit ISDN u-law */
  141.             p_sys->fmt.i_codec               = VLC_FOURCC( 'u','l','a','w' );
  142.             p_sys->fmt.audio.i_bitspersample = 8;
  143.             p_sys->fmt.audio.i_blockalign    = 1 * p_sys->fmt.audio.i_channels;
  144.             i_cat                    = AU_CAT_PCM;
  145.             break;
  146.         case AU_LINEAR_8:      /* 8-bit linear PCM */
  147.             p_sys->fmt.i_codec               = VLC_FOURCC( 't','w','o','s' );
  148.             p_sys->fmt.audio.i_bitspersample = 8;
  149.             p_sys->fmt.audio.i_blockalign    = 1 * p_sys->fmt.audio.i_channels;
  150.             i_cat                    = AU_CAT_PCM;
  151.             break;
  152.         case AU_LINEAR_16:     /* 16-bit linear PCM */
  153.             p_sys->fmt.i_codec               = VLC_FOURCC( 't','w','o','s' );
  154.             p_sys->fmt.audio.i_bitspersample = 16;
  155.             p_sys->fmt.audio.i_blockalign    = 2 * p_sys->fmt.audio.i_channels;
  156.             i_cat                    = AU_CAT_PCM;
  157.             break;
  158.         case AU_LINEAR_24:     /* 24-bit linear PCM */
  159.             p_sys->fmt.i_codec               = VLC_FOURCC( 't','w','o','s' );
  160.             p_sys->fmt.audio.i_bitspersample = 24;
  161.             p_sys->fmt.audio.i_blockalign    = 3 * p_sys->fmt.audio.i_channels;
  162.             i_cat                    = AU_CAT_PCM;
  163.             break;
  164.         case AU_LINEAR_32:     /* 32-bit linear PCM */
  165.             p_sys->fmt.i_codec               = VLC_FOURCC( 't','w','o','s' );
  166.             p_sys->fmt.audio.i_bitspersample = 32;
  167.             p_sys->fmt.audio.i_blockalign    = 4 * p_sys->fmt.audio.i_channels;
  168.             i_cat                    = AU_CAT_PCM;
  169.             break;
  170.         case AU_FLOAT:         /* 32-bit IEEE floating point */
  171.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_FLOAT );
  172.             p_sys->fmt.audio.i_bitspersample = 32;
  173.             p_sys->fmt.audio.i_blockalign    = 4 * p_sys->fmt.audio.i_channels;
  174.             i_cat                    = AU_CAT_PCM;
  175.             break;
  176.         case AU_DOUBLE:        /* 64-bit IEEE floating point */
  177.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_DOUBLE );
  178.             p_sys->fmt.audio.i_bitspersample = 64;
  179.             p_sys->fmt.audio.i_blockalign    = 8 * p_sys->fmt.audio.i_channels;
  180.             i_cat                    = AU_CAT_PCM;
  181.             break;
  182.         case AU_ADPCM_G721:    /* 4-bit CCITT g.721 ADPCM */
  183.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_ADPCM_G721 );
  184.             p_sys->fmt.audio.i_bitspersample = 0;
  185.             p_sys->fmt.audio.i_blockalign    = 0 * p_sys->fmt.audio.i_channels;
  186.             i_cat                    = AU_CAT_ADPCM;
  187.             break;
  188.         case AU_ADPCM_G722:    /* CCITT g.722 ADPCM */
  189.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_ADPCM_G722 );
  190.             p_sys->fmt.audio.i_bitspersample = 0;
  191.             p_sys->fmt.audio.i_blockalign    = 0 * p_sys->fmt.audio.i_channels;
  192.             i_cat                    = AU_CAT_ADPCM;
  193.             break;
  194.         case AU_ADPCM_G723_3:  /* CCITT g.723 3-bit ADPCM */
  195.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_ADPCM_G723_3 );
  196.             p_sys->fmt.audio.i_bitspersample = 0;
  197.             p_sys->fmt.audio.i_blockalign    = 0 * p_sys->fmt.audio.i_channels;
  198.             i_cat                    = AU_CAT_ADPCM;
  199.             break;
  200.         case AU_ADPCM_G723_5:  /* CCITT g.723 5-bit ADPCM */
  201.             p_sys->fmt.i_codec               = VLC_FOURCC( 'a', 'u', 0, AU_ADPCM_G723_5 );
  202.             p_sys->fmt.audio.i_bitspersample = 0;
  203.             p_sys->fmt.audio.i_blockalign    = 0 * p_sys->fmt.audio.i_channels;
  204.             i_cat                    = AU_CAT_ADPCM;
  205.             break;
  206.         default:
  207.             msg_Warn( p_demux, "unknow encoding=0x%x", GetDWBE( &hdr[8] ) );
  208.             p_sys->fmt.audio.i_bitspersample = 0;
  209.             p_sys->fmt.audio.i_blockalign    = 0;
  210.             i_cat                    = AU_CAT_UNKNOWN;
  211.             break;
  212.     }
  213.     p_sys->fmt.i_bitrate = p_sys->fmt.audio.i_rate *
  214.                            p_sys->fmt.audio.i_channels *
  215.                            p_sys->fmt.audio.i_bitspersample;
  216.     if( i_cat == AU_CAT_UNKNOWN || i_cat == AU_CAT_ADPCM )
  217.     {
  218.         p_sys->i_frame_size = 0;
  219.         p_sys->i_frame_length = 0;
  220.         msg_Err( p_demux, "unsupported codec/type (Please report it)" );
  221.         free( p_sys );
  222.         return VLC_EGENERIC;
  223.     }
  224.     /* add the es */
  225.     p_sys->es = es_out_Add( p_demux->out, &p_sys->fmt );
  226.     /* calculate 50ms frame size/time */
  227.     i_samples = __MAX( p_sys->fmt.audio.i_rate / 20, 1 );
  228.     p_sys->i_frame_size = i_samples * p_sys->fmt.audio.i_channels *
  229.                           ( (p_sys->fmt.audio.i_bitspersample + 7) / 8 );
  230.     if( p_sys->fmt.audio.i_blockalign > 0 )
  231.     {
  232.         if( ( i_modulo = p_sys->i_frame_size % p_sys->fmt.audio.i_blockalign ) != 0 )
  233.         {
  234.             p_sys->i_frame_size += p_sys->fmt.audio.i_blockalign - i_modulo;
  235.         }
  236.     }
  237.     p_sys->i_frame_length = (mtime_t)1000000 *
  238.                             (mtime_t)i_samples /
  239.                             (mtime_t)p_sys->fmt.audio.i_rate;
  240.     return VLC_SUCCESS;
  241. }
  242. /*****************************************************************************
  243.  * Demux: read packet and send them to decoders
  244.  *****************************************************************************
  245.  * Returns -1 in case of error, 0 in case of EOF, 1 otherwise
  246.  *****************************************************************************/
  247. static int Demux( demux_t *p_demux )
  248. {
  249.     demux_sys_t *p_sys = p_demux->p_sys;
  250.     block_t     *p_block;
  251.     /* set PCR */
  252.     es_out_Control( p_demux->out, ES_OUT_SET_PCR, p_sys->i_time );
  253.     if( ( p_block = stream_Block( p_demux->s, p_sys->i_frame_size ) ) == NULL )
  254.     {
  255.         msg_Warn( p_demux, "cannot read data" );
  256.         return 0;
  257.     }
  258.     p_block->i_dts =
  259.     p_block->i_pts = p_sys->i_time;
  260.     es_out_Send( p_demux->out, p_sys->es, p_block );
  261.     p_sys->i_time += p_sys->i_frame_length;
  262.     return 1;
  263. }
  264. /*****************************************************************************
  265.  * Close: frees unused data
  266.  *****************************************************************************/
  267. static void Close( vlc_object_t * p_this )
  268. {
  269.     demux_t     *p_demux = (demux_t*)p_this;
  270.     demux_sys_t *p_sys = p_demux->p_sys;
  271.     free( p_sys );
  272. }
  273. /*****************************************************************************
  274.  * Control:
  275.  *****************************************************************************/
  276. static int Control( demux_t *p_demux, int i_query, va_list args )
  277. {
  278.     demux_sys_t *p_sys = p_demux->p_sys;
  279.     return demux_vaControlHelper( p_demux->s, p_sys->i_header_size, -1,
  280.                                    p_sys->fmt.i_bitrate, p_sys->fmt.audio.i_blockalign,
  281.                                    i_query, args );
  282. }