This source file includes following definitions.
- gf_isom_ismacryp_new_sample
- gf_isom_ismacryp_delete_sample
- gf_isom_ismacryp_sample_from_data
- gf_isom_ismacryp_sample_to_sample
- isom_get_sinf_entry
- gf_isom_get_ismacryp_sample
- gf_isom_is_media_encrypted
- gf_isom_is_ismacryp_media
- gf_isom_is_omadrm_media
- gf_isom_get_ismacryp_info
- gf_isom_get_omadrm_info
- gf_isom_remove_track_protection
- gf_isom_change_ismacryp_protection
- isom_set_protected_entry
- gf_isom_set_ismacryp_protection
- gf_isom_set_oma_protection
- gf_isom_get_original_format_type
- gf_isom_is_cenc_media
- gf_isom_get_cenc_info
- gf_isom_set_cenc_protection
- gf_isom_set_cenc_saio
- gf_isom_remove_cenc_saiz
- gf_isom_remove_cenc_saio
- gf_cenc_set_pssh
- gf_isom_remove_samp_enc_box
- gf_isom_remove_samp_group_box
- gf_isom_remove_pssh_box
- gf_isom_create_piff_psec_box
- gf_isom_create_samp_enc_box
- gf_isom_cenc_allocate_storage
- gf_isom_cenc_set_saiz_saio
- gf_isom_cenc_merge_saiz_saio
- gf_isom_track_cenc_add_sample_info
- gf_isom_cenc_samp_aux_info_del
- gf_isom_cenc_has_saiz_saio_full
- gf_isom_cenc_has_saiz_saio_track
- gf_isom_cenc_has_saiz_saio_traf
- isom_cenc_get_sai_by_saiz_saio
- gf_isom_cenc_get_sample_aux_info
- gf_isom_cenc_get_default_info_ex
- gf_isom_cenc_get_default_info
- gf_isom_cenc_get_default_pattern_info_ex
- gf_isom_cenc_get_default_pattern_info
- gf_isom_set_adobe_protection
- gf_isom_is_adobe_protection_media
- gf_isom_get_adobe_protection_info
- gf_isom_cenc_is_pattern_mode
- gf_isom_ipmpx_remove_tool_list
#include <gpac/internal/isomedia_dev.h>
#ifndef GPAC_DISABLE_ISOM
GF_ISMASample *gf_isom_ismacryp_new_sample()
{
GF_ISMASample *tmp = (GF_ISMASample *) gf_malloc(sizeof(GF_ISMASample));
if (!tmp) return NULL;
memset(tmp, 0, sizeof(GF_ISMASample));
return tmp;
}
GF_EXPORT
void gf_isom_ismacryp_delete_sample(GF_ISMASample *samp)
{
if (!samp) return;
if (samp->data && samp->dataLength) gf_free(samp->data);
if (samp->key_indicator) gf_free(samp->key_indicator);
gf_free(samp);
}
GF_ISMASample *gf_isom_ismacryp_sample_from_data(char *data, u32 dataLength, Bool use_selective_encryption, u8 KI_length, u8 IV_length)
{
GF_ISMASample *s;
GF_BitStream *bs;
if (!data || !dataLength) {
return gf_isom_ismacryp_new_sample();
}
s = gf_isom_ismacryp_new_sample();
if (!data || !dataLength) return s;
bs = gf_bs_new(data, dataLength, GF_BITSTREAM_READ);
s->dataLength = dataLength;
s->IV_length = IV_length;
s->KI_length = KI_length;
if (use_selective_encryption) {
s->flags = GF_ISOM_ISMA_USE_SEL_ENC;
if (s->dataLength < 1) goto exit;
if (gf_bs_read_int(bs, 1)) s->flags |= GF_ISOM_ISMA_IS_ENCRYPTED;
gf_bs_read_int(bs, 7);
s->dataLength -= 1;
} else {
s->flags = GF_ISOM_ISMA_IS_ENCRYPTED;
}
if (s->flags & GF_ISOM_ISMA_IS_ENCRYPTED) {
if (IV_length != 0) {
if (s->dataLength < IV_length) goto exit;
s->IV = gf_bs_read_long_int(bs, 8*IV_length);
s->dataLength -= IV_length;
}
if (KI_length) {
if (s->dataLength < KI_length) goto exit;
s->key_indicator = (u8 *)gf_malloc(KI_length);
gf_bs_read_data(bs, (char*)s->key_indicator, KI_length);
s->dataLength -= KI_length;
}
}
s->data = (char*)gf_malloc(sizeof(char)*s->dataLength);
gf_bs_read_data(bs, s->data, s->dataLength);
gf_bs_del(bs);
return s;
exit:
gf_isom_ismacryp_delete_sample(s);
return NULL;
}
GF_Err gf_isom_ismacryp_sample_to_sample(GF_ISMASample *s, GF_ISOSample *dest)
{
GF_BitStream *bs;
if (!s || !dest) return GF_BAD_PARAM;
bs = gf_bs_new(NULL, 0, GF_BITSTREAM_WRITE);
if (s->flags & GF_ISOM_ISMA_USE_SEL_ENC) {
gf_bs_write_int(bs, (s->flags & GF_ISOM_ISMA_IS_ENCRYPTED) ? 1 : 0, 1);
gf_bs_write_int(bs, 0, 7);
}
if (s->flags & GF_ISOM_ISMA_IS_ENCRYPTED) {
if (s->IV_length) gf_bs_write_long_int(bs, (s64) s->IV, 8*s->IV_length);
if (s->KI_length) gf_bs_write_data(bs, (char*)s->key_indicator, s->KI_length);
}
gf_bs_write_data(bs, s->data, s->dataLength);
if (dest->data) gf_free(dest->data);
dest->data = NULL;
dest->dataLength = 0;
gf_bs_get_content(bs, &dest->data, &dest->dataLength);
gf_bs_del(bs);
return GF_OK;
}
static GF_ProtectionSchemeInfoBox *isom_get_sinf_entry(GF_TrackBox *trak, u32 sampleDescriptionIndex, u32 scheme_type, GF_SampleEntryBox **out_sea)
{
u32 i=0;
GF_SampleEntryBox *sea=NULL;
GF_ProtectionSchemeInfoBox *sinf;
Media_GetSampleDesc(trak->Media, sampleDescriptionIndex, &sea, NULL);
if (!sea) return NULL;
i = 0;
while ((sinf = (GF_ProtectionSchemeInfoBox*)gf_list_enum(sea->protections, &i))) {
if (sinf->original_format && sinf->scheme_type && sinf->info) {
if (!scheme_type || (sinf->scheme_type->scheme_type == scheme_type)) {
if (out_sea)
*out_sea = sea;
return sinf;
}
}
}
return NULL;
}
GF_EXPORT
GF_ISMASample *gf_isom_get_ismacryp_sample(GF_ISOFile *the_file, u32 trackNumber, GF_ISOSample *samp, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ISMASampleFormatBox *fmt;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return NULL;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, 0, NULL);
if (!sinf) return NULL;
if (sinf->scheme_type->scheme_type == GF_ISOM_ISMACRYP_SCHEME) {
fmt = sinf->info->isfm;
if (!fmt) return NULL;
return gf_isom_ismacryp_sample_from_data(samp->data, samp->dataLength, sinf->info->isfm->selective_encryption, sinf->info->isfm->key_indicator_length, sinf->info->isfm->IV_length);
}
else if (sinf->scheme_type->scheme_type == GF_4CC('o','d','k','m') ) {
if (!sinf->info->odkm) return NULL;
fmt = sinf->info->odkm->fmt;
if (fmt) {
return gf_isom_ismacryp_sample_from_data(samp->data, samp->dataLength, fmt->selective_encryption, fmt->key_indicator_length, fmt->IV_length);
}
return gf_isom_ismacryp_sample_from_data(samp->data, samp->dataLength, GF_FALSE, 0, 128);
}
return NULL;
}
GF_EXPORT
u32 gf_isom_is_media_encrypted(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return 0;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, 0, NULL);
if (!sinf) return 0;
if (!sinf || !sinf->scheme_type) return 0;
return sinf->scheme_type->scheme_type;
}
GF_EXPORT
Bool gf_isom_is_ismacryp_media(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ISMACRYP_SCHEME, NULL);
if (!sinf) return GF_FALSE;
if (!sinf->info || !sinf->info->ikms || !sinf->info->isfm )
return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
Bool gf_isom_is_omadrm_media(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_OMADRM_SCHEME, NULL);
if (!sinf) return GF_FALSE;
if (!sinf->info || !sinf->info->odkm || !sinf->info->odkm->hdr)
return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
GF_Err gf_isom_get_ismacryp_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *outOriginalFormat, u32 *outSchemeType, u32 *outSchemeVersion, const char **outSchemeURI, const char **outKMS_URI, Bool *outSelectiveEncryption, u32 *outIVLength, u32 *outKeyIndicationLength)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ISMACRYP_SCHEME, NULL);
if (!sinf) return GF_OK;
if (outOriginalFormat) {
*outOriginalFormat = sinf->original_format->data_format;
if (IsMP4Description(sinf->original_format->data_format)) *outOriginalFormat = GF_ISOM_SUBTYPE_MPEG4;
}
if (outSchemeType) *outSchemeType = sinf->scheme_type->scheme_type;
if (outSchemeVersion) *outSchemeVersion = sinf->scheme_type->scheme_version;
if (outSchemeURI) *outSchemeURI = sinf->scheme_type->URI;
if (sinf->info && sinf->info->ikms) {
if (outKMS_URI) *outKMS_URI = sinf->info->ikms->URI;
} else {
if (outKMS_URI) *outKMS_URI = NULL;
}
if (sinf->info && sinf->info->isfm) {
if (outSelectiveEncryption) *outSelectiveEncryption = sinf->info->isfm->selective_encryption;
if (outIVLength) *outIVLength = sinf->info->isfm->IV_length;
if (outKeyIndicationLength) *outKeyIndicationLength = sinf->info->isfm->key_indicator_length;
} else {
if (outSelectiveEncryption) *outSelectiveEncryption = GF_FALSE;
if (outIVLength) *outIVLength = 0;
if (outKeyIndicationLength) *outKeyIndicationLength = 0;
}
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_get_omadrm_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *outOriginalFormat,
u32 *outSchemeType, u32 *outSchemeVersion,
const char **outContentID, const char **outRightsIssuerURL, const char **outTextualHeaders, u32 *outTextualHeadersLen, u64 *outPlaintextLength, u32 *outEncryptionType, Bool *outSelectiveEncryption, u32 *outIVLength, u32 *outKeyIndicationLength)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_OMADRM_SCHEME, NULL);
if (!sinf) return GF_OK;
if (!sinf->info || !sinf->info->odkm || !sinf->info->odkm->hdr) return GF_NON_COMPLIANT_BITSTREAM;
if (outOriginalFormat) {
*outOriginalFormat = sinf->original_format->data_format;
if (IsMP4Description(sinf->original_format->data_format)) *outOriginalFormat = GF_ISOM_SUBTYPE_MPEG4;
}
if (outSchemeType) *outSchemeType = sinf->scheme_type->scheme_type;
if (outSchemeVersion) *outSchemeVersion = sinf->scheme_type->scheme_version;
if (outContentID) *outContentID = sinf->info->odkm->hdr->ContentID;
if (outRightsIssuerURL) *outRightsIssuerURL = sinf->info->odkm->hdr->RightsIssuerURL;
if (outTextualHeaders) {
*outTextualHeaders = sinf->info->odkm->hdr->TextualHeaders;
if (outTextualHeadersLen) *outTextualHeadersLen = sinf->info->odkm->hdr->TextualHeadersLen;
}
if (outPlaintextLength) *outPlaintextLength = sinf->info->odkm->hdr->PlaintextLength;
if (outEncryptionType) *outEncryptionType = sinf->info->odkm->hdr->EncryptionMethod;
if (sinf->info && sinf->info->odkm && sinf->info->odkm->fmt) {
if (outSelectiveEncryption) *outSelectiveEncryption = sinf->info->odkm->fmt->selective_encryption;
if (outIVLength) *outIVLength = sinf->info->odkm->fmt->IV_length;
if (outKeyIndicationLength) *outKeyIndicationLength = sinf->info->odkm->fmt->key_indicator_length;
} else {
if (outSelectiveEncryption) *outSelectiveEncryption = GF_FALSE;
if (outIVLength) *outIVLength = 0;
if (outKeyIndicationLength) *outKeyIndicationLength = 0;
}
return GF_OK;
}
#ifndef GPAC_DISABLE_ISOM_WRITE
GF_Err gf_isom_remove_track_protection(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_Err e;
GF_SampleEntryBox *sea;
GF_ProtectionSchemeInfoBox *sinf;
e = CanAccessMovie(the_file, GF_ISOM_OPEN_WRITE);
if (e) return e;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Media || !sampleDescriptionIndex) return GF_BAD_PARAM;
sea = NULL;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENC_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBC_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ISMACRYP_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_OMADRM_SCHEME, &sea);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ADOBE_SCHEME, &sea);
if (!sinf) return GF_OK;
sea->type = sinf->original_format->data_format;
gf_isom_box_array_del(sea->protections);
sea->protections = gf_list_new();
if (sea->type == GF_4CC('2','6','4','b')) sea->type = GF_ISOM_BOX_TYPE_AVC1;
if (sea->type == GF_4CC('2','6','5','b')) sea->type = GF_ISOM_BOX_TYPE_HVC1;
return GF_OK;
}
GF_EXPORT
GF_Err gf_isom_change_ismacryp_protection(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, char *scheme_uri, char *kms_uri)
{
GF_TrackBox *trak;
GF_Err e;
GF_SampleEntryBox *sea;
GF_ProtectionSchemeInfoBox *sinf;
e = CanAccessMovie(the_file, GF_ISOM_OPEN_WRITE);
if (e) return e;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak || !trak->Media || !sampleDescriptionIndex) return GF_BAD_PARAM;
sea = NULL;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ISMACRYP_SCHEME, &sea);
if (!sinf) return GF_OK;
if (scheme_uri) {
gf_free(sinf->scheme_type->URI);
sinf->scheme_type->URI = gf_strdup(scheme_uri);
}
if (kms_uri) {
gf_free(sinf->info->ikms->URI);
sinf->info->ikms->URI = gf_strdup(kms_uri);
}
return GF_OK;
}
static GF_Err isom_set_protected_entry(GF_ISOFile *the_file, u32 trackNumber, u32 desc_index, u8 version, u32 flags,
u32 scheme_type, u32 scheme_version, char *scheme_uri, Bool is_isma, GF_ProtectionSchemeInfoBox **out_sinf)
{
u32 original_format;
GF_Err e;
GF_SampleEntryBox *sea;
GF_ProtectionSchemeInfoBox *sinf;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
e = Media_GetSampleDesc(trak->Media, desc_index, &sea, NULL);
if (e) return e;
original_format = sea->type;
switch (sea->type) {
case GF_ISOM_BOX_TYPE_MP4A:
case GF_ISOM_BOX_TYPE_DAMR:
case GF_ISOM_BOX_TYPE_DEVC:
case GF_ISOM_BOX_TYPE_DQCP:
case GF_ISOM_BOX_TYPE_DSMV:
case GF_ISOM_BOX_TYPE_AC3:
case GF_ISOM_BOX_TYPE_EC3:
sea->type = GF_ISOM_BOX_TYPE_ENCA;
break;
case GF_ISOM_BOX_TYPE_MP4V:
case GF_ISOM_BOX_TYPE_D263:
sea->type = GF_ISOM_BOX_TYPE_ENCV;
break;
case GF_ISOM_BOX_TYPE_AVC1:
case GF_ISOM_BOX_TYPE_AVC2:
case GF_ISOM_BOX_TYPE_AVC3:
case GF_ISOM_BOX_TYPE_AVC4:
case GF_ISOM_BOX_TYPE_SVC1:
case GF_ISOM_BOX_TYPE_MVC1:
if (is_isma)
original_format = GF_4CC('2','6','4','b');
sea->type = GF_ISOM_BOX_TYPE_ENCV;
break;
case GF_ISOM_BOX_TYPE_HVC1:
case GF_ISOM_BOX_TYPE_HEV1:
case GF_ISOM_BOX_TYPE_HVC2:
case GF_ISOM_BOX_TYPE_HEV2:
case GF_ISOM_BOX_TYPE_LHE1:
case GF_ISOM_BOX_TYPE_LHV1:
case GF_ISOM_BOX_TYPE_HVT1:
if (is_isma)
original_format = GF_4CC('2','6','5','b');
sea->type = GF_ISOM_BOX_TYPE_ENCV;
break;
case GF_ISOM_BOX_TYPE_MP4S:
case GF_ISOM_BOX_TYPE_LSR1:
sea->type = GF_ISOM_BOX_TYPE_ENCS;
break;
case GF_ISOM_BOX_TYPE_STXT:
case GF_ISOM_BOX_TYPE_WVTT:
case GF_ISOM_BOX_TYPE_STPP:
sea->type = GF_ISOM_BOX_TYPE_ENCT;
break;
default:
return GF_BAD_PARAM;
}
sinf = (GF_ProtectionSchemeInfoBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_SINF);
gf_list_add(sea->protections, sinf);
sinf->scheme_type = (GF_SchemeTypeBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_SCHM);
sinf->scheme_type->version = version;
sinf->scheme_type->flags = flags;
sinf->scheme_type->scheme_type = scheme_type;
sinf->scheme_type->scheme_version = scheme_version;
if (sinf->scheme_type->flags == 1) {
sinf->scheme_type->URI = (char *)gf_malloc(sizeof(char)*strlen(scheme_uri));
memmove(sinf->scheme_type->URI, scheme_uri, strlen(scheme_uri));
}
sinf->original_format = (GF_OriginalFormatBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_FRMA);
sinf->original_format->data_format = original_format;
sinf->info = (GF_SchemeInformationBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_SCHI);
*out_sinf = sinf;
return GF_OK;
}
GF_Err gf_isom_set_ismacryp_protection(GF_ISOFile *the_file, u32 trackNumber, u32 desc_index, u32 scheme_type,
u32 scheme_version, char *scheme_uri, char *kms_URI,
Bool selective_encryption, u32 KI_length, u32 IV_length)
{
GF_Err e;
GF_ProtectionSchemeInfoBox *sinf;
e = isom_set_protected_entry(the_file, trackNumber, desc_index, 0, 0, scheme_type, scheme_version, NULL, GF_TRUE, &sinf);
if (e) return e;
if (scheme_uri) {
sinf->scheme_type->flags |= 0x000001;
sinf->scheme_type->URI = gf_strdup(scheme_uri);
}
sinf->info->ikms = (GF_ISMAKMSBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_IKMS);
sinf->info->ikms->URI = gf_strdup(kms_URI);
sinf->info->isfm = (GF_ISMASampleFormatBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_ISFM);
sinf->info->isfm->selective_encryption = selective_encryption;
sinf->info->isfm->key_indicator_length = KI_length;
sinf->info->isfm->IV_length = IV_length;
return GF_OK;
}
GF_Err gf_isom_set_oma_protection(GF_ISOFile *the_file, u32 trackNumber, u32 desc_index,
char *contentID, char *kms_URI, u32 encryption_type, u64 plainTextLength, char *textual_headers, u32 textual_headers_len,
Bool selective_encryption, u32 KI_length, u32 IV_length)
{
GF_ProtectionSchemeInfoBox *sinf;
GF_Err e;
e = isom_set_protected_entry(the_file, trackNumber, desc_index, 0, 0, GF_ISOM_OMADRM_SCHEME, 0x00000200, NULL, GF_FALSE, &sinf);
if (e) return e;
sinf->info->odkm = (GF_OMADRMKMSBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_ODKM);
sinf->info->odkm->fmt = (GF_OMADRMAUFormatBox*)gf_isom_box_new(GF_ISOM_BOX_TYPE_ODAF);
sinf->info->odkm->fmt->selective_encryption = selective_encryption;
sinf->info->odkm->fmt->key_indicator_length = KI_length;
sinf->info->odkm->fmt->IV_length = IV_length;
sinf->info->odkm->hdr = (GF_OMADRMCommonHeaderBox*)gf_isom_box_new(GF_ISOM_BOX_TYPE_OHDR);
sinf->info->odkm->hdr->EncryptionMethod = encryption_type;
sinf->info->odkm->hdr->PaddingScheme = (encryption_type==0x01) ? 1 : 0;
sinf->info->odkm->hdr->PlaintextLength = plainTextLength;
if (contentID) sinf->info->odkm->hdr->ContentID = gf_strdup(contentID);
if (kms_URI) sinf->info->odkm->hdr->RightsIssuerURL = gf_strdup(kms_URI);
if (textual_headers) {
sinf->info->odkm->hdr->TextualHeaders = (char*)gf_malloc(sizeof(char)*textual_headers_len);
memcpy(sinf->info->odkm->hdr->TextualHeaders, textual_headers, sizeof(char)*textual_headers_len);
sinf->info->odkm->hdr->TextualHeadersLen = textual_headers_len;
}
return GF_OK;
}
#endif
GF_EXPORT
GF_Err gf_isom_get_original_format_type(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *outOriginalFormat)
{
GF_TrackBox *trak;
GF_SampleEntryBox *sea;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
Media_GetSampleDesc(trak->Media, sampleDescriptionIndex, &sea, NULL);
if (!sea) return GF_BAD_PARAM;
sinf = (GF_ProtectionSchemeInfoBox*)gf_list_get(sea->protections, 0);
if (outOriginalFormat && sinf->original_format) {
*outOriginalFormat = sinf->original_format->data_format;
}
return GF_OK;
}
GF_EXPORT
Bool gf_isom_is_cenc_media(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
if (trak->sample_encryption) return GF_TRUE;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, NULL);
if (!sinf) return GF_FALSE;
if (!sinf->info || !sinf->info->tenc)
return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
GF_Err gf_isom_get_cenc_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *outOriginalFormat, u32 *outSchemeType, u32 *outSchemeVersion, u32 *outIVLength)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, NULL);
if (!sinf) return GF_BAD_PARAM;
if (outOriginalFormat) {
*outOriginalFormat = sinf->original_format->data_format;
if (IsMP4Description(sinf->original_format->data_format)) *outOriginalFormat = GF_ISOM_SUBTYPE_MPEG4;
}
if (outSchemeType) *outSchemeType = sinf->scheme_type->scheme_type;
if (outSchemeVersion) *outSchemeVersion = sinf->scheme_type->scheme_version;
if (outIVLength) {
if (sinf->info && sinf->info->tenc)
*outIVLength = sinf->info->tenc->Per_Sample_IV_Size;
else
*outIVLength = 0;
}
return GF_OK;
}
#ifndef GPAC_DISABLE_ISOM_WRITE
GF_Err gf_isom_set_cenc_protection(GF_ISOFile *the_file, u32 trackNumber, u32 desc_index, u32 scheme_type,
u32 scheme_version, u32 default_IsEncrypted, u8 default_IV_size, bin128 default_KID,
u8 default_crypt_byte_block, u8 default_skip_byte_block,
u8 default_constant_IV_size, bin128 default_constant_IV)
{
GF_Err e;
GF_ProtectionSchemeInfoBox *sinf;
e = isom_set_protected_entry(the_file, trackNumber, desc_index, 0, 0, scheme_type, scheme_version, NULL, GF_FALSE, &sinf);
if (e) return e;
sinf->info->tenc = (GF_TrackEncryptionBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_TENC);
sinf->info->tenc->isProtected = default_IsEncrypted;
sinf->info->tenc->Per_Sample_IV_Size = default_IV_size;
memcpy(sinf->info->tenc->KID, default_KID, 16*sizeof(char));
if ((scheme_type == GF_ISOM_CENS_SCHEME) || (scheme_type == GF_ISOM_CBCS_SCHEME)) {
sinf->info->tenc->version = 1;
sinf->info->tenc->crypt_byte_block = default_crypt_byte_block;
sinf->info->tenc->skip_byte_block = default_skip_byte_block;
}
if (scheme_type == GF_ISOM_CBCS_SCHEME) {
sinf->info->tenc->constant_IV_size = default_constant_IV_size;
memcpy(sinf->info->tenc->constant_IV, default_constant_IV, 16*sizeof(char));
}
return GF_OK;
}
#if 0
GF_Err gf_isom_set_cenc_saio(GF_ISOFile *the_file, u32 trackNumber)
{
GF_SampleTableBox *stbl;
GF_SampleAuxiliaryInfoOffsetBox *saio;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
saio = (GF_SampleAuxiliaryInfoOffsetBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_SAIO);
saio->aux_info_type = GF_4CC('c', 'e', 'n', 'c');
saio->aux_info_type_parameter = 0;
if (!stbl->sai_offsets) stbl->sai_offsets = gf_list_new();
gf_list_add(stbl->sai_offsets, saio);
return GF_OK;
}
#endif
GF_Err gf_isom_remove_cenc_saiz(GF_ISOFile *the_file, u32 trackNumber)
{
u32 i;
GF_SampleTableBox *stbl;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
for (i = 0; i < gf_list_count(stbl->sai_sizes); i++) {
GF_SampleAuxiliaryInfoSizeBox *saiz = (GF_SampleAuxiliaryInfoSizeBox *)gf_list_get(stbl->sai_sizes, i);
if (saiz->aux_info_type != GF_4CC('c', 'e', 'n', 'c'))
continue;
gf_isom_box_del((GF_Box *)saiz);
gf_list_rem(stbl->sai_sizes, i);
i--;
}
if (!gf_list_count(stbl->sai_sizes)) {
gf_list_del(stbl->sai_sizes);
stbl->sai_sizes = NULL;
}
return GF_OK;
}
GF_Err gf_isom_remove_cenc_saio(GF_ISOFile *the_file, u32 trackNumber)
{
u32 i;
GF_SampleTableBox *stbl;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
for (i = 0; i < gf_list_count(stbl->sai_offsets); i++) {
GF_SampleAuxiliaryInfoOffsetBox *saio = (GF_SampleAuxiliaryInfoOffsetBox *)gf_list_get(stbl->sai_offsets, i);
if (saio->aux_info_type != GF_4CC('c', 'e', 'n', 'c'))
continue;
gf_isom_box_del((GF_Box *)saio);
gf_list_rem(stbl->sai_offsets, i);
i--;
}
if (!gf_list_count(stbl->sai_offsets)) {
gf_list_del(stbl->sai_offsets);
stbl->sai_offsets = NULL;
}
return GF_OK;
}
GF_Err gf_cenc_set_pssh(GF_ISOFile *mp4, bin128 systemID, u32 version, u32 KID_count, bin128 *KIDs, char *data, u32 len) {
GF_ProtectionSystemHeaderBox *pssh;
pssh = (GF_ProtectionSystemHeaderBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_PSSH);
if (!pssh)
return GF_IO_ERR;
memmove((char *)pssh->SystemID, systemID, 16);
pssh->version = version;
if (version) {
pssh->KID_count = KID_count;
if (KID_count) {
if (!pssh->KIDs) pssh->KIDs = (bin128 *)gf_malloc(pssh->KID_count*sizeof(bin128));
memmove(pssh->KIDs, KIDs, pssh->KID_count*sizeof(bin128));
}
}
pssh->private_data_size = len;
if (!pssh->private_data)
pssh->private_data = (u8 *)gf_malloc(pssh->private_data_size*sizeof(char));
memmove((char *)pssh->private_data, data, pssh->private_data_size);
if (!mp4->moov->other_boxes) mp4->moov->other_boxes = gf_list_new();
gf_list_add(mp4->moov->other_boxes, pssh);
return GF_OK;
}
GF_Err gf_isom_remove_samp_enc_box(GF_ISOFile *the_file, u32 trackNumber)
{
u32 i;
GF_SampleTableBox *stbl;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
for (i = 0; i < gf_list_count(stbl->other_boxes); i++) {
GF_Box *a = (GF_Box *)gf_list_get(stbl->other_boxes, i);
if ((a->type ==GF_ISOM_BOX_TYPE_UUID) && (((GF_UUIDBox *)a)->internal_4cc == GF_ISOM_BOX_UUID_PSEC)) {
gf_isom_box_del(a);
gf_list_rem(stbl->other_boxes, i);
i--;
}
else if (a->type == GF_ISOM_BOX_TYPE_SENC) {
gf_isom_box_del(a);
gf_list_rem(stbl->other_boxes, i);
i--;
}
}
if (!gf_list_count(stbl->other_boxes)) {
gf_list_del(stbl->other_boxes);
stbl->other_boxes = NULL;
}
return GF_OK;
}
GF_Err gf_isom_remove_samp_group_box(GF_ISOFile *the_file, u32 trackNumber)
{
u32 i;
GF_SampleTableBox *stbl;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
for (i = 0; i < gf_list_count(stbl->sampleGroupsDescription); i++) {
GF_SampleGroupDescriptionBox *a = (GF_SampleGroupDescriptionBox *)gf_list_get(stbl->sampleGroupsDescription, i);
if (a->grouping_type == GF_4CC( 's', 'e', 'i', 'g' )) {
gf_list_rem(stbl->sampleGroupsDescription, i);
gf_isom_box_del((GF_Box *) a);
i--;
}
}
if (!gf_list_count(stbl->sampleGroupsDescription)) {
gf_list_del(stbl->sampleGroupsDescription);
stbl->sampleGroupsDescription = NULL;
}
for (i = 0; i < gf_list_count(stbl->sampleGroups); i++) {
GF_SampleGroupBox *a = (GF_SampleGroupBox *)gf_list_get(stbl->sampleGroups, i);
if (a->grouping_type == GF_4CC( 's', 'e', 'i', 'g' )) {
gf_list_rem(stbl->sampleGroups, i);
gf_isom_box_del((GF_Box *) a);
i--;
}
}
if (!gf_list_count(stbl->sampleGroups)) {
gf_list_del(stbl->sampleGroups);
stbl->sampleGroups = NULL;
}
return GF_OK;
}
GF_Err gf_isom_remove_pssh_box(GF_ISOFile *the_file)
{
u32 i;
for (i = 0; i < gf_list_count(the_file->moov->other_boxes); i++) {
GF_Box *a = (GF_Box *)gf_list_get(the_file->moov->other_boxes, i);
if (a->type == GF_4CC('p', 's', 's', 'h')) {
gf_list_rem(the_file->moov->other_boxes, i);
gf_isom_box_del(a);
i--;
}
}
if (!gf_list_count(the_file->moov->other_boxes)) {
gf_list_del(the_file->moov->other_boxes);
the_file->moov->other_boxes = NULL;
}
return GF_OK;
}
#endif
GF_SampleEncryptionBox * gf_isom_create_piff_psec_box(u8 version, u32 flags, u32 AlgorithmID, u8 IV_size, bin128 KID)
{
GF_SampleEncryptionBox *psec;
psec = (GF_SampleEncryptionBox *) gf_isom_box_new(GF_ISOM_BOX_UUID_PSEC);
if (!psec)
return NULL;
psec->version = version;
psec->flags = flags;
psec->is_piff = GF_TRUE;
if (psec->flags & 0x1) {
psec->AlgorithmID = AlgorithmID;
psec->IV_size = IV_size;
strcpy((char *)psec->KID, (const char *)KID);
}
psec->samp_aux_info = gf_list_new();
return psec;
}
GF_SampleEncryptionBox * gf_isom_create_samp_enc_box(u8 version, u32 flags)
{
GF_SampleEncryptionBox *senc;
senc = (GF_SampleEncryptionBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_SENC);
if (!senc)
return NULL;
senc->version = version;
senc->flags = flags;
senc->samp_aux_info = gf_list_new();
return senc;
}
GF_Err gf_isom_cenc_allocate_storage(GF_ISOFile *the_file, u32 trackNumber, u32 container_type, u32 AlgorithmID, u8 IV_size, bin128 KID)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
if (trak->sample_encryption) return GF_OK;
switch (container_type) {
case GF_ISOM_BOX_UUID_PSEC:
trak->sample_encryption = (GF_SampleEncryptionBox *)gf_isom_create_piff_psec_box(1, 0, AlgorithmID, IV_size, KID);
break;
case GF_ISOM_BOX_TYPE_SENC:
trak->sample_encryption = (GF_SampleEncryptionBox *)gf_isom_create_samp_enc_box(0, 0);
break;
default:
return GF_NOT_SUPPORTED;
}
return gf_isom_box_add_default((GF_Box *) trak, (GF_Box *) trak->sample_encryption);
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
void gf_isom_cenc_set_saiz_saio(GF_SampleEncryptionBox *senc, GF_SampleTableBox *stbl, GF_TrackFragmentBox *traf, u32 len)
{
u32 i;
if (!senc->cenc_saiz) {
senc->cenc_saiz = (GF_SampleAuxiliaryInfoSizeBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_SAIZ);
senc->cenc_saiz->aux_info_type = GF_4CC('c', 'e', 'n', 'c');
senc->cenc_saiz->aux_info_type_parameter = 0;
if (stbl)
stbl_AddBox((GF_Box*)stbl, (GF_Box *)senc->cenc_saiz);
else
traf_AddBox((GF_Box*)traf, (GF_Box *)senc->cenc_saiz);
}
if (!senc->cenc_saio) {
senc->cenc_saio = (GF_SampleAuxiliaryInfoOffsetBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_SAIO);
senc->cenc_saio->version = 1;
senc->cenc_saio->aux_info_type = GF_4CC('c', 'e', 'n', 'c');
senc->cenc_saio->aux_info_type_parameter = 0;
senc->cenc_saio->entry_count = 1;
if (stbl)
stbl_AddBox((GF_Box*)stbl, (GF_Box *)senc->cenc_saio);
else
traf_AddBox((GF_Box*)traf, (GF_Box *)senc->cenc_saio);
}
if (!senc->cenc_saiz->sample_count || ((senc->cenc_saiz->default_sample_info_size==len) && len) ) {
senc->cenc_saiz->sample_count ++;
senc->cenc_saiz->default_sample_info_size = len;
} else {
senc->cenc_saiz->sample_info_size = (u8*)gf_realloc(senc->cenc_saiz->sample_info_size, sizeof(u8)*(senc->cenc_saiz->sample_count+1));
if (senc->cenc_saiz->default_sample_info_size) {
for (i=0; i<senc->cenc_saiz->sample_count; i++)
senc->cenc_saiz->sample_info_size[i] = senc->cenc_saiz->default_sample_info_size;
senc->cenc_saiz->default_sample_info_size = 0;
}
senc->cenc_saiz->sample_info_size[senc->cenc_saiz->sample_count] = len;
senc->cenc_saiz->sample_count++;
}
}
void gf_isom_cenc_merge_saiz_saio(GF_SampleEncryptionBox *senc, GF_SampleTableBox *stbl, u64 offset, u32 len)
{
u32 i;
if (!senc->cenc_saiz) {
senc->cenc_saiz = (GF_SampleAuxiliaryInfoSizeBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_SAIZ);
senc->cenc_saiz->aux_info_type = GF_4CC('c', 'e', 'n', 'c');
senc->cenc_saiz->aux_info_type_parameter = 0;
if (stbl)
stbl_AddBox((GF_Box*)stbl, (GF_Box *)senc->cenc_saiz);
}
if (!senc->cenc_saio) {
senc->cenc_saio = (GF_SampleAuxiliaryInfoOffsetBox *) gf_isom_box_new(GF_ISOM_BOX_TYPE_SAIO);
senc->cenc_saio->version = 1;
senc->cenc_saio->aux_info_type = GF_4CC('c', 'e', 'n', 'c');
senc->cenc_saio->aux_info_type_parameter = 0;
if (stbl)
stbl_AddBox((GF_Box*)stbl, (GF_Box *)senc->cenc_saio);
}
if (!senc->cenc_saiz->sample_count || (senc->cenc_saiz->default_sample_info_size==len)) {
senc->cenc_saiz->sample_count ++;
senc->cenc_saiz->default_sample_info_size = len;
} else {
senc->cenc_saiz->sample_info_size = (u8*)gf_realloc(senc->cenc_saiz->sample_info_size, sizeof(u8)*(senc->cenc_saiz->sample_count+1));
if (senc->cenc_saiz->default_sample_info_size) {
for (i=0; i<senc->cenc_saiz->sample_count; i++)
senc->cenc_saiz->sample_info_size[i] = senc->cenc_saiz->default_sample_info_size;
senc->cenc_saiz->default_sample_info_size = 0;
}
senc->cenc_saiz->sample_info_size[senc->cenc_saiz->sample_count] = len;
senc->cenc_saiz->sample_count++;
}
if (!senc->cenc_saio->entry_count) {
senc->cenc_saio->offsets_large = (u64 *)gf_malloc(sizeof(u64));
senc->cenc_saio->offsets_large[0] = offset;
senc->cenc_saio->entry_count ++;
} else {
senc->cenc_saio->offsets_large = (u64*)gf_realloc(senc->cenc_saio->offsets_large, sizeof(u64)*(senc->cenc_saio->entry_count+1));
senc->cenc_saio->offsets_large[senc->cenc_saio->entry_count] = offset;
senc->cenc_saio->entry_count++;
}
}
#endif
GF_Err gf_isom_track_cenc_add_sample_info(GF_ISOFile *the_file, u32 trackNumber, u32 container_type, u8 IV_size, char *buf, u32 len)
{
u32 i;
GF_SampleEncryptionBox *senc;
GF_CENCSampleAuxInfo *sai;
GF_SampleTableBox *stbl;
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl) return GF_BAD_PARAM;
switch (container_type) {
case GF_ISOM_BOX_UUID_PSEC:
case GF_ISOM_BOX_TYPE_SENC:
senc = trak->sample_encryption;
break;
default:
return GF_NOT_SUPPORTED;
}
sai = (GF_CENCSampleAuxInfo *)gf_malloc(sizeof(GF_CENCSampleAuxInfo));
if (!sai) return GF_OUT_OF_MEM;
memset(sai, 0, sizeof(GF_CENCSampleAuxInfo));
if (len) {
GF_BitStream *bs = gf_bs_new(buf, len, GF_BITSTREAM_READ);
sai->IV_size = IV_size;
gf_bs_read_data(bs, (char *)sai->IV, IV_size);
sai->subsample_count = gf_bs_read_u16(bs);
if (sai->subsample_count) senc->flags = 0x00000002;
sai->subsamples = (GF_CENCSubSampleEntry *)gf_malloc(sai->subsample_count*sizeof(GF_CENCSubSampleEntry));
for (i = 0; i < sai->subsample_count; i++) {
sai->subsamples[i].bytes_clear_data = gf_bs_read_u16(bs);
sai->subsamples[i].bytes_encrypted_data = gf_bs_read_u32(bs);
}
gf_bs_del(bs);
}
gf_list_add(senc->samp_aux_info, sai);
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
gf_isom_cenc_set_saiz_saio(senc, stbl, NULL, len);
#endif
return GF_OK;
}
GF_EXPORT
void gf_isom_cenc_samp_aux_info_del(GF_CENCSampleAuxInfo *samp)
{
if (samp->subsamples) gf_free(samp->subsamples);
gf_free(samp);
}
Bool gf_isom_cenc_has_saiz_saio_full(GF_SampleTableBox *stbl, void *_traf)
{
u32 i;
GF_List *sai_sizes, *sai_offsets;
Bool has_saiz, has_saio;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
GF_TrackFragmentBox *traf=(GF_TrackFragmentBox *)_traf;
#endif
has_saiz = has_saio = GF_FALSE;
if (stbl) {
sai_sizes = stbl->sai_sizes;
sai_offsets = stbl->sai_offsets;
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
else if (_traf) {
sai_sizes = traf->sai_sizes;
sai_offsets = traf->sai_offsets;
}
#endif
else
return GF_FALSE;
for (i = 0; i < gf_list_count(sai_sizes); i++) {
GF_SampleAuxiliaryInfoSizeBox *saiz = (GF_SampleAuxiliaryInfoSizeBox *)gf_list_get(sai_sizes, i);
if (saiz->aux_info_type == GF_4CC('c', 'e', 'n', 'c')) {
has_saiz = GF_TRUE;
break;
}
}
for (i = 0; i < gf_list_count(sai_offsets); i++) {
GF_SampleAuxiliaryInfoOffsetBox *saio = (GF_SampleAuxiliaryInfoOffsetBox *)gf_list_get(sai_offsets, i);
if (saio->aux_info_type == GF_4CC('c', 'e', 'n', 'c')) {
has_saio = GF_TRUE;
break;
}
}
return (has_saiz && has_saio);
}
Bool gf_isom_cenc_has_saiz_saio_track(GF_SampleTableBox *stbl)
{
return gf_isom_cenc_has_saiz_saio_full(stbl, NULL);
}
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
Bool gf_isom_cenc_has_saiz_saio_traf(GF_TrackFragmentBox *traf)
{
return gf_isom_cenc_has_saiz_saio_full(NULL, traf);
}
#endif
static GF_Err isom_cenc_get_sai_by_saiz_saio(GF_MediaBox *mdia, u32 sampleNumber, u8 IV_size, GF_CENCSampleAuxInfo **sai)
{
GF_BitStream *bs;
u32 prev_sai_size, size, i, j, nb_saio;
u64 cur_position, offset;
GF_Err e = GF_OK;
char *buffer;
nb_saio = size = prev_sai_size = 0;
offset = 0;
for (i = 0; i < gf_list_count(mdia->information->sampleTable->sai_offsets); i++) {
GF_SampleAuxiliaryInfoOffsetBox *saio = (GF_SampleAuxiliaryInfoOffsetBox *)gf_list_get(mdia->information->sampleTable->sai_offsets, i);
if (saio->aux_info_type == GF_4CC('c', 'e', 'n', 'c')) {
if (saio->entry_count == 1)
offset = saio->version ? saio->offsets_large[0] : saio->offsets[0];
else
offset = saio->version ? saio->offsets_large[sampleNumber-1]: saio->offsets[sampleNumber-1];
nb_saio = saio->entry_count;
break;
}
}
for (i = 0; i < gf_list_count(mdia->information->sampleTable->sai_sizes); i++) {
GF_SampleAuxiliaryInfoSizeBox *saiz = (GF_SampleAuxiliaryInfoSizeBox *)gf_list_get(mdia->information->sampleTable->sai_sizes, i);
if (saiz->aux_info_type == GF_4CC('c', 'e', 'n', 'c')) {
for (j = 0; j < sampleNumber-1; j++)
prev_sai_size += saiz->default_sample_info_size ? saiz->default_sample_info_size : saiz->sample_info_size[j];
size = saiz->default_sample_info_size ? saiz->default_sample_info_size : saiz->sample_info_size[sampleNumber-1];
break;
}
}
offset += (nb_saio == 1) ? prev_sai_size : 0;
cur_position = gf_bs_get_position(mdia->information->dataHandler->bs);
gf_bs_seek(mdia->information->dataHandler->bs, offset);
buffer = (char *)gf_malloc(size);
gf_bs_read_data(mdia->information->dataHandler->bs, buffer, size);
gf_bs_seek(mdia->information->dataHandler->bs, cur_position);
*sai = (GF_CENCSampleAuxInfo *)gf_malloc(sizeof(GF_CENCSampleAuxInfo));
memset(*sai, 0, sizeof(GF_CENCSampleAuxInfo));
bs = gf_bs_new(buffer, size, GF_BITSTREAM_READ);
gf_bs_read_data(bs, (char *)(*sai)->IV, IV_size);
if (size > IV_size) {
(*sai)->subsample_count = gf_bs_read_u16(bs);
(*sai)->subsamples = (GF_CENCSubSampleEntry *)gf_malloc(sizeof(GF_CENCSubSampleEntry)*(*sai)->subsample_count);
for (i = 0; i < (*sai)->subsample_count; i++) {
(*sai)->subsamples[i].bytes_clear_data = gf_bs_read_u16(bs);
(*sai)->subsamples[i].bytes_encrypted_data = gf_bs_read_u32(bs);
}
}
gf_bs_del(bs);
gf_free(buffer);
return e;
}
GF_EXPORT
GF_Err gf_isom_cenc_get_sample_aux_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleNumber, GF_CENCSampleAuxInfo **sai, u32 *container_type)
{
GF_TrackBox *trak;
GF_SampleTableBox *stbl;
GF_SampleEncryptionBox *senc = NULL;
u32 type;
GF_CENCSampleAuxInfo *a_sai;
u8 IV_size;
u32 is_Protected;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
stbl = trak->Media->information->sampleTable;
if (!stbl)
return GF_BAD_PARAM;
senc = trak->sample_encryption;
if (!senc)
return GF_BAD_PARAM;
if ((senc->type == GF_ISOM_BOX_TYPE_UUID) && (((GF_UUIDBox *)senc)->internal_4cc == GF_ISOM_BOX_UUID_PSEC)) {
type = GF_ISOM_BOX_UUID_PSEC;
} else if (senc->type == GF_ISOM_BOX_TYPE_SENC) {
type = GF_ISOM_BOX_TYPE_SENC;
} else {
type = 0;
}
if (container_type) *container_type = type;
if (!sai) return GF_OK;
#ifndef GPAC_DISABLE_ISOM_FRAGMENTS
sampleNumber -= trak->sample_count_at_seg_start;
#endif
if (*sai) {
gf_isom_cenc_samp_aux_info_del(*sai);
*sai = NULL;
}
gf_isom_get_sample_cenc_info_ex(trak, NULL, senc, sampleNumber, &is_Protected, &IV_size, NULL, NULL, NULL, NULL, NULL);
if (!is_Protected) {
GF_SAFEALLOC( (*sai), GF_CENCSampleAuxInfo);
return GF_OK;
}
if (IV_size && gf_isom_cenc_has_saiz_saio_track(stbl)) {
return isom_cenc_get_sai_by_saiz_saio(trak->Media, sampleNumber, IV_size, sai);
}
a_sai = NULL;
switch (type) {
case GF_ISOM_BOX_UUID_PSEC:
if (senc)
a_sai = (GF_CENCSampleAuxInfo *)gf_list_get(((GF_SampleEncryptionBox *)senc)->samp_aux_info, sampleNumber-1);
break;
case GF_ISOM_BOX_TYPE_SENC:
if (senc)
a_sai = (GF_CENCSampleAuxInfo *)gf_list_get(((GF_SampleEncryptionBox *)senc)->samp_aux_info, sampleNumber-1);
break;
}
if (!a_sai)
return GF_NOT_SUPPORTED;
GF_SAFEALLOC((*sai), GF_CENCSampleAuxInfo);
if (!(*sai) ) return GF_OUT_OF_MEM;
if (senc) {
u8 size = ((*sai)->IV_size != 0) ? (*sai)->IV_size : 8;
memmove((*sai)->IV, a_sai->IV, size);
(*sai)->subsample_count = a_sai->subsample_count;
if ((*sai)->subsample_count > 0) {
(*sai)->subsamples = (GF_CENCSubSampleEntry*)gf_malloc(sizeof(GF_CENCSubSampleEntry)*(*sai)->subsample_count);
memmove((*sai)->subsamples, a_sai->subsamples, sizeof(GF_CENCSubSampleEntry)*(*sai)->subsample_count);
}
}
return GF_OK;
}
void gf_isom_cenc_get_default_info_ex(GF_TrackBox *trak, u32 sampleDescriptionIndex, u32 *default_IsEncrypted, u8 *default_IV_size, bin128 *default_KID)
{
GF_ProtectionSchemeInfoBox *sinf;
if (default_IsEncrypted) *default_IsEncrypted = 0;
if (default_IV_size) *default_IV_size = 0;
if (default_KID) memset(*default_KID, 0, 16);
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, NULL);
if (sinf && sinf->info && sinf->info->tenc) {
if (default_IsEncrypted) *default_IsEncrypted = sinf->info->tenc->isProtected;
if (default_IV_size) *default_IV_size = sinf->info->tenc->Per_Sample_IV_Size;
if (default_KID) memmove(*default_KID, sinf->info->tenc->KID, 16);
} else {
if (! trak->moov->mov->is_smooth) {
trak->moov->mov->is_smooth = GF_TRUE;
GF_LOG(GF_LOG_WARNING, GF_LOG_CONTAINER, ("[iso file] senc box without tenc, assuming MS smooth+piff\n"));
}
if (default_IsEncrypted) *default_IsEncrypted = GF_TRUE;
if (default_IV_size) *default_IV_size = 0;
}
}
GF_EXPORT
void gf_isom_cenc_get_default_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *default_IsEncrypted, u8 *default_IV_size, bin128 *default_KID)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return;
gf_isom_cenc_get_default_info_ex(trak, sampleDescriptionIndex, default_IsEncrypted, default_IV_size, default_KID);
}
void gf_isom_cenc_get_default_pattern_info_ex(GF_TrackBox *trak, u32 sampleDescriptionIndex, u8 *default_crypt_byte_block, u8 *default_skip_byte_block)
{
GF_ProtectionSchemeInfoBox *sinf;
if (default_crypt_byte_block) *default_crypt_byte_block = 0;
if (default_skip_byte_block) *default_skip_byte_block = 0;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBC_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, NULL);
if (sinf && sinf->info && sinf->info->tenc) {
}
}
GF_EXPORT
void gf_isom_cenc_get_default_pattern_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u8 *default_crypt_byte_block, u8 *default_skip_byte_block)
{
GF_TrackBox *trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return;
gf_isom_cenc_get_default_pattern_info_ex(trak, sampleDescriptionIndex, default_crypt_byte_block, default_skip_byte_block);
}
GF_Err gf_isom_set_adobe_protection(GF_ISOFile *the_file, u32 trackNumber, u32 desc_index, u32 scheme_type, u32 scheme_version, Bool is_selective_enc, char *metadata, u32 len)
{
GF_ProtectionSchemeInfoBox *sinf;
#ifndef GPAC_DISABLE_ISOM_WRITE
GF_Err e;
e = isom_set_protected_entry(the_file, trackNumber, desc_index, 1, 0, scheme_type, scheme_version, NULL, GF_FALSE, &sinf);
if (e) return e;
#else
return GF_NOT_SUPPORTED;
#endif
sinf->info->adkm = (GF_AdobeDRMKeyManagementSystemBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_ADKM);
sinf->info->adkm->header = (GF_AdobeDRMHeaderBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_AHDR);
sinf->info->adkm->header->std_enc_params = (GF_AdobeStdEncryptionParamsBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_APRM);
sinf->info->adkm->header->std_enc_params->enc_info = (GF_AdobeEncryptionInfoBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_AEIB);
if (sinf->info->adkm->header->std_enc_params->enc_info->enc_algo)
gf_free(sinf->info->adkm->header->std_enc_params->enc_info->enc_algo);
sinf->info->adkm->header->std_enc_params->enc_info->enc_algo = (char *)gf_malloc(8*sizeof(char));
strncpy(sinf->info->adkm->header->std_enc_params->enc_info->enc_algo, "AES-CBC", 7);
sinf->info->adkm->header->std_enc_params->enc_info->enc_algo[7] = 0;
sinf->info->adkm->header->std_enc_params->enc_info->key_length = 16;
sinf->info->adkm->header->std_enc_params->key_info = (GF_AdobeKeyInfoBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_AKEY);
sinf->info->adkm->header->std_enc_params->key_info->params = (GF_AdobeFlashAccessParamsBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_FLXS);
if (metadata && len) {
if (sinf->info->adkm->header->std_enc_params->key_info->params->metadata)
gf_free(sinf->info->adkm->header->std_enc_params->key_info->params->metadata);
sinf->info->adkm->header->std_enc_params->key_info->params->metadata = (char *)gf_malloc((len+1)*sizeof(char));
strncpy(sinf->info->adkm->header->std_enc_params->key_info->params->metadata, metadata, len);
sinf->info->adkm->header->std_enc_params->key_info->params->metadata[len] = 0;
}
sinf->info->adkm->au_format = (GF_AdobeDRMAUFormatBox *)gf_isom_box_new(GF_ISOM_BOX_TYPE_ADAF);
sinf->info->adkm->au_format->selective_enc = is_selective_enc ? 0x10 : 0x00;
sinf->info->adkm->au_format->IV_length = 16;
return GF_OK;
}
GF_EXPORT
Bool gf_isom_is_adobe_protection_media(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ADOBE_SCHEME, NULL);
if (!sinf) return GF_FALSE;
if (!sinf->info || !sinf->info->adkm)
return GF_FALSE;
return GF_TRUE;
}
GF_EXPORT
GF_Err gf_isom_get_adobe_protection_info(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex, u32 *outOriginalFormat, u32 *outSchemeType, u32 *outSchemeVersion)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_BAD_PARAM;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_ADOBE_SCHEME, NULL);
if (!sinf) return GF_BAD_PARAM;
if (outOriginalFormat) {
*outOriginalFormat = sinf->original_format->data_format;
if (IsMP4Description(sinf->original_format->data_format)) *outOriginalFormat = GF_ISOM_SUBTYPE_MPEG4;
}
if (outSchemeType) *outSchemeType = sinf->scheme_type->scheme_type;
if (outSchemeVersion) *outSchemeVersion = sinf->scheme_type->scheme_version;
return GF_OK;
}
GF_EXPORT
Bool gf_isom_cenc_is_pattern_mode(GF_ISOFile *the_file, u32 trackNumber, u32 sampleDescriptionIndex)
{
GF_TrackBox *trak;
GF_ProtectionSchemeInfoBox *sinf;
trak = gf_isom_get_track_from_file(the_file, trackNumber);
if (!trak) return GF_FALSE;
sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CENS_SCHEME, NULL);
if (!sinf) sinf = isom_get_sinf_entry(trak, sampleDescriptionIndex, GF_ISOM_CBCS_SCHEME, NULL);
if (!sinf) return GF_FALSE;
if (!sinf->info || !sinf->info->tenc || !sinf->info->tenc->crypt_byte_block || !sinf->info->tenc->skip_byte_block)
return GF_FALSE;
return GF_TRUE;
}
void gf_isom_ipmpx_remove_tool_list(GF_ISOFile *the_file)
{
if (the_file && the_file->moov && the_file->moov->iods && (the_file ->moov->iods->descriptor->tag == GF_ODF_ISOM_IOD_TAG) ) {
GF_IsomInitialObjectDescriptor *iod = (GF_IsomInitialObjectDescriptor *)the_file ->moov->iods->descriptor;
if (iod->IPMPToolList) gf_odf_desc_del((GF_Descriptor*) iod->IPMPToolList);
iod->IPMPToolList = NULL;
}
}
#endif