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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * clpi.c: BluRay Disc CLPI
  3.  *****************************************************************************
  4.  * Copyright (C) 2009 the VideoLAN team
  5.  * $Id: 5f88fa4227d0c0ca038f878bda56f5222fe83503 $
  6.  *
  7.  * Authors: Laurent Aimar <fenrir _AT_ videolan _DOT_ org>
  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. #ifdef HAVE_CONFIG_H
  24. # include "config.h"
  25. #endif
  26. #include <limits.h>
  27. #include <vlc_common.h>
  28. #include <vlc_bits.h>
  29. #include "clpi.h"
  30. /* */
  31. void bd_clpi_stc_Parse( bd_clpi_stc_t *p_stc, bs_t *s )
  32. {
  33.     p_stc->i_pcr_pid = bs_read( s, 16 );
  34.     p_stc->i_packet = bs_read( s, 32 );
  35.     p_stc->i_start = bs_read( s, 32 );
  36.     p_stc->i_end = bs_read( s, 32 );
  37. }
  38. void bd_clpi_stream_Parse( bd_clpi_stream_t *p_stream, bs_t *s )
  39. {
  40.     p_stream->i_pid = bs_read( s, 16 );
  41.     const int i_length = bs_read( s, 8 );
  42.     p_stream->i_type = bs_read( s, 8 );
  43.     /* Ignore the rest */
  44.     if( i_length > 1 )
  45.         bs_skip( s, 8*i_length - 8 );
  46. }
  47. void bd_clpi_ep_map_Clean( bd_clpi_ep_map_t *p_ep_map )
  48. {
  49.     free( p_ep_map->p_ep );
  50. }
  51. int bd_clpi_ep_map_Parse( bd_clpi_ep_map_t *p_ep_map,
  52.                           bs_t *s, const int i_ep_map_start )
  53. {
  54.     p_ep_map->i_pid = bs_read( s, 16 );
  55.     bs_skip( s, 10 );
  56.     p_ep_map->i_type = bs_read( s, 4 );
  57.     const int i_coarse = bs_read( s, 16 );
  58.     const int i_fine = bs_read( s, 18 );
  59.     const uint32_t i_coarse_start = bs_read( s, 32 );
  60.     p_ep_map->i_ep = i_fine;
  61.     p_ep_map->p_ep = calloc( i_fine, sizeof(*p_ep_map->p_ep) );
  62.     if( !p_ep_map->p_ep )
  63.         return VLC_EGENERIC;
  64.     bs_t cs = *s;
  65.     bs_skip( &cs, 8*(i_ep_map_start + i_coarse_start) - bs_pos( s ) );
  66.     const uint32_t i_fine_start = bs_read( &cs, 32 );
  67.     for( int i = 0; i < i_coarse; i++ )
  68.     {
  69.         const int      i_fine_id = bs_read( &cs, 18 );
  70.         const int      i_pts = bs_read( &cs, 14 );
  71.         const uint32_t i_packet = bs_read( &cs, 32 );
  72.         for( int j = i_fine_id; j < p_ep_map->i_ep; j++ )
  73.         {
  74.             p_ep_map->p_ep[j].i_pts = (int64_t)(i_pts & ~1) << 19;
  75.             p_ep_map->p_ep[j].i_packet = i_packet & ~( (1 << 17) - 1 );
  76.         }
  77.     }
  78.     bs_t fs = *s;
  79.     bs_skip( &fs, 8*(i_ep_map_start + i_coarse_start + i_fine_start) - bs_pos( s ) );
  80.     for( int i = 0; i < i_fine; i++ )
  81.     {
  82.         const bool b_angle_point = bs_read( &fs, 1 );
  83.         bs_skip( &fs, 3 );  /* I end position offset */
  84.         const int i_pts = bs_read( &fs, 11 );
  85.         const int i_packet = bs_read( &fs, 17 );
  86.         p_ep_map->p_ep[i].b_angle_point = b_angle_point;
  87.         p_ep_map->p_ep[i].i_pts |= i_pts << 9;
  88.         p_ep_map->p_ep[i].i_packet |= i_packet;
  89.     }
  90.     return VLC_SUCCESS;
  91. }
  92. void bd_clpi_Clean( bd_clpi_t *p_clpi )
  93. {
  94.     free( p_clpi->p_stc );
  95.     free( p_clpi->p_stream );
  96.     for( int i = 0; i < p_clpi->i_ep_map; i++ )
  97.         bd_clpi_ep_map_Clean( &p_clpi->p_ep_map[i] );
  98.     free( p_clpi->p_ep_map );
  99. }
  100. int bd_clpi_Parse( bd_clpi_t *p_clpi, bs_t *s, int i_id )
  101. {
  102.     const int i_start = bs_pos( s ) / 8;
  103.     /* */
  104.     if( bs_read( s, 32 ) != 0x48444D56 )
  105.         return VLC_EGENERIC;
  106.     const uint32_t i_version = bs_read( s, 32 );
  107.     if( i_version != 0x30313030 && i_version != 0x30323030 )
  108.         return VLC_EGENERIC;
  109.     /* */
  110.     const uint32_t i_sequence_start = bs_read( s, 32 );
  111.     const uint32_t i_program_start = bs_read( s, 32 );
  112.     const uint32_t i_cpi_start = bs_read( s, 32 );
  113.     bs_skip( s, 32 );   /* mark start */
  114.     bs_skip( s, 32 );   /* extension start */
  115.     /* */
  116.     p_clpi->i_id = i_id;
  117.     /* Read sequence */
  118.     bs_t ss = *s;
  119.     bs_skip( &ss, 8 * ( i_start + i_sequence_start ) - bs_pos( s ) );
  120.     bs_skip( &ss, 32 ); /* Length */
  121.     bs_skip( &ss, 8 );
  122.     bs_skip( &ss, 8 );  /* ATC sequence count (MUST be 1 ?) */
  123.     bs_skip( &ss, 32 ); /* ATC start (MUST be 0) */
  124.     const int i_stc = bs_read( &ss, 8 );
  125.     bs_skip( &ss, 8 );  /* STC ID offset (MUST be 0 ? */
  126.     p_clpi->p_stc = calloc( i_stc, sizeof(*p_clpi->p_stc) );
  127.     for( p_clpi->i_stc = 0; p_clpi->i_stc < i_stc; p_clpi->i_stc++ )
  128.     {
  129.         if( !p_clpi->p_stc )
  130.             break;
  131.         bd_clpi_stc_Parse( &p_clpi->p_stc[p_clpi->i_stc], &ss );
  132.     }
  133.     /* Program */
  134.     bs_t ps = *s;
  135.     bs_skip( &ps, 8 * ( i_start + i_program_start ) - bs_pos( s ) );
  136.     bs_skip( &ps, 32 ); /* Length */
  137.     bs_skip( &ps, 8 );
  138.     bs_skip( &ps, 8 );  /* Program count (MUST be 1 ?) */
  139.     bs_skip( &ps, 32 ); /* Program sequence start (MUST be 0) */
  140.     p_clpi->i_pmt_pid = bs_read( &ps, 16 );
  141.     const int i_stream = bs_read( &ps, 8 );
  142.     bs_skip( &ps, 8 );  /* Group count (MUST be 1 ?) */
  143.     p_clpi->p_stream = calloc( i_stream, sizeof(*p_clpi->p_stream) );
  144.     for( p_clpi->i_stream = 0; p_clpi->i_stream < i_stream; p_clpi->i_stream++ )
  145.     {
  146.         if( !p_clpi->p_stream )
  147.             break;
  148.         bd_clpi_stream_Parse( &p_clpi->p_stream[p_clpi->i_stream], &ps );
  149.     }
  150.     /* Read CPI */
  151.     bs_t cs = *s;
  152.     bs_skip( &cs, 8 * ( i_start + i_cpi_start ) - bs_pos( s ) );
  153.     const uint32_t i_cpi_length = bs_read( &cs, 32 );
  154.     if( i_cpi_length > 0 )
  155.     {
  156.         bs_skip( &cs, 12 );
  157.         bs_skip( &cs, 4 );  /* Type (MUST be 1) */
  158.         /* EPMap */
  159.         const int i_epmap_start = bs_pos( &cs ) / 8;
  160.         bs_skip( &cs, 8 );
  161.         const int i_ep_map = bs_read( &cs, 8 );
  162.         p_clpi->p_ep_map = calloc( i_ep_map, sizeof(*p_clpi->p_ep_map) );
  163.         for( p_clpi->i_ep_map = 0; p_clpi->i_ep_map < i_ep_map; p_clpi->i_ep_map++ )
  164.         {
  165.             if( !p_clpi->p_ep_map )
  166.                 break;
  167.             if( bd_clpi_ep_map_Parse( &p_clpi->p_ep_map[p_clpi->i_ep_map],
  168.                                       &cs, i_epmap_start ) )
  169.                 break;
  170.         }
  171.     }
  172.     else
  173.     {
  174.         p_clpi->i_ep_map = 0;
  175.         p_clpi->p_ep_map = NULL;
  176.     }
  177.     return VLC_SUCCESS;
  178. }