#include "mpeg4-aac.h" #include #include int mpeg4_aac_adts_pce_load(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac); int mpeg4_aac_adts_pce_save(uint8_t* data, size_t bytes, const struct mpeg4_aac_t* aac); int mpeg4_aac_audio_specific_config_load2(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac); int mpeg4_aac_audio_specific_config_save2(const struct mpeg4_aac_t* aac, uint8_t* data, size_t bytes); int mpeg4_aac_stream_mux_config_load2(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac); /* // ISO-14496-3 adts_frame (p122) adts_fixed_header() { syncword; 12 bslbf ID; 1 bslbf layer; 2 uimsbf protection_absent; 1 bslbf profile_ObjectType; 2 uimsbf sampling_frequency_index; 4 uimsbf private_bit; 1 bslbf channel_configuration; 3 uimsbf original_copy; 1 bslbf home; 1 bslbf } adts_variable_header() { copyright_identification_bit; 1 bslbf copyright_identification_start; 1 bslbf aac_frame_length; 13 bslbf adts_buffer_fullness; 11 bslbf number_of_raw_data_blocks_in_frame; 2 uimsbf } */ /// @return >=0-adts header length, <0-error int mpeg4_aac_adts_load(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac) { if (bytes < 7) return -1; memset(aac, 0, sizeof(struct mpeg4_aac_t)); assert(0xFF == data[0] && 0xF0 == (data[1] & 0xF0)); /* syncword */ aac->profile = ((data[2] >> 6) & 0x03) + 1; // 2 bits: the MPEG-2 Audio Object Type add 1 aac->sampling_frequency_index = (data[2] >> 2) & 0x0F; // 4 bits: MPEG-4 Sampling Frequency Index (15 is forbidden) aac->channel_configuration = ((data[2] & 0x01) << 2) | ((data[3] >> 6) & 0x03); // 3 bits: MPEG-4 Channel Configuration assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); aac->channels = mpeg4_aac_channel_count(aac->channel_configuration); aac->sampling_frequency = mpeg4_aac_audio_frequency_to(aac->sampling_frequency_index); aac->extension_frequency = aac->sampling_frequency; if (0 == aac->channel_configuration) return mpeg4_aac_adts_pce_load(data, bytes, aac); return 7; } /// @return >=0-adts header length, <0-error int mpeg4_aac_adts_save(const struct mpeg4_aac_t* aac, size_t payload, uint8_t* data, size_t bytes) { const uint8_t ID = 0; // 0-MPEG4/1-MPEG2 size_t len = payload + 7; if (bytes < 7 || len >= (1 << 12)) return -1; if (0 == aac->channel_configuration && aac->npce > 0) len += mpeg4_aac_adts_pce_save(data, bytes, aac); assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); data[0] = 0xFF; /* 12-syncword */ data[1] = 0xF0 /* 12-syncword */ | (ID << 3)/*1-ID*/ | (0x00 << 2) /*2-layer*/ | 0x01 /*1-protection_absent*/; data[2] = ((aac->profile - 1) << 6) | ((aac->sampling_frequency_index & 0x0F) << 2) | ((aac->channel_configuration >> 2) & 0x01); data[3] = ((aac->channel_configuration & 0x03) << 6) | ((len >> 11) & 0x03); /*0-original_copy*/ /*0-home*/ /*0-copyright_identification_bit*/ /*0-copyright_identification_start*/ data[4] = (uint8_t)(len >> 3); data[5] = ((len & 0x07) << 5) | 0x1F; data[6] = 0xFC /*| ((len / (1024 * aac->channels)) & 0x03)*/; return (int)(len - payload); } int mpeg4_aac_adts_frame_length(const uint8_t* data, size_t bytes) { uint16_t len; if (bytes < 7) return -1; assert(0xFF == data[0] && 0xF0 == (data[1] & 0xF0)); /* syncword */ len = ((uint16_t)(data[3] & 0x03) << 11) | ((uint16_t)data[4] << 3) | ((uint16_t)(data[5] >> 5) & 0x07); return len; } // ISO-14496-3 AudioSpecificConfig (p52) /* audioObjectType; 5 uimsbf if (audioObjectType == 31) { audioObjectType = 32 + audioObjectTypeExt; 6 uimsbf } samplingFrequencyIndex; 4 bslbf if ( samplingFrequencyIndex == 0xf ) { samplingFrequency; 24 uimsbf } channelConfiguration; 4 bslbf */ /// @return >=0-adts header length, <0-error int mpeg4_aac_audio_specific_config_load(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac) { if (bytes < 2) return -1; memset(aac, 0, sizeof(struct mpeg4_aac_t)); aac->profile = (data[0] >> 3) & 0x1F; aac->sampling_frequency_index = ((data[0] & 0x7) << 1) | ((data[1] >> 7) & 0x01); aac->channel_configuration = (data[1] >> 3) & 0x0F; assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); aac->channels = mpeg4_aac_channel_count(aac->channel_configuration); aac->sampling_frequency = mpeg4_aac_audio_frequency_to(aac->sampling_frequency_index); aac->extension_frequency = aac->sampling_frequency; if (bytes > 2) return mpeg4_aac_audio_specific_config_load2(data, bytes, aac); return 2; } // ISO-14496-3 AudioSpecificConfig int mpeg4_aac_audio_specific_config_save(const struct mpeg4_aac_t* aac, uint8_t* data, size_t bytes) { uint8_t channel_configuration; if (bytes < 2+ (size_t)aac->npce) return -1; channel_configuration = aac->npce > 0 ? 0 : aac->channel_configuration; assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); data[0] = (aac->profile << 3) | ((aac->sampling_frequency_index >> 1) & 0x07); data[1] = ((aac->sampling_frequency_index & 0x01) << 7) | ((channel_configuration & 0xF) << 3) | (0 << 2) /* frame length-1024 samples*/ | (0 << 1) /* don't depend on core */ | 0 /* not extension */; if (0 == aac->channel_configuration && aac->npce > 0) return mpeg4_aac_audio_specific_config_save2(aac, data, bytes); return 2; } // ISO/IEC 14496-3:2009(E) Table 1.42 - Syntax of StreamMuxConfig() (p83) int mpeg4_aac_stream_mux_config_load(const uint8_t* data, size_t bytes, struct mpeg4_aac_t* aac) { if (bytes < 6) return -1; memset(aac, 0, sizeof(*aac)); if (6 == bytes && 0x40 == data[0] && 0 == (data[1] & 0xFE)) { // fast path // [0] 0-audioMuxVersion(1), 1-allStreamsSameTimeFraming(1), 0-numSubFrames(6) assert(0 == (0x80 & data[0])); // audioMuxVersion: 0 aac->profile = ((data[1] & 0x01) << 4) | (data[2] >> 4); // 0-numProgram(4), 0-numLayer(3), 1-ASC(1) aac->sampling_frequency_index = data[2] & 0x0F; aac->channel_configuration = data[3] >> 4; assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); aac->channels = mpeg4_aac_channel_count(aac->channel_configuration); aac->sampling_frequency = mpeg4_aac_audio_frequency_to(aac->sampling_frequency_index); aac->extension_frequency = aac->sampling_frequency; return 6; } return mpeg4_aac_stream_mux_config_load2(data, bytes, aac); } // ISO/IEC 14496-3:2009(E) Table 1.42 - Syntax of StreamMuxConfig() (p83) int mpeg4_aac_stream_mux_config_save(const struct mpeg4_aac_t* aac, uint8_t* data, size_t bytes) { int profile; int frequncy; if (bytes < 6) return -1; profile = aac->ps ? MPEG4_AAC_PS : aac->profile; frequncy = mpeg4_aac_audio_frequency_from(aac->extension_frequency); frequncy = (aac->sbr || aac->ps) && -1 != frequncy ? frequncy : 0; assert(aac->profile > 0 && aac->profile < 31); assert(aac->channel_configuration >= 0 && aac->channel_configuration <= 7); assert(aac->sampling_frequency_index >= 0 && aac->sampling_frequency_index <= 0xc); data[0] = 0x40; // 0-audioMuxVersion(1), 1-allStreamsSameTimeFraming(1), 0-numSubFrames(6) data[1] = 0x00 | ((profile >> 4) & 0x01); // 0-numProgram(4), 0-numLayer(3) data[2] = ((profile & 0x0F) << 4) | (aac->sampling_frequency_index & 0x0F); data[3] = ((aac->channel_configuration & 0x0F) << 4) | (-1 != frequncy ? (frequncy & 0x0F) : 0); // 0-GASpecificConfig(3), 0-frameLengthType(1) data[4] = 0x3F; // 0-frameLengthType(2), 111111-latmBufferFullness(6) data[5] = 0xC0; // 11-latmBufferFullness(2), 0-otherDataPresent, 0-crcCheckPresent return 6; } // Table 1.6 ¨C Levels for the High Quality Audio Profile static int mpeg4_aac_high_quality_level(const struct mpeg4_aac_t* aac) { if (aac->sampling_frequency <= 22050) { if (aac->channel_configuration <= 2) return 1; // Level 1/5 } else if (aac->sampling_frequency <= 48000) { if (aac->channel_configuration <= 2) return 2; // Level 2/6 else if (aac->channel_configuration <= 5) return 3; // Level 3/4/7/8 } return 8; } // Table 1.10 ¨C Levels for the AAC Profile static int mpeg4_aac_level(const struct mpeg4_aac_t* aac) { if (aac->sampling_frequency <= 24000) { if (aac->channel_configuration <= 2) return 1; // AAC Profile, Level 1 } else if (aac->sampling_frequency <= 48000) { if (aac->channel_configuration <= 2) return 2; // Level 2 else if (aac->channel_configuration <= 5) return 4; // Level 4 } else if (aac->sampling_frequency <= 96000) { if (aac->channel_configuration <= 5) return 5; // Level 5 } return 5; } static int mpeg4_aac_he_level(const struct mpeg4_aac_t* aac) { if (aac->sampling_frequency <= 48000) { if (aac->channel_configuration <= 2) return aac->sbr ? 3 : 2; // Level 2/3 else if (aac->channel_configuration <= 5) return 4; // Level 4 } else if (aac->sampling_frequency <= 96000) { if (aac->channel_configuration <= 5) return 5; // Level 5 } return 5; } // ISO/IEC 14496-3:2009(E) Table 1.14 - audioProfileLevelIndication values (p51) int mpeg4_aac_profile_level(const struct mpeg4_aac_t* aac) { // Table 1.10 - Levels for the AAC Profile (p49) // Table 1.14 - audioProfileLevelIndication values (p51) switch (aac->profile) { case MPEG4_AAC_LC: return mpeg4_aac_level(aac) - 1 + 0x28; // AAC Profile case MPEG4_AAC_SBR: return mpeg4_aac_he_level(aac) - 2 + 0x2C; // High Efficiency AAC Profile case MPEG4_AAC_PS: return mpeg4_aac_he_level(aac) - 2 + 0x30; // High Efficiency AAC v2 Profile case MPEG4_AAC_CELP: return mpeg4_aac_high_quality_level(aac) - 1 + 0x0E; // High Quality Audio Profile default: return 1; // Main Audio Profile, Level 1 } } #define ARRAYOF(arr) sizeof(arr)/sizeof(arr[0]) static const int s_frequency[] = { 96000, 88200, 64000, 48000, 44100, 32000, 24000, 22050, 16000, 12000, 11025, 8000, 7350 }; int mpeg4_aac_audio_frequency_to(enum mpeg4_aac_frequency index) { if (index < 0 || index >= ARRAYOF(s_frequency)) return 0; return s_frequency[index]; } int mpeg4_aac_audio_frequency_from(int frequence) { int i = 0; while (i < ARRAYOF(s_frequency) && s_frequency[i] != frequence) i++; return i >= ARRAYOF(s_frequency) ? -1 : i; } uint8_t mpeg4_aac_channel_count(uint8_t channel_configuration) { static const uint8_t s_channels[] = { 0, 1, 2, 3, 4, 5, 6, 8 }; if (channel_configuration < 0 || channel_configuration >= ARRAYOF(s_channels)) return 0; return s_channels[channel_configuration]; } #undef ARRAYOF #if defined(_DEBUG) || defined(DEBUG) void mpeg4_aac_test(void) { struct mpeg4_aac_t aac, aac2; const unsigned char asc[] = { 0x13, 0x88 }; const unsigned char adts[] = { 0xFF, 0xF1, 0x5C, 0x40, 0x01, 0x1F, 0xFC }; // const unsigned char ascsbr[] = { 0x13, 0x10, 0x56, 0xe5, 0x9d, 0x48, 0x00 }; const unsigned char ascsbr[] = { 0x2b, 0x92, 0x08, 0x00 }; const unsigned char asc8ch[] = { 0x12, 0x00, 0x05, 0x08, 0x48, 0x00, 0x20, 0x00, 0xC6, 0x40, 0x0D, 0x4C, 0x61, 0x76, 0x63, 0x35, 0x38, 0x2E, 0x39, 0x37, 0x2E, 0x31, 0x30, 0x32, 0x56, 0xE5, 0x00 }; // https://datatracker.ietf.org/doc/html/rfc6416#page-25 const unsigned char mux1[] = { 0x40, 0x00, 0x8B, 0x18, 0x38, 0x83, 0x80 }; // 6 kbit/s CELP const unsigned char mux2[] = { 0x40, 0x00, 0x26, 0x20, 0x3f, 0xc0 }; // 64 kbit/s AAC LC Stereo const unsigned char mux3[] = { 0x40, 0x00, 0x56, 0x23, 0x10, 0x1f, 0xe0 }; // Hierarchical Signaling of SBR const unsigned char mux4[] = { 0x40, 0x00, 0x26, 0x10, 0x3f, 0xc0 }; // HE AAC v2 Signaling const unsigned char mux5[] = { 0x40, 0x01, 0xd6, 0x13, 0x10, 0x1f, 0xe0 }; // Hierarchical Signaling of PS const unsigned char mux6[] = { 0x8F, 0xF8, 0x00, 0x41, 0x92, 0xB1, 0x18, 0x80, 0xFF, 0x0D, 0xDE, 0x36, 0x99, 0xF2, 0x40, 0x8C, 0x00, 0x53, 0x6C, 0x02, 0x31, 0x3C, 0xF3, 0xCE, 0x0F, 0xF0 }; // MPEG Surround const unsigned char mux7[] = { 0x40, 0x00, 0x56, 0x23, 0x10, 0x1f, 0xe0 }; // MPEG Surround with Extended SDP Parameters const unsigned char mux8[] = { 0x8F, 0xF8, 0x00, 0x06, 0x52, 0xB9, 0x20, 0x87, 0x6A, 0x83, 0xA1, 0xF4, 0x40, 0x88, 0x40, 0x53, 0x62, 0x0F, 0xF0 }; // MPEG Surround with Single-Layer Configuration unsigned char data[32]; assert(sizeof(ascsbr) == mpeg4_aac_audio_specific_config_load(ascsbr, sizeof(ascsbr), &aac)); assert(2 == aac.profile && 7 == aac.sampling_frequency_index && 2 == aac.channel_configuration); //assert(sizeof(ascsbr) == mpeg4_aac_audio_specific_config_save(&aac, data, sizeof(data))); //assert(0 == memcmp(ascsbr, data, sizeof(ascsbr))); assert(sizeof(asc) == mpeg4_aac_audio_specific_config_load(asc, sizeof(asc), &aac)); assert(2 == aac.profile && 7 == aac.sampling_frequency_index && 1 == aac.channel_configuration); assert(sizeof(asc) == mpeg4_aac_audio_specific_config_save(&aac, data, sizeof(data))); assert(0 == memcmp(asc, data, sizeof(asc))); assert(sizeof(adts) == mpeg4_aac_adts_save(&aac, 1, data, sizeof(data))); assert(0 == memcmp(adts, data, sizeof(adts))); assert(7 == mpeg4_aac_adts_load(data, sizeof(adts), &aac2)); assert(0 == memcmp(&aac, &aac2, sizeof(aac))); assert(22050 == mpeg4_aac_audio_frequency_to(aac.sampling_frequency_index)); assert(aac.sampling_frequency_index == mpeg4_aac_audio_frequency_from(22050)); //assert(sizeof(ascsbr) == mpeg4_aac_audio_specific_config_load(ascsbr, sizeof(ascsbr), &aac)); //assert(2 == aac.profile && 6 == aac.sampling_frequency_index && 1 == aac.channel_configuration); assert(sizeof(asc8ch) == mpeg4_aac_audio_specific_config_load(asc8ch, sizeof(asc8ch), &aac)); assert(2 == aac.profile && 4 == aac.sampling_frequency_index && 8 == aac.channels); assert(29 == mpeg4_aac_adts_save(&aac, 1, data, sizeof(data))); memset(&aac, 0, sizeof(aac)); mpeg4_aac_stream_mux_config_load(mux1, sizeof(mux1), &aac); mpeg4_aac_stream_mux_config_load(mux2, sizeof(mux2), &aac); mpeg4_aac_stream_mux_config_load(mux3, sizeof(mux3), &aac); mpeg4_aac_stream_mux_config_load(mux4, sizeof(mux4), &aac); mpeg4_aac_stream_mux_config_load(mux5, sizeof(mux5), &aac); //mpeg4_aac_stream_mux_config_load(mux6, sizeof(mux6), &aac); //mpeg4_aac_stream_mux_config_load(mux7, sizeof(mux7), &aac); //mpeg4_aac_stream_mux_config_load(mux8, sizeof(mux8), &aac); mpeg4_aac_stream_mux_config_save(&aac, data, sizeof(data)); //assert(0 == memcmp(data, mux1, sizeof(mux1))); } #endif