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

多媒体

开发平台:

MultiPlatform

  1. /*****************************************************************************
  2.  * copy.c
  3.  *****************************************************************************
  4.  * Copyright (C) 2001, 2002 VideoLAN
  5.  * $Id: copy.c 7812 2004-05-29 14:00:32Z fenrir $
  6.  *
  7.  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
  8.  *          Eric Petit <titer@videolan.org>
  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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
  23.  *****************************************************************************/
  24. /*****************************************************************************
  25.  * Preamble
  26.  *****************************************************************************/
  27. #include <stdlib.h>                                      /* malloc(), free() */
  28. #include <vlc/vlc.h>
  29. #include <vlc/decoder.h>
  30. #include <vlc/input.h>
  31. /*****************************************************************************
  32.  * Module descriptor
  33.  *****************************************************************************/
  34. static int  Open ( vlc_object_t * );
  35. static void Close( vlc_object_t * );
  36. vlc_module_begin();
  37.     set_description( _("Copy packetizer") );
  38.     set_capability( "packetizer", 1 );
  39.     set_callbacks( Open, Close );
  40. vlc_module_end();
  41. /*****************************************************************************
  42.  * Local prototypes
  43.  *****************************************************************************/
  44. struct decoder_sys_t
  45. {
  46.     block_t *p_block;
  47. };
  48. static block_t *Packetize   ( decoder_t *, block_t ** );
  49. static block_t *PacketizeSub( decoder_t *, block_t ** );
  50. /*****************************************************************************
  51.  * Open: probe the packetizer and return score
  52.  *****************************************************************************
  53.  * Tries to launch a decoder and return score so that the interface is able
  54.  * to choose.
  55.  *****************************************************************************/
  56. static int Open( vlc_object_t *p_this )
  57. {
  58.     decoder_t     *p_dec = (decoder_t*)p_this;
  59.     decoder_sys_t *p_sys;
  60.     if( p_dec->fmt_in.i_cat != AUDIO_ES &&
  61.         p_dec->fmt_in.i_cat != VIDEO_ES &&
  62.         p_dec->fmt_in.i_cat != SPU_ES )
  63.     {
  64.         msg_Err( p_dec, "invalid ES type" );
  65.         return VLC_EGENERIC;
  66.     }
  67.     if( p_dec->fmt_in.i_cat == SPU_ES )
  68.         p_dec->pf_packetize = PacketizeSub;
  69.     else
  70.         p_dec->pf_packetize = Packetize;
  71.     /* Create the output format */
  72.     es_format_Copy( &p_dec->fmt_out, &p_dec->fmt_in );
  73.     /* Fix the value of the fourcc */
  74.     switch( p_dec->fmt_in.i_codec )
  75.     {
  76.         /* video */
  77.         case VLC_FOURCC( 'm', '4', 's', '2'):
  78.         case VLC_FOURCC( 'M', '4', 'S', '2'):
  79.         case VLC_FOURCC( 'm', 'p', '4', 's'):
  80.         case VLC_FOURCC( 'M', 'P', '4', 'S'):
  81.         case VLC_FOURCC( 'D', 'I', 'V', 'X'):
  82.         case VLC_FOURCC( 'd', 'i', 'v', 'x'):
  83.         case VLC_FOURCC( 'X', 'V', 'I', 'D'):
  84.         case VLC_FOURCC( 'X', 'v', 'i', 'D'):
  85.         case VLC_FOURCC( 'x', 'v', 'i', 'd'):
  86.         case VLC_FOURCC( 'D', 'X', '5', '0'):
  87.         case VLC_FOURCC( 0x04, 0,   0,   0):
  88.         case VLC_FOURCC( '3', 'I', 'V', '2'):
  89.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'm', 'p', '4', 'v');
  90.             break;
  91.         case VLC_FOURCC( 'm', 'p', 'g', '1' ):
  92.         case VLC_FOURCC( 'm', 'p', 'g', '2' ):
  93.         case VLC_FOURCC( 'm', 'p', '1', 'v' ):
  94.         case VLC_FOURCC( 'm', 'p', '2', 'v' ):
  95.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'm', 'p', 'g', 'v' );
  96.             break;
  97.         case VLC_FOURCC( 'd', 'i', 'v', '1' ):
  98.         case VLC_FOURCC( 'M', 'P', 'G', '4' ):
  99.         case VLC_FOURCC( 'm', 'p', 'g', '4' ):
  100.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'D', 'I', 'V', '1' );
  101.             break;
  102.         case VLC_FOURCC( 'd', 'i', 'v', '2' ):
  103.         case VLC_FOURCC( 'M', 'P', '4', '2' ):
  104.         case VLC_FOURCC( 'm', 'p', '4', '2' ):
  105.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'D', 'I', 'V', '2' );
  106.             break;
  107.         case VLC_FOURCC( 'd', 'i', 'v', '3' ):
  108.         case VLC_FOURCC( 'd', 'i', 'v', '4' ):
  109.         case VLC_FOURCC( 'D', 'I', 'V', '4' ):
  110.         case VLC_FOURCC( 'd', 'i', 'v', '5' ):
  111.         case VLC_FOURCC( 'D', 'I', 'V', '5' ):
  112.         case VLC_FOURCC( 'd', 'i', 'v', '6' ):
  113.         case VLC_FOURCC( 'D', 'I', 'V', '6' ):
  114.         case VLC_FOURCC( 'M', 'P', '4', '3' ):
  115.         case VLC_FOURCC( 'm', 'p', '4', '3' ):
  116.         case VLC_FOURCC( 'm', 'p', 'g', '3' ):
  117.         case VLC_FOURCC( 'M', 'P', 'G', '3' ):
  118.         case VLC_FOURCC( 'A', 'P', '4', '1' ):
  119.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'D', 'I', 'V', '3' );
  120.             break;
  121.         case VLC_FOURCC( 'h', '2', '6', '3' ):
  122.         case VLC_FOURCC( 'U', '2', '6', '3' ):
  123.         case VLC_FOURCC( 'u', '2', '6', '3' ):
  124.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'H', '2', '6', '3' );
  125.             break;
  126.         case VLC_FOURCC( 'i', '2', '6', '3' ):
  127.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'I', '2', '6', '3' );
  128.             break;
  129.         case VLC_FOURCC( 'm', 'j', 'p', 'g' ):
  130.         case VLC_FOURCC( 'm', 'j', 'p', 'a' ):
  131.         case VLC_FOURCC( 'j', 'p', 'e', 'g' ):
  132.         case VLC_FOURCC( 'J', 'P', 'E', 'G' ):
  133.         case VLC_FOURCC( 'J', 'F', 'I', 'F' ):
  134.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'M', 'J', 'P', 'G' );
  135.             break;
  136.         case VLC_FOURCC( 'd', 'v', 's', 'd' ):
  137.         case VLC_FOURCC( 'D', 'V', 'S', 'D' ):
  138.         case VLC_FOURCC( 'd', 'v', 'h', 'd' ):
  139.             p_dec->fmt_out.i_codec = VLC_FOURCC( 'd', 'v', 's', 'l' );
  140.             break;
  141.         /* audio */
  142.         case VLC_FOURCC( 'a', 'r', 'a', 'w' ):
  143.             switch( ( p_dec->fmt_in.audio.i_bitspersample + 7 ) / 8 )
  144.             {
  145.                 case 1:
  146.                     p_dec->fmt_out.i_codec = VLC_FOURCC('u','8',' ',' ');
  147.                     break;
  148.                 case 2:
  149.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','1','6','l');
  150.                     break;
  151.                 case 3:
  152.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','2','4','l');
  153.                     break;
  154.                 case 4:
  155.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','3','2','l');
  156.                     break;
  157.                 default:
  158.                     msg_Err( p_dec, "unknown raw audio sample size" );
  159.                     return VLC_EGENERIC;
  160.             }
  161.             break;
  162.         case VLC_FOURCC( 't', 'w', 'o', 's' ):
  163.             switch( ( p_dec->fmt_in.audio.i_bitspersample + 7 ) / 8 )
  164.             {
  165.                 case 1:
  166.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','8',' ',' ');
  167.                     break;
  168.                 case 2:
  169.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','1','6','b');
  170.                     break;
  171.                 case 3:
  172.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','2','4','b');
  173.                     break;
  174.                 case 4:
  175.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','3','2','b');
  176.                     break;
  177.                 default:
  178.                     msg_Err( p_dec, "unknown raw audio sample size" );
  179.                     return VLC_EGENERIC;
  180.             }
  181.             break;
  182.         case VLC_FOURCC( 's', 'o', 'w', 't' ):
  183.             switch( ( p_dec->fmt_in.audio.i_bitspersample + 7 ) / 8 )
  184.             {
  185.                 case 1:
  186.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','8',' ',' ');
  187.                     break;
  188.                 case 2:
  189.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','1','6','l');
  190.                     break;
  191.                 case 3:
  192.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','2','4','l');
  193.                     break;
  194.                 case 4:
  195.                     p_dec->fmt_out.i_codec = VLC_FOURCC('s','3','2','l');
  196.                     break;
  197.                 default:
  198.                     msg_Err( p_dec, "unknown raw audio sample size" );
  199.                     return VLC_EGENERIC;
  200.             }
  201.             break;
  202.     }
  203.     p_dec->p_sys = p_sys = malloc( sizeof( block_t ) );
  204.     p_sys->p_block    = NULL;
  205.     return VLC_SUCCESS;
  206. }
  207. /*****************************************************************************
  208.  * Close:
  209.  *****************************************************************************/
  210. static void Close( vlc_object_t *p_this )
  211. {
  212.     decoder_t     *p_dec = (decoder_t*)p_this;
  213.     if( p_dec->p_sys->p_block )
  214.     {
  215.         block_ChainRelease( p_dec->p_sys->p_block );
  216.     }
  217.     free( p_dec->p_sys );
  218. }
  219. /*****************************************************************************
  220.  * Packetize: packetize an unit (here copy a complete block )
  221.  *****************************************************************************/
  222. static block_t *Packetize ( decoder_t *p_dec, block_t **pp_block )
  223. {
  224.     block_t *p_block;
  225.     block_t *p_ret = p_dec->p_sys->p_block;
  226.     if( pp_block == NULL || *pp_block == NULL )
  227.     {
  228.         return NULL;
  229.     }
  230.     p_block = *pp_block;
  231.     *pp_block = NULL;
  232.     if( p_block->i_dts <= 0 )
  233.     {
  234.         p_block->i_dts = p_block->i_pts;
  235.     }
  236.     if( p_block->i_dts <= 0 )
  237.     {
  238.         msg_Dbg( p_dec, "need dts > 0" );
  239.         block_Release( p_block );
  240.         return NULL;
  241.     }
  242.     if( p_ret != NULL && p_block->i_pts > p_ret->i_pts )
  243.     {
  244.         p_ret->i_length = p_block->i_pts - p_ret->i_pts;
  245.     }
  246.     p_dec->p_sys->p_block = p_block;
  247.     return p_ret;
  248. }
  249. /*****************************************************************************
  250.  * PacketizeSub: packetize an unit (here copy a complete block )
  251.  *****************************************************************************/
  252. static block_t *PacketizeSub( decoder_t *p_dec, block_t **pp_block )
  253. {
  254.     block_t *p_block;
  255.     if( pp_block == NULL || *pp_block == NULL )
  256.     {
  257.         return NULL;
  258.     }
  259.     p_block = *pp_block;
  260.     *pp_block = NULL;
  261.     if( p_block->i_dts <= 0 )
  262.     {
  263.         p_block->i_dts = p_block->i_pts;
  264.     }
  265.     if( p_block->i_dts <= 0 )
  266.     {
  267.         msg_Dbg( p_dec, "need dts > 0" );
  268.         block_Release( p_block );
  269.         return NULL;
  270.     }
  271.     return p_block;
  272. }