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

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * mod.c: MOD file demuxer (using libmodplug)
  3.  *****************************************************************************
  4.  * Copyright (C) 2004-2009 the VideoLAN team
  5.  * $Id: 745e2673c326014ba357692a814028e1e9c7106f $
  6.  *
  7.  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
  8.  * Konstanty Bialkowski <konstanty@ieee.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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  23.  *****************************************************************************/
  24. /*****************************************************************************
  25.  * Preamble
  26.  *****************************************************************************/
  27. #ifdef HAVE_CONFIG_H
  28. # include "config.h"
  29. #endif
  30. #include <vlc_common.h>
  31. #include <vlc_plugin.h>
  32. #include <vlc_demux.h>
  33. #include <vlc_meta.h>
  34. #include <assert.h>
  35. #include <libmodplug/modplug.h>
  36. /* TODO:
  37.  *  - extend demux control to query meta data (demuxer should NEVER touch
  38.  *      playlist itself)
  39.  *  - FIXME test endian of samples
  40.  *  - ...
  41.  */
  42. /*****************************************************************************
  43.  * Module descriptor
  44.  *****************************************************************************/
  45. static int  Open    ( vlc_object_t * );
  46. static void Close  ( vlc_object_t * );
  47. #define NOISE_LONGTEXT N_("Enable noise reduction algorithm.")
  48. #define REVERB_LONGTEXT N_("Enable reverberation" )
  49. #define REVERB_LEVEL_LONGTEXT N_( "Reverberation level (from 0 " 
  50.                 "to 100, default value is 0)." )
  51. #define REVERB_DELAY_LONGTEXT N_("Reverberation delay, in ms." 
  52.                 " Usual values are from to 40 to 200ms." )
  53. #define MEGABASS_LONGTEXT N_( "Enable megabass mode" )
  54. #define MEGABASS_LEVEL_LONGTEXT N_("Megabass mode level (from 0 to 100, " 
  55.                 "default value is 0)." )
  56. #define MEGABASS_RANGE_LONGTEXT N_("Megabass mode cutoff frequency, in Hz. " 
  57.                 "This is the maximum frequency for which the megabass " 
  58.                 "effect applies. Valid values are from 10 to 100 Hz." )
  59. #define SURROUND_LEVEL_LONGTEXT N_( "Surround effect level (from 0 to 100, " 
  60.                 "default value is 0)." )
  61. #define SURROUND_DELAY_LONGTEXT N_("Surround delay, in ms. Usual values are " 
  62.                 "from 5 to 40 ms." )
  63. vlc_module_begin ()
  64.     set_shortname( "MOD")
  65.     set_description( N_("MOD demuxer (libmodplug)" ) )
  66.     set_capability( "demux", 10 )
  67.     set_category( CAT_INPUT )
  68.     set_subcategory( SUBCAT_INPUT_DEMUX )
  69.     add_bool( "mod-noisereduction", true, NULL, N_("Noise reduction"),
  70.               NOISE_LONGTEXT, false )
  71.     add_bool( "mod-reverb", false, NULL, N_("Reverb"),
  72.               REVERB_LONGTEXT, false )
  73.     add_integer_with_range( "mod-reverb-level", 0, 0, 100, NULL,
  74.              N_("Reverberation level"), REVERB_LEVEL_LONGTEXT, true )
  75.     add_integer_with_range( "mod-reverb-delay", 40, 0, 1000, NULL,
  76.              N_("Reverberation delay"), REVERB_DELAY_LONGTEXT, true )
  77.     add_bool( "mod-megabass", false, NULL, N_("Mega bass"),
  78.                     MEGABASS_LONGTEXT, false )
  79.     add_integer_with_range( "mod-megabass-level", 0, 0, 100, NULL,
  80.               N_("Mega bass level"), MEGABASS_LEVEL_LONGTEXT, true )
  81.     add_integer_with_range( "mod-megabass-range", 10, 10, 100, NULL,
  82.               N_("Mega bass cutoff"), MEGABASS_RANGE_LONGTEXT, true )
  83.     add_bool( "mod-surround", false, NULL, N_("Surround"), N_("Surround"),
  84.                false )
  85.     add_integer_with_range( "mod-surround-level", 0, 0, 100, NULL,
  86.               N_("Surround level"), SURROUND_LEVEL_LONGTEXT, true )
  87.     add_integer_with_range( "mod-surround-delay", 5, 0, 1000, NULL,
  88.               N_("Surround delay (ms)"), SURROUND_DELAY_LONGTEXT, true )
  89.     set_callbacks( Open, Close )
  90.     add_shortcut( "mod" )
  91. vlc_module_end ()
  92. /*****************************************************************************
  93.  * Local prototypes
  94.  *****************************************************************************/
  95. struct demux_sys_t
  96. {
  97.     es_format_t  fmt;
  98.     es_out_id_t *es;
  99.     date_t      pts;
  100.     int64_t     i_length;
  101.     int         i_data;
  102.     uint8_t     *p_data;
  103.     ModPlugFile *f;
  104. };
  105. static int Demux  ( demux_t *p_demux );
  106. static int Control( demux_t *p_demux, int i_query, va_list args );
  107. static int Validate( demux_t *p_demux, const char *psz_ext );
  108. static const char *ppsz_mod_ext[] =
  109. {
  110.     "mod", "s3m", "xm",  "it",  "669", "amf", "ams", "dbm", "dmf", "dsm",
  111.     "far", "mdl", "med", "mtm", "okt", "ptm", "stm", "ult", "umx", "mt2",
  112.     "psm", "abc", NULL
  113. };
  114. /* We load the complete file in memory, put a higher bound
  115.  * of 500 Mo (which is really big anyway) */
  116. #define MOD_MAX_FILE_SIZE (500*1000*1000)
  117. /*****************************************************************************
  118.  * Open
  119.  *****************************************************************************/
  120. static int Open( vlc_object_t *p_this )
  121. {
  122.     demux_t     *p_demux = (demux_t*)p_this;
  123.     demux_sys_t *p_sys;
  124.     ModPlug_Settings settings;
  125.     /* We accept file based on extension match */
  126.     if( !p_demux->b_force )
  127.     {
  128.         const char *psz_ext = strrchr( p_demux->psz_path, '.' );
  129.         int i;
  130.         if( !psz_ext )
  131.             return VLC_EGENERIC;
  132.         psz_ext++;  /* skip . */
  133.         for( i = 0; ppsz_mod_ext[i] != NULL; i++ )
  134.         {
  135.             if( !strcasecmp( psz_ext, ppsz_mod_ext[i] ) )
  136.                 break;
  137.         }
  138.         if( ppsz_mod_ext[i] == NULL )
  139.             return VLC_EGENERIC;
  140.         if( Validate( p_demux, ppsz_mod_ext[i] ) )
  141.         {
  142.             msg_Warn( p_demux, "MOD validation failed (ext=%s)", ppsz_mod_ext[i]);
  143.             return VLC_EGENERIC;
  144.         }
  145.         msg_Dbg( p_demux, "running MOD demuxer (ext=%s)", ppsz_mod_ext[i] );
  146.     }
  147.     const int64_t i_size = stream_Size( p_demux->s );
  148.     if( i_size <= 0 || i_size >= MOD_MAX_FILE_SIZE )
  149.         return VLC_EGENERIC;
  150.     /* Fill p_demux field */
  151.     p_demux->pf_demux = Demux;
  152.     p_demux->pf_control = Control;
  153.     p_demux->p_sys = p_sys = malloc( sizeof( *p_sys ) );
  154.     if( !p_sys )
  155.         return VLC_ENOMEM;
  156.     msg_Dbg( p_demux, "loading complete file (could be long)" );
  157.     p_sys->i_data = i_size;
  158.     p_sys->p_data = malloc( p_sys->i_data );
  159.     if( p_sys->p_data )
  160.         p_sys->i_data = stream_Read( p_demux->s, p_sys->p_data, p_sys->i_data );
  161.     if( p_sys->i_data <= 0 || !p_sys->p_data )
  162.     {
  163.         msg_Err( p_demux, "failed to read the complete file" );
  164.         free( p_sys->p_data );
  165.         free( p_sys );
  166.         return VLC_EGENERIC;
  167.     }
  168.     /* Configure modplug before loading the file */
  169.     ModPlug_GetSettings( &settings );
  170.     settings.mFlags = MODPLUG_ENABLE_OVERSAMPLING;
  171.     settings.mChannels = 2;
  172.     settings.mBits = 16;
  173.     settings.mFrequency = 44100;
  174.     settings.mResamplingMode = MODPLUG_RESAMPLE_FIR;
  175.     if( var_CreateGetBool( p_demux, "mod-noisereduction" ) )
  176.         settings.mFlags |= MODPLUG_ENABLE_NOISE_REDUCTION;
  177.     if( var_CreateGetBool( p_demux, "mod-reverb" ) )
  178.         settings.mFlags |= MODPLUG_ENABLE_REVERB;
  179.     settings.mReverbDepth = var_CreateGetInteger( p_demux, "mod-reverb-level" );
  180.     settings.mReverbDelay = var_CreateGetInteger( p_demux, "mod-reverb-delay" );
  181.     if( var_CreateGetBool( p_demux, "mod-megabass" ) )
  182.         settings.mFlags |= MODPLUG_ENABLE_MEGABASS;
  183.     settings.mBassAmount = var_CreateGetInteger( p_demux, "mod-megabass-level" );
  184.     settings.mBassRange = var_CreateGetInteger( p_demux, "mod-megabass-range" );
  185.     if( var_CreateGetBool( p_demux, "mod-surround" ) )
  186.         settings.mFlags |= MODPLUG_ENABLE_SURROUND;
  187.     settings.mSurroundDepth = var_CreateGetInteger( p_demux, "mod-surround-level" );
  188.     settings.mSurroundDelay = var_CreateGetInteger( p_demux, "mod-surround-delay" );
  189.     ModPlug_SetSettings( &settings );
  190.     if( ( p_sys->f = ModPlug_Load( p_sys->p_data, p_sys->i_data ) ) == NULL )
  191.     {
  192.         msg_Err( p_demux, "failed to understand the file" );
  193.         /* we try to seek to recover for other plugin */
  194.         stream_Seek( p_demux->s, 0 );
  195.         free( p_sys->p_data );
  196.         free( p_sys );
  197.         return VLC_EGENERIC;
  198.     }
  199.     /* init time */
  200.     date_Init( &p_sys->pts, settings.mFrequency, 1 );
  201.     date_Set( &p_sys->pts, 1 );
  202.     p_sys->i_length = ModPlug_GetLength( p_sys->f ) * INT64_C(1000);
  203.     msg_Dbg( p_demux, "MOD loaded name=%s lenght=%"PRId64"ms",
  204.              ModPlug_GetName( p_sys->f ),
  205.              p_sys->i_length );
  206. #ifdef WORDS_BIGENDIAN
  207.     es_format_Init( &p_sys->fmt, AUDIO_ES, VLC_FOURCC( 't', 'w', 'o', 's' ) );
  208. #else
  209.     es_format_Init( &p_sys->fmt, AUDIO_ES, VLC_FOURCC( 'a', 'r', 'a', 'w' ) );
  210. #endif
  211.     p_sys->fmt.audio.i_rate = settings.mFrequency;
  212.     p_sys->fmt.audio.i_channels = settings.mChannels;
  213.     p_sys->fmt.audio.i_bitspersample = settings.mBits;
  214.     p_sys->es = es_out_Add( p_demux->out, &p_sys->fmt );
  215.     return VLC_SUCCESS;
  216. }
  217. /*****************************************************************************
  218.  * Close
  219.  *****************************************************************************/
  220. static void Close( vlc_object_t *p_this )
  221. {
  222.     demux_t     *p_demux = (demux_t*)p_this;
  223.     demux_sys_t *p_sys = p_demux->p_sys;
  224.     ModPlug_Unload( p_sys->f );
  225.     free( p_sys->p_data );
  226.     free( p_sys );
  227. }
  228. /*****************************************************************************
  229.  * Demux:
  230.  *****************************************************************************/
  231. static int Demux( demux_t *p_demux )
  232. {
  233.     demux_sys_t *p_sys = p_demux->p_sys;
  234.     block_t     *p_frame;
  235.     const int i_bk = ( p_sys->fmt.audio.i_bitspersample / 8 ) *
  236.                        p_sys->fmt.audio.i_channels;
  237.     p_frame = block_New( p_demux, p_sys->fmt.audio.i_rate / 10 * i_bk );
  238.     if( !p_frame )
  239.         return -1;
  240.     const int i_read = ModPlug_Read( p_sys->f, p_frame->p_buffer, p_frame->i_buffer );
  241.     if( i_read <= 0 )
  242.     {
  243.         /* EOF */
  244.         block_Release( p_frame );
  245.         return 0;
  246.     }
  247.     p_frame->i_buffer = i_read;
  248.     p_frame->i_dts =
  249.     p_frame->i_pts = date_Increment( &p_sys->pts, p_frame->i_buffer / i_bk );
  250.     /* Set PCR */
  251.     es_out_Control( p_demux->out, ES_OUT_SET_PCR, p_frame->i_pts );
  252.     /* Send data */
  253.     es_out_Send( p_demux->out, p_sys->es, p_frame );
  254.     return 1;
  255. }
  256. /*****************************************************************************
  257.  * Control:
  258.  *****************************************************************************/
  259. static int Control( demux_t *p_demux, int i_query, va_list args )
  260. {
  261.     demux_sys_t *p_sys = p_demux->p_sys;
  262.     double f, *pf;
  263.     int64_t i64, *pi64;
  264.     switch( i_query )
  265.     {
  266.     case DEMUX_GET_POSITION:
  267.         pf = (double*) va_arg( args, double* );
  268.         if( p_sys->i_length > 0 )
  269.         {
  270.             *pf = (double)date_Get( &p_sys->pts ) / (double)p_sys->i_length;
  271.             return VLC_SUCCESS;
  272.         }
  273.         return VLC_EGENERIC;
  274.     case DEMUX_SET_POSITION:
  275.         f = (double) va_arg( args, double );
  276.         i64 = f * p_sys->i_length;
  277.         if( i64 >= 0 && i64 <= p_sys->i_length )
  278.         {
  279.             ModPlug_Seek( p_sys->f, i64 / 1000 );
  280.             date_Set( &p_sys->pts, i64 + 1 );
  281.             return VLC_SUCCESS;
  282.         }
  283.         return VLC_EGENERIC;
  284.     case DEMUX_GET_TIME:
  285.         pi64 = (int64_t*)va_arg( args, int64_t * );
  286.         *pi64 = date_Get( &p_sys->pts );
  287.         return VLC_SUCCESS;
  288.     case DEMUX_GET_LENGTH:
  289.         pi64 = (int64_t*)va_arg( args, int64_t * );
  290.         *pi64 = p_sys->i_length;
  291.         return VLC_SUCCESS;
  292.     case DEMUX_SET_TIME:
  293.         i64 = (int64_t)va_arg( args, int64_t );
  294.         if( i64 >= 0 && i64 <= p_sys->i_length )
  295.         {
  296.             ModPlug_Seek( p_sys->f, i64 / 1000 );
  297.             date_Set( &p_sys->pts, i64 + 1 );
  298.             return VLC_SUCCESS;
  299.         }
  300.         return VLC_EGENERIC;
  301.     case DEMUX_HAS_UNSUPPORTED_META:
  302.     {
  303.         bool *pb_bool = (bool*)va_arg( args, bool* );
  304.         *pb_bool = false; /* FIXME I am not sure of this one */
  305.         return VLC_SUCCESS;
  306.     }
  307.     case DEMUX_GET_META:
  308.     {
  309.         vlc_meta_t *p_meta = (vlc_meta_t *)va_arg( args, vlc_meta_t* );
  310.         unsigned i_num_samples = ModPlug_NumSamples( p_sys->f ),
  311.                  i_num_instruments = ModPlug_NumInstruments( p_sys->f );
  312.         unsigned i_num_patterns = ModPlug_NumPatterns( p_sys->f ),
  313.                  i_num_channels = ModPlug_NumChannels( p_sys->f );
  314.         //      unsigned modType = ModPlug_GetModuleType( p_sys->f );
  315.         char psz_temp[2048]; /* 32 * 240 max, but only need start  */
  316.         char *psz_module_info, *psz_instrument_info;
  317.         unsigned i_temp_index = 0;
  318.         const char *psz_name = ModPlug_GetName( p_sys->f );
  319.         if( psz_name && *psz_name )
  320.             vlc_meta_SetTitle( p_meta, psz_name );
  321.         /* Comment field from artist - not in every type of MOD */
  322.         psz_name = ModPlug_GetMessage( p_sys->f );
  323.         if( psz_name && *psz_name )
  324.             vlc_meta_SetDescription( p_meta, psz_name );
  325.         /* Instruments only in newer MODs - so don't show if 0 */
  326.         if( asprintf( &psz_instrument_info, ", %i Instruments",
  327.                       i_num_instruments ) >= 0 )
  328.         {
  329.             if( asprintf( &psz_module_info,
  330.                           "%i Channels, %i Patternsn"
  331.                           "%i Samples%sn",
  332.                           i_num_channels, i_num_patterns, i_num_samples,
  333.                           ( i_num_instruments ? psz_instrument_info : "" ) ) >= 0 )
  334.             {
  335.                 vlc_meta_AddExtra( p_meta, "Module Information",
  336.                                    psz_module_info );
  337.                 free( psz_module_info );
  338.             }
  339.             free( psz_instrument_info );
  340.         }
  341.         /* Make list of instruments (XM, IT, etc) */
  342.         if( i_num_instruments )
  343.         {
  344.             i_temp_index = 0;
  345.             for( int i = 0; i < i_num_instruments && i_temp_index < sizeof(psz_temp); i++ )
  346.             {
  347.                 char lBuffer[33];
  348.                 ModPlug_InstrumentName( p_sys->f, i, lBuffer );
  349.                 if ( !lBuffer[0] ) continue; // don't add empty fields.
  350.                 i_temp_index += snprintf( &psz_temp[i_temp_index], sizeof(psz_temp) - i_temp_index, "%sn", lBuffer );
  351.             }
  352.             vlc_meta_AddExtra( p_meta, "Instruments", psz_temp );
  353.         }
  354.         /* Make list of samples */
  355.         for( int i = 0; i < i_num_samples && i_temp_index < sizeof(psz_temp); i++ )
  356.         {
  357.             char psz_buffer[33];
  358.             ModPlug_SampleName( p_sys->f, i, psz_buffer );
  359.             if ( !psz_buffer[0] ) continue; // don't add empty fields.
  360.             i_temp_index += snprintf( &psz_temp[i_temp_index], sizeof(psz_temp) - i_temp_index, "%sn", psz_buffer );
  361.         }
  362.         vlc_meta_AddExtra( p_meta, "Samples", psz_temp );
  363.         return VLC_SUCCESS;
  364.     }
  365.     case DEMUX_GET_FPS: /* meaningless */
  366.     default:
  367.         return VLC_EGENERIC;
  368.     }
  369. }
  370. /*****************************************************************************
  371.  * Validate: try to ensure it is really a mod file.
  372.  * The tests are not robust enough to replace extension checks in the general
  373.  * cases.
  374.  * TODO: maybe it should return a score, which will be used to bypass the
  375.  * extension checks when high enough.
  376.  *****************************************************************************/
  377. static int Validate( demux_t *p_demux, const char *psz_ext )
  378. {
  379.     static const struct
  380.     {
  381.         int i_offset;
  382.         const char *psz_marker;
  383.     } p_marker[] = {
  384.         {  0, "ziRCONia" },             /* MMCMP files */
  385.         {  0, "Extended Module" },      /* XM */
  386.         { 44, "SCRM" },                 /* S3M */
  387.         {  0, "IMPM" },                 /* IT */
  388.         {  0, "MThd" },                 /* MID */
  389.         {  0, "GF1PATCH110" },          /* PAT */
  390.         { 20, "!SCREAM!" },             /* STM */
  391.         { 20, "!Scream!" },             /* STM */
  392.         { 20, "BMOD2STM" },             /* STM */
  393.         {  0, "MMD0" },                 /* MED v0 */
  394.         {  0, "MMD1" },                 /* MED v1 */
  395.         {  0, "MMD2" },                 /* MED v2 */
  396.         {  0, "MMD3" },                 /* MED v3 */
  397.         {  0, "MTM" },                  /* MTM */
  398.         {  0, "DMDL" },                 /* MDL */
  399.         {  0, "DBM0" },                 /* DBM */
  400.         {  0, "if" },                   /* 669 */
  401.         {  0, "JN" },                   /* 669 */
  402.         {  0, "FARxfe" },              /* FAR */
  403.         {  0, "Extreme" },              /* AMS */
  404.         {  0, "OKTASONGCMOD" },         /* OKT */
  405.         { 44, "PTMF" },                 /* PTM */
  406.         {  0, "MAS_UTrack_V00" },       /* Ult */
  407.         {  0, "DDMF" },                 /* DMF */
  408.         {  8, "DSMFSONG" },             /* DSM */
  409.         {  0, "xc1x83x2ax9e" },     /* UMX */
  410.         {  0, "ASYLUM Music Format V1.0" }, /* AMF Type 0 */
  411.         {  0, "AMF" },                  /* AMF */
  412.         {  0, "PSMxfe" },              /* PSM */
  413.         {  0, "PSM " },                 /* PSM */
  414.         {  0, "MT20" },                 /* MT2 */
  415.         { 1080, "M.K." },               /* MOD */
  416.         { 1080, "M!K!" },
  417.         { 1080, "M&K!" },
  418.         { 1080, "N.T." },
  419.         { 1080, "CD81" },
  420.         { 1080, "OKTA" },
  421.         { 1080, "16CN" },
  422.         { 1080, "32CN" },
  423.         { 1080, "FLT" },
  424.         { 1080, "TDZ" },
  425.         { 1081, "CHN" },
  426.         { 1082, "CH" },
  427.         {  -1, NULL }
  428.     };
  429.     const uint8_t *p_peek;
  430.     const int i_peek = stream_Peek( p_demux->s, &p_peek, 2048 );
  431.     if( i_peek < 4 )
  432.         return VLC_EGENERIC;
  433.     for( int i = 0; p_marker[i].i_offset >= 0; i++ )
  434.     {
  435.         const char *psz_marker = p_marker[i].psz_marker;
  436.         const int i_size = strlen( psz_marker );
  437.         const int i_offset = p_marker[i].i_offset;
  438.         if( i_peek < i_offset + i_size )
  439.             continue;
  440.         if( !memcmp( &p_peek[i_offset], psz_marker, i_size ) )
  441.             return VLC_SUCCESS;
  442.     }
  443.     /* The only two format left untested are ABC and MOD(old version)
  444.      * ant they are difficult to test :( */
  445.     /* Check for ABC
  446.      * TODO i_peek = 2048 is too big for such files */
  447.     if( !strcasecmp( psz_ext, "abc" ) )
  448.     {
  449.         bool b_k = false;
  450.         bool b_tx = false;
  451.         for( int i = 0; i < i_peek-1; i++ )
  452.         {
  453.             b_k |= p_peek[i+0] == 'K' && p_peek[i+1] == ':';
  454.             b_tx |= ( p_peek[i+0] == 'X' || p_peek[i+0] == 'T') && p_peek[i+1] == ':';
  455.         }
  456.         if( !b_k || !b_tx )
  457.             return VLC_EGENERIC;
  458.         return VLC_SUCCESS;
  459.     }
  460.     /* Check for MOD */
  461.     if( !strcasecmp( psz_ext, "mod" ) && i_peek >= 20 + 15 * 30 )
  462.     {
  463.         /* Check that the name is correctly null padded */
  464.         const uint8_t *p = memchr( p_peek, '', 20 );
  465.         if( p )
  466.         {
  467.             for( ; p < &p_peek[20]; p++ )
  468.             {
  469.                 if( *p )
  470.                     return VLC_EGENERIC;
  471.             }
  472.         }
  473.         for( int i = 0; i < 15; i++ )
  474.         {
  475.             const uint8_t *p_sample = &p_peek[20 + i*30];
  476.             /* Check correct null padding */
  477.             const uint8_t *p = memchr( &p_sample[0], '', 22 );
  478.             if( p )
  479.             {
  480.                 for( ; p < &p_sample[22]; p++ )
  481.                 {
  482.                     if( *p )
  483.                         return VLC_EGENERIC;
  484.                 }
  485.             }
  486.             if( p_sample[25] > 64 ) /* Volume value */
  487.                 return VLC_EGENERIC;
  488.         }
  489.         return VLC_SUCCESS;
  490.     }
  491.     return VLC_EGENERIC;
  492. }