This source file includes following definitions.
- gf_isom_set_last_error
- gf_isom_last_error
- gf_isom_get_mode
- gf_isom_get_file_size
- gf_isom_sample_new
- gf_isom_sample_del
- gf_isom_probe_file
- isom_create_init_from_mem
- gf_isom_open_progressive
- gf_isom_open
- gf_isom_close
- gf_isom_has_root_od
- gf_isom_get_root_od
- gf_isom_get_track_count
- gf_isom_get_track_id
- gf_isom_get_track_by_id
- gf_isom_get_track_original_id
- gf_isom_has_movie
- gf_isom_has_segment
- gf_isom_segment_get_fragment_count
- gf_isom_get_moof
- gf_isom_segment_get_track_fragment_count
- gf_isom_segment_get_track_fragment_decode_time
- gf_isom_get_timescale
- gf_isom_get_duration
- gf_isom_get_original_duration
- gf_isom_get_creation_time
- gf_isom_is_track_in_root_od
- gf_isom_is_track_enabled
- gf_isom_get_track_duration
- gf_isom_get_media_language
- gf_isom_get_track_kind_count
- gf_isom_get_track_kind
- gf_isom_get_reference_count
- gf_isom_get_reference
- gf_isom_get_reference_ID
- gf_isom_has_track_reference
- gf_isom_get_media_time
- gf_isom_get_sample_description_index
- gf_isom_get_sample_description_count
- gf_isom_get_esd
- gf_isom_get_decoder_config
- gf_isom_get_media_duration
- gf_isom_get_media_original_duration
- gf_isom_get_media_timescale
- gf_isom_get_copyright_count
- gf_isom_get_copyright
- gf_isom_get_watermark
- gf_isom_get_chapter_count
- gf_isom_get_chapter
- gf_isom_get_media_type
- IsMP4Description
- IsMP4EncryptedDescription
- gf_isom_is_track_encrypted
- gf_isom_get_media_subtype
- gf_isom_get_mpeg4_subtype
- gf_isom_get_handler_name
- gf_isom_check_data_reference
- gf_isom_get_data_reference
- gf_isom_get_sample_count
- gf_isom_get_constant_sample_size
- gf_isom_has_time_offset
- gf_isom_get_cts_to_dts_shift
- gf_isom_has_sync_shadows
- gf_isom_has_sample_dependency
- gf_isom_get_sample_flags
- gf_isom_get_sample
- gf_isom_get_sample_duration
- gf_isom_get_sample_size
- gf_isom_get_sample_sync
- gf_isom_get_sample_info
- gf_isom_get_sample_dts
- gf_isom_is_self_contained
- gf_isom_get_sample_from_dts
- gf_isom_get_sample_for_media_time
- gf_isom_get_sample_for_movie_time
- gf_isom_get_missing_bytes
- gf_isom_set_sample_padding
- gf_isom_get_edit_list_type
- gf_isom_get_edit_segment_count
- gf_isom_get_edit_segment
- gf_isom_has_sync_points
- gf_isom_get_sync_point_count
- gf_isom_get_brand_info
- gf_isom_get_alternate_brand
- gf_isom_get_sample_padding_bits
- gf_isom_has_padding_bits
- gf_isom_get_udta_count
- gf_isom_get_udta_type
- gf_isom_get_user_data_count
- gf_isom_get_user_data
- gf_isom_delete
- gf_isom_get_chunks_infos
- gf_isom_get_sample_fragment_count
- gf_isom_get_sample_fragment_size
- gf_isom_get_fragment_defaults
- gf_isom_refresh_fragmented
- gf_isom_set_single_moof_mode
- gf_isom_reset_data_offset
- gf_isom_reset_tables
- gf_isom_release_segment
- gf_isom_open_segment
- gf_isom_get_highest_track_in_scalable_segment
- gf_isom_text_set_streaming_mode
- gf_isom_get_generic_sample_description
- gf_isom_get_visual_info
- gf_isom_get_audio_info
- gf_isom_get_pixel_aspect_ratio
- gf_isom_get_filename
- gf_isom_get_pl_indication
- gf_isom_get_track_matrix
- gf_isom_get_track_layout_info
- gf_isom_get_media_data_size
- gf_isom_set_default_sync_track
- gf_isom_is_single_av
- gf_isom_is_JPEG2000
- gf_isom_guess_specification
- gf_ismo_locate_box
- gf_isom_apple_get_tag
- gf_isom_get_track_switch_group_count
- gf_isom_get_track_switch_parameter
- gf_isom_get_next_alternate_group_id
- gf_isom_sample_has_subsamples
- gf_isom_sample_get_subsample
- gf_isom_get_rvc_config
- gf_isom_moov_first
- gf_isom_reset_fragment_info
- gf_isom_get_sample_rap_roll_info
- gf_isom_get_sample_group_info_entry
- gf_isom_get_sample_group_info
- gf_isom_get_fragmented_duration
- gf_isom_get_fragments_count
- gf_isom_get_fragmented_samples_info
- gf_isom_set_nalu_extract_mode
- gf_isom_get_nalu_extract_mode
- gf_isom_get_composition_offset_shift
- gf_isom_needs_layer_reconstruction
- gf_isom_keep_utc_times
- gf_isom_no_version_date_info
- gf_isom_get_pssh_count
- gf_isom_get_pssh_info
- gf_isom_get_sample_cenc_info_ex
- gf_isom_get_sample_cenc_info
- gf_isom_get_last_producer_time_box
- gf_isom_get_current_tfdt
- gf_isom_parse_trif_info
- gf_isom_get_tile_info
- gf_isom_drop_date_version_info_enabled
- gf_isom_get_oinf_info
#include <gpac/internal/isomedia_dev.h>
#include <gpac/constants.h>
#ifndef GPAC_DISABLE_ISOM
static GF_Err MP4_API_IO_Err;
void gf_isom_set_last_error(GF_ISOFile *movie, GF_Err error)
{
if (!movie) {
MP4_API_IO_Err = error;
} else {
movie->LastError = error;
}
}
GF_EXPORT
GF_Err gf_isom_last_error(GF_ISOFile *the_file)
{
if (!the_file) return MP4_API_IO_Err;
return the_file->LastError;
}
GF_EXPORT
u8 gf_isom_get_mode(GF_ISOFile *the_file)
{
if (!the_file) return 0;
return the_file->openMode;
}
GF_EXPORT
u64 gf_isom_get_file_size(GF_ISOFile *the_file)
{
if (!the_file) return 0;
if (the_file->movieFileMap) return gf_bs_get_size(the_file->movieFileMap->bs);
#ifndef GPAC_DISABLE_ISOM_WRITE
if (the_file->editFileMap) return gf_bs_get_size(the_file->editFileMap->bs);
#endif
return 0;
}
GF_EXPORT
GF_ISOSample *gf_isom_sample_new()
{
GF_ISOSample *tmp;
GF_SAFEALLOC(tmp, GF_ISOSample);
return tmp;
}
GF_EXPORT
void gf_isom_sample_del(GF_ISOSample **samp)
{
if (! *samp) return;
if ((*samp)->data && (*samp)->dataLength) gf_free((*samp)->data);
gf_free(*samp);
*samp = NULL;
}
GF_EXPORT
u32 gf_isom_probe_file(const char *fileName)
{
u32 type = 0;
if (!strncmp(fileName, "gmem://", 7)) {
u32 size;
u8 *mem_address;
if (sscanf(fileName, "gmem://%d@%p", &size, &mem_address) != 2) {
return 0;
}
if (size>8)
type = GF_4CC(mem_address[4], mem_address[5], mem_address[6], mem_address[7]);
} else {
unsigned char data[4];
FILE *f = gf_fopen(fileName, "rb");
if (!f) return 0;
type = 0;
if (fread(data, 1, 4, f) == 4) {
if (fread(data, 1, 4, f) == 4) {
type = GF_4CC(data[0], data[1], data[2], data[3]);
}
}
gf_fclose(f);
}
switch (type) {
case GF_ISOM_BOX_TYPE_FTYP:
case GF_ISOM_BOX_TYPE_MOOV:
return 2;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
case GF_ISOM_BOX_TYPE_MOOF:
case GF_ISOM_BOX_TYPE_STYP:
return 3;
#ifndef GPAC_DISABLE_ISOM_ADOBE
case GF_ISOM_BOX_TYPE_AFRA:
case GF_ISOM_BOX_TYPE_ABST:
#endif
#endif
case GF_ISOM_BOX_TYPE_MDAT:
case GF_ISOM_BOX_TYPE_FREE:
case GF_ISOM_BOX_TYPE_SKIP:
case GF_ISOM_BOX_TYPE_UDTA:
case GF_ISOM_BOX_TYPE_META:
case GF_ISOM_BOX_TYPE_VOID:
case GF_4CC('j','P',' ',' '):
case GF_4CC('w','i','d','e'):
return 1;
default:
return 0;
}
}
#ifndef GPAC_DISABLE_AV_PARSERS
#include <gpac/internal/media_dev.h>
#endif
static GF_Err isom_create_init_from_mem(const char *fileName, GF_ISOFile *file)
{
u32 sample_rate=0;
u32 nb_channels=0;
u32 bps=0;
u32 atag=0;
u32 nal_len=4;
u32 width = 0;
u32 height = 0;
u32 timescale = 10000000;
u64 tfdt = 0;
char sz4cc[5];
char CodecParams[2048];
u32 CodecParamLen=0;
char *sep, *val;
GF_TrackBox *trak;
GF_TrackExtendsBox *trex;
sz4cc[0] = 0;
val = (char*) ( fileName + strlen("isobmff://") );
while (1) {
sep = strchr(val, ' ');
if (sep) sep[0] = 0;
if (!strncmp(val, "4cc=", 4)) strcpy(sz4cc, val+4);
else if (!strncmp(val, "init=", 5)) {
char szH[3], *data = val+5;
u32 i, len = (u32) strlen(data);
for (i=0; i<len; i+=2) {
u32 v;
szH[0] = data[i];
szH[1] = data[i+1];
szH[2] = 0;
sscanf(szH, "%X", &v);
CodecParams[CodecParamLen] = (char) v;
CodecParamLen++;
}
}
else if (!strncmp(val, "nal=", 4)) nal_len = atoi(val+4);
else if (!strncmp(val, "bps=", 4)) bps = atoi(val+4);
else if (!strncmp(val, "atag=", 5)) atag = atoi(val+5);
else if (!strncmp(val, "ch=", 3)) nb_channels = atoi(val+3);
else if (!strncmp(val, "srate=", 6)) sample_rate = atoi(val+6);
else if (!strncmp(val, "w=", 2)) width = atoi(val+2);
else if (!strncmp(val, "h=", 2)) height = atoi(val+2);
else if (!strncmp(val, "scale=", 6)) timescale = atoi(val+6);
else if (!strncmp(val, "tfdt=", 5)) tfdt = atoi(val+5);
if (!sep) break;
sep[0] = ' ';
val = sep+1;
}
if (!stricmp(sz4cc, "H264")) {
}
else if (!stricmp(sz4cc, "AACL")) {
}
else {
GF_LOG(GF_LOG_WARNING, GF_LOG_CONTAINER, ("[iso file] Cannot convert smooth media type %s to ISO init segment\n", sz4cc));
return GF_NOT_SUPPORTED;
}
file->moov = (GF_MovieBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MOOV);
file->moov->mov = file;
file->is_smooth = GF_TRUE;
file->moov->mvhd = (GF_MovieHeaderBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MVHD);
file->moov->mvhd->timeScale = timescale;
file->moov->mvex = (GF_MovieExtendsBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MVEX);
trex = (GF_TrackExtendsBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_TREX);
trex->def_sample_desc_index = 1;
trex->trackID = 1;
gf_list_add(file->moov->mvex->TrackExList, trex);
trak = (GF_TrackBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_TRAK);
trak->moov = file->moov;
gf_list_add(file->moov->trackList, trak);
trak->Header = (GF_TrackHeaderBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_TKHD);
trak->Header->trackID = 1;
trak->Header->flags |= 1;
trak->Header->width = width;
trak->Header->height = height;
trak->Media = (GF_MediaBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MDIA);
trak->Media->mediaTrack = trak;
trak->Media->mediaHeader = (GF_MediaHeaderBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MDHD);
trak->Media->mediaHeader->timeScale = timescale;
trak->Media->handler = (GF_HandlerBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_HDLR);
trak->Media->handler->handlerType = width ? GF_ISOM_MEDIA_VISUAL : GF_ISOM_MEDIA_AUDIO;
trak->Media->information = (GF_MediaInformationBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MINF);
trak->Media->information->sampleTable = (GF_SampleTableBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STBL);
trak->Media->information->sampleTable->SampleSize = (GF_SampleSizeBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STSZ);
trak->Media->information->sampleTable->TimeToSample = (GF_TimeToSampleBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STTS);
trak->Media->information->sampleTable->ChunkOffset = (GF_Box *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STCO);
trak->Media->information->sampleTable->SampleToChunk = (GF_SampleToChunkBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STSC);
trak->Media->information->sampleTable->SyncSample = (GF_SyncSampleBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STSS);
trak->Media->information->sampleTable->SampleDescription = (GF_SampleDescriptionBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_STSD);
trak->dts_at_seg_start = tfdt;
if (!stricmp(sz4cc, "H264")) {
u32 pos = 0;
u32 end, sc_size=0;
GF_MPEGVisualSampleEntryBox *avc = (GF_MPEGVisualSampleEntryBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_AVC1);
avc->avc_config = (GF_AVCConfigurationBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_AVCC);
avc->Width = width;
avc->Height = height;
avc->avc_config->config = gf_odf_avc_cfg_new();
avc->avc_config->config->nal_unit_size = nal_len;
avc->avc_config->config->configurationVersion = 1;
#ifndef GPAC_DISABLE_AV_PARSERS
end = gf_media_nalu_next_start_code((u8 *) CodecParams, CodecParamLen, &sc_size);
pos += sc_size;
while (pos<CodecParamLen) {
GF_AVCConfigSlot *slc;
u8 nal_type;
char *nal = &CodecParams[pos];
end = gf_media_nalu_next_start_code(nal, CodecParamLen-pos, &sc_size);
if (!end) end = CodecParamLen;
GF_SAFEALLOC(slc, GF_AVCConfigSlot);
slc->size = end;
slc->data = gf_malloc(sizeof(char)*slc->size);
memcpy(slc->data, nal, sizeof(char)*slc->size);
nal_type = nal[0] & 0x1F;
if (nal_type == GF_AVC_NALU_SEQ_PARAM) {
gf_list_add(avc->avc_config->config->sequenceParameterSets, slc);
} else {
gf_list_add(avc->avc_config->config->pictureParameterSets, slc);
}
pos += slc->size + sc_size;
}
#endif
AVC_RewriteESDescriptor(avc);
gf_list_add(trak->Media->information->sampleTable->SampleDescription->other_boxes, avc);
}
else if (!stricmp(sz4cc, "AACL")) {
#ifndef GPAC_DISABLE_AV_PARSERS
GF_M4ADecSpecInfo aacinfo;
#endif
GF_MPEGAudioSampleEntryBox *aac = (GF_MPEGAudioSampleEntryBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_MP4A);
aac->esd = (GF_ESDBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_ESDS);
aac->esd->desc = gf_odf_desc_esd_new(2);
#ifndef GPAC_DISABLE_AV_PARSERS
memset(&aacinfo, 0, sizeof(GF_M4ADecSpecInfo));
aacinfo.nb_chan = nb_channels;
aacinfo.base_object_type = GF_M4A_AAC_LC;
aacinfo.base_sr = sample_rate;
gf_m4a_write_config(&aacinfo, &aac->esd->desc->decoderConfig->decoderSpecificInfo->data, &aac->esd->desc->decoderConfig->decoderSpecificInfo->dataLength);
#endif
aac->esd->desc->decoderConfig->streamType = GF_STREAM_AUDIO;
aac->esd->desc->decoderConfig->objectTypeIndication = GPAC_OTI_AUDIO_AAC_MPEG4;
aac->bitspersample = bps;
aac->samplerate_hi = sample_rate;
aac->channel_count = nb_channels;
gf_list_add(trak->Media->information->sampleTable->SampleDescription->other_boxes, aac);
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_open_progressive(const char *fileName, u64 start_range, u64 end_range, GF_ISOFile **the_file, u64 *BytesMissing)
{
GF_Err e;
GF_ISOFile *movie;
if (!BytesMissing || !the_file)
return GF_BAD_PARAM;
*BytesMissing = 0;
*the_file = NULL;
movie = gf_isom_new_movie();
if (!movie) return GF_OUT_OF_MEM;
movie->fileName = gf_strdup(fileName);
movie->openMode = GF_ISOM_OPEN_READ;
#ifndef GPAC_DISABLE_ISOM_WRITE
movie->editFileMap = NULL;
movie->finalName = NULL;
#endif
if (!strncmp(fileName, "isobmff://", 10)) {
movie->movieFileMap = NULL;
e = isom_create_init_from_mem(fileName, movie);
} else {
e = gf_isom_datamap_new(fileName, NULL, GF_ISOM_DATA_MAP_READ, &movie->movieFileMap);
if (e) {
gf_isom_delete_movie(movie);
return e;
}
if (end_range>start_range) {
gf_bs_seek(movie->movieFileMap->bs, end_range+1);
gf_bs_truncate(movie->movieFileMap->bs);
gf_bs_seek(movie->movieFileMap->bs, start_range);
}
e = gf_isom_parse_movie_boxes(movie, BytesMissing, GF_TRUE);
}
if (e == GF_ISOM_INCOMPLETE_FILE) {
if (movie->moov) {
*the_file = (GF_ISOFile *)movie;
return GF_OK;
}
gf_isom_delete_movie(movie);
return e;
} else if (e) {
gf_isom_delete_movie(movie);
return e;
}
*the_file = (GF_ISOFile *)movie;
return GF_OK;
}
GF_EXPORT
GF_ISOFile *gf_isom_open(const char *fileName, u32 OpenMode, const char *tmp_dir)
{
GF_ISOFile *movie;
MP4_API_IO_Err = GF_OK;
switch (OpenMode & 0xFF) {
case GF_ISOM_OPEN_READ_DUMP:
case GF_ISOM_OPEN_READ:
movie = gf_isom_open_file(fileName, OpenMode, NULL);
break;
#ifndef GPAC_DISABLE_ISOM_WRITE
case GF_ISOM_OPEN_WRITE:
movie = gf_isom_create_movie(fileName, OpenMode, tmp_dir);
break;
case GF_ISOM_OPEN_EDIT:
case GF_ISOM_OPEN_CAT_FRAGMENTS:
movie = gf_isom_open_file(fileName, OpenMode, tmp_dir);
break;
case GF_ISOM_WRITE_EDIT:
movie = gf_isom_create_movie(fileName, OpenMode, tmp_dir);
break;
#endif
default:
return NULL;
}
return (GF_ISOFile *) movie;
}
GF_EXPORT
GF_Err gf_isom_close(GF_ISOFile *movie)
{
GF_Err e;
if (movie == NULL) return GF_ISOM_INVALID_FILE;
e = GF_OK;
#ifndef GPAC_DISABLE_ISOM_WRITE
if (movie->openMode != GF_ISOM_OPEN_READ) {
gf_isom_get_duration(movie);
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if ( (movie->openMode == GF_ISOM_OPEN_WRITE) && (movie->FragmentsFlags & GF_ISOM_FRAG_WRITE_READY) ) {
e = gf_isom_close_fragments(movie);
if (e) return e;
} else
#endif
e = WriteToFile(movie);
}
#endif
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (movie->moov) {
u32 i;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
if (trak->Media && trak->Media->information && trak->Media->information->scalableDataHandler && (trak->Media->information->scalableDataHandler != movie->movieFileMap))
gf_isom_datamap_del(trak->Media->information->scalableDataHandler);
}
}
#endif
gf_isom_delete_movie(movie);
return e;
}
GF_EXPORT
Bool gf_isom_has_root_od(GF_ISOFile *movie)
{
if (!movie || !movie->moov || !movie->moov->iods || !movie->moov->iods->descriptor) return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
GF_Descriptor *gf_isom_get_root_od(GF_ISOFile *movie)
{
GF_Descriptor *desc;
GF_ObjectDescriptor *od;
GF_InitialObjectDescriptor *iod;
GF_IsomObjectDescriptor *isom_od;
GF_IsomInitialObjectDescriptor *isom_iod;
GF_ESD *esd;
GF_ES_ID_Inc *inc;
u32 i;
u8 useIOD;
if (!movie || !movie->moov) return NULL;
if (!movie->moov->iods) return NULL;
od = NULL;
iod = NULL;
switch (movie->moov->iods->descriptor->tag) {
case GF_ODF_ISOM_OD_TAG:
od = (GF_ObjectDescriptor*)gf_malloc(sizeof(GF_ObjectDescriptor));
memset(od, 0, sizeof(GF_ObjectDescriptor));
od->ESDescriptors = gf_list_new();
useIOD = 0;
break;
case GF_ODF_ISOM_IOD_TAG:
iod = (GF_InitialObjectDescriptor*)gf_malloc(sizeof(GF_InitialObjectDescriptor));
memset(iod, 0, sizeof(GF_InitialObjectDescriptor));
iod->ESDescriptors = gf_list_new();
useIOD = 1;
break;
default:
return NULL;
}
movie->LastError = gf_odf_desc_copy((GF_Descriptor *) movie->moov->iods->descriptor, &desc);
if (movie->LastError) return NULL;
if (!useIOD) {
isom_od = (GF_IsomObjectDescriptor *)desc;
od->objectDescriptorID = isom_od->objectDescriptorID;
od->extensionDescriptors = isom_od->extensionDescriptors;
isom_od->extensionDescriptors = NULL;
od->IPMP_Descriptors = isom_od->IPMP_Descriptors;
isom_od->IPMP_Descriptors = NULL;
od->OCIDescriptors = isom_od->OCIDescriptors;
isom_od->OCIDescriptors = NULL;
od->URLString = isom_od->URLString;
isom_od->URLString = NULL;
od->tag = GF_ODF_OD_TAG;
i=0;
while ((inc = (GF_ES_ID_Inc*)gf_list_enum(isom_od->ES_ID_IncDescriptors, &i))) {
movie->LastError = GetESDForTime(movie->moov, inc->trackID, 0, &esd);
if (!movie->LastError) movie->LastError = gf_list_add(od->ESDescriptors, esd);
if (movie->LastError) {
gf_odf_desc_del(desc);
gf_odf_desc_del((GF_Descriptor *) od);
return NULL;
}
}
gf_odf_desc_del(desc);
return (GF_Descriptor *)od;
} else {
isom_iod = (GF_IsomInitialObjectDescriptor *)desc;
iod->objectDescriptorID = isom_iod->objectDescriptorID;
iod->extensionDescriptors = isom_iod->extensionDescriptors;
isom_iod->extensionDescriptors = NULL;
iod->IPMP_Descriptors = isom_iod->IPMP_Descriptors;
isom_iod->IPMP_Descriptors = NULL;
iod->OCIDescriptors = isom_iod->OCIDescriptors;
isom_iod->OCIDescriptors = NULL;
iod->URLString = isom_iod->URLString;
isom_iod->URLString = NULL;
iod->tag = GF_ODF_IOD_TAG;
iod->audio_profileAndLevel = isom_iod->audio_profileAndLevel;
iod->graphics_profileAndLevel = isom_iod->graphics_profileAndLevel;
iod->inlineProfileFlag = isom_iod->inlineProfileFlag;
iod->OD_profileAndLevel = isom_iod->OD_profileAndLevel;
iod->scene_profileAndLevel = isom_iod->scene_profileAndLevel;
iod->visual_profileAndLevel = isom_iod->visual_profileAndLevel;
iod->IPMPToolList = isom_iod->IPMPToolList;
isom_iod->IPMPToolList = NULL;
i=0;
while ((inc = (GF_ES_ID_Inc*)gf_list_enum(isom_iod->ES_ID_IncDescriptors, &i))) {
movie->LastError = GetESDForTime(movie->moov, inc->trackID, 0, &esd);
if (!movie->LastError) movie->LastError = gf_list_add(iod->ESDescriptors, esd);
if (movie->LastError) {
gf_odf_desc_del(desc);
gf_odf_desc_del((GF_Descriptor *) iod);
return NULL;
}
}
gf_odf_desc_del(desc);
return (GF_Descriptor *)iod;
}
}
GF_EXPORT
u32 gf_isom_get_track_count(GF_ISOFile *movie)
{
if (!movie || !movie->moov) return 0;
if (!movie->moov->trackList) {
movie->LastError = GF_ISOM_INVALID_FILE;
return 0;
}
return gf_list_count(movie->moov->trackList);
}
GF_EXPORT
u32 gf_isom_get_track_id(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
if (!movie) return 0;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
return trak->Header->trackID;
}
GF_EXPORT
u32 gf_isom_get_track_by_id(GF_ISOFile *the_file, u32 trackID)
{
GF_TrackBox *trak;
u32 count;
u32 i;
if (the_file == NULL) return 0;
count = gf_isom_get_track_count(the_file);
if (!count) return 0;
for (i = 0; i < count; i++) {
trak = gf_isom_get_track_from_file(the_file, i+1);
if (!trak) return 0;
if (trak->Header->trackID == trackID) return i+1;
}
return 0;
}
GF_EXPORT
u32 gf_isom_get_track_original_id(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
if (!movie) return 0;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
return trak->originalID;
}
GF_EXPORT
Bool gf_isom_has_movie(GF_ISOFile *file)
{
if (file && file->moov) return GF_TRUE;
return GF_FALSE;
}
#ifndef GPAC_DISABLE_ISOM
GF_EXPORT
Bool gf_isom_has_segment(GF_ISOFile *file, u32 *brand, u32 *version)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
u32 i;
GF_Box *a;
i = 0;
while (NULL != (a = (GF_Box*)gf_list_enum(file->TopBoxes, &i))) {
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (a->type == GF_ISOM_BOX_TYPE_STYP) {
GF_SegmentTypeBox *styp = (GF_SegmentTypeBox *)a;
*brand = styp->majorBrand;
*version = styp->minorVersion;
return GF_TRUE;
}
#endif
}
#endif
return GF_FALSE;
}
GF_EXPORT
u32 gf_isom_segment_get_fragment_count(GF_ISOFile *file)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (file) {
u32 i, count = 0;
for (i=0; i<gf_list_count(file->TopBoxes); i++) {
GF_Box *a = (GF_Box*)gf_list_get(file->TopBoxes, i);
if (a->type==GF_ISOM_BOX_TYPE_MOOF) count++;
}
return count;
}
#endif
return 0;
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
static GF_MovieFragmentBox *gf_isom_get_moof(GF_ISOFile *file, u32 moof_index)
{
u32 i;
for (i=0; i<gf_list_count(file->TopBoxes); i++) {
GF_Box *a = (GF_Box*)gf_list_get(file->TopBoxes, i);
if (a->type==GF_ISOM_BOX_TYPE_MOOF) {
moof_index--;
if (!moof_index) return (GF_MovieFragmentBox *) a;
}
}
return NULL;
}
#endif
GF_EXPORT
u32 gf_isom_segment_get_track_fragment_count(GF_ISOFile *file, u32 moof_index)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_MovieFragmentBox *moof;
if (!file) return 0;
gf_list_count(file->TopBoxes);
moof = gf_isom_get_moof(file, moof_index);
return moof ? gf_list_count(moof->TrackList) : 0;
#endif
return 0;
}
GF_EXPORT
u32 gf_isom_segment_get_track_fragment_decode_time(GF_ISOFile *file, u32 moof_index, u32 traf_index, u64 *decode_time)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_MovieFragmentBox *moof;
GF_TrackFragmentBox *traf;
if (!file) return 0;
gf_list_count(file->TopBoxes);
moof = gf_isom_get_moof(file, moof_index);
traf = moof ? (GF_TrackFragmentBox*)gf_list_get(moof->TrackList, traf_index-1) : NULL;
if (!traf) return 0;
if (decode_time) {
*decode_time = traf->tfdt ? traf->tfdt->baseMediaDecodeTime : 0;
}
return traf->tfhd->trackID;
#endif
return 0;
}
#endif
GF_EXPORT
u32 gf_isom_get_timescale(GF_ISOFile *movie)
{
if (!movie || !movie->moov) return 0;
return movie->moov->mvhd->timeScale;
}
GF_EXPORT
u64 gf_isom_get_duration(GF_ISOFile *movie)
{
if (!movie || !movie->moov) return 0;
#ifndef GPAC_DISABLE_ISOM_WRITE
gf_isom_update_duration(movie);
#endif
return movie->moov->mvhd->duration;
}
GF_EXPORT
u64 gf_isom_get_original_duration(GF_ISOFile *movie)
{
if (!movie || !movie->moov) return 0;
return movie->moov->mvhd->original_duration;
}
GF_EXPORT
GF_Err gf_isom_get_creation_time(GF_ISOFile *movie, u64 *creationTime, u64 *modificationTime)
{
if (!movie || !movie->moov) return GF_BAD_PARAM;
if (creationTime) *creationTime = movie->moov->mvhd->creationTime;
if (creationTime) *modificationTime = movie->moov->mvhd->modificationTime;
return GF_OK;
}
GF_EXPORT
u8 gf_isom_is_track_in_root_od(GF_ISOFile *movie, u32 trackNumber)
{
u32 i;
u32 trackID;
GF_Descriptor *desc;
GF_ES_ID_Inc *inc;
GF_List *inc_list;
if (!movie) return 2;
if (!movie->moov || !movie->moov->iods) return 0;
desc = movie->moov->iods->descriptor;
switch (desc->tag) {
case GF_ODF_ISOM_IOD_TAG:
inc_list = ((GF_IsomInitialObjectDescriptor *)desc)->ES_ID_IncDescriptors;
break;
case GF_ODF_ISOM_OD_TAG:
inc_list = ((GF_IsomObjectDescriptor *)desc)->ES_ID_IncDescriptors;
break;
default:
return 0;
}
trackID = gf_isom_get_track_id(movie, trackNumber);
if (!trackID) return 2;
i=0;
while ((inc = (GF_ES_ID_Inc*)gf_list_enum(inc_list, &i))) {
if (inc->trackID == trackID) return 1;
}
return 0;
}
GF_EXPORT
u8 gf_isom_is_track_enabled(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 2;
return (trak->Header->flags & 1) ? 1 : 0;
}
GF_EXPORT
u64 gf_isom_get_track_duration(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
#ifndef GPAC_DISABLE_ISOM_WRITE
if (movie->openMode != GF_ISOM_OPEN_READ_DUMP) {
SetTrackDuration(trak);
}
#endif
return trak->Header->duration;
}
GF_EXPORT
GF_Err gf_isom_get_media_language(GF_ISOFile *the_file, u32 trackNumber, char **lang)
{
u32 count;
Bool elng_found = GF_FALSE;
GF_TrackBox *trak;
if (!lang) {
return GF_BAD_PARAM;
}
*lang = NULL;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
count = gf_list_count(trak->Media->other_boxes);
if (count>0) {
u32 i;
for (i = 0; i < count; i++) {
GF_Box *box = (GF_Box *)gf_list_get(trak->Media->other_boxes, i);
if (box->type == GF_ISOM_BOX_TYPE_ELNG) {
*lang = gf_strdup(((GF_ExtendedLanguageBox *)box)->extended_language);
return GF_OK;
}
}
}
if (!elng_found) {
*lang = gf_strdup(trak->Media->mediaHeader->packedLanguage);
}
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_track_kind_count(GF_ISOFile *the_file, u32 trackNumber)
{
GF_UserDataBox *udta;
GF_UserDataMap *map;
if (trackNumber) {
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
if (!trak->udta) {
return 0;
}
udta = trak->udta;
} else {
return 0;
}
map = udta_getEntry(udta, GF_ISOM_BOX_TYPE_KIND, NULL);
if (!map) return 0;
return gf_list_count(map->other_boxes);
}
GF_EXPORT
GF_Err gf_isom_get_track_kind(GF_ISOFile *the_file, u32 trackNumber, u32 index, char **scheme, char **value)
{
GF_Err e;
GF_UserDataBox *udta;
GF_UserDataMap *map;
GF_KindBox *kindBox;
if (!scheme || !value) {
return GF_BAD_PARAM;
}
*scheme = NULL;
*value = NULL;
if (trackNumber) {
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (!trak->udta) {
e = trak_AddBox((GF_Box*)trak, gf_isom_box_new(GF_ISOM_BOX_TYPE_UDTA));
if (e) return e;
}
udta = trak->udta;
} else {
return GF_BAD_PARAM;
}
map = udta_getEntry(udta, GF_ISOM_BOX_TYPE_KIND, NULL);
if (!map) return GF_BAD_PARAM;
kindBox = (GF_KindBox *)gf_list_get(map->other_boxes, index);
if (!kindBox) return GF_BAD_PARAM;
*scheme = gf_strdup(kindBox->schemeURI);
if (kindBox->value) {
*value = gf_strdup(kindBox->value);
}
return GF_OK;
}
GF_EXPORT
s32 gf_isom_get_reference_count(GF_ISOFile *movie, u32 trackNumber, u32 referenceType)
{
GF_TrackBox *trak;
GF_TrackReferenceTypeBox *dpnd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return -1;
if (!trak->References) return 0;
if (movie->openMode == GF_ISOM_OPEN_WRITE) {
movie->LastError = GF_ISOM_INVALID_MODE;
return -1;
}
dpnd = NULL;
if ( (movie->LastError = Track_FindRef(trak, referenceType, &dpnd)) ) return -1;
if (!dpnd) return 0;
return dpnd->trackIDCount;
}
GF_EXPORT
GF_Err gf_isom_get_reference(GF_ISOFile *movie, u32 trackNumber, u32 referenceType, u32 referenceIndex, u32 *refTrack)
{
GF_Err e;
GF_TrackBox *trak;
GF_TrackReferenceTypeBox *dpnd;
u32 refTrackNum;
trak = gf_isom_get_track_from_file(movie, trackNumber);
*refTrack = 0;
if (!trak || !trak->References) return GF_BAD_PARAM;
dpnd = NULL;
e = Track_FindRef(trak, referenceType, &dpnd);
if (e) return e;
if (!dpnd) return GF_BAD_PARAM;
if (referenceIndex > dpnd->trackIDCount) return GF_BAD_PARAM;
if (dpnd->trackIDs[referenceIndex - 1] == 0) return GF_OK;
refTrackNum = gf_isom_get_tracknum_from_id(movie->moov, dpnd->trackIDs[referenceIndex-1]);
if (! refTrackNum) return GF_ISOM_INVALID_FILE;
*refTrack = refTrackNum;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_reference_ID(GF_ISOFile *movie, u32 trackNumber, u32 referenceType, u32 referenceIndex, u32 *refTrackID)
{
GF_Err e;
GF_TrackBox *trak;
GF_TrackReferenceTypeBox *dpnd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
*refTrackID = 0;
if (!trak || !trak->References) return GF_BAD_PARAM;
dpnd = NULL;
e = Track_FindRef(trak, referenceType, &dpnd);
if (e) return e;
if (!dpnd) return GF_BAD_PARAM;
if (referenceIndex > dpnd->trackIDCount) return GF_BAD_PARAM;
*refTrackID = dpnd->trackIDs[referenceIndex-1];
return GF_OK;
}
GF_EXPORT
u32 gf_isom_has_track_reference(GF_ISOFile *movie, u32 trackNumber, u32 referenceType, u32 refTrackID)
{
u32 i;
GF_TrackBox *trak;
GF_TrackReferenceTypeBox *dpnd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
if (!trak->References) return 0;
dpnd = NULL;
if ( (movie->LastError = Track_FindRef(trak, referenceType, &dpnd)) ) return 0;
if (!dpnd) return 0;
for (i=0; i<dpnd->trackIDCount; i++) {
if (dpnd->trackIDs[i]==refTrackID) return i+1;
}
return 0;
}
GF_EXPORT
GF_Err gf_isom_get_media_time(GF_ISOFile *the_file, u32 trackNumber, u32 movieTime, u64 *MediaTime)
{
GF_TrackBox *trak;
u8 useEdit;
s64 SegmentStartTime, mediaOffset;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !MediaTime) return GF_BAD_PARAM;;
SegmentStartTime = 0;
return GetMediaTime(trak, GF_FALSE, movieTime, MediaTime, &SegmentStartTime, &mediaOffset, &useEdit, NULL);
}
GF_EXPORT
u32 gf_isom_get_sample_description_index(GF_ISOFile *movie, u32 trackNumber, u64 for_time)
{
u32 streamDescIndex;
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
if ( (movie->LastError = Media_GetSampleDescIndex(trak->Media, for_time, &streamDescIndex)) ) {
return 0;
}
return streamDescIndex;
}
GF_EXPORT
u32 gf_isom_get_sample_description_count(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return gf_list_count(trak->Media->information->sampleTable->SampleDescription->other_boxes);
}
GF_EXPORT
GF_ESD *gf_isom_get_esd(GF_ISOFile *movie, u32 trackNumber, u32 StreamDescriptionIndex)
{
GF_ESD *esd;
GF_Err e;
e = GetESD(movie->moov, gf_isom_get_track_id(movie, trackNumber), StreamDescriptionIndex, &esd);
if (e && (e!= GF_ISOM_INVALID_MEDIA)) {
movie->LastError = e;
return NULL;
}
return esd;
}
GF_EXPORT
GF_DecoderConfig *gf_isom_get_decoder_config(GF_ISOFile *the_file, u32 trackNumber, u32 StreamDescriptionIndex)
{
GF_TrackBox *trak;
GF_ESD *esd;
GF_Descriptor *decInfo;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return NULL;
Media_GetESD(trak->Media, StreamDescriptionIndex, &esd, GF_FALSE);
if (!esd) return NULL;
decInfo = (GF_Descriptor *) esd->decoderConfig;
esd->decoderConfig = NULL;
gf_odf_desc_del((GF_Descriptor *) esd);
return (GF_DecoderConfig *)decInfo;
}
GF_EXPORT
u64 gf_isom_get_media_duration(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
#ifndef GPAC_DISABLE_ISOM_WRITE
if (movie->openMode != GF_ISOM_OPEN_READ_DUMP) {
if ( (movie->LastError = Media_SetDuration(trak)) ) return 0;
}
#endif
return trak->Media->mediaHeader->duration;
}
GF_EXPORT
u64 gf_isom_get_media_original_duration(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
return trak->Media->mediaHeader->original_duration;
}
GF_EXPORT
u32 gf_isom_get_media_timescale(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return trak->Media->mediaHeader->timeScale;
}
GF_EXPORT
u32 gf_isom_get_copyright_count(GF_ISOFile *mov)
{
GF_UserDataMap *map;
if (!mov || !mov->moov || !mov->moov->udta) return 0;
map = udta_getEntry(mov->moov->udta, GF_ISOM_BOX_TYPE_CPRT, NULL);
if (!map) return 0;
return gf_list_count(map->other_boxes);
}
GF_EXPORT
GF_Err gf_isom_get_copyright(GF_ISOFile *mov, u32 Index, const char **threeCharCode, const char **notice)
{
GF_UserDataMap *map;
GF_CopyrightBox *cprt;
if (!mov || !mov->moov || !Index) return GF_BAD_PARAM;
if (!mov->moov->udta) return GF_OK;
map = udta_getEntry(mov->moov->udta, GF_ISOM_BOX_TYPE_CPRT, NULL);
if (!map) return GF_OK;
if (Index > gf_list_count(map->other_boxes)) return GF_BAD_PARAM;
cprt = (GF_CopyrightBox*)gf_list_get(map->other_boxes, Index-1);
(*threeCharCode) = cprt->packedLanguageCode;
(*notice) = cprt->notice;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_watermark(GF_ISOFile *mov, bin128 UUID, u8** data, u32* length)
{
GF_UserDataMap *map;
GF_UnknownUUIDBox *wm;
if (!mov) return GF_BAD_PARAM;
if (!mov->moov || !mov->moov->udta) return GF_NOT_SUPPORTED;
map = udta_getEntry(mov->moov->udta, GF_ISOM_BOX_TYPE_UUID, (bin128 *) & UUID);
if (!map) return GF_NOT_SUPPORTED;
wm = (GF_UnknownUUIDBox*)gf_list_get(map->other_boxes, 0);
if (!wm) return GF_NOT_SUPPORTED;
*data = (u8 *) gf_malloc(sizeof(char)*wm->dataSize);
memcpy(*data, wm->data, wm->dataSize);
*length = wm->dataSize;
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_chapter_count(GF_ISOFile *movie, u32 trackNumber)
{
GF_UserDataMap *map;
GF_ChapterListBox *lst;
GF_UserDataBox *udta;
if (!movie || !movie->moov) return 0;
udta = NULL;
if (trackNumber) {
GF_TrackBox *trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (!udta) return 0;
map = udta_getEntry(udta, GF_ISOM_BOX_TYPE_CHPL, NULL);
if (!map) return 0;
lst = (GF_ChapterListBox *)gf_list_get(map->other_boxes, 0);
if (!lst) return 0;
return gf_list_count(lst->list);
}
GF_EXPORT
GF_Err gf_isom_get_chapter(GF_ISOFile *movie, u32 trackNumber, u32 Index, u64 *chapter_time, const char **name)
{
GF_UserDataMap *map;
GF_ChapterListBox *lst;
GF_ChapterEntry *ce;
GF_UserDataBox *udta;
if (!movie || !movie->moov) return GF_BAD_PARAM;
udta = NULL;
if (trackNumber) {
GF_TrackBox *trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (!udta) return GF_BAD_PARAM;
map = udta_getEntry(movie->moov->udta, GF_ISOM_BOX_TYPE_CHPL, NULL);
if (!map) return GF_BAD_PARAM;
lst = (GF_ChapterListBox *)gf_list_get(map->other_boxes, 0);
if (!lst) return GF_BAD_PARAM;
ce = (GF_ChapterEntry *)gf_list_get(lst->list, Index-1);
if (!ce) return GF_BAD_PARAM;
if (chapter_time) {
*chapter_time = ce->start_time;
*chapter_time /= 10000L;
}
if (name) *name = ce->name;
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_media_type(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
return (trak->Media && trak->Media->handler) ? trak->Media->handler->handlerType : 0;
}
Bool IsMP4Description(u32 entryType)
{
switch (entryType) {
case GF_ISOM_BOX_TYPE_MP4S:
case GF_ISOM_BOX_TYPE_MP4A:
case GF_ISOM_BOX_TYPE_MP4V:
case GF_ISOM_BOX_TYPE_ENCA:
case GF_ISOM_BOX_TYPE_ENCV:
case GF_ISOM_BOX_TYPE_RESV:
case GF_ISOM_BOX_TYPE_ENCS:
return GF_TRUE;
default:
return GF_FALSE;
}
}
Bool IsMP4EncryptedDescription(u32 entryType)
{
switch (entryType) {
case GF_ISOM_BOX_TYPE_ENCA:
case GF_ISOM_BOX_TYPE_ENCV:
case GF_ISOM_BOX_TYPE_ENCS:
return GF_TRUE;
default:
return GF_FALSE;
}
}
GF_EXPORT
Bool gf_isom_is_track_encrypted(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
GF_Box *entry;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 2;
entry = (GF_Box*)gf_list_get(trak->Media->information->sampleTable->SampleDescription->other_boxes, 0);
if (!entry) return 2;
if (IsMP4EncryptedDescription(entry->type)) return GF_TRUE;
if (gf_isom_is_cenc_media(the_file, trackNumber, 1))
return GF_TRUE;
return GF_FALSE;
}
GF_EXPORT
u32 gf_isom_get_media_subtype(GF_ISOFile *the_file, u32 trackNumber, u32 DescriptionIndex)
{
GF_TrackBox *trak;
GF_Box *entry;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !DescriptionIndex || !trak->Media || !trak->Media->information || !trak->Media->information->sampleTable) return 0;
entry = (GF_Box*)gf_list_get(trak->Media->information->sampleTable->SampleDescription->other_boxes, DescriptionIndex-1);
if (!entry) return 0;
if (IsMP4Description(entry->type)) {
if (IsMP4EncryptedDescription(entry->type)) return GF_ISOM_SUBTYPE_MPEG4_CRYP;
else return GF_ISOM_SUBTYPE_MPEG4;
}
if (entry->type == GF_ISOM_BOX_TYPE_GNRV) {
return ((GF_GenericVisualSampleEntryBox *)entry)->EntryType;
}
else if (entry->type == GF_ISOM_BOX_TYPE_GNRA) {
return ((GF_GenericAudioSampleEntryBox *)entry)->EntryType;
}
else if (entry->type == GF_ISOM_BOX_TYPE_GNRM) {
return ((GF_GenericSampleEntryBox *)entry)->EntryType;
}
return entry->type;
}
GF_EXPORT
u32 gf_isom_get_mpeg4_subtype(GF_ISOFile *the_file, u32 trackNumber, u32 DescriptionIndex)
{
GF_TrackBox *trak;
GF_Box *entry;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !DescriptionIndex) return 0;
entry = (GF_Box*)gf_list_get(trak->Media->information->sampleTable->SampleDescription->other_boxes, DescriptionIndex-1);
if (!entry) return 0;
if (!IsMP4Description(entry->type)) return 0;
return entry->type;
}
GF_EXPORT
GF_Err gf_isom_get_handler_name(GF_ISOFile *the_file, u32 trackNumber, const char **outName)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !outName) return GF_BAD_PARAM;
*outName = trak->Media->handler->nameUTF8;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_check_data_reference(GF_ISOFile *the_file, u32 trackNumber, u32 StreamDescriptionIndex)
{
GF_Err e;
u32 drefIndex;
GF_TrackBox *trak;
if (!StreamDescriptionIndex || !trackNumber) return GF_BAD_PARAM;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
e = Media_GetSampleDesc(trak->Media, StreamDescriptionIndex , NULL, &drefIndex);
if (e) return e;
if (!drefIndex) return GF_BAD_PARAM;
return Media_CheckDataEntry(trak->Media, drefIndex);
}
GF_EXPORT
GF_Err gf_isom_get_data_reference(GF_ISOFile *the_file, u32 trackNumber, u32 StreamDescriptionIndex, const char **outURL, const char **outURN)
{
GF_TrackBox *trak;
GF_DataEntryURLBox *url;
GF_DataEntryURNBox *urn;
u32 drefIndex;
GF_Err e;
*outURL = *outURN = NULL;
if (!StreamDescriptionIndex || !trackNumber) return GF_BAD_PARAM;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
e = Media_GetSampleDesc(trak->Media, StreamDescriptionIndex , NULL, &drefIndex);
if (e) return e;
if (!drefIndex) return GF_BAD_PARAM;
url = (GF_DataEntryURLBox*)gf_list_get(trak->Media->information->dataInformation->dref->other_boxes, drefIndex - 1);
if (!url) return GF_ISOM_INVALID_FILE;
*outURL = *outURN = NULL;
if (url->type == GF_ISOM_BOX_TYPE_URL) {
*outURL = url->location;
*outURN = NULL;
} else if (url->type == GF_ISOM_BOX_TYPE_URN) {
urn = (GF_DataEntryURNBox *) url;
*outURN = urn->nameURN;
*outURL = urn->location;
} else {
*outURN = NULL;
*outURL = NULL;
}
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_sample_count(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Media->information->sampleTable->SampleSize) return 0;
return trak->Media->information->sampleTable->SampleSize->sampleCount
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
+ trak->sample_count_at_seg_start
#endif
;
}
u32 gf_isom_get_constant_sample_size(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return trak->Media->information->sampleTable->SampleSize->sampleSize;
}
GF_EXPORT
u32 gf_isom_has_time_offset(GF_ISOFile *the_file, u32 trackNumber)
{
u32 i;
GF_CompositionOffsetBox *ctts;
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Media->information->sampleTable->CompositionOffset) return 0;
ctts = trak->Media->information->sampleTable->CompositionOffset;
for (i=0; i<ctts->nb_entries; i++) {
if (ctts->entries[i].decodingOffset && ctts->entries[i].sampleCount) return ctts->version ? 2 : 1;
}
return 0;
}
GF_EXPORT
s64 gf_isom_get_cts_to_dts_shift(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Media->information->sampleTable->CompositionToDecode) return 0;
return trak->Media->information->sampleTable->CompositionToDecode->compositionToDTSShift;
}
GF_EXPORT
Bool gf_isom_has_sync_shadows(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
if (!trak->Media->information->sampleTable->ShadowSync) return GF_FALSE;
if (gf_list_count(trak->Media->information->sampleTable->ShadowSync->entries) ) return GF_TRUE;
return GF_FALSE;
}
GF_EXPORT
Bool gf_isom_has_sample_dependency(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
if (!trak->Media->information->sampleTable->SampleDep) return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
GF_Err gf_isom_get_sample_flags(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, u32 *isLeading, u32 *dependsOn, u32 *dependedOn, u32 *redundant)
{
GF_TrackBox *trak = NULL;
*isLeading = 0;
*dependsOn = 0;
*dependedOn = 0;
*redundant = 0;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (!trak->Media->information->sampleTable->SampleDep) return GF_BAD_PARAM;
return stbl_GetSampleDepType(trak->Media->information->sampleTable->SampleDep, sampleNumber, isLeading, dependsOn, dependedOn, redundant);
}
GF_EXPORT
GF_ISOSample *gf_isom_get_sample(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, u32 *sampleDescriptionIndex)
{
GF_Err e;
u32 descIndex;
GF_TrackBox *trak;
GF_ISOSample *samp;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return NULL;
if (!sampleNumber) return NULL;
samp = gf_isom_sample_new();
if (!samp) return NULL;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start)
return NULL;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
e = Media_GetSample(trak->Media, sampleNumber, &samp, &descIndex, GF_FALSE, NULL);
if (e) {
gf_isom_set_last_error(the_file, e);
gf_isom_sample_del(&samp);
return NULL;
}
if (sampleDescriptionIndex) *sampleDescriptionIndex = descIndex;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (samp) samp->DTS += trak->dts_at_seg_start;
#endif
return samp;
}
GF_EXPORT
u32 gf_isom_get_sample_duration(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber)
{
u32 dur;
u64 dts;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !sampleNumber) return 0;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start) return 0;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
stbl_GetSampleDTS_and_Duration(trak->Media->information->sampleTable->TimeToSample, sampleNumber, &dts, &dur);
return dur;
}
GF_EXPORT
u32 gf_isom_get_sample_size(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber)
{
u32 size = 0;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !sampleNumber) return 0;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start) return 0;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
stbl_GetSampleSize(trak->Media->information->sampleTable->SampleSize, sampleNumber, &size);
return size;
}
GF_EXPORT
u8 gf_isom_get_sample_sync(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber)
{
SAPType is_rap;
GF_Err e;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !sampleNumber) return 0;
if (! trak->Media->information->sampleTable->SyncSample) return 1;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start) return 0;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
e = stbl_GetSampleRAP(trak->Media->information->sampleTable->SyncSample, sampleNumber, &is_rap, NULL, NULL);
if (e) return 0;
return is_rap;
}
GF_EXPORT
GF_ISOSample *gf_isom_get_sample_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, u32 *sampleDescriptionIndex, u64 *data_offset)
{
GF_Err e;
GF_TrackBox *trak;
GF_ISOSample *samp;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return NULL;
if (!sampleNumber) return NULL;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start) return NULL;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
samp = gf_isom_sample_new();
if (!samp) return NULL;
e = Media_GetSample(trak->Media, sampleNumber, &samp, sampleDescriptionIndex, GF_TRUE, data_offset);
if (e) {
gf_isom_set_last_error(the_file, e);
gf_isom_sample_del(&samp);
return NULL;
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (samp) samp->DTS += trak->dts_at_seg_start;
#endif
return samp;
}
GF_EXPORT
u64 gf_isom_get_sample_dts(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber)
{
u64 dts;
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
if (!sampleNumber) return 0;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (sampleNumber<=trak->sample_count_at_seg_start) return 0;
sampleNumber -= trak->sample_count_at_seg_start;
#endif
if (stbl_GetSampleDTS(trak->Media->information->sampleTable->TimeToSample, sampleNumber, &dts) != GF_OK) return 0;
return dts;
}
GF_EXPORT
Bool gf_isom_is_self_contained(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
return Media_IsSelfContained(trak->Media, sampleDescriptionIndex);
}
u32 gf_isom_get_sample_from_dts(GF_ISOFile *the_file, u32 trackNumber, u64 dts)
{
GF_Err e;
u32 sampleNumber, prevSampleNumber;
GF_TrackBox *trak;
GF_SampleTableBox *stbl;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
stbl = trak->Media->information->sampleTable;
e = stbl_findEntryForTime(stbl, dts, 1, &sampleNumber, &prevSampleNumber);
if (e) return 0;
return sampleNumber;
}
GF_EXPORT
GF_Err gf_isom_get_sample_for_media_time(GF_ISOFile *the_file, u32 trackNumber, u64 desiredTime, u32 *StreamDescriptionIndex, u8 SearchMode, GF_ISOSample **sample, u32 *SampleNum)
{
GF_Err e;
u32 sampleNumber, prevSampleNumber, syncNum, shadowSync;
GF_TrackBox *trak;
GF_ISOSample *shadow;
GF_SampleTableBox *stbl;
u8 useShadow, IsSync;
if (!sample) return GF_BAD_PARAM;
if (SampleNum) *SampleNum = 0;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (desiredTime < trak->dts_at_seg_start) {
desiredTime = 0;
} else {
desiredTime -= trak->dts_at_seg_start;
}
#endif
e = stbl_findEntryForTime(stbl, desiredTime, 0, &sampleNumber, &prevSampleNumber);
if (e) return e;
useShadow = 0;
if (!stbl->ShadowSync && (SearchMode == GF_ISOM_SEARCH_SYNC_SHADOW))
SearchMode = GF_ISOM_SEARCH_SYNC_BACKWARD;
if (! trak->Media->information->sampleTable->SyncSample) {
if (SearchMode == GF_ISOM_SEARCH_SYNC_FORWARD) SearchMode = GF_ISOM_SEARCH_FORWARD;
if (SearchMode == GF_ISOM_SEARCH_SYNC_BACKWARD) SearchMode = GF_ISOM_SEARCH_BACKWARD;
}
if (!sampleNumber && !prevSampleNumber) {
if (SearchMode == GF_ISOM_SEARCH_SYNC_BACKWARD || SearchMode == GF_ISOM_SEARCH_BACKWARD) {
sampleNumber = trak->Media->information->sampleTable->SampleSize->sampleCount;
}
if (!sampleNumber) return GF_EOS;
}
IsSync = 0;
switch (SearchMode) {
case GF_ISOM_SEARCH_SYNC_FORWARD:
IsSync = 1;
case GF_ISOM_SEARCH_FORWARD:
if (!sampleNumber) {
if (prevSampleNumber != stbl->SampleSize->sampleCount) {
sampleNumber = prevSampleNumber + 1;
} else {
sampleNumber = prevSampleNumber;
}
}
break;
case GF_ISOM_SEARCH_SYNC_BACKWARD:
IsSync = 1;
case GF_ISOM_SEARCH_SYNC_SHADOW:
case GF_ISOM_SEARCH_BACKWARD:
default:
if (!sampleNumber && !prevSampleNumber) {
sampleNumber = stbl->SampleSize->sampleCount;
} else if (!sampleNumber) {
sampleNumber = prevSampleNumber;
}
break;
}
if (IsSync) {
e = Media_FindSyncSample(trak->Media->information->sampleTable,
sampleNumber, &syncNum, SearchMode);
if (e) return e;
if (syncNum) sampleNumber = syncNum;
syncNum = 0;
}
else if (SearchMode == GF_ISOM_SEARCH_SYNC_SHADOW) {
e = Media_FindSyncSample(trak->Media->information->sampleTable,
sampleNumber, &syncNum, GF_ISOM_SEARCH_SYNC_BACKWARD);
if (e) return e;
}
*sample = gf_isom_sample_new();
if (*sample == NULL) return GF_OUT_OF_MEM;
if (SearchMode == GF_ISOM_SEARCH_SYNC_SHADOW) {
stbl_GetSampleShadow(stbl->ShadowSync, &sampleNumber, &shadowSync);
if ((sampleNumber < syncNum) || (!shadowSync)) {
sampleNumber = syncNum;
} else {
useShadow = 1;
}
}
e = Media_GetSample(trak->Media, sampleNumber, sample, StreamDescriptionIndex, GF_FALSE, NULL);
if (e) {
gf_isom_sample_del(sample);
return e;
}
if (! (*sample)->IsRAP) {
Bool has_roll, is_rap;
e = gf_isom_get_sample_rap_roll_info(the_file, trackNumber, sampleNumber, &is_rap, &has_roll, NULL);
if (e) return e;
if (is_rap) (*sample)->IsRAP = SAP_TYPE_3;
}
if (SampleNum) {
*SampleNum = sampleNumber;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
*SampleNum += trak->sample_count_at_seg_start;
#endif
}
if (useShadow) {
shadow = gf_isom_get_sample(the_file, trackNumber, shadowSync, StreamDescriptionIndex);
if (!shadow) return GF_OK;
(*sample)->IsRAP = RAP;
gf_free((*sample)->data);
(*sample)->dataLength = shadow->dataLength;
(*sample)->data = shadow->data;
shadow->dataLength = 0;
gf_isom_sample_del(&shadow);
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_sample_for_movie_time(GF_ISOFile *the_file, u32 trackNumber, u64 movieTime, u32 *StreamDescriptionIndex, u8 SearchMode, GF_ISOSample **sample, u32 *sampleNumber)
{
Double tsscale;
GF_Err e;
GF_TrackBox *trak;
u64 mediaTime, nextMediaTime;
s64 segStartTime, mediaOffset;
u32 sampNum;
u8 useEdit;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (*sample || !sample) return GF_BAD_PARAM;
if (!trak->Header->duration) {
if (movieTime && ( (SearchMode == GF_ISOM_SEARCH_SYNC_FORWARD) || (SearchMode == GF_ISOM_SEARCH_FORWARD)) ) {
*sample = NULL;
if (sampleNumber) *sampleNumber = 0;
*StreamDescriptionIndex = 0;
return GF_EOS;
}
}
else if ((movieTime * trak->moov->mvhd->timeScale > trak->Header->duration * trak->Media->mediaHeader->timeScale)
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
&& !trak->dts_at_seg_start
#endif
) {
*sample = NULL;
if (sampleNumber) *sampleNumber = 0;
*StreamDescriptionIndex = 0;
return GF_EOS;
}
mediaTime = segStartTime = 0;
*StreamDescriptionIndex = 0;
nextMediaTime = 0;
e = GetMediaTime(trak, (SearchMode==GF_ISOM_SEARCH_SYNC_FORWARD) ? GF_TRUE : GF_FALSE, movieTime, &mediaTime, &segStartTime, &mediaOffset, &useEdit, &nextMediaTime);
if (e) return e;
if (useEdit && mediaOffset == -1) {
if ((SearchMode==GF_ISOM_SEARCH_FORWARD) || (SearchMode==GF_ISOM_SEARCH_BACKWARD)) {
if (SearchMode==GF_ISOM_SEARCH_FORWARD)
e = GetNextMediaTime(trak, movieTime, &mediaTime);
else
e = GetPrevMediaTime(trak, movieTime, &mediaTime);
if (e) return e;
return gf_isom_get_sample_for_movie_time(the_file, trackNumber, (u32) mediaTime, StreamDescriptionIndex, GF_ISOM_SEARCH_SYNC_FORWARD, sample, sampleNumber);
}
if (sampleNumber) *sampleNumber = 0;
*sample = gf_isom_sample_new();
if (! *sample) return GF_OUT_OF_MEM;
(*sample)->DTS = movieTime;
return GF_OK;
}
if (useEdit==2) {
if ((SearchMode==GF_ISOM_SEARCH_FORWARD) || (SearchMode==GF_ISOM_SEARCH_BACKWARD)) {
if (SearchMode==GF_ISOM_SEARCH_FORWARD)
e = GetNextMediaTime(trak, movieTime, &mediaTime);
else
e = GetPrevMediaTime(trak, movieTime, &mediaTime);
if (e) return e;
return gf_isom_get_sample_for_movie_time(the_file, trackNumber, (u32) mediaTime, StreamDescriptionIndex, GF_ISOM_SEARCH_SYNC_FORWARD, sample, sampleNumber);
}
}
tsscale = trak->Media->mediaHeader->timeScale;
tsscale /= trak->moov->mvhd->timeScale;
e = gf_isom_get_sample_for_media_time(the_file, trackNumber, mediaTime, StreamDescriptionIndex, SearchMode, sample, &sampNum);
if (e) {
if (e==GF_EOS) {
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (the_file->moov->mvex && !trak->Media->information->sampleTable->SampleSize->sampleCount)
return e;
#endif
if (nextMediaTime)
return gf_isom_get_sample_for_movie_time(the_file, trackNumber, nextMediaTime-1, StreamDescriptionIndex, SearchMode, sample, sampleNumber);
}
return e;
}
if (useEdit) {
u64 _ts = (u64)(segStartTime * tsscale);
(*sample)->DTS += _ts;
if ( (*sample)->DTS > (u64) mediaOffset) {
(*sample)->DTS -= (u64) mediaOffset;
} else {
(*sample)->DTS = 0;
}
}
if (sampleNumber) *sampleNumber = sampNum;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if ( (*sample) ) (*sample)->DTS += trak->dts_at_seg_start;
#endif
return GF_OK;
}
GF_EXPORT
u64 gf_isom_get_missing_bytes(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return trak->Media->BytesMissing;
}
GF_EXPORT
GF_Err gf_isom_set_sample_padding(GF_ISOFile *the_file, u32 trackNumber, u32 padding_bytes)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
trak->padding_bytes = padding_bytes;
return GF_OK;
}
GF_EXPORT
Bool gf_isom_get_edit_list_type(GF_ISOFile *the_file, u32 trackNumber, s64 *mediaOffset)
{
GF_EdtsEntry *ent;
GF_TrackBox *trak;
u32 count;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
*mediaOffset = 0;
if (!trak->editBox || !trak->editBox->editList) return GF_FALSE;
count = gf_list_count(trak->editBox->editList->entryList);
ent = (GF_EdtsEntry*)gf_list_get(trak->editBox->editList->entryList, 0);
if (!ent) return GF_TRUE;
if ((count==1) && (ent->mediaRate==1)) {
*mediaOffset = - ent->mediaTime;
return GF_FALSE;
} else if (count==2) {
if ((ent->mediaRate==-1) || (ent->mediaTime==-1)) {
Double time = (Double) ent->segmentDuration;
time /= trak->moov->mvhd->timeScale;
time *= trak->Media->mediaHeader->timeScale;
*mediaOffset = (s64) time;
return GF_FALSE;
}
}
return GF_TRUE;
}
GF_EXPORT
u32 gf_isom_get_edit_segment_count(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
if (!trak->editBox || !trak->editBox->editList) return 0;
return gf_list_count(trak->editBox->editList->entryList);
}
GF_EXPORT
GF_Err gf_isom_get_edit_segment(GF_ISOFile *the_file, u32 trackNumber, u32 SegmentIndex, u64 *EditTime, u64 *SegmentDuration, u64 *MediaTime, u8 *EditMode)
{
u32 i;
u64 startTime;
GF_TrackBox *trak;
GF_EditListBox *elst;
GF_EdtsEntry *ent;
ent = NULL;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (!trak->editBox ||
!trak->editBox->editList ||
(SegmentIndex > gf_list_count(trak->editBox->editList->entryList)) ||
!SegmentIndex)
return GF_BAD_PARAM;
elst = trak->editBox->editList;
startTime = 0;
for (i = 0; i < SegmentIndex; i++) {
ent = (GF_EdtsEntry*)gf_list_get(elst->entryList, i);
if (i < SegmentIndex-1) startTime += ent->segmentDuration;
}
*EditTime = startTime;
*SegmentDuration = ent->segmentDuration;
if (ent->mediaTime < 0) {
*MediaTime = 0;
*EditMode = GF_ISOM_EDIT_EMPTY;
return GF_OK;
}
if (ent->mediaRate == 0) {
*MediaTime = ent->mediaTime;
*EditMode = GF_ISOM_EDIT_DWELL;
return GF_OK;
}
*MediaTime = ent->mediaTime;
*EditMode = GF_ISOM_EDIT_NORMAL;
return GF_OK;
}
GF_EXPORT
u8 gf_isom_has_sync_points(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
if (trak->Media->information->sampleTable->SyncSample) {
if (!trak->Media->information->sampleTable->SyncSample->nb_entries) return 2;
return 1;
}
return 0;
}
GF_EXPORT
u32 gf_isom_get_sync_point_count(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
if (trak->Media->information->sampleTable->SyncSample) {
return trak->Media->information->sampleTable->SyncSample->nb_entries;
}
return 0;
}
GF_EXPORT
GF_Err gf_isom_get_brand_info(GF_ISOFile *movie, u32 *brand, u32 *minorVersion, u32 *AlternateBrandsCount)
{
if (!movie || !brand) return GF_BAD_PARAM;
if (!movie->brand) {
*brand = 0;
if (minorVersion) *minorVersion = 0;
if (AlternateBrandsCount) *AlternateBrandsCount = 0;
return GF_OK;
}
*brand = movie->brand->majorBrand;
if (minorVersion) *minorVersion = movie->brand->minorVersion;
if (AlternateBrandsCount) *AlternateBrandsCount = movie->brand->altCount;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_alternate_brand(GF_ISOFile *movie, u32 BrandIndex, u32 *brand)
{
if (!movie || !movie->brand || !brand) return GF_BAD_PARAM;
if (BrandIndex > movie->brand->altCount || !BrandIndex) return GF_BAD_PARAM;
*brand = movie->brand->altBrand[BrandIndex-1];
return GF_OK;
}
GF_Err gf_isom_get_sample_padding_bits(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, u8 *NbBits)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
return stbl_GetPaddingBits(trak->Media->information->sampleTable->PaddingBits,
sampleNumber, NbBits);
}
GF_EXPORT
Bool gf_isom_has_padding_bits(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
if (trak->Media->information->sampleTable->PaddingBits) return GF_TRUE;
return GF_FALSE;
}
GF_EXPORT
u32 gf_isom_get_udta_count(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *trak;
GF_UserDataBox *udta;
if (!movie || !movie->moov) return 0;
if (trackNumber) {
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (udta) return gf_list_count(udta->recordList);
return 0;
}
GF_EXPORT
GF_Err gf_isom_get_udta_type(GF_ISOFile *movie, u32 trackNumber, u32 udta_idx, u32 *UserDataType, bin128 *UUID)
{
GF_TrackBox *trak;
GF_UserDataBox *udta;
GF_UserDataMap *map;
if (!movie || !movie->moov || !udta_idx) return GF_BAD_PARAM;
if (trackNumber) {
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_OK;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (!udta) return GF_BAD_PARAM;
if (udta_idx>gf_list_count(udta->recordList)) return GF_BAD_PARAM;
map = (GF_UserDataMap*)gf_list_get(udta->recordList, udta_idx - 1);
if (UserDataType) *UserDataType = map->boxType;
if (UUID) memcpy(*UUID, map->uuid, 16);
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_user_data_count(GF_ISOFile *movie, u32 trackNumber, u32 UserDataType, bin128 UUID)
{
GF_UserDataMap *map;
GF_TrackBox *trak;
GF_UserDataBox *udta;
bin128 t;
u32 i, count;
if (!movie || !movie->moov) return 0;
if (UserDataType == GF_ISOM_BOX_TYPE_UUID) UserDataType = 0;
memset(t, 1, 16);
if (trackNumber) {
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return 0;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (!udta) return 0;
i=0;
while ((map = (GF_UserDataMap*)gf_list_enum(udta->recordList, &i))) {
count = gf_list_count(map->other_boxes);
if ((map->boxType == GF_ISOM_BOX_TYPE_UUID) && !memcmp(map->uuid, UUID, 16)) return count;
else if (map->boxType == UserDataType) return count;
}
return 0;
}
GF_EXPORT
GF_Err gf_isom_get_user_data(GF_ISOFile *movie, u32 trackNumber, u32 UserDataType, bin128 UUID, u32 UserDataIndex, char **userData, u32 *userDataSize)
{
GF_UserDataMap *map;
GF_UnknownBox *ptr;
GF_BitStream *bs;
u32 i;
bin128 t;
GF_TrackBox *trak;
GF_UserDataBox *udta;
if (!movie || !movie->moov) return GF_BAD_PARAM;
if (trackNumber) {
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
udta = trak->udta;
} else {
udta = movie->moov->udta;
}
if (!udta) return GF_BAD_PARAM;
if (UserDataType == GF_ISOM_BOX_TYPE_UUID) UserDataType = 0;
memset(t, 1, 16);
if (!userData || !userDataSize || *userData) return GF_BAD_PARAM;
i=0;
while ((map = (GF_UserDataMap*)gf_list_enum(udta->recordList, &i))) {
if ((map->boxType == GF_ISOM_BOX_TYPE_UUID) && !memcmp(map->uuid, UUID, 16)) goto found;
else if (map->boxType == UserDataType) goto found;
}
return GF_BAD_PARAM;
found:
if (UserDataIndex) {
if (UserDataIndex > gf_list_count(map->other_boxes) ) return GF_BAD_PARAM;
ptr = (GF_UnknownBox*)gf_list_get(map->other_boxes, UserDataIndex-1);
if (ptr->type == GF_ISOM_BOX_TYPE_UNKNOWN) {
*userData = (char *)gf_malloc(sizeof(char)*ptr->dataSize);
if (!*userData) return GF_OUT_OF_MEM;
memcpy(*userData, ptr->data, sizeof(char)*ptr->dataSize);
*userDataSize = ptr->dataSize;
return GF_OK;
} else if (ptr->type != GF_ISOM_BOX_TYPE_UUID) {
GF_UnknownUUIDBox *p_uuid = (GF_UnknownUUIDBox *)ptr;
*userData = (char *)gf_malloc(sizeof(char)*p_uuid->dataSize);
if (!*userData) return GF_OUT_OF_MEM;
memcpy(*userData, p_uuid->data, sizeof(char)*p_uuid->dataSize);
*userDataSize = p_uuid->dataSize;
return GF_OK;
} else {
return GF_ISOM_INVALID_FILE;
}
}
bs = gf_bs_new(NULL, 0, GF_BITSTREAM_WRITE);
i=0;
while ( (ptr = (GF_UnknownBox*)gf_list_enum(map->other_boxes, &i))) {
u32 type, s, data_size;
char *data=NULL;
if (ptr->type == GF_ISOM_BOX_TYPE_UNKNOWN) {
type = ptr->original_4cc;
data_size = ptr->dataSize;
data = ptr->data;
} else if (ptr->type == GF_ISOM_BOX_TYPE_UUID) {
GF_UnknownUUIDBox *p_uuid = (GF_UnknownUUIDBox *)ptr;
type = p_uuid->type;
data_size = p_uuid->dataSize;
data = p_uuid->data;
} else {
gf_isom_box_write((GF_Box *)ptr, bs);
continue;
}
s = data_size+8;
if (ptr->type==GF_ISOM_BOX_TYPE_UUID) s += 16;
gf_bs_write_u32(bs, s);
gf_bs_write_u32(bs, type);
if (type==GF_ISOM_BOX_TYPE_UUID) gf_bs_write_data(bs, (char *) map->uuid, 16);
if (data) {
gf_bs_write_data(bs, data, data_size);
} else if (ptr->other_boxes) {
#ifndef GPAC_DISABLE_ISOM_WRITE
gf_isom_box_array_write((GF_Box *)ptr, ptr->other_boxes, bs);
#else
GF_LOG(GF_LOG_WARNING, GF_LOG_CONTAINER, ("ISOBMF: udta is a box-list - cannot export in read-only version of libisom in GPAC\n" ));
#endif
}
}
gf_bs_get_content(bs, userData, userDataSize);
gf_bs_del(bs);
return GF_OK;
}
GF_EXPORT
void gf_isom_delete(GF_ISOFile *movie)
{
gf_isom_delete_movie(movie);
}
GF_EXPORT
GF_Err gf_isom_get_chunks_infos(GF_ISOFile *movie, u32 trackNumber, u32 *dur_min, u32 *dur_avg, u32 *dur_max, u32 *size_min, u32 *size_avg, u32 *size_max)
{
GF_TrackBox *trak;
u32 i, k, sample_idx, dmin, dmax, smin, smax, tot_chunks;
u64 davg, savg;
GF_SampleToChunkBox *stsc;
GF_TimeToSampleBox *stts;
if (!movie || !trackNumber || !movie->moov) return GF_BAD_PARAM;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
stsc = trak->Media->information->sampleTable->SampleToChunk;
stts = trak->Media->information->sampleTable->TimeToSample;
if (!stsc || !stts) return GF_ISOM_INVALID_FILE;
dmin = smin = (u32) -1;
dmax = smax = 0;
davg = savg = 0;
sample_idx = 1;
tot_chunks = 0;
for (i=0; i<stsc->nb_entries; i++) {
u32 nb_chunk = 0;
if (stsc->entries[i].samplesPerChunk > 2*trak->Media->information->sampleTable->SampleSize->sampleCount) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CONTAINER, ("[iso file] likely broken stco entry (%u samples per chunk but %u samples total)\n", stsc->entries[i].samplesPerChunk, trak->Media->information->sampleTable->SampleSize->sampleCount));
return GF_ISOM_INVALID_FILE;
}
while (1) {
u32 chunk_dur = 0;
u32 chunk_size = 0;
for (k=0; k<stsc->entries[i].samplesPerChunk; k++) {
u64 dts;
u32 dur;
u32 size;
stbl_GetSampleDTS_and_Duration(stts, k+sample_idx, &dts, &dur);
chunk_dur += dur;
stbl_GetSampleSize(trak->Media->information->sampleTable->SampleSize, k+sample_idx, &size);
chunk_size += size;
}
if (dmin>chunk_dur) dmin = chunk_dur;
if (dmax<chunk_dur) dmax = chunk_dur;
davg += chunk_dur;
if (smin>chunk_size) smin = chunk_size;
if (smax<chunk_size) smax = chunk_size;
savg += chunk_size;
tot_chunks ++;
sample_idx += stsc->entries[i].samplesPerChunk;
if (i+1==stsc->nb_entries) break;
nb_chunk ++;
if (stsc->entries[i].firstChunk + nb_chunk == stsc->entries[i+1].firstChunk) break;
}
}
if (tot_chunks) {
davg /= tot_chunks;
savg /= tot_chunks;
}
if (dur_min) *dur_min = dmin;
if (dur_avg) *dur_avg = (u32) davg;
if (dur_max) *dur_max = dmax;
if (size_min) *size_min = smin;
if (size_avg) *size_avg = (u32) savg;
if (size_max) *size_max = smax;
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_sample_fragment_count(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return stbl_GetSampleFragmentCount(trak->Media->information->sampleTable->Fragments, sampleNumber);
}
GF_EXPORT
u16 gf_isom_get_sample_fragment_size(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, u32 FragmentIndex)
{
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !FragmentIndex) return 0;
return stbl_GetSampleFragmentSize(trak->Media->information->sampleTable->Fragments, sampleNumber, FragmentIndex);
}
GF_EXPORT
GF_Err gf_isom_get_fragment_defaults(GF_ISOFile *the_file, u32 trackNumber,
u32 *defaultDuration, u32 *defaultSize, u32 *defaultDescriptionIndex,
u32 *defaultRandomAccess, u8 *defaultPadding, u16 *defaultDegradationPriority)
{
GF_TrackBox *trak;
GF_StscEntry *sc_ent;
u32 i, j, maxValue, value;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_TrackExtendsBox *trex;
#endif
GF_SampleTableBox *stbl;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
trex = the_file->moov->mvex ? GetTrex(the_file->moov, gf_isom_get_track_id(the_file,trackNumber) ) : NULL;
if (trex) {
trex->track = trak;
if (defaultDuration) *defaultDuration = trex->def_sample_duration;
if (defaultSize) *defaultSize = trex->def_sample_size;
if (defaultDescriptionIndex) *defaultDescriptionIndex = trex->def_sample_desc_index;
if (defaultRandomAccess) *defaultRandomAccess = GF_ISOM_GET_FRAG_SYNC(trex->def_sample_flags);
if (defaultPadding) *defaultPadding = GF_ISOM_GET_FRAG_PAD(trex->def_sample_flags);
if (defaultDegradationPriority) *defaultDegradationPriority = GF_ISOM_GET_FRAG_DEG(trex->def_sample_flags);
return GF_OK;
}
#endif
stbl = trak->Media->information->sampleTable;
if (!stbl->TimeToSample || !stbl->SampleSize || !stbl->SampleToChunk) return GF_ISOM_INVALID_FILE;
if (defaultDuration) {
maxValue = value = 0;
for (i=0; i<stbl->TimeToSample->nb_entries; i++) {
if (stbl->TimeToSample->entries[i].sampleCount>maxValue) {
value = stbl->TimeToSample->entries[i].sampleDelta;
maxValue = stbl->TimeToSample->entries[i].sampleCount;
}
}
*defaultDuration = value;
}
if (defaultSize) {
*defaultSize = stbl->SampleSize->sampleSize;
}
if (defaultDescriptionIndex) {
GF_SampleToChunkBox *stsc= stbl->SampleToChunk;
maxValue = value = 0;
for (i=0; i<stsc->nb_entries; i++) {
sc_ent = &stsc->entries[i];
if ((sc_ent->nextChunk - sc_ent->firstChunk) * sc_ent->samplesPerChunk > maxValue) {
value = sc_ent->sampleDescriptionIndex;
maxValue = (sc_ent->nextChunk - sc_ent->firstChunk) * sc_ent->samplesPerChunk;
}
}
*defaultDescriptionIndex = value ? value : 1;
}
if (defaultRandomAccess) {
*defaultRandomAccess = stbl->SyncSample ? 0 : 1;
if (stbl->SyncSample
&& (stbl->SyncSample->nb_entries >= stbl->SampleSize->sampleCount/2)) {
*defaultRandomAccess = 1;
}
}
if (defaultPadding) {
*defaultPadding = 0;
if (stbl->PaddingBits) {
maxValue = 0;
for (i=0; i<stbl->PaddingBits->SampleCount; i++) {
value = 0;
for (j=0; j<stbl->PaddingBits->SampleCount; j++) {
if (stbl->PaddingBits->padbits[i]==stbl->PaddingBits->padbits[j]) {
value ++;
}
}
if (value>maxValue) {
maxValue = value;
*defaultPadding = stbl->PaddingBits->padbits[i];
}
}
}
}
if (defaultDegradationPriority) {
*defaultDegradationPriority = 0;
if (stbl->DegradationPriority) {
maxValue = 0;
for (i=0; i<stbl->DegradationPriority->nb_entries; i++) {
value = 0;
for (j=0; j<stbl->DegradationPriority->nb_entries; j++) {
if (stbl->DegradationPriority->priorities[i]==stbl->DegradationPriority->priorities[j]) {
value ++;
}
}
if (value>maxValue) {
maxValue = value;
*defaultDegradationPriority = stbl->DegradationPriority->priorities[i];
}
}
}
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_refresh_fragmented(GF_ISOFile *movie, u64 *MissingBytes, const char *new_location)
{
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
return GF_NOT_SUPPORTED;
#else
u64 prevsize, size;
u32 i;
if (!movie || !movie->movieFileMap || !movie->moov || !movie->moov->mvex) return GF_BAD_PARAM;
if (movie->openMode != GF_ISOM_OPEN_READ) return GF_BAD_PARAM;
size = movie->movieFileMap ? gf_bs_get_size(movie->movieFileMap->bs) : 0;
if (new_location) {
Bool delete_map;
GF_DataMap *previous_movie_fileMap_address = movie->movieFileMap;
GF_Err e;
e = gf_isom_datamap_new(new_location, NULL, GF_ISOM_DATA_MAP_READ_ONLY, &movie->movieFileMap);
if (e) {
movie->movieFileMap = previous_movie_fileMap_address;
return e;
}
delete_map = (previous_movie_fileMap_address != NULL ? GF_TRUE: GF_FALSE);
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox *)gf_list_get(movie->moov->trackList, i);
if (trak->Media->information->dataHandler == previous_movie_fileMap_address) {
trak->Media->information->scalableDataHandler = movie->movieFileMap;
trak->Media->information->dataHandler = movie->movieFileMap;
} else if (trak->Media->information->scalableDataHandler == previous_movie_fileMap_address) {
delete_map = GF_FALSE;
}
}
if (delete_map) {
gf_isom_datamap_del(previous_movie_fileMap_address);
}
}
prevsize = gf_bs_get_refreshed_size(movie->movieFileMap->bs);
if (prevsize==size) return GF_OK;
return gf_isom_parse_movie_boxes(movie, MissingBytes, GF_TRUE);
#endif
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_EXPORT
void gf_isom_set_single_moof_mode(GF_ISOFile *movie, Bool mode)
{
movie->single_moof_mode = mode;
}
#endif
GF_EXPORT
GF_Err gf_isom_reset_data_offset(GF_ISOFile *movie, u64 *top_box_start)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (!movie || !movie->moov) return GF_BAD_PARAM;
*top_box_start = movie->current_top_box_start;
movie->current_top_box_start = 0;
if (movie->moov->mvex && movie->single_moof_mode) {
movie->single_moof_state = 0;
}
#endif
return GF_OK;
}
#define RECREATE_BOX(_a, __cast) \
if (_a) { \
type = _a->type;\
gf_isom_box_del((GF_Box *)_a);\
_a = __cast gf_isom_box_new(type);\
}\
GF_EXPORT
GF_Err gf_isom_reset_tables(GF_ISOFile *movie, Bool reset_sample_count)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
u32 i, j;
GF_Box *a;
if (!movie || !movie->moov || !movie->moov->mvex) return GF_BAD_PARAM;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox *)gf_list_get(movie->moov->trackList, i);
u32 type, dur;
u64 dts;
GF_SampleTableBox *stbl = trak->Media->information->sampleTable;
trak->sample_count_at_seg_start += stbl->SampleSize->sampleCount;
if (trak->sample_count_at_seg_start) {
GF_Err e;
e = stbl_GetSampleDTS_and_Duration(stbl->TimeToSample, stbl->SampleSize->sampleCount, &dts, &dur);
if (e == GF_OK) {
trak->dts_at_seg_start += dts + dur;
}
}
RECREATE_BOX(stbl->ChunkOffset, (GF_Box *));
RECREATE_BOX(stbl->CompositionOffset, (GF_CompositionOffsetBox *));
RECREATE_BOX(stbl->DegradationPriority, (GF_DegradationPriorityBox *));
RECREATE_BOX(stbl->PaddingBits, (GF_PaddingBitsBox *));
RECREATE_BOX(stbl->SampleDep, (GF_SampleDependencyTypeBox *));
RECREATE_BOX(stbl->SampleSize, (GF_SampleSizeBox *));
RECREATE_BOX(stbl->SampleToChunk, (GF_SampleToChunkBox *));
RECREATE_BOX(stbl->ShadowSync, (GF_ShadowSyncBox *));
RECREATE_BOX(stbl->SyncSample, (GF_SyncSampleBox *));
RECREATE_BOX(stbl->TimeToSample, (GF_TimeToSampleBox *));
gf_isom_box_array_del(stbl->sai_offsets);
stbl->sai_offsets = NULL;
gf_isom_box_array_del(stbl->sai_sizes);
stbl->sai_sizes = NULL;
gf_isom_box_array_del(stbl->sampleGroups);
stbl->sampleGroups = NULL;
j = stbl->nb_sgpd_in_stbl;
while ((a = (GF_Box *)gf_list_enum(stbl->sampleGroupsDescription, &j))) {
gf_isom_box_del(a);
j--;
gf_list_rem(stbl->sampleGroupsDescription, j);
}
j = stbl->nb_other_boxes_in_stbl;
while ((a = (GF_Box *)gf_list_enum(stbl->other_boxes, &j))) {
gf_isom_box_del(a);
j--;
gf_list_rem(stbl->other_boxes, j);
}
if (reset_sample_count) {
trak->Media->information->sampleTable->SampleSize->sampleCount = 0;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
trak->sample_count_at_seg_start = 0;
#endif
}
}
#endif
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_release_segment(GF_ISOFile *movie, Bool reset_tables)
{
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
u32 i, j, base_track_sample_count;
Bool has_scalable;
GF_Box *a;
if (!movie || !movie->moov || !movie->moov->mvex) return GF_BAD_PARAM;
has_scalable = gf_isom_needs_layer_reconstruction(movie);
base_track_sample_count = 0;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
trak->first_traf_merged = GF_FALSE;
if (trak->Media->information->dataHandler == movie->movieFileMap) {
trak->Media->information->dataHandler = NULL;
}
if (trak->Media->information->scalableDataHandler == movie->movieFileMap) {
trak->Media->information->scalableDataHandler = NULL;
} else {
if (trak->Media->information->scalableDataHandler==trak->Media->information->dataHandler)
trak->Media->information->dataHandler = NULL;
gf_isom_datamap_del(trak->Media->information->scalableDataHandler);
trak->Media->information->scalableDataHandler = NULL;
}
if (reset_tables) {
u32 type, dur;
u64 dts;
GF_SampleTableBox *stbl = trak->Media->information->sampleTable;
if (has_scalable) {
if (gf_isom_get_reference_count(movie, i+1, GF_ISOM_REF_BASE) > 0) {
u32 on_track=0;
GF_TrackBox *base;
gf_isom_get_reference(movie, i+1, GF_ISOM_REF_BASE, 1, &on_track);
base = gf_isom_get_track_from_file(movie, on_track);
if (!base) {
base_track_sample_count=0;
} else {
base_track_sample_count = base->Media->information->sampleTable->SampleSize->sampleCount;
}
}
}
trak->sample_count_at_seg_start += base_track_sample_count ? base_track_sample_count : stbl->SampleSize->sampleCount;
if (trak->sample_count_at_seg_start) {
GF_Err e;
e = stbl_GetSampleDTS_and_Duration(stbl->TimeToSample, stbl->SampleSize->sampleCount, &dts, &dur);
if (e == GF_OK) {
trak->dts_at_seg_start += dts + dur;
}
}
RECREATE_BOX(stbl->ChunkOffset, (GF_Box *));
RECREATE_BOX(stbl->CompositionOffset, (GF_CompositionOffsetBox *));
RECREATE_BOX(stbl->DegradationPriority, (GF_DegradationPriorityBox *));
RECREATE_BOX(stbl->PaddingBits, (GF_PaddingBitsBox *));
RECREATE_BOX(stbl->SampleDep, (GF_SampleDependencyTypeBox *));
RECREATE_BOX(stbl->SampleSize, (GF_SampleSizeBox *));
RECREATE_BOX(stbl->SampleToChunk, (GF_SampleToChunkBox *));
RECREATE_BOX(stbl->ShadowSync, (GF_ShadowSyncBox *));
RECREATE_BOX(stbl->SyncSample, (GF_SyncSampleBox *));
RECREATE_BOX(stbl->TimeToSample, (GF_TimeToSampleBox *));
gf_isom_box_array_del(stbl->sai_offsets);
stbl->sai_offsets = NULL;
gf_isom_box_array_del(stbl->sai_sizes);
stbl->sai_sizes = NULL;
gf_isom_box_array_del(stbl->sampleGroups);
stbl->sampleGroups = NULL;
j = stbl->nb_sgpd_in_stbl;
while ((a = (GF_Box *)gf_list_enum(stbl->sampleGroupsDescription, &j))) {
gf_isom_box_del(a);
j--;
gf_list_rem(stbl->sampleGroupsDescription, j);
}
j = stbl->nb_other_boxes_in_stbl;
while ((a = (GF_Box *)gf_list_enum(stbl->other_boxes, &j))) {
gf_isom_box_del(a);
j--;
gf_list_rem(stbl->other_boxes, j);
}
}
j = 0;
while ((a = (GF_Box *)gf_list_enum(movie->moov->other_boxes, &j))) {
if (a->type == GF_ISOM_BOX_TYPE_PSSH) {
gf_isom_box_del(a);
j--;
gf_list_rem(movie->moov->other_boxes, j);
}
}
}
gf_isom_datamap_del(movie->movieFileMap);
movie->movieFileMap = NULL;
#endif
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_open_segment(GF_ISOFile *movie, const char *fileName, u64 start_range, u64 end_range, u32 flags)
{
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
return GF_NOT_SUPPORTED;
#else
u64 MissingBytes;
GF_Err e;
u32 i;
Bool segment_map_assigned = GF_FALSE;
Bool is_scalable_segment = (flags & GF_ISOM_SEGMENT_SCALABLE_FLAG) ? GF_TRUE : GF_FALSE;
Bool no_order_check = (flags & GF_ISOM_SEGMENT_NO_ORDER_FLAG) ? GF_TRUE: GF_FALSE;
GF_DataMap *tmp = NULL;
GF_DataMap *orig_file_map = NULL;
if (!movie || !movie->moov || !movie->moov->mvex) return GF_BAD_PARAM;
if (movie->openMode != GF_ISOM_OPEN_READ) return GF_BAD_PARAM;
if (is_scalable_segment) {
tmp = NULL;
e = gf_isom_datamap_new(fileName, NULL, GF_ISOM_DATA_MAP_READ_ONLY, &tmp);
if (e) return e;
orig_file_map = movie->movieFileMap;
movie->movieFileMap = tmp;
} else {
if (movie->movieFileMap)
gf_isom_release_segment(movie, GF_FALSE);
e = gf_isom_datamap_new(fileName, NULL, GF_ISOM_DATA_MAP_READ_ONLY, &movie->movieFileMap);
if (e) return e;
}
movie->current_top_box_start = 0;
if (end_range > start_range) {
gf_bs_seek(movie->movieFileMap->bs, end_range+1);
gf_bs_truncate(movie->movieFileMap->bs);
gf_bs_seek(movie->movieFileMap->bs, start_range);
movie->current_top_box_start = start_range;
}
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
if (!is_scalable_segment) {
if (trak->Media->information->dataHandler == NULL) {
trak->Media->information->dataHandler = movie->movieFileMap;
}
} else {
trak->present_in_scalable_segment = GF_FALSE;
}
}
if (no_order_check) movie->NextMoofNumber = 0;
e = gf_isom_parse_movie_boxes(movie, &MissingBytes, GF_TRUE);
if (!is_scalable_segment)
return e;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
if (trak->present_in_scalable_segment) {
trak->Media->information->scalableDataHandler = tmp;
if (!segment_map_assigned) {
trak->Media->information->scalableDataHandler = tmp;
segment_map_assigned = GF_TRUE;
}
trak->Media->information->dataHandler = tmp;
}
}
movie->movieFileMap = orig_file_map;
return e;
#endif
}
GF_EXPORT
u32 gf_isom_get_highest_track_in_scalable_segment(GF_ISOFile *movie, u32 for_base_track)
{
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
return 0;
#else
s32 max_ref;
u32 i, j, track_id;
max_ref = 0;
track_id = 0;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
s32 ref;
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
if (! trak->present_in_scalable_segment) continue;
ref = gf_isom_get_reference_count(movie, i+1, GF_ISOM_REF_SCAL);
if (ref<=0) continue;
if (ref<=max_ref) continue;
for (j=0; j< (u32) ref; j++) {
u32 on_track=0;
gf_isom_get_reference(movie, i+1, GF_ISOM_REF_SCAL, j+1, &on_track);
if (on_track==for_base_track) {
max_ref = ref;
track_id = trak->Header->trackID;
}
}
}
return track_id;
#endif
}
GF_EXPORT
GF_Err gf_isom_text_set_streaming_mode(GF_ISOFile *movie, Bool do_convert)
{
if (!movie) return GF_BAD_PARAM;
movie->convert_streaming_text = do_convert;
return GF_OK;
}
GF_EXPORT
GF_GenericSampleDescription *gf_isom_get_generic_sample_description(GF_ISOFile *movie, u32 trackNumber, u32 StreamDescriptionIndex)
{
GF_GenericVisualSampleEntryBox *entry;
GF_GenericAudioSampleEntryBox *gena;
GF_GenericSampleEntryBox *genm;
GF_TrackBox *trak;
GF_GenericSampleDescription *udesc;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak || !StreamDescriptionIndex || !trak->Media || !trak->Media->information || !trak->Media->information->sampleTable) return 0;
entry = (GF_GenericVisualSampleEntryBox *)gf_list_get(trak->Media->information->sampleTable->SampleDescription->other_boxes, StreamDescriptionIndex-1);
if (!entry || IsMP4Description(entry->type) ) return NULL;
switch (entry->type) {
case GF_ISOM_SUBTYPE_3GP_AMR:
case GF_ISOM_SUBTYPE_3GP_AMR_WB:
case GF_ISOM_SUBTYPE_3GP_EVRC:
case GF_ISOM_SUBTYPE_3GP_QCELP:
case GF_ISOM_SUBTYPE_3GP_SMV:
case GF_ISOM_SUBTYPE_3GP_H263:
return NULL;
case GF_ISOM_BOX_TYPE_GNRV:
GF_SAFEALLOC(udesc, GF_GenericSampleDescription);
if (!udesc) return NULL;
if (entry->EntryType == GF_ISOM_BOX_TYPE_UUID) {
memcpy(udesc->UUID, ((GF_UUIDBox*)entry)->uuid, sizeof(bin128));
} else {
udesc->codec_tag = entry->EntryType;
}
udesc->version = entry->version;
udesc->revision = entry->revision;
udesc->vendor_code = entry->vendor;
udesc->temporal_quality = entry->temporal_quality;
udesc->spatial_quality = entry->spatial_quality;
udesc->width = entry->Width;
udesc->height = entry->Height;
udesc->h_res = entry->horiz_res;
udesc->v_res = entry->vert_res;
strcpy(udesc->compressor_name, entry->compressor_name);
udesc->depth = entry->bit_depth;
udesc->color_table_index = entry->color_table_index;
if (entry->data_size) {
udesc->extension_buf_size = entry->data_size;
udesc->extension_buf = (char*)gf_malloc(sizeof(char) * entry->data_size);
if (!udesc->extension_buf) {
gf_free(udesc);
return NULL;
}
memcpy(udesc->extension_buf, entry->data, entry->data_size);
}
return udesc;
case GF_ISOM_BOX_TYPE_GNRA:
gena = (GF_GenericAudioSampleEntryBox *)entry;
GF_SAFEALLOC(udesc, GF_GenericSampleDescription);
if (!udesc) return NULL;
if (gena->EntryType == GF_ISOM_BOX_TYPE_UUID) {
memcpy(udesc->UUID, ((GF_UUIDBox*)gena)->uuid, sizeof(bin128));
} else {
udesc->codec_tag = gena->EntryType;
}
udesc->version = gena->version;
udesc->revision = gena->revision;
udesc->vendor_code = gena->vendor;
udesc->samplerate = gena->samplerate_hi;
udesc->bits_per_sample = gena->bitspersample;
udesc->nb_channels = gena->channel_count;
if (gena->data_size) {
udesc->extension_buf_size = gena->data_size;
udesc->extension_buf = (char*)gf_malloc(sizeof(char) * gena->data_size);
if (!udesc->extension_buf) {
gf_free(udesc);
return NULL;
}
memcpy(udesc->extension_buf, gena->data, gena->data_size);
}
return udesc;
case GF_ISOM_BOX_TYPE_GNRM:
genm = (GF_GenericSampleEntryBox *)entry;
GF_SAFEALLOC(udesc, GF_GenericSampleDescription);
if (!udesc) return NULL;
if (genm->EntryType == GF_ISOM_BOX_TYPE_UUID) {
memcpy(udesc->UUID, ((GF_UUIDBox*)genm)->uuid, sizeof(bin128));
} else {
udesc->codec_tag = genm->EntryType;
}
if (genm->data_size) {
udesc->extension_buf_size = genm->data_size;
udesc->extension_buf = (char*)gf_malloc(sizeof(char) * genm->data_size);
if (!udesc->extension_buf) {
gf_free(udesc);
return NULL;
}
memcpy(udesc->extension_buf, genm->data, genm->data_size);
}
return udesc;
}
return NULL;
}
GF_EXPORT
GF_Err gf_isom_get_visual_info(GF_ISOFile *movie, u32 trackNumber, u32 StreamDescriptionIndex, u32 *Width, u32 *Height)
{
GF_TrackBox *trak;
GF_SampleEntryBox *entry;
GF_SampleDescriptionBox *stsd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
stsd = trak->Media->information->sampleTable->SampleDescription;
if (!stsd) return movie->LastError = GF_ISOM_INVALID_FILE;
if (!StreamDescriptionIndex || StreamDescriptionIndex > gf_list_count(stsd->other_boxes)) return movie->LastError = GF_BAD_PARAM;
entry = (GF_SampleEntryBox *)gf_list_get(stsd->other_boxes, StreamDescriptionIndex - 1);
if (entry == NULL) return GF_BAD_PARAM;
if (entry->internal_type == GF_ISOM_SAMPLE_ENTRY_VIDEO) {
*Width = ((GF_VisualSampleEntryBox*)entry)->Width;
*Height = ((GF_VisualSampleEntryBox*)entry)->Height;
} else if (trak->Media->handler->handlerType==GF_ISOM_MEDIA_SCENE) {
*Width = trak->Header->width>>16;
*Height = trak->Header->height>>16;
} else {
return GF_BAD_PARAM;
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_audio_info(GF_ISOFile *movie, u32 trackNumber, u32 StreamDescriptionIndex, u32 *SampleRate, u32 *Channels, u8 *bitsPerSample)
{
GF_TrackBox *trak;
GF_SampleEntryBox *entry;
GF_SampleDescriptionBox *stsd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
stsd = trak->Media->information->sampleTable->SampleDescription;
if (!stsd) return movie->LastError = GF_ISOM_INVALID_FILE;
if (!StreamDescriptionIndex || StreamDescriptionIndex > gf_list_count(stsd->other_boxes)) return movie->LastError = GF_BAD_PARAM;
entry = (GF_SampleEntryBox *)gf_list_get(stsd->other_boxes, StreamDescriptionIndex - 1);
if (entry == NULL) return GF_BAD_PARAM;
if (entry->internal_type != GF_ISOM_SAMPLE_ENTRY_AUDIO) return GF_BAD_PARAM;
if (SampleRate) (*SampleRate) = ((GF_AudioSampleEntryBox*)entry)->samplerate_hi;
if (Channels) (*Channels) = ((GF_AudioSampleEntryBox*)entry)->channel_count;
if (bitsPerSample) (*bitsPerSample) = (u8) ((GF_AudioSampleEntryBox*)entry)->bitspersample;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_pixel_aspect_ratio(GF_ISOFile *movie, u32 trackNumber, u32 StreamDescriptionIndex, u32 *hSpacing, u32 *vSpacing)
{
GF_TrackBox *trak;
GF_VisualSampleEntryBox *entry;
GF_SampleDescriptionBox *stsd;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak || !hSpacing || !vSpacing) return GF_BAD_PARAM;
*hSpacing = 1;
*vSpacing = 1;
stsd = trak->Media->information->sampleTable->SampleDescription;
if (!stsd) return movie->LastError = GF_ISOM_INVALID_FILE;
if (!StreamDescriptionIndex || StreamDescriptionIndex > gf_list_count(stsd->other_boxes)) return movie->LastError = GF_BAD_PARAM;
entry = (GF_VisualSampleEntryBox *)gf_list_get(stsd->other_boxes, StreamDescriptionIndex - 1);
if (entry == NULL) return GF_OK;
if (entry->internal_type==GF_ISOM_SAMPLE_ENTRY_VIDEO) {
if (entry->pasp) {
*hSpacing = entry->pasp->hSpacing;
*vSpacing = entry->pasp->vSpacing;
}
return GF_OK;
} else {
return GF_BAD_PARAM;
}
}
GF_EXPORT
const char *gf_isom_get_filename(GF_ISOFile *movie)
{
if (!movie) return NULL;
#ifndef GPAC_DISABLE_ISOM_WRITE
if (movie->finalName && !movie->fileName) return movie->finalName;
#endif
return movie->fileName;
}
GF_EXPORT
u8 gf_isom_get_pl_indication(GF_ISOFile *movie, u8 PL_Code)
{
GF_IsomInitialObjectDescriptor *iod;
if (!movie || !movie->moov) return 0;
if (!movie->moov->iods || !movie->moov->iods->descriptor) return 0xFF;
if (movie->moov->iods->descriptor->tag != GF_ODF_ISOM_IOD_TAG) return 0xFF;
iod = (GF_IsomInitialObjectDescriptor *)movie->moov->iods->descriptor;
switch (PL_Code) {
case GF_ISOM_PL_AUDIO:
return iod->audio_profileAndLevel;
case GF_ISOM_PL_VISUAL:
return iod->visual_profileAndLevel;
case GF_ISOM_PL_GRAPHICS:
return iod->graphics_profileAndLevel;
case GF_ISOM_PL_SCENE:
return iod->scene_profileAndLevel;
case GF_ISOM_PL_OD:
return iod->OD_profileAndLevel;
case GF_ISOM_PL_INLINE:
return iod->inlineProfileFlag;
case GF_ISOM_PL_MPEGJ:
default:
return 0xFF;
}
}
GF_EXPORT
GF_Err gf_isom_get_track_matrix(GF_ISOFile *the_file, u32 trackNumber, u32 matrix[9])
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Header) return GF_BAD_PARAM;
memcpy(matrix, trak->Header->matrix, sizeof(trak->Header->matrix));
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_track_layout_info(GF_ISOFile *movie, u32 trackNumber, u32 *width, u32 *height, s32 *translation_x, s32 *translation_y, s16 *layer)
{
GF_TrackBox *tk = gf_isom_get_track_from_file(movie, trackNumber);
if (!tk) return GF_BAD_PARAM;
if (width) *width = tk->Header->width>>16;
if (height) *height = tk->Header->height>>16;
if (layer) *layer = tk->Header->layer;
if (translation_x) *translation_x = tk->Header->matrix[6] >> 16;
if (translation_y) *translation_y = tk->Header->matrix[7] >> 16;
return GF_OK;
}
GF_EXPORT
u64 gf_isom_get_media_data_size(GF_ISOFile *movie, u32 trackNumber)
{
u32 i, size;
GF_SampleSizeBox *stsz;
GF_TrackBox *tk = gf_isom_get_track_from_file(movie, trackNumber);
if (!tk) return 0;
stsz = tk->Media->information->sampleTable->SampleSize;
if (stsz->sampleSize) return stsz->sampleSize*stsz->sampleCount;
size = 0;
for (i=0; i<stsz->sampleCount; i++) size += stsz->sizes[i];
return size;
}
GF_EXPORT
void gf_isom_set_default_sync_track(GF_ISOFile *movie, u32 trackNumber)
{
GF_TrackBox *tk = gf_isom_get_track_from_file(movie, trackNumber);
if (!tk) movie->es_id_default_sync = -1;
else movie->es_id_default_sync = tk->Header->trackID;
}
GF_EXPORT
Bool gf_isom_is_single_av(GF_ISOFile *file)
{
u32 count, i, nb_any, nb_a, nb_v, nb_scene, nb_od, nb_text;
nb_a = nb_v = nb_any = nb_scene = nb_od = nb_text = 0;
if (!file->moov) return GF_FALSE;
count = gf_isom_get_track_count(file);
for (i=0; i<count; i++) {
u32 mtype = gf_isom_get_media_type(file, i+1);
switch (mtype) {
case GF_ISOM_MEDIA_SCENE:
if (gf_isom_get_sample_count(file, i+1)>1) nb_any++;
else nb_scene++;
break;
case GF_ISOM_MEDIA_OD:
if (gf_isom_get_sample_count(file, i+1)>1) nb_any++;
else nb_od++;
break;
case GF_ISOM_MEDIA_TEXT:
case GF_ISOM_MEDIA_SUBT:
nb_text++;
break;
case GF_ISOM_MEDIA_AUDIO:
nb_a++;
break;
case GF_ISOM_MEDIA_VISUAL:
if (gf_isom_get_sample_count(file, i+1)==1) nb_any++;
else nb_v++;
break;
default:
nb_any++;
break;
}
}
if (nb_any) return GF_FALSE;
if ((nb_scene<=1) && (nb_od<=1) && (nb_a<=1) && (nb_v<=1) && (nb_text<=1) ) return GF_TRUE;
return GF_FALSE;
}
GF_EXPORT
Bool gf_isom_is_JPEG2000(GF_ISOFile *mov)
{
return (mov && mov->is_jp2) ? GF_TRUE : GF_FALSE;
}
GF_EXPORT
u32 gf_isom_guess_specification(GF_ISOFile *file)
{
u32 count, i, nb_any, nb_m4s, nb_a, nb_v, nb_scene, nb_od, nb_mp3, nb_aac, nb_m4v, nb_avc, nb_amr, nb_h263, nb_qcelp, nb_evrc, nb_smv, nb_text;
nb_m4s = nb_a = nb_v = nb_any = nb_scene = nb_od = nb_mp3 = nb_aac = nb_m4v = nb_avc = nb_amr = nb_h263 = nb_qcelp = nb_evrc = nb_smv = nb_text = 0;
if (file->is_jp2) {
if (file->moov) return GF_4CC('m','j','p','2');
return GF_4CC('j','p','2',' ');
}
if (!file->moov) {
if (!file->meta || !file->meta->handler) return 0;
return file->meta->handler->handlerType;
}
count = gf_isom_get_track_count(file);
for (i=0; i<count; i++) {
u32 mtype = gf_isom_get_media_type(file, i+1);
u32 mstype = gf_isom_get_media_subtype(file, i+1, 1);
if (mtype==GF_ISOM_MEDIA_SCENE) {
nb_scene++;
if (gf_isom_get_sample_count(file, i+1)>1) nb_m4s++;
} else if (mtype==GF_ISOM_MEDIA_OD) {
nb_od++;
if (gf_isom_get_sample_count(file, i+1)>1) nb_m4s++;
}
else if ((mtype==GF_ISOM_MEDIA_TEXT) || (mtype==GF_ISOM_MEDIA_SUBT)) nb_text++;
else if ((mtype==GF_ISOM_MEDIA_AUDIO) || (mtype==GF_ISOM_MEDIA_VISUAL)) {
switch (mstype) {
case GF_ISOM_SUBTYPE_3GP_AMR:
case GF_ISOM_SUBTYPE_3GP_AMR_WB:
nb_amr++;
break;
case GF_ISOM_SUBTYPE_3GP_H263:
nb_h263++;
break;
case GF_ISOM_SUBTYPE_3GP_EVRC:
nb_evrc++;
break;
case GF_ISOM_SUBTYPE_3GP_QCELP:
nb_qcelp++;
break;
case GF_ISOM_SUBTYPE_3GP_SMV:
nb_smv++;
break;
case GF_ISOM_SUBTYPE_AVC_H264:
case GF_ISOM_SUBTYPE_AVC2_H264:
case GF_ISOM_SUBTYPE_AVC3_H264:
case GF_ISOM_SUBTYPE_AVC4_H264:
nb_avc++;
break;
case GF_ISOM_SUBTYPE_SVC_H264:
case GF_ISOM_SUBTYPE_MVC_H264:
nb_avc++;
break;
case GF_ISOM_SUBTYPE_MPEG4:
case GF_ISOM_SUBTYPE_MPEG4_CRYP:
{
GF_DecoderConfig *dcd = gf_isom_get_decoder_config(file, i+1, 1);
switch (dcd->streamType) {
case GF_STREAM_VISUAL:
if (dcd->objectTypeIndication==GPAC_OTI_VIDEO_MPEG4_PART2) nb_m4v++;
else if ((dcd->objectTypeIndication==GPAC_OTI_VIDEO_AVC) || (dcd->objectTypeIndication==GPAC_OTI_VIDEO_SVC) || (dcd->objectTypeIndication==GPAC_OTI_VIDEO_MVC)) nb_avc++;
else nb_v++;
break;
case GF_STREAM_AUDIO:
switch (dcd->objectTypeIndication) {
case GPAC_OTI_AUDIO_AAC_MPEG2_MP:
case GPAC_OTI_AUDIO_AAC_MPEG2_LCP:
case GPAC_OTI_AUDIO_AAC_MPEG2_SSRP:
case GPAC_OTI_AUDIO_AAC_MPEG4:
nb_aac++;
break;
case GPAC_OTI_AUDIO_MPEG2_PART3:
case GPAC_OTI_AUDIO_MPEG1:
nb_mp3++;
break;
case GPAC_OTI_AUDIO_EVRC_VOICE:
nb_evrc++;
break;
case GPAC_OTI_AUDIO_SMV_VOICE:
nb_smv++;
break;
case GPAC_OTI_AUDIO_13K_VOICE:
nb_qcelp++;
break;
default:
nb_a++;
break;
}
break;
default:
nb_any++;
break;
}
gf_odf_desc_del((GF_Descriptor *)dcd);
}
break;
default:
if (mtype==GF_ISOM_MEDIA_VISUAL) nb_v++;
else nb_a++;
break;
}
} else if ((mtype==GF_ISOM_SUBTYPE_MPEG4) || (mtype==GF_ISOM_SUBTYPE_MPEG4_CRYP)) nb_m4s++;
else nb_any++;
}
if (nb_any) return GF_ISOM_BRAND_ISOM;
if (nb_qcelp || nb_evrc || nb_smv) {
if (nb_m4s || nb_avc || nb_scene || nb_od || nb_mp3 || nb_a || nb_v) return GF_ISOM_BRAND_ISOM;
return GF_ISOM_BRAND_3G2A;
}
if (nb_v || nb_a || nb_m4s) return GF_ISOM_BRAND_MP42;
nb_v = nb_m4v + nb_avc + nb_h263;
nb_a = nb_mp3 + nb_aac + nb_amr;
if (nb_avc) {
if (!nb_scene && !nb_od) return GF_ISOM_BRAND_AVC1;
return GF_ISOM_BRAND_MP42;
}
if (nb_mp3) {
if (!nb_text && (nb_v<=1) && (nb_a<=1) && (nb_scene==1) && (nb_od==1))
return GF_4CC('I', 'S', 'M', 'A');
return GF_ISOM_BRAND_MP42;
}
if (nb_scene || nb_od) {
if (nb_amr || nb_h263) return GF_ISOM_BRAND_ISOM;
return GF_ISOM_BRAND_MP42;
}
if (!nb_amr && !nb_h263 && !nb_text) {
if ((nb_v<=1) && (nb_a<=1)) return GF_4CC('I', 'S', 'M', 'A');
return GF_ISOM_BRAND_MP42;
}
if ((nb_v<=1) && (nb_a<=1) && (nb_text<=1)) return nb_text ? GF_ISOM_BRAND_3GP6 : GF_ISOM_BRAND_3GP5;
return GF_ISOM_BRAND_3GG6;
}
GF_ItemListBox *gf_ismo_locate_box(GF_List *list, u32 boxType, bin128 UUID)
{
u32 i;
GF_Box *box;
i=0;
while ((box = (GF_Box *)gf_list_enum(list, &i))) {
if (box->type == boxType) {
GF_UUIDBox* box2 = (GF_UUIDBox* )box;
if (boxType != GF_ISOM_BOX_TYPE_UUID) return (GF_ItemListBox *)box;
if (!memcmp(box2->uuid, UUID, 16)) return (GF_ItemListBox *)box;
}
}
return NULL;
}
GF_EXPORT
GF_Err gf_isom_apple_get_tag(GF_ISOFile *mov, u32 tag, const char **data, u32 *data_len)
{
u32 i;
GF_ListItemBox *info;
GF_ItemListBox *ilst;
GF_MetaBox *meta;
*data = NULL;
*data_len = 0;
meta = gf_isom_apple_get_meta_extensions(mov);
if (!meta) return GF_URL_ERROR;
ilst = gf_ismo_locate_box(meta->other_boxes, GF_ISOM_BOX_TYPE_ILST, NULL);
if (!ilst) return GF_URL_ERROR;
if (tag==GF_ISOM_ITUNE_PROBE) return GF_OK;
i=0;
while ( (info=(GF_ListItemBox*)gf_list_enum(ilst->other_boxes, &i))) {
if (info->type==tag) break;
if ((tag==GF_ISOM_ITUNE_GENRE) && (info->type==(u32) GF_ISOM_BOX_TYPE_0xA9GEN)) break;
info = NULL;
}
if (!info || !info->data || !info->data->data) return GF_URL_ERROR;
if ((tag == GF_ISOM_ITUNE_GENRE) && (info->data->flags == 0)) {
if (info->data->dataSize && (info->data->dataSize>2) && (info->data->dataSize < 5)) {
GF_BitStream* bs = gf_bs_new(info->data->data, info->data->dataSize, GF_BITSTREAM_READ);
*data_len = gf_bs_read_int(bs, info->data->dataSize * 8);
gf_bs_del(bs);
return GF_OK;
}
}
*data = info->data->data;
*data_len = info->data->dataSize;
if ((tag==GF_ISOM_ITUNE_COVER_ART) && (info->data->flags==14)) *data_len |= (1<<31);
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_track_switch_group_count(GF_ISOFile *movie, u32 trackNumber, u32 *alternateGroupID, u32 *nb_groups)
{
GF_UserDataMap *map;
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!trak) return GF_BAD_PARAM;
*alternateGroupID = trak->Header->alternate_group;
*nb_groups = 0;
if (!trak->udta) return GF_OK;
map = udta_getEntry(trak->udta, GF_ISOM_BOX_TYPE_TSEL, NULL);
if (!map) return GF_OK;
*nb_groups = gf_list_count(map->other_boxes);
return GF_OK;
}
GF_EXPORT
const u32 *gf_isom_get_track_switch_parameter(GF_ISOFile *movie, u32 trackNumber, u32 group_index, u32 *switchGroupID, u32 *criteriaListSize)
{
GF_TrackBox *trak;
GF_UserDataMap *map;
GF_TrackSelectionBox *tsel;
trak = gf_isom_get_track_from_file(movie, trackNumber);
if (!group_index || !trak || !trak->udta) return NULL;
map = udta_getEntry(trak->udta, GF_ISOM_BOX_TYPE_TSEL, NULL);
if (!map) return NULL;
tsel = (GF_TrackSelectionBox*)gf_list_get(map->other_boxes, group_index-1);
*switchGroupID = tsel->switchGroup;
*criteriaListSize = tsel->attributeListCount;
return (const u32 *) tsel->attributeList;
}
GF_EXPORT
u32 gf_isom_get_next_alternate_group_id(GF_ISOFile *movie)
{
u32 id = 0;
u32 i=0;
while (i< gf_isom_get_track_count(movie) ) {
GF_TrackBox *trak = gf_isom_get_track_from_file(movie, i+1);
if (trak->Header->alternate_group > id)
id = trak->Header->alternate_group;
i++;
}
return id+1;
}
u32 gf_isom_sample_has_subsamples(GF_ISOFile *movie, u32 track, u32 sampleNumber, u32 flags)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(movie, track);
if (!trak) return GF_BAD_PARAM;
if (!trak->Media->information->sampleTable->sub_samples) return 0;
return gf_isom_sample_get_subsample_entry(movie, track, sampleNumber, flags, NULL);
}
GF_Err gf_isom_sample_get_subsample(GF_ISOFile *movie, u32 track, u32 sampleNumber, u32 flags, u32 subSampleNumber, u32 *size, u8 *priority, u32 *reserved, Bool *discardable)
{
GF_SubSampleEntry *entry;
GF_SubSampleInfoEntry *sub_sample;
u32 count = gf_isom_sample_get_subsample_entry(movie, track, sampleNumber, flags, &sub_sample);
if (!size || !priority || !discardable) return GF_BAD_PARAM;
if (!subSampleNumber || (subSampleNumber>count)) return GF_BAD_PARAM;
entry = (GF_SubSampleEntry*)gf_list_get(sub_sample->SubSamples, subSampleNumber-1);
*size = entry->subsample_size;
*priority = entry->subsample_priority;
*reserved = entry->reserved;
*discardable = entry->discardable ? GF_TRUE : GF_FALSE;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_rvc_config(GF_ISOFile *movie, u32 track, u32 sampleDescriptionIndex, u16 *rvc_predefined, char **data, u32 *size, const char **mime)
{
GF_MPEGVisualSampleEntryBox *entry;
GF_TrackBox *trak;
if (!rvc_predefined || !data || !size) return GF_BAD_PARAM;
*rvc_predefined = 0;
trak = gf_isom_get_track_from_file(movie, track);
if (!trak) return GF_BAD_PARAM;
entry = (GF_MPEGVisualSampleEntryBox *) gf_list_get(trak->Media->information->sampleTable->SampleDescription->other_boxes, sampleDescriptionIndex-1);
if (!entry ) return GF_BAD_PARAM;
if (entry->internal_type != GF_ISOM_SAMPLE_ENTRY_VIDEO) return GF_BAD_PARAM;
if (!entry->rvcc) return GF_BAD_PARAM;
*rvc_predefined = entry->rvcc->predefined_rvc_config;
if (entry->rvcc->rvc_meta_idx) {
if (!data || !size) return GF_OK;
return gf_isom_extract_meta_item_mem(movie, GF_FALSE, track, entry->rvcc->rvc_meta_idx, data, size, mime);
}
return GF_OK;
}
GF_EXPORT
Bool gf_isom_moov_first(GF_ISOFile *movie)
{
u32 i;
for (i=0; i<gf_list_count(movie->TopBoxes); i++) {
GF_Box *b = (GF_Box*)gf_list_get(movie->TopBoxes, i);
if (b->type == GF_ISOM_BOX_TYPE_MOOV) return GF_TRUE;
if (b->type == GF_ISOM_BOX_TYPE_MDAT) return GF_FALSE;
}
return GF_FALSE;
}
GF_EXPORT
void gf_isom_reset_fragment_info(GF_ISOFile *movie, Bool keep_sample_count)
{
u32 i;
if (!movie) return;
for (i=0; i<gf_list_count(movie->moov->trackList); i++) {
GF_TrackBox *trak = (GF_TrackBox*)gf_list_get(movie->moov->trackList, i);
trak->Media->information->sampleTable->SampleSize->sampleCount = 0;
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
}
#else
trak->dts_at_seg_start = 0;
if (!keep_sample_count)
trak->sample_count_at_seg_start = 0;
}
movie->NextMoofNumber = 0;
#endif
}
GF_EXPORT
GF_Err gf_isom_get_sample_rap_roll_info(GF_ISOFile *the_file, u32 trackNumber, u32 sample_number, Bool *is_rap, Bool *has_roll, s32 *roll_distance)
{
GF_TrackBox *trak;
u32 i, count;
if (is_rap) *is_rap = GF_FALSE;
if (has_roll) *has_roll = GF_FALSE;
if (roll_distance) *roll_distance = 0;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (!trak->Media->information->sampleTable->sampleGroups) return GF_OK;
if (!sample_number) {
count = gf_list_count(trak->Media->information->sampleTable->sampleGroupsDescription);
for (i=0; i<count; i++) {
GF_SampleGroupDescriptionBox *sgdesc = (GF_SampleGroupDescriptionBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroupsDescription, i);
switch (sgdesc->grouping_type) {
case GF_4CC('r','a','p',' '):
if (is_rap) *is_rap = GF_TRUE;
break;
case GF_4CC('r','o','l','l'):
if (has_roll) *has_roll = GF_TRUE;
if (roll_distance) {
s32 max_roll = 0;
u32 j;
for (j=0; j<gf_list_count(sgdesc->group_descriptions); j++) {
GF_RollRecoveryEntry *roll_entry = (GF_RollRecoveryEntry*)gf_list_get(sgdesc->group_descriptions, j);
if (max_roll < roll_entry->roll_distance)
max_roll = roll_entry->roll_distance;
}
if (*roll_distance < max_roll) *roll_distance = max_roll;
}
break;
}
}
return GF_OK;
}
count = gf_list_count(trak->Media->information->sampleTable->sampleGroups);
for (i=0; i<count; i++) {
GF_SampleGroupBox *sg;
u32 j, group_desc_index;
GF_SampleGroupDescriptionBox *sgdesc;
u32 first_sample_in_entry, last_sample_in_entry;
first_sample_in_entry = 1;
group_desc_index = 0;
sg = (GF_SampleGroupBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroups, i);
for (j=0; j<sg->entry_count; j++) {
last_sample_in_entry = first_sample_in_entry + sg->sample_entries[j].sample_count - 1;
if ((sample_number<first_sample_in_entry) || (sample_number>last_sample_in_entry)) {
first_sample_in_entry = last_sample_in_entry+1;
continue;
}
group_desc_index = 1 + sg->sample_entries[j].group_description_index;
break;
}
if (!group_desc_index) continue;
sgdesc = NULL;
for (j=0; j<gf_list_count(trak->Media->information->sampleTable->sampleGroupsDescription); j++) {
sgdesc = (GF_SampleGroupDescriptionBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroupsDescription, j);
if (sgdesc->grouping_type==sg->grouping_type) break;
sgdesc = NULL;
}
if (!sgdesc) continue;
switch (sgdesc->grouping_type) {
case GF_4CC('r','a','p',' '):
if (is_rap) *is_rap = GF_TRUE;
break;
case GF_4CC('r','o','l','l'):
if (has_roll) *has_roll = GF_TRUE;
if (roll_distance) {
GF_RollRecoveryEntry *roll_entry = (GF_RollRecoveryEntry *) gf_list_get(sgdesc->group_descriptions, group_desc_index - 1);
if (roll_entry) *roll_distance = roll_entry->roll_distance;
}
break;
}
}
return GF_OK;
}
GF_DefaultSampleGroupDescriptionEntry * gf_isom_get_sample_group_info_entry(GF_ISOFile *the_file, GF_TrackBox *trak, u32 grouping_type, u32 sample_description_index, u32 *default_index, GF_SampleGroupDescriptionBox **out_sgdp)
{
u32 i, count;
if (!trak || !sample_description_index) return NULL;
if (!trak->Media->information->sampleTable->sampleGroupsDescription) return NULL;
count = gf_list_count(trak->Media->information->sampleTable->sampleGroupsDescription);
for (i=0; i<count; i++) {
GF_SampleGroupDescriptionBox *sgdesc = (GF_SampleGroupDescriptionBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroupsDescription, i);
if (sgdesc->grouping_type != grouping_type) continue;
if (sgdesc->default_description_index && !sample_description_index) sample_description_index = sgdesc->default_description_index;
if (default_index) *default_index = sgdesc->default_description_index ;
if (out_sgdp) *out_sgdp = sgdesc;
if (!sample_description_index) return NULL;
return (GF_DefaultSampleGroupDescriptionEntry*)gf_list_get(sgdesc->group_descriptions, sample_description_index-1);
}
return NULL;
}
GF_EXPORT
Bool gf_isom_get_sample_group_info(GF_ISOFile *the_file, u32 trackNumber, u32 sample_description_index, u32 grouping_type, u32 *default_index, const char **data, u32 *size)
{
GF_DefaultSampleGroupDescriptionEntry *sg_entry;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (default_index) *default_index = 0;
if (size) *size = 0;
if (data) *data = NULL;
sg_entry = gf_isom_get_sample_group_info_entry(the_file, trak, grouping_type, sample_description_index, default_index, NULL);
if (!sg_entry) return GF_FALSE;
switch (grouping_type) {
case GF_4CC('r','a','p',' '):
case GF_4CC('r','o','l','l'):
case GF_4CC( 's', 'e', 'i', 'g' ):
case GF_4CC( 'o', 'i', 'n', 'f' ):
case GF_4CC( 'l', 'i', 'n', 'f' ):
return GF_TRUE;
default:
if (sg_entry && data) *data = (char *) sg_entry->data;
if (sg_entry && size) *size = sg_entry->length;
return GF_TRUE;
}
return GF_FALSE;
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_EXPORT
u64 gf_isom_get_fragmented_duration(GF_ISOFile *movie)
{
if (movie->moov->mvex && movie->moov->mvex->mehd)
return movie->moov->mvex->mehd->fragment_duration;
return 0;
}
GF_EXPORT
u32 gf_isom_get_fragments_count(GF_ISOFile *movie, Bool segments_only)
{
u32 i=0;
u32 nb_frags = 0;
GF_Box *b;
while ((b=(GF_Box*)gf_list_enum(movie->TopBoxes, &i))) {
if (segments_only) {
if (b->type==GF_ISOM_BOX_TYPE_SIDX)
nb_frags++;
} else {
if (b->type==GF_ISOM_BOX_TYPE_MOOF)
nb_frags++;
}
}
return nb_frags;
}
GF_EXPORT
GF_Err gf_isom_get_fragmented_samples_info(GF_ISOFile *movie, u32 trackID, u32 *nb_samples, u64 *duration)
{
u32 i=0;
u32 k, l;
u64 def_duration, samp_dur;
GF_MovieFragmentBox *moof;
GF_TrackFragmentBox *traf;
*nb_samples = 0;
*duration = 0;
while ((moof=(GF_MovieFragmentBox*)gf_list_enum(movie->TopBoxes, &i))) {
if (moof->type==GF_ISOM_BOX_TYPE_MOOF) {
u32 j=0;
while ((traf=(GF_TrackFragmentBox*)gf_list_enum( moof->TrackList, &j))) {
if (traf->tfhd->trackID != trackID)
continue;
def_duration = 0;
if (traf->tfhd->flags & GF_ISOM_TRAF_SAMPLE_DUR) def_duration = traf->tfhd->def_sample_duration;
else if (traf->trex) def_duration = traf->trex->def_sample_duration;
for (k=0; k<gf_list_count(traf->TrackRuns); k++) {
GF_TrackFragmentRunBox *trun = (GF_TrackFragmentRunBox*)gf_list_get(traf->TrackRuns, k);
*nb_samples += gf_list_count(trun->entries);
for (l=0; l<gf_list_count(trun->entries); l++) {
GF_TrunEntry *ent = (GF_TrunEntry*)gf_list_get(trun->entries, l);
samp_dur = def_duration;
if (trun->flags & GF_ISOM_TRUN_DURATION) samp_dur = ent->Duration;
*duration += samp_dur;
}
}
}
}
}
return GF_OK;
}
#endif
GF_EXPORT
GF_Err gf_isom_set_nalu_extract_mode(GF_ISOFile *the_file, u32 trackNumber, u32 nalu_extract_mode)
{
GF_TrackReferenceTypeBox *dpnd;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
trak->extractor_mode = nalu_extract_mode;
if (!trak->References) return GF_OK;
dpnd = NULL;
trak->has_base_layer = GF_FALSE;
Track_FindRef(trak, GF_ISOM_REF_SCAL, &dpnd);
if (dpnd) trak->has_base_layer = GF_TRUE;
return GF_OK;
}
GF_EXPORT
u32 gf_isom_get_nalu_extract_mode(GF_ISOFile *the_file, u32 trackNumber)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return trak->extractor_mode;
}
GF_EXPORT
s32 gf_isom_get_composition_offset_shift(GF_ISOFile *file, u32 track)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(file, track);
if (!trak) return 0;
if (!trak->Media || !trak->Media->information || !trak->Media->information->sampleTable || !trak->Media->information->sampleTable->CompositionToDecode) return 0;
return trak->Media->information->sampleTable->CompositionToDecode->compositionToDTSShift;
}
GF_EXPORT
Bool gf_isom_needs_layer_reconstruction(GF_ISOFile *file)
{
u32 count, i;
if (!file)
return GF_FALSE;
count = gf_isom_get_track_count(file);
for (i = 0; i < count; i++) {
if (gf_isom_get_reference_count(file, i+1, GF_ISOM_REF_SCAL) > 0) {
return GF_TRUE;
}
if (gf_isom_get_reference_count(file, i+1, GF_ISOM_REF_SABT) > 0) {
return GF_TRUE;
}
switch (gf_isom_get_media_subtype(file, i+1, 1)) {
case GF_ISOM_SUBTYPE_LHV1:
case GF_ISOM_SUBTYPE_LHE1:
if (gf_isom_get_reference_count(file, i+1, GF_ISOM_REF_BASE) > 0) {
return GF_TRUE;
}
}
}
return GF_FALSE;
}
GF_EXPORT
void gf_isom_keep_utc_times(GF_ISOFile *file, Bool keep_utc)
{
if (!file) return;
file->keep_utc = keep_utc;
}
GF_EXPORT
void gf_isom_no_version_date_info(GF_ISOFile *file, Bool drop_info)
{
if (!file) return;
file->drop_date_version_info = drop_info;
}
GF_EXPORT
u32 gf_isom_get_pssh_count(GF_ISOFile *file)
{
u32 count=0;
u32 i=0;
GF_Box *a_box;
while ((a_box = (GF_Box*)gf_list_enum(file->moov->other_boxes, &i))) {
if (a_box->type != GF_ISOM_BOX_TYPE_PSSH) continue;
count++;
}
return count;
}
GF_EXPORT
GF_Err gf_isom_get_pssh_info(GF_ISOFile *file, u32 pssh_index, bin128 SystemID, u32 *KID_count, const bin128 **KIDs, const u8 **private_data, u32 *private_data_size)
{
u32 count=1;
u32 i=0;
GF_ProtectionSystemHeaderBox *pssh;
while ((pssh = (GF_ProtectionSystemHeaderBox *)gf_list_enum(file->moov->other_boxes, &i))) {
if (pssh->type != GF_ISOM_BOX_TYPE_PSSH) continue;
if (count == pssh_index) break;
count++;
}
if (!pssh) return GF_BAD_PARAM;
memcpy(SystemID, pssh->SystemID, 16);
*KID_count = pssh->KID_count;
*KIDs = (const bin128 *) pssh->KIDs;
*private_data_size = pssh->private_data_size;
*private_data = pssh->private_data;
return GF_OK;
}
GF_EXPORT
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_Err gf_isom_get_sample_cenc_info_ex(GF_TrackBox *trak, GF_TrackFragmentBox *traf, GF_SampleEncryptionBox *senc, u32 sample_number, u32 *IsEncrypted, u8 *IV_size, bin128 *KID, u8 *crypt_byte_block, u8 *skip_byte_block, u8 *constant_IV_size, bin128 *constant_IV)
#else
GF_Err gf_isom_get_sample_cenc_info_ex(GF_TrackBox *trak, void *traf, GF_SampleEncryptionBox *senc, u32 sample_number, u32 *IsEncrypted, u8 *IV_size, bin128 *KID, u8 *crypt_byte_block, u8 *skip_byte_block, u8 *constant_IV_size, bin128 *constant_IV)
#endif
{
GF_SampleGroupBox *sample_group;
u32 j, group_desc_index;
GF_SampleGroupDescriptionBox *sgdesc;
u32 i, count;
u32 descIndex, chunkNum;
u64 offset;
u8 edit;
u32 first_sample_in_entry, last_sample_in_entry;
GF_CENCSampleEncryptionGroupEntry *entry;
if (IsEncrypted) *IsEncrypted = 0;
if (IV_size) *IV_size = 0;
if (KID) memset(*KID, 0, 16);
if (crypt_byte_block) *crypt_byte_block = 0;
if (skip_byte_block) *skip_byte_block = 0;
if (constant_IV_size) *constant_IV_size = 0;
if (constant_IV) memset(*constant_IV, 0, 16);
if (!senc) return GF_BAD_PARAM;
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
if (traf)
return GF_BAD_PARAM;
#endif
if (trak->Media->information->sampleTable->SampleSize && trak->Media->information->sampleTable->SampleSize->sampleCount>=sample_number) {
stbl_GetSampleInfos(trak->Media->information->sampleTable, sample_number, &offset, &chunkNum, &descIndex, &edit);
} else {
descIndex = 1;
}
gf_isom_cenc_get_default_info_ex(trak, descIndex, IsEncrypted, IV_size, KID);
sample_group = NULL;
group_desc_index = 0;
if (trak->Media->information->sampleTable->sampleGroups) {
count = gf_list_count(trak->Media->information->sampleTable->sampleGroups);
for (i=0; i<count; i++) {
sample_group = (GF_SampleGroupBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroups, i);
if (sample_group->grouping_type == GF_4CC( 's', 'e', 'i', 'g' ))
break;
sample_group = NULL;
}
if (sample_group) {
first_sample_in_entry = 1;
for (j=0; j<sample_group->entry_count; j++) {
last_sample_in_entry = first_sample_in_entry + sample_group->sample_entries[j].sample_count - 1;
if ((sample_number<first_sample_in_entry) || (sample_number>last_sample_in_entry)) {
first_sample_in_entry = last_sample_in_entry+1;
continue;
}
group_desc_index = sample_group->sample_entries[j].group_description_index;
break;
}
}
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
if (!group_desc_index && traf && traf->sampleGroups) {
count = gf_list_count(traf->sampleGroups);
for (i=0; i<count; i++) {
group_desc_index = 0;
sample_group = (GF_SampleGroupBox*)gf_list_get(traf->sampleGroups, i);
if (sample_group->grouping_type == GF_4CC( 's', 'e', 'i', 'g' ))
break;
sample_group = NULL;
}
if (sample_group) {
first_sample_in_entry = 1;
for (j=0; j<sample_group->entry_count; j++) {
last_sample_in_entry = first_sample_in_entry + sample_group->sample_entries[j].sample_count - 1;
if ((sample_number<first_sample_in_entry) || (sample_number>last_sample_in_entry)) {
first_sample_in_entry = last_sample_in_entry+1;
continue;
}
group_desc_index = sample_group->sample_entries[j].group_description_index;
break;
}
}
}
#endif
if (!group_desc_index) goto exit;
sgdesc = NULL;
if (group_desc_index<=0x10000) {
for (j=0; j<gf_list_count(trak->Media->information->sampleTable->sampleGroupsDescription); j++) {
sgdesc = (GF_SampleGroupDescriptionBox*)gf_list_get(trak->Media->information->sampleTable->sampleGroupsDescription, j);
if (sgdesc->grouping_type==sample_group->grouping_type) break;
sgdesc = NULL;
}
} else if (traf) {
group_desc_index -= 0x10000;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
for (j=0; j<gf_list_count(traf->sampleGroupsDescription); j++) {
sgdesc = (GF_SampleGroupDescriptionBox*)gf_list_get(traf->sampleGroupsDescription, j);
if (sgdesc->grouping_type==sample_group->grouping_type) break;
sgdesc = NULL;
}
#endif
}
if (!sgdesc) return GF_ISOM_INVALID_FILE;
entry = (GF_CENCSampleEncryptionGroupEntry *) gf_list_get(sgdesc->group_descriptions, group_desc_index - 1);
if (!entry) return GF_ISOM_INVALID_FILE;
if (IsEncrypted) *IsEncrypted = entry->IsProtected;
if (IV_size) *IV_size = entry->Per_Sample_IV_size;
if (KID) memmove(*KID, entry->KID, 16);
if (crypt_byte_block) *crypt_byte_block = entry->crypt_byte_block;
if (skip_byte_block) *skip_byte_block = entry->skip_byte_block;
if (constant_IV_size) *constant_IV_size = entry->constant_IV_size;
if (constant_IV) memmove(*constant_IV, entry->constant_IV, 16);
exit:
if ((senc->is_piff || (trak->moov && trak->moov->mov->is_smooth) ) && !(*IV_size) ) {
if (!senc->is_piff) {
senc->is_piff = GF_TRUE;
senc->IV_size=8;
}
assert(senc->IV_size);
*IV_size = senc->IV_size;
*IsEncrypted = GF_TRUE;
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_sample_cenc_info(GF_ISOFile *movie, u32 track, u32 sample_number, u32 *IsEncrypted, u8 *IV_size, bin128 *KID,
u8 *crypt_byte_block, u8 *skip_byte_block, u8 *constant_IV_size, bin128 *constant_IV)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(movie, track);
GF_SampleEncryptionBox *senc = trak->sample_encryption;
return gf_isom_get_sample_cenc_info_ex(trak, NULL, senc, sample_number, IsEncrypted, IV_size, KID, crypt_byte_block, skip_byte_block, constant_IV_size, constant_IV);
}
GF_EXPORT
Bool gf_isom_get_last_producer_time_box(GF_ISOFile *file, u32 *refTrackID, u64 *ntp, u64 *timestamp, Bool reset_info)
{
if (!file) return GF_FALSE;
if (refTrackID) *refTrackID = 0;
if (ntp) *ntp = 0;
if (timestamp) *timestamp = 0;
if (file->last_producer_ref_time) {
if (refTrackID) *refTrackID = file->last_producer_ref_time->refTrackID;
if (ntp) *ntp = file->last_producer_ref_time->ntp;
if (timestamp) *timestamp = file->last_producer_ref_time->timestamp;
if (reset_info) {
file->last_producer_ref_time = NULL;
}
return GF_TRUE;
}
return GF_FALSE;
}
GF_EXPORT
u64 gf_isom_get_current_tfdt(GF_ISOFile *the_file, u32 trackNumber)
{
#ifdef GPAC_DISABLE_ISOM_FRAGMENTS
return 0;
#else
GF_TrackBox *trak;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
return trak->dts_at_seg_start;
#endif
}
void gf_isom_parse_trif_info(const char *data, u32 size, u32 *id, u32 *independent, Bool *full_picture, u32 *x, u32 *y, u32 *w, u32 *h)
{
GF_BitStream *bs;
bs = gf_bs_new(data, size, GF_BITSTREAM_READ);
*id = gf_bs_read_u16(bs);
if (! gf_bs_read_int(bs, 1)) {
*independent=0;
*full_picture=0;
*x = *y = *w = *h = 0;
} else {
*independent = gf_bs_read_int(bs, 2);
*full_picture = (Bool)gf_bs_read_int(bs, 1);
gf_bs_read_int(bs, 1);
gf_bs_read_int(bs, 1);
gf_bs_read_int(bs, 2);
*x = *full_picture ? 0 : gf_bs_read_u16(bs);
*y = *full_picture ? 0 : gf_bs_read_u16(bs);
*w = gf_bs_read_u16(bs);
*h = gf_bs_read_u16(bs);
}
gf_bs_del(bs);
}
GF_EXPORT
Bool gf_isom_get_tile_info(GF_ISOFile *file, u32 trackNumber, u32 sample_description_index, u32 *default_sample_group_index, u32 *id, u32 *independent, Bool *full_picture, u32 *x, u32 *y, u32 *w, u32 *h)
{
const char *data;
u32 size;
if (!gf_isom_get_sample_group_info(file, trackNumber, sample_description_index, GF_4CC('t','r','i','f'), default_sample_group_index, &data, &size))
return GF_FALSE;
gf_isom_parse_trif_info(data, size, id, independent, full_picture, x, y, w, h);
return GF_TRUE;
}
GF_EXPORT
Bool gf_isom_drop_date_version_info_enabled(GF_ISOFile *file)
{
return file->drop_date_version_info;
}
GF_EXPORT
Bool gf_isom_get_oinf_info(GF_ISOFile *file, u32 trackNumber, GF_OperatingPointsInformation **ptr)
{
u32 oref_track, def_index=0;
GF_TrackBox *trak = gf_isom_get_track_from_file(file, trackNumber);
if (!ptr) return GF_FALSE;
oref_track=0;
gf_isom_get_reference(file, trackNumber, GF_ISOM_REF_OREF, 1, &oref_track);
if (oref_track) {
trak = gf_isom_get_track_from_file(file, oref_track);
if (!trak) return GF_FALSE;
}
*ptr = (GF_OperatingPointsInformation *) gf_isom_get_sample_group_info_entry(file, trak, GF_4CC('o','i','n','f'), 1, &def_index, NULL);
return *ptr ? GF_TRUE : GF_FALSE;
}
#endif