#include "mpeg4-avc.h" #include #include #include /* ISO/IEC 14496-15:2010(E) 5.2.4.1.1 Syntax (p16) aligned(8) class AVCDecoderConfigurationRecord { unsigned int(8) configurationVersion = 1; unsigned int(8) AVCProfileIndication; unsigned int(8) profile_compatibility; unsigned int(8) AVCLevelIndication; bit(6) reserved = '111111'b; unsigned int(2) lengthSizeMinusOne; bit(3) reserved = '111'b; unsigned int(5) numOfSequenceParameterSets; for (i=0; i< numOfSequenceParameterSets; i++) { unsigned int(16) sequenceParameterSetLength ; bit(8*sequenceParameterSetLength) sequenceParameterSetNALUnit; } unsigned int(8) numOfPictureParameterSets; for (i=0; i< numOfPictureParameterSets; i++) { unsigned int(16) pictureParameterSetLength; bit(8*pictureParameterSetLength) pictureParameterSetNALUnit; } if( profile_idc == 100 || profile_idc == 110 || profile_idc == 122 || profile_idc == 144 ) { bit(6) reserved = '111111'b; unsigned int(2) chroma_format; bit(5) reserved = '11111'b; unsigned int(3) bit_depth_luma_minus8; bit(5) reserved = '11111'b; unsigned int(3) bit_depth_chroma_minus8; unsigned int(8) numOfSequenceParameterSetExt; for (i=0; i< numOfSequenceParameterSetExt; i++) { unsigned int(16) sequenceParameterSetExtLength; bit(8*sequenceParameterSetExtLength) sequenceParameterSetExtNALUnit; } } } */ static int _mpeg4_avc_decoder_configuration_record_load(const uint8_t* data, size_t bytes, struct mpeg4_avc_t* avc) { uint8_t i; uint32_t j; uint16_t len; uint8_t *p, *end; if (bytes < 7) return -1; assert(1 == data[0]); // avc->version = data[0]; avc->profile = data[1]; avc->compatibility = data[2]; avc->level = data[3]; avc->nalu = (data[4] & 0x03) + 1; avc->nb_sps = data[5] & 0x1F; if (avc->nb_sps > sizeof(avc->sps) / sizeof(avc->sps[0])) { assert(0); return -1; // sps <= 32 } j = 6; p = avc->data; end = avc->data + sizeof(avc->data); for (i = 0; i < avc->nb_sps && j + 2 < bytes; ++i) { len = (data[j] << 8) | data[j + 1]; if (j + 2 + len >= bytes || p + len > end) { assert(0); return -1; } memcpy(p, data + j + 2, len); avc->sps[i].data = p; avc->sps[i].bytes = len; j += len + 2; p += len; } if (j >= bytes || (unsigned int)data[j] > sizeof(avc->pps) / sizeof(avc->pps[0])) { assert(0); return -1; } avc->nb_pps = data[j++]; for (i = 0; i < avc->nb_pps && j + 2 < bytes; i++) { len = (data[j] << 8) | data[j + 1]; if (j + 2 + len > bytes || p + len > end) { assert(0); return -1; } memcpy(p, data + j + 2, len); avc->pps[i].data = p; avc->pps[i].bytes = len; j += len + 2; p += len; } avc->off = (int)(p - avc->data); return j; } int mpeg4_avc_decoder_configuration_record_save(const struct mpeg4_avc_t* avc, uint8_t* data, size_t bytes) { uint8_t i; uint8_t *p = data; assert(0 < avc->nalu && avc->nalu <= 4); if (bytes < 7 || avc->nb_sps > 32) return -1; bytes -= 7; // AVCDecoderConfigurationRecord // ISO/IEC 14496-15:2010 // 5.2.4.1.1 Syntax p[0] = 1; // configurationVersion p[1] = avc->profile; // AVCProfileIndication p[2] = avc->compatibility; // profile_compatibility p[3] = avc->level; // AVCLevelIndication p[4] = 0xFC | (avc->nalu - 1); // lengthSizeMinusOne: 3 p += 5; // sps *p++ = 0xE0 | avc->nb_sps; for (i = 0; i < avc->nb_sps && bytes >= (size_t)avc->sps[i].bytes + 2; i++) { *p++ = (avc->sps[i].bytes >> 8) & 0xFF; *p++ = avc->sps[i].bytes & 0xFF; memcpy(p, avc->sps[i].data, avc->sps[i].bytes); p += avc->sps[i].bytes; bytes -= avc->sps[i].bytes + 2; } if (i < avc->nb_sps) return -1; // check length // pps *p++ = avc->nb_pps; for (i = 0; i < avc->nb_pps && bytes >= (size_t)avc->pps[i].bytes + 2; i++) { *p++ = (avc->pps[i].bytes >> 8) & 0xFF; *p++ = avc->pps[i].bytes & 0xFF; memcpy(p, avc->pps[i].data, avc->pps[i].bytes); p += avc->pps[i].bytes; bytes -= avc->pps[i].bytes + 2; } if (i < avc->nb_pps) return -1; // check length if (bytes >= 4) { if (avc->profile == 100 || avc->profile == 110 || avc->profile == 122 || avc->profile == 244 || avc->profile == 44 || avc->profile == 83 || avc->profile == 86 || avc->profile == 118 || avc->profile == 128 || avc->profile == 138 || avc->profile == 139 || avc->profile == 134) { *p++ = 0xFC | avc->chroma_format_idc; *p++ = 0xF8 | avc->bit_depth_luma_minus8; *p++ = 0xF8 | avc->bit_depth_chroma_minus8; *p++ = 0; // numOfSequenceParameterSetExt } } return (int)(p - data); } #define H264_STARTCODE(p) (p[0]==0 && p[1]==0 && (p[2]==1 || (p[2]==0 && p[3]==1))) int mpeg4_avc_from_nalu(const uint8_t* data, size_t bytes, struct mpeg4_avc_t* avc) { int r; r = h264_annexbtomp4(avc, data, bytes, NULL, 0, NULL, NULL); return avc->nb_sps > 0 && avc->nb_pps > 0 ? bytes : r; } int mpeg4_avc_to_nalu(const struct mpeg4_avc_t* avc, uint8_t* data, size_t bytes) { uint8_t i; size_t k = 0; uint8_t* h264 = data; // sps for (i = 0; i < avc->nb_sps && bytes >= k + avc->sps[i].bytes + 4; i++) { if (avc->sps[i].bytes < 4 || !H264_STARTCODE(avc->sps[i].data)) { h264[k++] = 0; h264[k++] = 0; h264[k++] = 0; h264[k++] = 1; } memcpy(h264 + k, avc->sps[i].data, avc->sps[i].bytes); k += avc->sps[i].bytes; } if (i < avc->nb_sps) return -1; // check length // pps for (i = 0; i < avc->nb_pps && bytes >= k + avc->pps[i].bytes + 2; i++) { if (avc->pps[i].bytes < 4 || !H264_STARTCODE(avc->pps[i].data)) { h264[k++] = 0; h264[k++] = 0; h264[k++] = 0; h264[k++] = 1; } memcpy(h264 + k, avc->pps[i].data, avc->pps[i].bytes); k += avc->pps[i].bytes; } if (i < avc->nb_pps) return -1; // check length assert(k < 0x7FFF); return (int)k; } int mpeg4_avc_codecs(const struct mpeg4_avc_t* avc, char* codecs, size_t bytes) { // https://tools.ietf.org/html/rfc6381#section-3.3 // https://developer.mozilla.org/en-US/docs/Web/Media/Formats/codecs_parameter return snprintf(codecs, bytes, "avc1.%02x%02x%02x", avc->profile, avc->compatibility, avc->level); } int mpeg4_avc_decoder_configuration_record_load(const uint8_t* data, size_t bytes, struct mpeg4_avc_t* avc) { int r; r = _mpeg4_avc_decoder_configuration_record_load(data, bytes, avc); if (r > 0 && avc->nb_sps > 0 && avc->nb_pps > 0) return r; // try annexb memset(avc, 0, sizeof(*avc)); return mpeg4_avc_from_nalu(data, bytes, avc); } #if defined(_DEBUG) || defined(DEBUG) void mpeg4_annexbtomp4_test(void); void mpeg4_avc_test(void) { const unsigned char src[] = { 0x01,0x42,0xe0,0x1e,0xff,0xe1,0x00,0x21,0x67,0x42,0xe0,0x1e,0xab,0x40,0xf0,0x28, 0xd0,0x80,0x00,0x00,0x00,0x80,0x00,0x00,0x19,0x70,0x20,0x00,0x78,0x00,0x00,0x0f, 0x00,0x16,0xb1,0xb0,0x3c,0x50,0xaa,0x80,0x80,0x01,0x00,0x04,0x28,0xce,0x3c,0x80 }; const unsigned char nalu[] = { 0x00,0x00,0x00,0x01,0x67,0x42,0xe0,0x1e,0xab,0x40,0xf0,0x28,0xd0,0x80,0x00,0x00, 0x00,0x80,0x00,0x00,0x19,0x70,0x20,0x00,0x78,0x00,0x00,0x0f,0x00,0x16,0xb1,0xb0, 0x3c,0x50,0xaa,0x80,0x80,0x00,0x00,0x00,0x01,0x28,0xce,0x3c,0x80 }; unsigned char data[sizeof(src)]; struct mpeg4_avc_t avc; assert(sizeof(src) == mpeg4_avc_decoder_configuration_record_load(src, sizeof(src), &avc)); assert(0x42 == avc.profile && 0xe0 == avc.compatibility && 0x1e == avc.level); assert(4 == avc.nalu && 1 == avc.nb_sps && 1 == avc.nb_pps); assert(sizeof(src) == mpeg4_avc_decoder_configuration_record_save(&avc, data, sizeof(data))); assert(0 == memcmp(src, data, sizeof(src))); mpeg4_avc_codecs(&avc, (char*)data, sizeof(data)); assert(0 == memcmp("avc1.42e01e", data, 11)); assert(sizeof(nalu) == mpeg4_avc_to_nalu(&avc, data, sizeof(data))); assert(0 == memcmp(nalu, data, sizeof(nalu))); mpeg4_annexbtomp4_test(); } #endif