Coverage Report

Created: 2025-07-18 07:04

/src/vlc/modules/demux/mp4/essetup.c
Line
Count
Source (jump to first uncovered line)
1
/*****************************************************************************
2
 * essetup.h: es setup from stsd and extensions parsing
3
 *****************************************************************************
4
 * Copyright (C) 2001-2004, 2010, 2014 VLC authors and VideoLAN
5
 *
6
 * Authors: Laurent Aimar <fenrir@via.ecp.fr>
7
 *
8
 * This program is free software; you can redistribute it and/or modify it
9
 * under the terms of the GNU Lesser General Public License as published by
10
 * the Free Software Foundation; either version 2.1 of the License, or
11
 * (at your option) any later version.
12
 *
13
 * This program is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16
 * GNU Lesser General Public License for more details.
17
 *
18
 * You should have received a copy of the GNU Lesser General Public License
19
 * along with this program; if not, write to the Free Software Foundation,
20
 * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
21
 *****************************************************************************/
22
23
#ifdef HAVE_CONFIG_H
24
# include "config.h"
25
#endif
26
27
#include <stdbit.h>
28
29
#include "mp4.h"
30
#include "avci.h"
31
#include "../xiph.h"
32
#include "../../packetizer/iso_color_tables.h"
33
#include "../../packetizer/mpeg4systems.h"
34
#include "qt_palette.h"
35
36
#include <vlc_demux.h>
37
#include <vlc_aout.h>
38
#include <assert.h>
39
40
41
static void CopyExtradata( const uint8_t *p_extra, size_t i_extra,
42
                           es_format_t *fmt )
43
0
{
44
0
    if( i_extra > 0 && !fmt->i_extra )
45
0
    {
46
0
        fmt->p_extra = malloc( i_extra );
47
0
        if( i_extra )
48
0
        {
49
0
            fmt->i_extra = i_extra;
50
0
            memcpy( fmt->p_extra, p_extra, i_extra );
51
0
        }
52
0
    }
53
0
}
54
55
static uint32_t GetSampleType( demux_t *p_demux, const MP4_Box_t *p_sample )
56
0
{
57
0
    const MP4_Box_t *p_frma;
58
0
    if( ( p_frma = MP4_BoxGet( p_sample, "sinf/frma" ) ) && BOXDATA(p_frma) )
59
0
    {
60
0
        msg_Warn( p_demux, "Original Format Box: %4.4s", (char *)&BOXDATA(p_frma)->i_type );
61
0
        return BOXDATA(p_frma)->i_type;
62
0
    }
63
0
    return p_sample->i_type;
64
0
}
65
66
static void SetupGlobalExtensions( const MP4_Box_t *p_sample,
67
                                   es_format_t *p_fmt )
68
0
{
69
0
    if( !p_fmt->i_bitrate )
70
0
    {
71
0
        const MP4_Box_t *p_btrt = MP4_BoxGet( p_sample, "btrt" );
72
0
        if( p_btrt && BOXDATA(p_btrt) )
73
0
        {
74
0
            p_fmt->i_bitrate = BOXDATA(p_btrt)->i_avg_bitrate;
75
0
        }
76
0
    }
77
78
0
    const MP4_Box_t *p_glbl = MP4_BoxGet( p_sample, "glbl" );
79
0
    if( p_glbl && p_glbl->data.p_binary && p_glbl->data.p_binary->p_blob )
80
0
    {
81
0
        CopyExtradata( p_glbl->data.p_binary->p_blob,
82
0
                       p_glbl->data.p_binary->i_blob, p_fmt );
83
0
    }
84
0
}
85
86
static void SetupESDS( demux_t *p_demux, const mp4_track_t *p_track,
87
                       const MP4_descriptor_decoder_config_t *p_decconfig,
88
                       es_format_t *p_fmt )
89
0
{
90
    /* First update information based on i_objectTypeIndication */
91
0
    switch( p_decconfig->i_objectProfileIndication )
92
0
    {
93
        /* Private ID */
94
0
    case( 0xe0 ): /* NeroDigital: dvd subs */
95
0
        if( p_fmt->i_cat == SPU_ES )
96
0
        {
97
0
            p_fmt->i_codec = VLC_CODEC_SPU;
98
0
            if( p_track->i_width > 0 )
99
0
                p_fmt->subs.spu.i_original_frame_width = p_track->i_width;
100
0
            if( p_track->i_height > 0 )
101
0
                p_fmt->subs.spu.i_original_frame_height = p_track->i_height;
102
0
        }
103
0
        break;
104
0
    case( 0xe1 ): /* QCelp for 3gp */
105
0
        if( p_fmt->i_cat == AUDIO_ES )
106
0
        {
107
0
            p_fmt->i_codec = VLC_CODEC_QCELP;
108
0
        }
109
0
        break;
110
111
        /* Fallback */
112
0
    default:
113
0
        if( MPEG4_get_codec_by_ObjectType( p_decconfig->i_objectProfileIndication,
114
0
                                       p_decconfig->p_decoder_specific_info,
115
0
                                       p_decconfig->i_decoder_specific_info_len,
116
0
                                       &p_fmt->i_codec,
117
0
                                       &p_fmt->i_profile ) )
118
0
            break;
119
        /* Unknown entry, but don't touch i_fourcc */
120
0
        msg_Warn( p_demux,
121
0
                  "unknown objectProfileIndication(0x%x) (Track[ID 0x%x])",
122
0
                  p_decconfig->i_objectProfileIndication,
123
0
                  p_track->i_track_ID );
124
0
        return;
125
0
    }
126
127
0
    p_fmt->i_original_fourcc = 0; /* so we don't have MP4A as original fourcc */
128
0
    p_fmt->i_bitrate = p_decconfig->i_avg_bitrate;
129
130
0
    CopyExtradata( p_decconfig->p_decoder_specific_info,
131
0
                   p_decconfig->i_decoder_specific_info_len,
132
0
                   p_fmt );
133
134
0
    if( p_fmt->i_codec == VLC_CODEC_SPU &&
135
0
            p_fmt->i_extra >= sizeof(p_fmt->subs.spu.palette) )
136
0
    {
137
0
        for (size_t i = 0; i < ARRAY_SIZE(p_fmt->subs.spu.palette); i++)
138
0
        {
139
0
            p_fmt->subs.spu.palette[i] = GetDWBE((char*)p_fmt->p_extra + i * 4);
140
0
        }
141
0
        p_fmt->subs.spu.b_palette = true;
142
0
    }
143
0
}
144
145
static int SetupRTPReceptionHintTrack( demux_t *p_demux, const mp4_track_t *p_track,
146
                                       const MP4_Box_t *p_sample, es_format_t *p_fmt,
147
                                       track_config_t *params )
148
0
{
149
0
    const uint32_t i_sample_type = GetSampleType( p_demux, p_sample );
150
0
    p_fmt->i_original_fourcc = i_sample_type;
151
152
0
    const MP4_Box_t *p_sdp = MP4_BoxGet( p_track->p_track, "udta/hnti/sdp " );
153
0
    if( !p_sdp )
154
0
    {
155
0
        msg_Err(p_demux, "Required 'sdp '-box not found");
156
0
        return 0;
157
0
    }
158
0
    char *strtok_state;
159
0
    char * pch = strtok_r(BOXDATA(p_sdp)->psz_text, " =\n", &strtok_state); /* media entry */
160
0
    if( pch && pch[0] != 'm' )
161
0
    {
162
0
        msg_Err(p_demux, "No Media entry found in SDP:%s", pch);
163
0
        return 0;
164
0
    }
165
166
0
    if( !( pch = strtok_r(NULL, " =\n", &strtok_state) ) ) /* media type */
167
0
        return 0;
168
    /* media type has already been checked */
169
0
    msg_Dbg(p_demux, "sdp: media type:%s", pch);
170
0
    if( !( pch = strtok_r(NULL, " =\n", &strtok_state) ) ) /* port */
171
0
        return 0;
172
0
    msg_Dbg(p_demux, "sdp: port:%s", pch);
173
0
    if( !( pch = strtok_r(NULL, " =\n", &strtok_state) ) ) /* protocol */
174
0
        return 0;
175
0
    msg_Dbg(p_demux, "sdp: protocol:%s", pch);
176
177
0
    if( !( pch = strtok_r(NULL, " =\n", &strtok_state) ) ) /* fmt */
178
0
        return 0;
179
180
0
    bool codec_set = false;
181
    /* process rtp types until we get an attribute field or end of sdp */
182
0
    while( pch && pch[0] != 'a' )
183
0
    {
184
0
        int rtp_payload = atoi(pch);
185
0
        msg_Dbg(p_demux, "sdp: payload type:%d", rtp_payload);
186
187
0
        if( !codec_set )
188
0
        {
189
            /* Payload types 34 and under have a set type and can be identified here */
190
0
            switch( rtp_payload )
191
0
            {
192
0
             case 3:
193
0
                p_fmt->i_codec = VLC_CODEC_GSM;
194
0
                codec_set = true;
195
0
                break;
196
0
             default:
197
0
                break;
198
0
            }
199
0
        }
200
0
        pch = strtok_r(NULL, " =\n", &strtok_state); /* attribute or additional payload type */
201
0
        if( !pch && !codec_set )
202
0
            return 0;
203
0
    }
204
205
0
    while( pch && pch[0] == 'a' )
206
0
    {
207
0
        if( !( pch = strtok_r(NULL, " :=\n", &strtok_state) ) ) /* attribute type */
208
0
            return 0;
209
0
        msg_Dbg(p_demux, "sdp: attribute type:%s", pch);
210
211
0
        if( !strcmp(pch, "rtpmap") )
212
0
        {
213
0
            if( !( pch = strtok_r(NULL, " :=\n", &strtok_state) ) ) /* payload type */
214
0
                return 0;
215
0
            msg_Dbg(p_demux, "sdp: payload type:%s", pch);
216
0
            if( !(pch = strtok_r(NULL, " /:=\n", &strtok_state) ) ) /* encoding name */
217
0
                return 0;
218
0
            msg_Dbg(p_demux, "sdp: encoding name:%s", pch);
219
220
            /* Simply adding codec recognition should work for most codecs */
221
            /* Codecs using slices need their picture constructed from sample */
222
0
            if( !strcmp(pch, "H264") )
223
0
            {
224
0
                p_fmt->i_codec = VLC_CODEC_H264;
225
                /* ******* sending AnnexB ! */
226
0
                p_fmt->b_packetized = false;
227
0
            }
228
0
            else if( !strcmp(pch, "GSM") )
229
0
            {
230
0
                p_fmt->i_codec = VLC_CODEC_GSM;
231
0
            }
232
0
            else if( !strcmp(pch, "Speex") )
233
0
            {
234
0
                p_fmt->i_codec = VLC_CODEC_SPEEX;
235
0
            }
236
0
            else if( !codec_set )
237
0
            {
238
0
                msg_Err(p_demux, "Support for codec contained in RTP \
239
0
                        Reception Hint Track RTP stream has not been added");
240
0
                return 0;
241
0
            }
242
243
0
            if( !( pch = strtok_r(NULL, " :=\n", &strtok_state) ) ) /* clock rate */
244
0
                return 0;
245
0
            int clock_rate = atoi(pch);
246
0
            msg_Dbg(p_demux, "sdp clock rate:%d", clock_rate);
247
0
            if( p_fmt->i_cat == AUDIO_ES )
248
0
                p_fmt->audio.i_rate = clock_rate;
249
0
        }
250
0
        pch = strtok_r(NULL, " =\n", &strtok_state); /* next attribute */
251
0
    }
252
253
0
    const MP4_Box_t *p_tims = MP4_BoxGet(p_sample, "tims");
254
0
    if( p_tims && BOXDATA(p_tims) && BOXDATA(p_tims)->i_timescale )
255
0
    {
256
0
        params->i_timescale_override = BOXDATA(p_tims)->i_timescale;
257
0
    }
258
0
    else
259
0
    {
260
0
        msg_Warn(p_demux, "Missing mandatory box tims");
261
0
        return 0;
262
0
    }
263
#if 0
264
    const MP4_Box_t *p_tssy = MP4_BoxGet(p_sample, "tssy");
265
    if( p_tssy && BOXDATA(p_tssy) )
266
    {
267
        /* take the 2 last bits which indicate the synchronization mode */
268
        params->sync_mode = (RTP_timstamp_synchronization_t)
269
                            BOXDATA(p_tssy)->i_reserved_timestamp_sync & 0x03;
270
    }
271
272
    const MP4_Box_t *p_tsro = MP4_BoxGet(p_sample, "tsro");
273
    if( p_tsro && BOXDATA(p_tsro) )
274
        params->i_tsro_offset = BOXDATA(p_tsro)->i_offset;
275
    else
276
        msg_Dbg(p_demux, "No tsro box present");
277
    msg_Dbg(p_demux, "setting tsro: %" PRId32, p_track->i_tsro_offset);
278
#endif
279
0
    return 1;
280
0
}
281
282
283
int SetupVideoES( demux_t *p_demux, const mp4_track_t *p_track, const MP4_Box_t *p_sample,
284
                  es_format_t *p_fmt, track_config_t *p_cfg )
285
0
{
286
0
    track_config_t trackparams, *params = &trackparams;
287
288
0
    const MP4_Box_data_sample_vide_t *p_vide = p_sample->data.p_sample_vide;
289
0
    if(!p_vide)
290
0
        return 0;
291
292
0
    const uint32_t i_sample_type = GetSampleType( p_demux, p_sample );
293
294
0
    p_fmt->video.i_width = p_vide->i_width;
295
0
    p_fmt->video.i_height = p_vide->i_height;
296
297
    /* fall on display size */
298
0
    if( p_fmt->video.i_width <= 0 )
299
0
        p_fmt->video.i_width = p_track->i_width;
300
0
    if( p_fmt->video.i_height <= 0 )
301
0
        p_fmt->video.i_height = p_track->i_height;
302
303
    /* Find out apect ratio from display size */
304
0
    if( p_track->i_width > 0 && p_track->i_height > 0 &&
305
        /* Work-around buggy muxed files */
306
0
        p_vide->i_width != p_track->i_width )
307
0
    {
308
0
        p_fmt->video.i_sar_num = p_track->i_width  * p_fmt->video.i_height;
309
0
        p_fmt->video.i_sar_den = p_track->i_height * p_fmt->video.i_width;
310
0
    }
311
312
    /* Support for cropping (eg. in H263 files) */
313
0
    p_fmt->video.i_visible_width = p_fmt->video.i_width;
314
0
    p_fmt->video.i_visible_height = p_fmt->video.i_height;
315
316
    /* Rotation */
317
0
    switch( (int)p_track->f_rotation ) {
318
0
        case 90:
319
0
            p_fmt->video.orientation = ORIENT_ROTATED_90;
320
0
            break;
321
0
        case 180:
322
0
            if (p_track->i_flip == 1) {
323
0
                p_fmt->video.orientation = ORIENT_VFLIPPED;
324
0
            } else {
325
0
                p_fmt->video.orientation = ORIENT_ROTATED_180;
326
0
            }
327
0
            break;
328
0
        case 270:
329
0
            p_fmt->video.orientation = ORIENT_ROTATED_270;
330
0
            break;
331
0
    }
332
333
    /* Flip, unless already flipped */
334
0
    if (p_track->i_flip == 1 && (int)p_track->f_rotation != 180) {
335
0
        video_transform_t transform = (video_transform_t)p_fmt->video.orientation;
336
        /* Flip first then rotate */
337
0
        p_fmt->video.orientation = ORIENT_HFLIPPED;
338
0
        video_format_TransformBy(&p_fmt->video, transform);
339
0
    }
340
341
    /* Set 360 video mode */
342
0
    p_fmt->video.projection_mode = PROJECTION_MODE_RECTANGULAR;
343
0
    const MP4_Box_t *p_uuid = MP4_BoxGet( p_track->p_track, "uuid" );
344
0
    for( ; p_uuid; p_uuid = p_uuid->p_next)
345
0
    {
346
0
        if( p_uuid->i_type == ATOM_uuid
347
0
            && !CmpUUID( &p_uuid->i_uuid, &XML360BoxUUID )
348
0
            && p_uuid->data.p_360 )
349
0
        {
350
0
            p_fmt->video.projection_mode = p_uuid->data.p_360->i_projection_mode;
351
0
            switch (p_uuid->data.p_360->e_stereo_mode)
352
0
            {
353
0
            case XML360_STEREOSCOPIC_TOP_BOTTOM:
354
0
                p_fmt->video.multiview_mode = MULTIVIEW_STEREO_TB;
355
0
                break;
356
0
            case XML360_STEREOSCOPIC_LEFT_RIGHT:
357
0
                p_fmt->video.multiview_mode = MULTIVIEW_STEREO_SBS;
358
0
                break;
359
0
            default:
360
0
                p_fmt->video.multiview_mode = MULTIVIEW_2D;
361
0
                break;
362
0
            }
363
0
        }
364
0
    }
365
366
0
    const MP4_Box_t *p_st3d = MP4_BoxGet( p_sample, "st3d" );
367
0
    if (p_st3d && BOXDATA(p_st3d))
368
0
    {
369
0
        switch( BOXDATA(p_st3d)->i_stereo_mode )
370
0
        {
371
0
        case ST3D_MONOSCOPIC:
372
0
            p_fmt->video.multiview_mode = MULTIVIEW_2D;
373
0
            break;
374
0
        case ST3D_STEREOSCOPIC_TOP_BOTTOM:
375
0
            p_fmt->video.multiview_mode = MULTIVIEW_STEREO_TB;
376
0
            break;
377
0
        case ST3D_STEREOSCOPIC_LEFT_RIGHT:
378
0
            p_fmt->video.multiview_mode = MULTIVIEW_STEREO_SBS;
379
0
            break;
380
0
        default:
381
0
            msg_Warn( p_demux, "Unknown stereo mode %d", BOXDATA(p_st3d)->i_stereo_mode );
382
0
            break;
383
0
        }
384
0
    }
385
0
    else
386
0
    {
387
0
        for( p_uuid = MP4_BoxGet( p_sample, "uuid" ); p_uuid;
388
0
             p_uuid = p_uuid->p_next )
389
0
        {
390
0
            if( p_uuid->i_type == ATOM_uuid &&
391
0
               !CmpUUID( &p_uuid->i_uuid, &PS3DDSBoxUUID ) &&
392
0
                p_uuid->data.p_binary &&
393
0
                p_uuid->data.p_binary->i_blob == 4 &&
394
0
                !memcmp( p_uuid->data.p_binary->p_blob, "\x82\x81\x10\x02", 4 ) )
395
0
            {
396
0
                p_fmt->video.multiview_mode = MULTIVIEW_STEREO_FRAME;
397
0
                break;
398
0
            }
399
0
        }
400
0
    }
401
402
0
    const MP4_Box_t *p_prhd = MP4_BoxGet( p_sample, "sv3d/proj/prhd" );
403
0
    if (p_prhd && BOXDATA(p_prhd))
404
0
    {
405
0
        vlc_viewpoint_from_euler(&p_fmt->video.pose,
406
0
                                  BOXDATA(p_prhd)->f_pose_yaw_degrees,
407
0
                                  BOXDATA(p_prhd)->f_pose_pitch_degrees,
408
0
                                  BOXDATA(p_prhd)->f_pose_roll_degrees);
409
0
    }
410
411
0
    const MP4_Box_t *p_equi = MP4_BoxGet( p_sample, "sv3d/proj/equi" );
412
0
    const MP4_Box_t *p_cbmp = MP4_BoxGet( p_sample, "sv3d/proj/cbmp" );
413
0
    if (p_equi && BOXDATA(p_equi))
414
0
        p_fmt->video.projection_mode = PROJECTION_MODE_EQUIRECTANGULAR;
415
0
    else if (p_cbmp && BOXDATA(p_cbmp))
416
0
        p_fmt->video.projection_mode = PROJECTION_MODE_CUBEMAP_LAYOUT_STANDARD;
417
418
    /* It's a little ugly but .. there are special cases */
419
0
    switch( i_sample_type )
420
0
    {
421
0
        case( VLC_FOURCC( 's', '2', '6', '3' ) ):
422
0
            p_fmt->i_codec = VLC_CODEC_H263;
423
0
            break;
424
0
        case VLC_FOURCC('y','v','1','2'):
425
0
            p_fmt->i_codec = VLC_CODEC_YV12;
426
0
            break;
427
0
        case VLC_FOURCC('y','u','v','2'):
428
0
            p_fmt->i_codec = VLC_CODEC_YUV2;
429
0
            break;
430
0
        case VLC_FOURCC('A','B','G','R'):
431
0
            p_fmt->i_codec = VLC_CODEC_ARGB;
432
0
            break;
433
0
        case VLC_FOURCC('2','4','B','G'):
434
0
            p_fmt->i_codec = VLC_CODEC_BGR24;
435
0
            break;
436
0
        case VLC_FOURCC('r','a','w',' '):
437
0
            switch( p_vide->i_depth ) {
438
0
                case 16:
439
0
                    p_fmt->i_codec = VLC_CODEC_RGB555BE;
440
0
                    break;
441
0
                case 24:
442
0
                    p_fmt->i_codec = VLC_CODEC_RGB24;
443
0
                    break;
444
0
                case 32:
445
0
                    p_fmt->i_codec = VLC_CODEC_ARGB;
446
0
                    break;
447
0
                case 32 + 8:
448
0
                    p_fmt->i_codec = VLC_CODEC_GREY;
449
0
                    break;
450
0
                default:
451
0
                    msg_Dbg( p_demux, "Unrecognized raw video format (depth = %d)",
452
0
                             p_vide->i_depth );
453
0
                    p_fmt->i_codec = i_sample_type;
454
0
                    break;
455
0
            }
456
0
            break;
457
0
        case( VLC_FOURCC( 'r', 'r', 't', 'p' ) ): /* RTP Reception Hint Track */
458
0
        {
459
0
            if( !SetupRTPReceptionHintTrack( p_demux, p_track, p_sample, p_fmt, params ) )
460
0
                p_fmt->i_codec = i_sample_type;
461
0
            break;
462
0
        }
463
0
        default:
464
0
            p_fmt->i_codec = i_sample_type;
465
0
            break;
466
0
    }
467
468
469
    /* Read extensions */
470
471
0
    const MP4_Box_t *p_clap = MP4_BoxGet( p_sample, "clap" );
472
0
    if( p_clap && BOXDATA(p_clap) &&
473
0
        BOXDATA(p_clap)->i_width + BOXDATA(p_clap)->i_x_offset <= p_fmt->video.i_width &&
474
0
        BOXDATA(p_clap)->i_height + BOXDATA(p_clap)->i_y_offset <= p_fmt->video.i_height )
475
0
    {
476
0
        p_fmt->video.i_visible_width = BOXDATA(p_clap)->i_width;
477
0
        p_fmt->video.i_visible_height = BOXDATA(p_clap)->i_height;
478
0
        p_fmt->video.i_x_offset = BOXDATA(p_clap)->i_x_offset;
479
0
        p_fmt->video.i_y_offset = BOXDATA(p_clap)->i_y_offset;
480
0
    }
481
482
    /* Set up A/R from extension atom */
483
0
    const MP4_Box_t *p_pasp = MP4_BoxGet( p_sample, "pasp" );
484
0
    if( p_pasp && BOXDATA(p_pasp) && BOXDATA(p_pasp)->i_horizontal_spacing > 0 &&
485
0
                  BOXDATA(p_pasp)->i_vertical_spacing > 0 )
486
0
    {
487
0
        p_fmt->video.i_sar_num = BOXDATA(p_pasp)->i_horizontal_spacing;
488
0
        p_fmt->video.i_sar_den = BOXDATA(p_pasp)->i_vertical_spacing;
489
0
    }
490
491
0
    const MP4_Box_t *p_fiel = MP4_BoxGet( p_sample, "fiel" );
492
0
    if( p_fiel && BOXDATA(p_fiel) )
493
0
    {
494
0
        p_cfg->i_block_flags = BOXDATA(p_fiel)->i_flags;
495
0
    }
496
497
0
    const MP4_Box_t *p_colr = MP4_BoxGet( p_sample, "colr" );
498
0
    if ( p_colr != NULL )
499
0
    {
500
0
        if ( BOXDATA(p_colr)->i_type == VLC_FOURCC( 'n', 'c', 'l', 'c' ) ||
501
0
             BOXDATA(p_colr)->i_type == VLC_FOURCC( 'n', 'c', 'l', 'x' ) )
502
0
        {
503
0
            p_fmt->video.primaries =
504
0
                    iso_23001_8_cp_to_vlc_primaries( BOXDATA( p_colr )->nclc.i_primary_idx );
505
0
            p_fmt->video.transfer =
506
0
                    iso_23001_8_tc_to_vlc_xfer( BOXDATA( p_colr )->nclc.i_transfer_function_idx );
507
0
            p_fmt->video.space =
508
0
                    iso_23001_8_mc_to_vlc_coeffs( BOXDATA( p_colr )->nclc.i_matrix_idx );
509
0
            if ( BOXDATA(p_colr)->i_type == VLC_FOURCC( 'n', 'c', 'l', 'x' ) &&
510
0
                    (BOXDATA(p_colr)->nclc.i_full_range >> 7) != 0 )
511
0
                p_fmt->video.color_range = COLOR_RANGE_FULL;
512
0
            else
513
0
                p_fmt->video.color_range = COLOR_RANGE_LIMITED;
514
0
        }
515
0
    }
516
517
0
    const MP4_Box_t *p_dvcC = MP4_BoxGet( p_sample, "dvcC" );
518
0
    if( !p_dvcC )
519
0
        p_dvcC = MP4_BoxGet( p_sample, "dvvC" );
520
0
    if( !p_dvcC )
521
0
        p_dvcC = MP4_BoxGet( p_sample, "dvwC" );
522
0
    if( p_dvcC && BOXDATA(p_dvcC) )
523
0
    {
524
0
        const MP4_Box_data_dvcC_t *p_data = BOXDATA( p_dvcC );
525
0
        p_fmt->video.dovi.version_major = p_data->i_version_major;
526
0
        p_fmt->video.dovi.version_minor = p_data->i_version_minor;
527
0
        p_fmt->video.dovi.profile = p_data->i_profile;
528
0
        p_fmt->video.dovi.level = p_data->i_level;
529
0
        p_fmt->video.dovi.rpu_present = p_data->i_rpu_present;
530
0
        p_fmt->video.dovi.bl_present = p_data->i_bl_present;
531
0
        p_fmt->video.dovi.el_present = p_data->i_el_present;
532
0
    }
533
534
0
    SetupGlobalExtensions( p_sample, p_fmt );
535
536
    /* now see if esds is present and if so create a data packet
537
        with decoder_specific_info  */
538
0
    MP4_Box_t *p_esds = MP4_BoxGet( p_sample, "esds" );
539
0
    if ( p_esds && BOXDATA(p_esds) && BOXDATA(p_esds)->es_descriptor.p_decConfigDescr )
540
0
    {
541
0
        assert(i_sample_type == ATOM_mp4v);
542
0
        SetupESDS( p_demux, p_track, BOXDATA(p_esds)->es_descriptor.p_decConfigDescr, p_fmt );
543
0
    }
544
0
    else switch( i_sample_type )
545
0
    {
546
        /* qt decoder, send the complete chunk */
547
0
        case VLC_FOURCC ('h', 'd', 'v', '1'): // HDV 720p30
548
0
        case VLC_FOURCC ('h', 'd', 'v', '2'): // HDV 1080i60
549
0
        case VLC_FOURCC ('h', 'd', 'v', '3'): // HDV 1080i50
550
0
        case VLC_FOURCC ('h', 'd', 'v', '5'): // HDV 720p25
551
0
        case VLC_FOURCC ('m', 'x', '5', 'n'): // MPEG2 IMX NTSC 525/60 50mb/s produced by FCP
552
0
        case VLC_FOURCC ('m', 'x', '5', 'p'): // MPEG2 IMX PAL 625/60 50mb/s produced by FCP
553
0
        case VLC_FOURCC ('m', 'x', '4', 'n'): // MPEG2 IMX NTSC 525/60 40mb/s produced by FCP
554
0
        case VLC_FOURCC ('m', 'x', '4', 'p'): // MPEG2 IMX PAL 625/60 40mb/s produced by FCP
555
0
        case VLC_FOURCC ('m', 'x', '3', 'n'): // MPEG2 IMX NTSC 525/60 30mb/s produced by FCP
556
0
        case VLC_FOURCC ('m', 'x', '3', 'p'): // MPEG2 IMX PAL 625/50 30mb/s produced by FCP
557
0
        case VLC_FOURCC ('x', 'd', 'v', '2'): // XDCAM HD 1080i60
558
0
        case VLC_FOURCC ('A', 'V', 'm', 'p'): // AVID IMX PAL
559
0
            p_fmt->i_codec = VLC_CODEC_MPGV;
560
0
            break;
561
        /* qt decoder, send the complete chunk */
562
0
        case VLC_CODEC_SVQ1:
563
0
        case VLC_CODEC_SVQ3:
564
0
        case VLC_FOURCC( 'V', 'P', '3', '1' ):
565
0
        case VLC_FOURCC( '3', 'I', 'V', '1' ):
566
0
        case VLC_FOURCC( 'Z', 'y', 'G', 'o' ):
567
0
        {
568
0
            CopyExtradata( p_sample->data.p_sample_vide->p_qt_image_description,
569
0
                           p_sample->data.p_sample_vide->i_qt_image_description,
570
0
                           p_fmt );
571
0
            break;
572
0
        }
573
574
0
        case VLC_FOURCC( 'A', 'V', 'j', '2' ):
575
0
            p_fmt->i_codec = VLC_CODEC_JPEG2000;
576
            /* final decoded resolution stored in ARES w, h, nbfields to group
577
             * but since avcodec can't tell... */
578
0
            break;
579
580
0
        case VLC_FOURCC('j', 'p', 'e', 'g'):
581
0
            p_fmt->i_codec = VLC_CODEC_MJPG;
582
0
           break;
583
584
0
        case VLC_CODEC_FFV1:
585
0
        {
586
0
            MP4_Box_t *p_binary = MP4_BoxGet( p_sample, "glbl" );
587
0
            if( p_binary && BOXDATA(p_binary) )
588
0
            {
589
0
                CopyExtradata( BOXDATA(p_binary)->p_blob,
590
0
                               BOXDATA(p_binary)->i_blob,
591
0
                               p_fmt );
592
0
            }
593
0
            break;
594
0
        }
595
596
0
        case VLC_FOURCC( 'v', 'c', '-', '1' ):
597
0
        {
598
0
            MP4_Box_t *p_dvc1 = MP4_BoxGet( p_sample, "dvc1" );
599
0
            if( p_dvc1 && BOXDATA(p_dvc1) )
600
0
            {
601
0
                CopyExtradata( BOXDATA(p_dvc1)->p_vc1,
602
0
                               BOXDATA(p_dvc1)->i_vc1,
603
0
                               p_fmt );
604
0
            }
605
0
            else
606
0
            {
607
0
                msg_Err( p_demux, "missing dvc1" );
608
0
            }
609
0
            break;
610
0
        }
611
612
0
        case ATOM_av01:
613
0
        {
614
0
            static_assert(ATOM_av01 == VLC_CODEC_AV1, "VLC_CODEC_AV1 != ATOM_av01");
615
0
            MP4_Box_t *p_av1C = MP4_BoxGet( p_sample, "av1C" );
616
0
            if( p_av1C && BOXDATA(p_av1C) )
617
0
            {
618
0
                p_fmt->i_profile = BOXDATA(p_av1C)->i_profile;
619
0
                p_fmt->i_level = BOXDATA(p_av1C)->i_level;
620
0
                CopyExtradata( BOXDATA(p_av1C)->p_av1C,
621
0
                               BOXDATA(p_av1C)->i_av1C,
622
0
                               p_fmt );
623
0
                if (p_fmt->i_extra <= 4)
624
0
                    p_fmt->b_packetized = false; // force full extradata by the packetizer
625
0
            }
626
0
            break;
627
0
        }
628
629
0
        case VLC_FOURCC( 'v', 'v', 'c', '1' ):
630
0
        {
631
0
            MP4_Box_t *p_vvcC = MP4_BoxGet( p_sample, "vvcC" );
632
0
            if( p_vvcC && p_vvcC->data.p_binary &&
633
0
                p_vvcC->data.p_binary->i_blob > 4 )
634
0
            {
635
0
                CopyExtradata( ((uint8_t *)p_vvcC->data.p_binary->p_blob) + 4,
636
0
                               p_vvcC->data.p_binary->i_blob - 4,
637
0
                               p_fmt );
638
0
            }
639
0
            break;
640
0
        }
641
642
        /* avc1: send avcC (h264 without annexe B, ie without start code)*/
643
0
        case VLC_FOURCC( 'a', 'v', 'c', '3' ):
644
0
        case VLC_FOURCC( 'a', 'v', 'c', '1' ):
645
0
        case VLC_FOURCC( 'd', 'v', 'a', '1' ): /* DolbyVision */
646
0
        case VLC_FOURCC( 'd', 'v', 'a', 'v' ): /* DolbyVision */
647
0
        {
648
0
            MP4_Box_t *p_avcC = MP4_BoxGet( p_sample, "avcC" );
649
650
0
            if( p_avcC && BOXDATA(p_avcC) )
651
0
            {
652
0
                p_fmt->i_profile = BOXDATA(p_avcC)->i_profile;
653
0
                p_fmt->i_level = BOXDATA(p_avcC)->i_level;
654
0
                CopyExtradata( BOXDATA(p_avcC)->p_avcC,
655
0
                               BOXDATA(p_avcC)->i_avcC,
656
0
                               p_fmt );
657
0
            }
658
0
            else
659
0
            {
660
0
                msg_Err( p_demux, "missing avcC" );
661
0
            }
662
0
            break;
663
0
        }
664
0
        case VLC_FOURCC( 'h', 'v', 'c', '1' ):
665
0
        case VLC_FOURCC( 'h', 'e', 'v', '1' ):
666
0
        case VLC_FOURCC( 'd', 'v', 'h', 'e' ): /* DolbyVision */
667
0
        case VLC_FOURCC( 'd', 'v', 'h', '1' ): /* DolbyVision */
668
0
        {
669
0
            MP4_Box_t *p_hvcC = MP4_BoxGet( p_sample, "hvcC" );
670
671
            /* Handle DV fourcc collision at demux level */
672
0
            if( i_sample_type == VLC_FOURCC( 'd', 'v', 'h', '1' ) )
673
0
                p_fmt->i_codec = VLC_FOURCC( 'd', 'v', 'h', 'e' );
674
675
0
            if( p_hvcC && p_hvcC->data.p_binary )
676
0
            {
677
0
                CopyExtradata( p_hvcC->data.p_binary->p_blob,
678
0
                               p_hvcC->data.p_binary->i_blob,
679
0
                               p_fmt );
680
0
            }
681
0
            else
682
0
            {
683
0
                msg_Err( p_demux, "missing hvcC" );
684
0
            }
685
0
            break;
686
0
        }
687
688
0
        case ATOM_vp08:
689
0
        case ATOM_vp09:
690
0
        case ATOM_vp10:
691
0
        {
692
0
            const MP4_Box_t *p_vpcC = MP4_BoxGet(  p_sample, "vpcC" );
693
0
            if( p_vpcC && BOXDATA(p_vpcC) )
694
0
            {
695
0
                const MP4_Box_data_vpcC_t *p_data = BOXDATA(p_vpcC);
696
0
                if( i_sample_type == ATOM_vp10 )
697
0
                    p_fmt->i_codec = VLC_CODEC_VP10;
698
0
                else if( i_sample_type == ATOM_vp09 )
699
0
                    p_fmt->i_codec = VLC_CODEC_VP9;
700
0
                else
701
0
                    p_fmt->i_codec = VLC_CODEC_VP8;
702
0
                p_fmt->i_profile = p_data->i_profile;
703
0
                p_fmt->i_level = p_data->i_level;
704
705
0
                if( p_data->i_version == 0 ) /* old deprecated */
706
0
                {
707
0
                    const uint8_t colorspacesmapping[] =
708
0
                    {
709
0
                        COLOR_SPACE_UNDEF,
710
0
                        COLOR_SPACE_BT601,
711
0
                        COLOR_SPACE_BT709,
712
0
                        COLOR_SPACE_SMPTE_170,
713
0
                        COLOR_SPACE_SMPTE_240,
714
0
                        COLOR_SPACE_BT2020,
715
0
                        COLOR_SPACE_BT2020,
716
0
                        COLOR_SPACE_SRGB,
717
0
                    };
718
0
                    if( p_data->i_color_primaries < ARRAY_SIZE(colorspacesmapping) )
719
0
                        p_fmt->video.space = colorspacesmapping[p_data->i_color_primaries];
720
721
0
                    if( p_data->i_xfer_function == 0 )
722
0
                        p_fmt->video.transfer = TRANSFER_FUNC_BT709;
723
0
                    else if ( p_data->i_xfer_function == 1 )
724
0
                        p_fmt->video.transfer = TRANSFER_FUNC_SMPTE_ST2084;
725
0
                }
726
0
                else
727
0
                {
728
0
                    p_fmt->video.primaries =
729
0
                            iso_23001_8_cp_to_vlc_primaries( p_data->i_color_primaries );
730
0
                    p_fmt->video.transfer =
731
0
                            iso_23001_8_tc_to_vlc_xfer( p_data->i_xfer_function );
732
0
                    p_fmt->video.space =
733
0
                            iso_23001_8_mc_to_vlc_coeffs( p_data->i_matrix_coeffs );
734
0
                }
735
736
0
                p_fmt->video.color_range = p_data->i_fullrange ? COLOR_RANGE_FULL : COLOR_RANGE_LIMITED;
737
0
                if (p_fmt->i_profile == -1 && p_fmt->i_level == -1)
738
                    // HACK: keep the bits per sample in i_level
739
0
                    p_fmt->i_level = p_data->i_bit_depth;
740
741
0
                CopyExtradata( p_data->p_codec_init_data,
742
0
                               p_data->i_codec_init_datasize,
743
0
                               p_fmt );
744
745
0
                const MP4_Box_t *p_SmDm = MP4_BoxGet( p_sample, "SmDm" );
746
0
                if( !p_SmDm )
747
0
                    p_SmDm = MP4_BoxGet( p_sample, "mdcv" );
748
0
                if( p_SmDm && BOXDATA(p_SmDm) )
749
0
                {
750
0
                    memcpy( p_fmt->video.mastering.primaries,
751
0
                            BOXDATA(p_SmDm)->primaries, sizeof(p_fmt->video.mastering.primaries) );
752
0
                    memcpy( p_fmt->video.mastering.white_point,
753
0
                            BOXDATA(p_SmDm)->white_point, sizeof(p_fmt->video.mastering.white_point) );
754
0
                    p_fmt->video.mastering.max_luminance = BOXDATA(p_SmDm)->i_luminanceMax;
755
0
                    p_fmt->video.mastering.min_luminance = BOXDATA(p_SmDm)->i_luminanceMin;
756
0
                }
757
758
0
                const MP4_Box_t *p_CoLL = MP4_BoxGet( p_sample, "CoLL" );
759
0
                if( !p_CoLL )
760
0
                    p_CoLL = MP4_BoxGet( p_sample, "clli" );
761
0
                if( p_CoLL && BOXDATA(p_CoLL) )
762
0
                {
763
0
                    p_fmt->video.lighting.MaxCLL = BOXDATA(p_CoLL)->i_maxCLL;
764
0
                    p_fmt->video.lighting.MaxFALL = BOXDATA(p_CoLL)->i_maxFALL;
765
0
                }
766
0
            }
767
0
        }
768
0
        break;
769
770
0
        case ATOM_WMV3:
771
0
            p_cfg->p_asf = MP4_BoxGet( p_sample, "ASF " );
772
            /* fallthrough */
773
0
        case ATOM_H264:
774
0
        case VLC_FOURCC('W','V','C','1'):
775
0
        {
776
0
            MP4_Box_t *p_strf = MP4_BoxGet(  p_sample, "strf" );
777
0
            if ( p_strf && BOXDATA(p_strf) )
778
0
            {
779
0
                p_fmt->video.i_width = BOXDATA(p_strf)->bmiHeader.biWidth;
780
0
                p_fmt->video.i_visible_width = p_fmt->video.i_width;
781
0
                p_fmt->video.i_height = BOXDATA(p_strf)->bmiHeader.biHeight;
782
0
                p_fmt->video.i_visible_height =p_fmt->video.i_height;
783
0
                CopyExtradata( BOXDATA(p_strf)->p_extra,
784
0
                               BOXDATA(p_strf)->i_extra,
785
0
                               p_fmt );
786
0
            }
787
            /* Mostly bogus muxs with old codecs
788
             * see 71c4cc66456facb59cd0eef1626be1be1befeb39 */
789
0
            p_cfg->b_ignore_implicit_pts = true;
790
0
            break;
791
0
        }
792
793
0
        case VLC_FOURCC( 'a', 'i', '5', 'p' ):
794
0
        case VLC_FOURCC( 'a', 'i', '5', 'q' ):
795
0
        case VLC_FOURCC( 'a', 'i', '5', '2' ):
796
0
        case VLC_FOURCC( 'a', 'i', '5', '3' ):
797
0
        case VLC_FOURCC( 'a', 'i', '5', '5' ):
798
0
        case VLC_FOURCC( 'a', 'i', '5', '6' ):
799
0
        case VLC_FOURCC( 'a', 'i', '1', 'p' ):
800
0
        case VLC_FOURCC( 'a', 'i', '1', 'q' ):
801
0
        case VLC_FOURCC( 'a', 'i', '1', '2' ):
802
0
        case VLC_FOURCC( 'a', 'i', '1', '3' ):
803
0
        case VLC_FOURCC( 'a', 'i', '1', '5' ):
804
0
        case VLC_FOURCC( 'a', 'i', '1', '6' ):
805
0
        {
806
0
            if( !p_fmt->i_extra && p_fmt->video.i_width < UINT16_MAX &&
807
0
                p_fiel && BOXDATA(p_fiel) )
808
0
            {
809
0
                p_fmt->p_extra =
810
0
                        AVCi_create_AnnexB( p_fmt->video.i_width,
811
0
                                            !!BOXDATA(p_fiel)->i_flags, &p_fmt->i_extra );
812
0
            }
813
0
            break;
814
0
        }
815
816
0
        case VLC_FOURCC('s','m','c',' '):
817
0
        case VLC_FOURCC('8','B','P','S'):
818
0
        {
819
0
            if( p_sample->data.p_sample_vide->p_palette )
820
0
            {
821
0
                p_fmt->video.p_palette = malloc(sizeof(video_palette_t));
822
0
                if( p_fmt->video.p_palette )
823
0
                    *p_fmt->video.p_palette = *p_sample->data.p_sample_vide->p_palette;
824
0
            }
825
0
            else if(qt_palette_depth_has_default(p_sample->data.p_sample_vide->i_depth))
826
0
            {
827
0
                p_fmt->video.p_palette = qt_make_palette(p_sample->data.p_sample_vide->i_depth);
828
0
            }
829
0
            break;
830
0
        }
831
832
0
        default:
833
0
            msg_Dbg( p_demux, "Unrecognized FourCC %4.4s", (char *)&i_sample_type );
834
0
            break;
835
0
    }
836
837
    /* Codec like QTRLE will need to provide depth. */
838
0
    if (p_fmt->i_profile == -1 && p_fmt->i_level == -1 &&
839
0
        p_fmt->video.i_chroma == 0)
840
0
        p_fmt->i_level = p_vide->i_depth;
841
0
    return 1;
842
0
}
843
844
static bool SetupAudioFromWaveFormatEx( const MP4_Box_t *p_WMA2, es_format_t *p_fmt )
845
0
{
846
0
    if( p_WMA2 && BOXDATA(p_WMA2) )
847
0
    {
848
0
        wf_tag_to_fourcc(BOXDATA(p_WMA2)->Format.wFormatTag, &p_fmt->i_codec, NULL);
849
0
        p_fmt->audio.i_channels = BOXDATA(p_WMA2)->Format.nChannels;
850
0
        p_fmt->audio.i_rate = BOXDATA(p_WMA2)->Format.nSamplesPerSec;
851
0
        p_fmt->i_bitrate = BOXDATA(p_WMA2)->Format.nAvgBytesPerSec * 8;
852
0
        p_fmt->audio.i_blockalign = BOXDATA(p_WMA2)->Format.nBlockAlign;
853
0
        p_fmt->audio.i_bitspersample = BOXDATA(p_WMA2)->Format.wBitsPerSample;
854
0
        CopyExtradata( BOXDATA(p_WMA2)->p_extra,
855
0
                       BOXDATA(p_WMA2)->i_extra,
856
0
                       p_fmt );
857
0
        return true;
858
0
    }
859
0
    return false;
860
0
}
861
862
int SetupAudioES( demux_t *p_demux, const mp4_track_t *p_track,
863
                  const MP4_Box_t *p_sample, es_format_t *p_fmt,
864
                  track_config_t *p_cfg )
865
0
{
866
0
    const MP4_Box_data_sample_soun_t *p_soun = p_sample->data.p_sample_soun;
867
0
    if(!p_soun)
868
0
        return 0;
869
870
0
    const uint32_t i_sample_type = GetSampleType( p_demux, p_sample );
871
0
    p_fmt->i_original_fourcc = i_sample_type;
872
873
0
    p_fmt->audio.i_channels = p_soun->i_channelcount;
874
0
    p_fmt->audio.i_rate = p_soun->i_sampleratehi;
875
0
    if( p_soun->i_qt_version == 0 ) /* otherwise defaults to meaningless 16 */
876
0
    {
877
0
        p_fmt->audio.i_bitspersample = p_soun->i_samplesize;
878
0
        p_fmt->i_bitrate = p_soun->i_channelcount * p_soun->i_sampleratehi *
879
0
                           p_soun->i_samplesize;
880
0
    }
881
882
    /* Endianness atom */
883
0
    const MP4_Box_t *p_enda = MP4_BoxGet( p_sample, "wave/enda" );
884
0
    if( !p_enda )
885
0
        p_enda = MP4_BoxGet( p_sample, "enda" );
886
887
    /* It's a little ugly but .. there are special cases */
888
0
    switch( i_sample_type )
889
0
    {
890
0
        case( VLC_FOURCC( 'r', 'r', 't', 'p' ) ): /* RTP Reception Hint Track */
891
0
        {
892
0
            if( !SetupRTPReceptionHintTrack( p_demux, p_track, p_sample, p_fmt, p_cfg ) )
893
0
                return 0;
894
0
            break;
895
0
        }
896
0
        case ATOM_agsm: /* Apple gsm 33 bytes != MS GSM (agsm fourcc, 65 bytes) */
897
0
            p_fmt->i_codec = VLC_CODEC_GSM;
898
0
            break;
899
0
        case( VLC_FOURCC( '.', 'm', 'p', '3' ) ):
900
0
        case( VLC_FOURCC( 'm', 's', 0x00, 0x55 ) ):
901
0
        {
902
0
            p_fmt->i_codec = VLC_CODEC_MP3;
903
0
            p_fmt->b_packetized = false;
904
0
            break;
905
0
        }
906
0
        case ATOM_XiVs:
907
0
        {
908
0
            const MP4_Box_t *p_vCtH = MP4_BoxGet( p_sample, "wave/vCtH" ); /* kCookieTypeVorbisHeader */
909
0
            const MP4_Box_t *p_vCtd = MP4_BoxGet( p_sample, "wave/vCt#" ); /* kCookieTypeVorbisComments */
910
0
            const MP4_Box_t *p_vCtC = MP4_BoxGet( p_sample, "wave/vCtC" ); /* kCookieTypeVorbisCodebooks */
911
0
            if( p_vCtH && p_vCtH->data.p_binary &&
912
0
                p_vCtd && p_vCtd->data.p_binary &&
913
0
                p_vCtC && p_vCtC->data.p_binary )
914
0
            {
915
0
                size_t headers_sizes[3] = {
916
0
                    p_vCtH->data.p_binary->i_blob,
917
0
                    p_vCtd->data.p_binary->i_blob,
918
0
                    p_vCtC->data.p_binary->i_blob
919
0
                };
920
921
0
                const void * headers[3] = {
922
0
                    p_vCtH->data.p_binary->p_blob,
923
0
                    p_vCtd->data.p_binary->p_blob,
924
0
                    p_vCtC->data.p_binary->p_blob
925
0
                };
926
927
0
                if( xiph_PackHeaders( &p_fmt->i_extra, &p_fmt->p_extra,
928
0
                                      headers_sizes, headers, 3 ) == VLC_SUCCESS )
929
0
                {
930
0
                    p_fmt->i_codec = VLC_CODEC_VORBIS;
931
0
                    p_fmt->b_packetized = false;
932
0
                }
933
0
            }
934
0
            break;
935
0
        }
936
0
        case ATOM_XiFL:
937
0
        {
938
0
            const MP4_Box_t *p_fCtS = MP4_BoxGet( p_sample, "wave/fCtS" ); /* kCookieTypeFLACStreaminfo */
939
0
            if( p_fCtS && p_fCtS->data.p_binary )
940
0
            {
941
0
                size_t i_extra = 8 + p_fCtS->data.p_binary->i_blob;
942
0
                uint8_t *p_extra = malloc(i_extra);
943
0
                if( p_extra )
944
0
                {
945
0
                    p_fmt->i_extra = i_extra;
946
0
                    p_fmt->p_extra = p_extra;
947
0
                    memcpy( p_extra, "fLaC", 4 );
948
0
                    SetDWBE( &p_extra[4], p_fCtS->data.p_binary->i_blob ); /* want the lowest 24bits */
949
0
                    p_extra[4] = 0x80; /* 0x80 Last metablock | 0x00 StreamInfo */
950
0
                    memcpy( &p_extra[8], p_fCtS->data.p_binary->p_blob, p_fCtS->data.p_binary->i_blob );
951
952
0
                    p_fmt->i_codec = VLC_CODEC_FLAC;
953
0
                    p_fmt->b_packetized = false;
954
0
                }
955
0
            }
956
0
            break;
957
0
        }
958
0
        case ATOM_fLaC:
959
0
        {
960
0
            const MP4_Box_t *p_dfLa = MP4_BoxGet(  p_sample, "dfLa" );
961
0
            if( p_dfLa && p_dfLa->data.p_binary->i_blob > 4 &&
962
0
                GetDWBE(p_dfLa->data.p_binary->p_blob) == 0 ) /* fullbox header, avoids creating dedicated parser */
963
0
            {
964
0
                size_t i_extra = p_dfLa->data.p_binary->i_blob;
965
0
                uint8_t *p_extra = malloc(i_extra);
966
0
                if( likely( p_extra ) )
967
0
                {
968
0
                    p_fmt->i_extra = i_extra;
969
0
                    p_fmt->p_extra = p_extra;
970
0
                    memcpy( p_extra, p_dfLa->data.p_binary->p_blob, p_dfLa->data.p_binary->i_blob);
971
0
                    memcpy( p_extra, "fLaC", 4 );
972
0
                    p_fmt->i_codec = VLC_CODEC_FLAC;
973
0
                }
974
0
            }
975
0
            break;
976
0
        }
977
0
        case ATOM_Opus:
978
0
        {
979
0
            const MP4_Box_t *p_dOps = MP4_BoxGet(  p_sample, "dOps" );
980
0
            if( p_dOps && p_dOps->data.p_binary->i_blob > 10 )
981
0
            {
982
0
                size_t i_src = p_dOps->data.p_binary->i_blob;
983
0
                const uint8_t *p_src = p_dOps->data.p_binary->p_blob;
984
0
                if(p_src[0] != 0x00 || (SIZE_MAX - p_dOps->data.p_binary->i_blob < 26))
985
0
                    break;
986
0
                size_t i_dst = 2 + 8 + p_dOps->data.p_binary->i_blob + 8 + 8;
987
0
                uint8_t *p_dst = malloc(i_dst);
988
0
                if( likely( p_dst ) )
989
0
                {
990
0
                    p_dst[0] = 0x01;
991
0
                    p_dst[1] = 8 + i_src;
992
0
                    memcpy(&p_dst[2], "OpusHead", 8);
993
0
                    memcpy(&p_dst[10], p_src, i_src);
994
0
                    p_dst[10] = 0x01; // set version != ISOBMFF mapping
995
0
                    SetWLE(&p_dst[12], GetWBE(&p_dst[12])); // swap endianness for PreSkip
996
0
                    SetDWLE(&p_dst[14], GetDWBE(&p_dst[14])); // swap endianness for InputSampleRate
997
0
                    SetWLE(&p_dst[18], GetWBE(&p_dst[18])); // swap endianness for OutputGain
998
0
                    memcpy(&p_dst[10 + i_src], "OpusTags\x00\x00\x00\x00\x00\x00\x00", 16);
999
0
                    p_fmt->i_extra = i_dst;
1000
0
                    p_fmt->p_extra = p_dst;
1001
0
                    p_fmt->i_codec = VLC_CODEC_OPUS;
1002
0
                }
1003
0
            }
1004
0
            break;
1005
0
        }
1006
0
        case( ATOM_eac3 ):
1007
0
        {
1008
0
            p_fmt->i_codec = VLC_CODEC_EAC3;
1009
            /* TS 102.366. F6 The values of the ChannelCount and SampleSize fields
1010
             *             within the EC3SampleEntry Box shall be ignored. */
1011
0
            p_fmt->audio.i_channels = 0;
1012
0
            p_fmt->audio.i_bitspersample = 0;
1013
1014
0
            const MP4_Box_t *p_dec3 = MP4_BoxGet(  p_sample, "dec3" );
1015
0
            if( p_dec3 && BOXDATA(p_dec3) )
1016
0
            {
1017
0
                p_fmt->i_bitrate = BOXDATA(p_dec3)->i_data_rate * 1000;
1018
0
            }
1019
0
            break;
1020
0
        }
1021
0
        case( ATOM_AC3 ):
1022
0
        case( ATOM_ac3 ):
1023
0
        {
1024
0
            p_fmt->i_codec = VLC_CODEC_A52;
1025
            /* TS 102.366. F3 The values of the ChannelCount and SampleSize fields
1026
             *             within the AC3SampleEntry Box shall be ignored */
1027
0
            p_fmt->audio.i_channels = 0;
1028
0
            p_fmt->audio.i_bitspersample = 0;
1029
1030
0
            MP4_Box_t *p_dac3 = MP4_BoxGet(  p_sample, "dac3" );
1031
0
            if( p_dac3 && BOXDATA(p_dac3) )
1032
0
            {
1033
0
                static const int pi_bitrate[] = {
1034
0
                     32,  40,  48,  56,
1035
0
                     64,  80,  96, 112,
1036
0
                    128, 160, 192, 224,
1037
0
                    256, 320, 384, 448,
1038
0
                    512, 576, 640,
1039
0
                };
1040
0
                p_fmt->i_bitrate = 0;
1041
0
                if( BOXDATA(p_dac3)->i_bitrate_code < sizeof(pi_bitrate)/sizeof(*pi_bitrate) )
1042
0
                {
1043
0
                    p_fmt->i_bitrate = pi_bitrate[BOXDATA(p_dac3)->i_bitrate_code] * 1000;
1044
1045
0
                    if (pi_bitrate[BOXDATA(p_dac3)->i_bitrate_code] == 640
1046
0
                     && BOXDATA(p_dac3)->i_acmod == 7
1047
0
                     && BOXDATA(p_dac3)->i_lfeon == 1)
1048
0
                    {
1049
                        /* DD+ can be an optional codec, and is deployed as an
1050
                         * extension to a "core" AC-3 5.1 640kbit/s audiotrack.
1051
                         * In that case, the AC-3 track might have an EAC3
1052
                         * extension, therefore trigger the A52 packetizer to
1053
                         * detect it (this is needed for aout passhthrough
1054
                         * configuration). */
1055
1056
0
                        p_fmt->b_packetized = false;
1057
0
                    }
1058
0
                }
1059
0
            }
1060
0
            break;
1061
0
        }
1062
0
        case VLC_FOURCC( 'm', 'l', 'p', 'a' ):
1063
            /* spec violation: 32 bits rate instead of fixed point */
1064
0
            p_fmt->i_codec = VLC_CODEC_TRUEHD;
1065
0
            p_fmt->audio.i_rate = (p_soun->i_sampleratehi << 16) | p_soun->i_sampleratelo;
1066
0
            break;
1067
1068
0
        case ATOM_dtsc: /* DTS */
1069
0
        {
1070
0
            p_fmt->i_codec = VLC_CODEC_DTS;
1071
0
            p_fmt->i_profile = PROFILE_DTS;
1072
0
            break;
1073
0
        }
1074
0
        case ATOM_dtse: /* DTS LBR */
1075
0
        {
1076
0
            p_fmt->i_codec = VLC_CODEC_DTS;
1077
0
            p_fmt->i_profile = PROFILE_DTS_EXPRESS;
1078
0
            break;
1079
0
        }
1080
0
        case ATOM_dtsh: /* DTS‐HD audio formats */
1081
0
        case ATOM_dtsl: /* DTS‐HD Lossless formats */
1082
0
        {
1083
0
            p_fmt->i_codec = VLC_CODEC_DTS;
1084
0
            p_fmt->i_profile = PROFILE_DTS_HD;
1085
0
            break;
1086
0
        }
1087
1088
0
        case VLC_FOURCC( 't', 'w', 'o', 's' ):
1089
0
            p_fmt->i_codec = VLC_CODEC_S16B;
1090
0
            p_fmt->i_original_fourcc = i_sample_type;
1091
0
            p_fmt->audio.i_bitspersample = 16;
1092
0
            break;
1093
1094
0
        case VLC_FOURCC( 's', 'o', 'w', 't' ):
1095
0
            p_fmt->i_codec = VLC_CODEC_S16L;
1096
0
            p_fmt->i_original_fourcc = i_sample_type;
1097
0
            p_fmt->audio.i_bitspersample = 16;
1098
0
            break;
1099
1100
0
        case 0x0000000:
1101
0
        case( VLC_FOURCC( 'r', 'a', 'w', ' ' ) ):
1102
0
        case( VLC_FOURCC( 'N', 'O', 'N', 'E' ) ):
1103
0
        {
1104
0
            if( (p_soun->i_samplesize+7)/8 == 1 )
1105
0
            {
1106
0
                p_fmt->i_codec = VLC_CODEC_U8;
1107
0
                p_fmt->audio.i_bitspersample = 8;
1108
0
            }
1109
0
            else
1110
0
            {
1111
0
                p_fmt->i_codec = VLC_CODEC_S16B;
1112
0
                p_fmt->audio.i_bitspersample = 16;
1113
0
            }
1114
0
            p_fmt->i_original_fourcc = p_fmt->i_codec;
1115
1116
            /* Buggy files workaround */
1117
0
            if( p_track->i_timescale != p_soun->i_sampleratehi && p_soun->i_qt_version == 0 )
1118
0
            {
1119
0
                msg_Warn( p_demux, "i_timescale (%"PRId32") != i_sampleratehi "
1120
0
                          "(%u), making both equal (report any problem).",
1121
0
                          p_track->i_timescale, p_soun->i_sampleratehi );
1122
1123
0
                if( p_soun->i_sampleratehi != 0 )
1124
0
                    p_cfg->i_timescale_override = p_soun->i_sampleratehi;
1125
0
                else
1126
0
                    p_fmt->audio.i_rate = p_track->i_timescale;
1127
0
            }
1128
0
            break;
1129
0
        }
1130
1131
0
        case ATOM_ipcm:
1132
0
        case ATOM_fpcm:
1133
0
        {
1134
0
            const MP4_Box_t *p_pcmC = MP4_BoxGet( p_sample, "pcmC" );
1135
0
            if( p_pcmC )
1136
0
            {
1137
0
                const vlc_fourcc_t lookup = i_sample_type +
1138
0
                        BOXDATA(p_pcmC)->i_sample_size +
1139
0
                       (BOXDATA(p_pcmC)->i_format_flags & 0x01);
1140
0
                const vlc_fourcc_t lookuptable[10][2] =
1141
0
                {
1142
0
                    { ATOM_fpcm + 32 + 0, VLC_CODEC_F32B },
1143
0
                    { ATOM_fpcm + 32 + 1, VLC_CODEC_F32L },
1144
0
                    { ATOM_fpcm + 64 + 0, VLC_CODEC_F64B },
1145
0
                    { ATOM_fpcm + 64 + 1, VLC_CODEC_F64L },
1146
0
                    { ATOM_ipcm + 16 + 0, VLC_CODEC_S16B },
1147
0
                    { ATOM_ipcm + 16 + 1, VLC_CODEC_S16L },
1148
0
                    { ATOM_ipcm + 24 + 0, VLC_CODEC_S24B },
1149
0
                    { ATOM_ipcm + 24 + 1, VLC_CODEC_S24L },
1150
0
                    { ATOM_ipcm + 32 + 0, VLC_CODEC_S32B },
1151
0
                    { ATOM_ipcm + 32 + 1, VLC_CODEC_S32L },
1152
0
                };
1153
0
                for( size_t i = 0; i<10; i++ )
1154
0
                    if( lookuptable[i][0] == lookup )
1155
0
                        p_fmt->i_codec = lookuptable[i][1];
1156
0
            }
1157
0
            break;
1158
0
        }
1159
1160
0
        case ATOM_in24:
1161
0
            p_fmt->i_original_fourcc =
1162
0
            p_fmt->i_codec = p_enda && BOXDATA(p_enda)->i_little_endian == 1 ?
1163
0
                                    VLC_CODEC_S24L : VLC_CODEC_S24B;
1164
0
            break;
1165
0
        case ATOM_in32:
1166
0
            p_fmt->i_original_fourcc =
1167
0
            p_fmt->i_codec = p_enda && BOXDATA(p_enda)->i_little_endian == 1 ?
1168
0
                                    VLC_CODEC_S32L : VLC_CODEC_S32B;
1169
0
            break;
1170
0
        case ATOM_fl32:
1171
0
            p_fmt->i_original_fourcc =
1172
0
            p_fmt->i_codec = p_enda && BOXDATA(p_enda)->i_little_endian == 1 ?
1173
0
                                    VLC_CODEC_F32L : VLC_CODEC_F32B;
1174
0
            break;
1175
0
        case ATOM_fl64:
1176
0
            p_fmt->i_original_fourcc =
1177
0
            p_fmt->i_codec = p_enda && BOXDATA(p_enda)->i_little_endian == 1 ?
1178
0
                                    VLC_CODEC_F64L : VLC_CODEC_F64B;
1179
0
            break;
1180
1181
0
        case VLC_CODEC_DVD_LPCM:
1182
0
        {
1183
0
            if( p_soun->i_qt_version == 2 )
1184
0
            {
1185
                /* Flags:
1186
                 *  0x01: IsFloat
1187
                 *  0x02: IsBigEndian
1188
                 *  0x04: IsSigned
1189
                 */
1190
0
                static const struct {
1191
0
                    unsigned     i_flags;
1192
0
                    unsigned     i_mask;
1193
0
                    unsigned     i_bits;
1194
0
                    vlc_fourcc_t i_codec;
1195
0
                } p_formats[] = {
1196
0
                    { 0x01,           0x03, 32, VLC_CODEC_F32L },
1197
0
                    { 0x01,           0x03, 64, VLC_CODEC_F64L },
1198
0
                    { 0x01|0x02,      0x03, 32, VLC_CODEC_F32B },
1199
0
                    { 0x01|0x02,      0x03, 64, VLC_CODEC_F64B },
1200
1201
0
                    { 0x00,           0x05,  8, VLC_CODEC_U8 },
1202
0
                    { 0x00|     0x04, 0x05,  8, VLC_CODEC_S8 },
1203
1204
0
                    { 0x00,           0x07, 16, VLC_CODEC_U16L },
1205
0
                    { 0x00|0x02,      0x07, 16, VLC_CODEC_U16B },
1206
0
                    { 0x00     |0x04, 0x07, 16, VLC_CODEC_S16L },
1207
0
                    { 0x00|0x02|0x04, 0x07, 16, VLC_CODEC_S16B },
1208
1209
0
                    { 0x00,           0x07, 24, VLC_CODEC_U24L },
1210
0
                    { 0x00|0x02,      0x07, 24, VLC_CODEC_U24B },
1211
0
                    { 0x00     |0x04, 0x07, 24, VLC_CODEC_S24L },
1212
0
                    { 0x00|0x02|0x04, 0x07, 24, VLC_CODEC_S24B },
1213
1214
0
                    { 0x00,           0x07, 32, VLC_CODEC_U32L },
1215
0
                    { 0x00|0x02,      0x07, 32, VLC_CODEC_U32B },
1216
0
                    { 0x00     |0x04, 0x07, 32, VLC_CODEC_S32L },
1217
0
                    { 0x00|0x02|0x04, 0x07, 32, VLC_CODEC_S32B },
1218
1219
0
                    {0, 0, 0, 0}
1220
0
                };
1221
1222
0
                for( int i = 0; p_formats[i].i_codec; i++ )
1223
0
                {
1224
0
                    if( p_formats[i].i_bits == p_soun->i_constbitsperchannel &&
1225
0
                        (p_soun->i_formatflags & p_formats[i].i_mask) == p_formats[i].i_flags )
1226
0
                    {
1227
0
                        p_fmt->i_codec = p_formats[i].i_codec;
1228
0
                        p_fmt->audio.i_bitspersample = p_soun->i_constbitsperchannel;
1229
0
                        p_fmt->audio.i_blockalign =
1230
0
                                p_soun->i_channelcount * p_soun->i_constbitsperchannel / 8;
1231
0
                        p_cfg->i_sample_size_override = p_fmt->audio.i_blockalign;
1232
0
                        break;
1233
0
                    }
1234
0
                }
1235
0
            }
1236
0
            break;
1237
0
        }
1238
0
        default:
1239
0
            p_fmt->i_codec = i_sample_type;
1240
0
            break;
1241
0
    }
1242
1243
1244
    /* Process extensions */
1245
1246
    /* Lookup for then channels extension */
1247
0
    const MP4_Box_t *p_chan = MP4_BoxGet( p_sample, "chan" );
1248
0
    if ( p_chan )
1249
0
    {
1250
0
        uint16_t i_vlc_mapping = 0;
1251
0
        uint8_t i_channels = 0;
1252
0
        const uint32_t *p_rg_chans_order = NULL;
1253
1254
0
        if ( CoreAudio_Layout_to_vlc( &BOXDATA(p_chan)->layout,
1255
0
                                      &i_vlc_mapping, &i_channels,
1256
0
                                      &p_rg_chans_order ) != VLC_SUCCESS )
1257
0
        {
1258
0
            msg_Warn( p_demux, "discarding chan mapping" );
1259
0
        }
1260
0
        else if( i_vlc_mapping )
1261
0
        {
1262
0
            const unsigned i_bps = aout_BitsPerSample( p_fmt->i_codec );
1263
            /* Uncompressed audio */
1264
0
            if( i_bps && aout_CheckChannelReorder( p_rg_chans_order, NULL,
1265
0
                                                   i_vlc_mapping,
1266
0
                                                   p_cfg->rgi_chans_reordering ) )
1267
0
                p_cfg->b_chans_reorder = true;
1268
1269
            /* we can only set bitmap for VLC mapping or [re]mapped pcm audio
1270
             * as vlc can't enumerate channels for compressed content */
1271
0
            if( i_bps )
1272
0
            {
1273
0
                p_fmt->audio.i_channels = stdc_count_ones(i_vlc_mapping);
1274
0
                p_fmt->audio.i_physical_channels = i_vlc_mapping;
1275
0
            }
1276
0
        }
1277
0
    }
1278
1279
0
    const MP4_Box_t *p_srat = MP4_BoxGet( p_sample, "srat" );
1280
0
    if ( p_srat )
1281
0
        p_fmt->audio.i_rate = BOXDATA(p_srat)->i_sample_rate;
1282
1283
0
    SetupGlobalExtensions( p_sample, p_fmt );
1284
1285
    /* now see if esds is present and if so create a data packet
1286
        with decoder_specific_info  */
1287
0
    MP4_Box_t *p_esds = MP4_BoxGet( p_sample, "esds" );
1288
0
    if ( !p_esds ) p_esds = MP4_BoxGet( p_sample, "wave/esds" );
1289
0
    if ( p_esds && BOXDATA(p_esds) && BOXDATA(p_esds)->es_descriptor.p_decConfigDescr )
1290
0
    {
1291
0
        assert(i_sample_type == ATOM_mp4a);
1292
0
        SetupESDS( p_demux, p_track, BOXDATA(p_esds)->es_descriptor.p_decConfigDescr, p_fmt );
1293
0
    }
1294
0
    else switch( i_sample_type )
1295
0
    {
1296
0
        case VLC_CODEC_AMR_NB:
1297
0
            p_fmt->audio.i_rate = 8000;
1298
0
            break;
1299
0
        case VLC_CODEC_AMR_WB:
1300
0
            p_fmt->audio.i_rate = 16000;
1301
0
            break;
1302
0
        case VLC_CODEC_QDMC:
1303
0
        case VLC_CODEC_QDM2:
1304
0
        case VLC_CODEC_ALAC:
1305
0
        {
1306
0
            CopyExtradata( p_sample->data.p_sample_soun->p_qt_description,
1307
0
                           p_sample->data.p_sample_soun->i_qt_description,
1308
0
                           p_fmt );
1309
0
            if( p_fmt->i_extra == 56 && i_sample_type == VLC_CODEC_ALAC )
1310
0
            {
1311
0
                p_fmt->audio.i_channels = *((uint8_t*)p_fmt->p_extra + 41);
1312
0
                p_fmt->audio.i_rate = GetDWBE((uint8_t*)p_fmt->p_extra + 52);
1313
0
            }
1314
0
            break;
1315
0
        }
1316
0
        case VLC_CODEC_ADPCM_MS:
1317
0
        case VLC_CODEC_ADPCM_IMA_WAV:
1318
0
        case VLC_CODEC_QCELP:
1319
0
        {
1320
0
            p_fmt->audio.i_blockalign = p_sample->data.p_sample_soun->i_bytes_per_frame;
1321
0
            break;
1322
0
        }
1323
0
        case ATOM_WMA2:
1324
0
        {
1325
0
            if( SetupAudioFromWaveFormatEx( MP4_BoxGet( p_sample, "wave/WMA2" ), p_fmt ) )
1326
0
            {
1327
0
                p_cfg->p_asf = MP4_BoxGet( p_sample, "wave/ASF " );
1328
0
            }
1329
0
            else
1330
0
            {
1331
0
                msg_Err( p_demux, "missing WMA2 %4.4s", (char*) &p_sample->p_father->i_type );
1332
0
            }
1333
0
            break;
1334
0
        }
1335
0
        case ATOM_wma: /* isml wmapro */
1336
0
        {
1337
0
            if( !SetupAudioFromWaveFormatEx( MP4_BoxGet( p_sample, "wfex" ), p_fmt ) )
1338
0
                msg_Err( p_demux, "missing wfex for wma" );
1339
0
            break;
1340
0
        }
1341
1342
0
        default:
1343
0
            if(p_fmt->i_codec == 0)
1344
0
                msg_Dbg( p_demux, "Unrecognized FourCC %4.4s", (char *)&i_sample_type );
1345
0
            break;
1346
0
    }
1347
1348
    /* Ambisonics */
1349
0
    const MP4_Box_t *p_SA3D = MP4_BoxGet(p_sample, "SA3D");
1350
0
    if (p_SA3D && BOXDATA(p_SA3D))
1351
0
        p_fmt->audio.channel_type = AUDIO_CHANNEL_TYPE_AMBISONICS;
1352
1353
0
    return 1;
1354
0
}
1355
1356
int SetupSpuES( demux_t *p_demux, const mp4_track_t *p_track,
1357
                const MP4_Box_t *p_sample, es_format_t *p_fmt,
1358
                track_config_t *p_cfg )
1359
0
{
1360
0
    const uint32_t i_sample_type = GetSampleType( p_demux, p_sample );
1361
1362
    /* It's a little ugly but .. there are special cases */
1363
0
    switch( i_sample_type )
1364
0
    {
1365
0
        case VLC_FOURCC('s','t','p','p'):
1366
0
            p_fmt->i_codec = VLC_CODEC_TTML;
1367
0
            break;
1368
0
        case ATOM_wvtt:
1369
0
            p_fmt->i_codec = VLC_CODEC_WEBVTT;
1370
0
            break;
1371
0
        case ATOM_c608: /* EIA608 closed captions */
1372
0
            p_fmt->i_codec = VLC_CODEC_CEA608;
1373
0
            p_fmt->subs.cc.i_reorder_depth = -1;
1374
0
            break;
1375
0
        case ATOM_c708: /* EIA708 closed captions */
1376
0
            p_fmt->i_codec = VLC_CODEC_CEA708;
1377
0
            p_fmt->subs.cc.i_reorder_depth = -1;
1378
0
            break;
1379
1380
0
        case( VLC_FOURCC( 't', 'e', 'x', 't' ) ):
1381
0
        case( VLC_FOURCC( 't', 'x', '3', 'g' ) ):
1382
0
        {
1383
0
            const MP4_Box_data_sample_generic_t *p_text = p_sample->data.p_sample_gen;
1384
0
            if(!p_text)
1385
0
                return 0;
1386
1387
0
            if( i_sample_type == VLC_FOURCC( 't', 'e', 'x', 't' ) )
1388
0
                p_fmt->i_codec = VLC_CODEC_QTXT;
1389
0
            else
1390
0
                p_fmt->i_codec = VLC_CODEC_TX3G;
1391
1392
0
            if( p_text->i_data > 4 && GetDWBE(p_text->p_data) & 0xC0000000 )
1393
0
            {
1394
0
                p_fmt->i_priority = ES_PRIORITY_SELECTABLE_MIN + 1;
1395
0
                p_cfg->b_forced_spu = true;
1396
0
            }
1397
1398
0
            CopyExtradata( p_text->p_data, p_text->i_data, p_fmt );
1399
1400
            /* FIXME UTF-8 doesn't work here ? */
1401
0
            if( p_track->b_mac_encoding )
1402
0
                p_fmt->subs.psz_encoding = strdup( "MAC" );
1403
0
            else
1404
0
                p_fmt->subs.psz_encoding = strdup( "UTF-8" );
1405
0
            break;
1406
0
        }
1407
1408
0
        default:
1409
0
            p_fmt->i_codec = i_sample_type;
1410
0
            break;
1411
0
    }
1412
1413
0
    SetupGlobalExtensions( p_sample, p_fmt );
1414
1415
    /* now see if esds is present and if so create a data packet
1416
        with decoder_specific_info  */
1417
0
    MP4_Box_t *p_esds = MP4_BoxGet( p_sample, "esds" );
1418
0
    if ( p_esds && BOXDATA(p_esds) && BOXDATA(p_esds)->es_descriptor.p_decConfigDescr )
1419
0
    {
1420
0
        SetupESDS( p_demux, p_track, BOXDATA(p_esds)->es_descriptor.p_decConfigDescr, p_fmt );
1421
0
    }
1422
1423
0
    return 1;
1424
0
}