[mod_opus] Fix data types, coding guidelines

This commit is contained in:
Jakub Karolczyk 2023-07-08 06:19:51 -05:00
parent 545538fb7a
commit 4f5e8fd60a
3 changed files with 59 additions and 18 deletions

View File

@ -1173,21 +1173,26 @@ static switch_status_t switch_opus_keep_fec_enabled(switch_codec_t *codec)
static switch_bool_t switch_opus_vad(struct opus_context *context, void *encoded_data, uint32_t encoded_data_len) { static switch_bool_t switch_opus_vad(struct opus_context *context, void *encoded_data, uint32_t encoded_data_len) {
const uint8_t *payload = (const uint8_t *) encoded_data; const uint8_t *payload = (const uint8_t *) encoded_data;
opus_packet_info_t opus_packet_info; opus_packet_info_t opus_packet_info;
switch_bool_t debug = (globals.debug || context->debug > 1); switch_bool_t debug = (globals.debug || context->debug > 1);
if (!switch_opus_packet_parse(payload, encoded_data_len, &opus_packet_info, debug)) { if (!switch_opus_packet_parse(payload, encoded_data_len, &opus_packet_info, debug)) {
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "OPUS PACKET PARSING ERROR len:%d bytes:%02x %02x\n", switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "OPUS PACKET PARSING ERROR len:%d bytes:%02x %02x\n",
(int)encoded_data_len, payload[0], payload[1]); (int)encoded_data_len, payload[0], payload[1]);
} }
return SWITCH_TRUE; return SWITCH_TRUE;
} }
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "OPUS EXTRACT PAYLOAD VAD len:%d vad_ms:%d bytes:%02x %02x\n", switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_INFO, "OPUS EXTRACT PAYLOAD VAD len:%d vad_ms:%d bytes:%02x %02x\n",
(int)encoded_data_len, opus_packet_info.vad_ms, payload[0], payload[1]); (int)encoded_data_len, opus_packet_info.vad_ms, payload[0], payload[1]);
} }
if (opus_packet_info.vad_ms == 0) { if (opus_packet_info.vad_ms == 0) {
return SWITCH_FALSE; return SWITCH_FALSE;
} }
return SWITCH_TRUE; return SWITCH_TRUE;
} }

View File

@ -44,6 +44,7 @@ static const opus_int16 silk_LBRR_flags_3_PDFCum[7] = {41, 61, 90, 131, 146, 174
static opus_int16 switch_opus_get_nb_flags_in_silk_frame(int16_t config) static opus_int16 switch_opus_get_nb_flags_in_silk_frame(int16_t config)
{ {
opus_int16 silk_frame_nb_flags; opus_int16 silk_frame_nb_flags;
if (config > 15) { if (config > 15) {
/* CELT_only frame no VAD flag nor LBRR flag */ /* CELT_only frame no VAD flag nor LBRR flag */
silk_frame_nb_flags = 0; silk_frame_nb_flags = 0;
@ -53,11 +54,13 @@ static opus_int16 switch_opus_get_nb_flags_in_silk_frame(int16_t config)
/* silk-only NB, MB or WB */ /* silk-only NB, MB or WB */
/* The least two significant bits give the number of VAD flags inside the silk frame 1, 2 or 3 */ /* The least two significant bits give the number of VAD flags inside the silk frame 1, 2 or 3 */
silk_frame_nb_flags = config & 0x3; silk_frame_nb_flags = config & 0x3;
if (silk_frame_nb_flags == 0) { /* 0 => 10ms frame : one VAD flag */ if (silk_frame_nb_flags == 0) { /* 0 => 10ms frame : one VAD flag */
silk_frame_nb_flags++; silk_frame_nb_flags++;
} }
} }
} }
return silk_frame_nb_flags; return silk_frame_nb_flags;
} }
@ -73,14 +76,17 @@ static opus_int16 switch_opus_get_silk_frame_ms_per_flag(int16_t config, opus_in
if (config > 15) { if (config > 15) {
/* CELT_only frame no VAD flag nor LBRR flag */ /* CELT_only frame no VAD flag nor LBRR flag */
/* switch_opus_get_silk_frame_ms_per_flag: code not written for CELT-only mode */ /* switch_opus_get_silk_frame_ms_per_flag: code not written for CELT-only mode */
return FALSE;
return 0;
} }
silk_size_frame_ms_per_flag = 20; /* default*/ silk_size_frame_ms_per_flag = 20; /* default*/
if (silk_frame_nb_flags == 1) { /* could be 10 or 20 ms */ if (silk_frame_nb_flags == 1) { /* could be 10 or 20 ms */
if ((config &0x01) == 0) { if ((config &0x01) == 0) {
silk_size_frame_ms_per_flag = 10; silk_size_frame_ms_per_flag = 10;
} }
} }
return silk_size_frame_ms_per_flag; return silk_size_frame_ms_per_flag;
} }
@ -89,7 +95,7 @@ static opus_int16 switch_opus_get_silk_frame_ms_per_flag(int16_t config, opus_in
/* for stereo : the mid frame VAD_flags and the LBRR_flag could be obtained */ /* for stereo : the mid frame VAD_flags and the LBRR_flag could be obtained */
/* yet, to get the LBRR_flags of the mid frame the routine should be modified */ /* yet, to get the LBRR_flags of the mid frame the routine should be modified */
/* to skip the side VAD flags and the side LBRR flag and to get the mid LBRR_symbol */ /* to skip the side VAD flags and the side LBRR flag and to get the mid LBRR_symbol */
static bool_t switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk_frame_nb_flags, static void switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk_frame_nb_flags,
opus_int16 *VAD_flags, opus_int16 *LBRR_flags, opus_int16 *nb_VAD1, opus_int16 *nb_FEC) opus_int16 *VAD_flags, opus_int16 *LBRR_flags, opus_int16 *nb_VAD1, opus_int16 *nb_FEC)
{ {
const opus_int16 *ptr_pdf_cum; const opus_int16 *ptr_pdf_cum;
@ -129,6 +135,7 @@ static bool_t switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk
val >>= 1; val >>= 1;
*(--ptr_flags) = val & 0x1; *(--ptr_flags) = val & 0x1;
} }
if (LBRR_flag != 0) { /* there is at least one LBRR frame */ if (LBRR_flag != 0) { /* there is at least one LBRR frame */
if (silk_frame_nb_flags == 1) { if (silk_frame_nb_flags == 1) {
LBRR_flags[0] = 1; LBRR_flags[0] = 1;
@ -148,6 +155,7 @@ static bool_t switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk
nb_pdf_symbol = 7; nb_pdf_symbol = 7;
ptr_pdf_cum = silk_LBRR_flags_3_PDFCum; ptr_pdf_cum = silk_LBRR_flags_3_PDFCum;
} }
LBRR_symbol = 0; LBRR_symbol = 0;
for (i = 1; i <= nb_pdf_symbol; i++) { for (i = 1; i <= nb_pdf_symbol; i++) {
if (val < *ptr_pdf_cum++) { if (val < *ptr_pdf_cum++) {
@ -155,6 +163,7 @@ static bool_t switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk
break; break;
} }
} }
for (i = 0; i < silk_frame_nb_flags; i++) { for (i = 0; i < silk_frame_nb_flags; i++) {
LBRR_flags[i] = LBRR_symbol & 0x01; LBRR_flags[i] = LBRR_symbol & 0x01;
LBRR_symbol >>= 1; LBRR_symbol >>= 1;
@ -162,20 +171,22 @@ static bool_t switch_opus_get_VAD_LBRR_flags(const uint8_t *buf, opus_int16 silk
} }
} }
} }
for (i = 0; i < silk_frame_nb_flags; i++) { for (i = 0; i < silk_frame_nb_flags; i++) {
nb_vad += VAD_flags[i]; nb_vad += VAD_flags[i];
} }
*nb_VAD1 = nb_vad; *nb_VAD1 = nb_vad;
*nb_FEC = nb_fec; *nb_FEC = nb_fec;
return TRUE;
return;
} }
/* Parse the packet to retrieve informations about its content /* Parse the packet to retrieve informations about its content
* RFC6716: Definition of the Opus Audio Codec * RFC6716: Definition of the Opus Audio Codec
* return: FALSE if there was a problem found parsing the packet, the info returned should be ignored. * return: SWITCH_FALSE if there was a problem found parsing the packet, the info returned should be ignored.
* */ * */
bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes, opus_packet_info_t *packet_info, bool_t debug) switch_bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes, opus_packet_info_t *packet_info, switch_bool_t debug)
{ {
int f; int f;
int32_t samplerate; int32_t samplerate;
@ -188,6 +199,7 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
opus_int16 silk_frame_nb_flags, silk_size_frame_ms_per_flag; opus_int16 silk_frame_nb_flags, silk_size_frame_ms_per_flag;
opus_int16 silk_frame_nb_fec, silk_frame_nb_vad1; opus_int16 silk_frame_nb_fec, silk_frame_nb_vad1;
opus_int sample_per_frame; opus_int sample_per_frame;
packet_info->config = 0; packet_info->config = 0;
packet_info->fec = 0; packet_info->fec = 0;
packet_info->fec_ms = 0; packet_info->fec_ms = 0;
@ -198,19 +210,23 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
packet_info->channels = 1; /* as stereo is set to FALSE */ packet_info->channels = 1; /* as stereo is set to FALSE */
packet_info->ms_per_frame = 0; packet_info->ms_per_frame = 0;
packet_info->ptime_ts = 0; packet_info->ptime_ts = 0;
if (payload == NULL || payload_length_bytes <= 0) { if (payload == NULL || payload_length_bytes <= 0) {
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: payload null."); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: payload null.");
} }
return FALSE;
return SWITCH_FALSE;
} }
/* In CELT_ONLY mode, packets should not have FEC. */ /* In CELT_ONLY mode, packets should not have FEC. */
if (payload[0] & 0x80) { if (payload[0] & 0x80) {
/* opus_packet_parse: CELT_ONLY mode, we do not support this mode. */ /* opus_packet_parse: CELT_ONLY mode, we do not support this mode. */
return FALSE;
return SWITCH_FALSE;
} else { } else {
int mode = (payload[0] >> 3); int mode = (payload[0] >> 3);
if (mode <= 3) { if (mode <= 3) {
samplerate = 8000; samplerate = 8000;
} else if (mode <= 7) { } else if (mode <= 7) {
@ -223,16 +239,19 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
samplerate = 48000; samplerate = 48000;
} else { } else {
/* opus_packet_parse: CELT_ONLY mode, we do not support this mode. */ /* opus_packet_parse: CELT_ONLY mode, we do not support this mode. */
return FALSE;
return SWITCH_FALSE;
} }
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: mode[%d]s[%d]c[%d] [%d]Hz\n", mode, (payload[0]>>2)&0x1 ,(payload[0])&0x3, samplerate); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: mode[%d]s[%d]c[%d] [%d]Hz\n", mode, (payload[0]>>2)&0x1 ,(payload[0])&0x3, samplerate);
} }
} }
if (payload[0] & 0x04) { if (payload[0] & 0x04) {
packet_info->stereo = TRUE; packet_info->stereo = TRUE;
packet_info->channels = 2; packet_info->channels = 2;
} }
packet_info->config = payload[0] >> 3; packet_info->config = payload[0] >> 3;
sample_per_frame = opus_packet_get_samples_per_frame(payload, samplerate); sample_per_frame = opus_packet_get_samples_per_frame(payload, samplerate);
packet_info->ms_per_frame = sample_per_frame * 1000 / samplerate; packet_info->ms_per_frame = sample_per_frame * 1000 / samplerate;
@ -240,7 +259,8 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: invalid packet."); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: invalid packet.");
} }
return FALSE;
return SWITCH_FALSE;
} }
packet_info->frames = opus_packet_parse(payload, payload_length_bytes, NULL, frame_data, frame_sizes, NULL); packet_info->frames = opus_packet_parse(payload, payload_length_bytes, NULL, frame_data, frame_sizes, NULL);
@ -249,15 +269,18 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: opus_packet_parse found no frame.\n"); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: opus_packet_parse found no frame.\n");
} }
return FALSE;
return SWITCH_FALSE;
} }
packet_info->ptime_ts = packet_info->frames * sample_per_frame; packet_info->ptime_ts = packet_info->frames * sample_per_frame;
if (frame_sizes[0] <= 1) { if (frame_sizes[0] <= 1) {
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: opus_packet_parse frame size too small.\n"); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: opus_packet_parse frame size too small.\n");
} }
return FALSE;
return SWITCH_FALSE;
} }
/* +---------------+-----------+-----------+-------------------+ */ /* +---------------+-----------+-----------+-------------------+ */
@ -281,14 +304,16 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
silk_frame_nb_flags = switch_opus_get_nb_flags_in_silk_frame(packet_info->config); /* =1 for 10 or 20 ms frame; = 2 for 40 ms; = 3 for 60 ms */ silk_frame_nb_flags = switch_opus_get_nb_flags_in_silk_frame(packet_info->config); /* =1 for 10 or 20 ms frame; = 2 for 40 ms; = 3 for 60 ms */
if (!silk_frame_nb_flags) { if (!silk_frame_nb_flags) {
/* We should not go there as CELT_ONLY is already tested above */ /* We should not go there as CELT_ONLY is already tested above */
return FALSE;
return SWITCH_FALSE;
} }
packet_info->frames_silk = silk_frame_nb_flags; packet_info->frames_silk = silk_frame_nb_flags;
silk_size_frame_ms_per_flag = switch_opus_get_silk_frame_ms_per_flag(packet_info->config, silk_frame_nb_flags); /* 10 or 20 ms frame*/ silk_size_frame_ms_per_flag = switch_opus_get_silk_frame_ms_per_flag(packet_info->config, silk_frame_nb_flags); /* 10 or 20 ms frame*/
if (!silk_size_frame_ms_per_flag) { if (!silk_size_frame_ms_per_flag) {
/* we should not go there as CELT_ONLY is already tested above */ /* we should not go there as CELT_ONLY is already tested above */
return FALSE;
return SWITCH_FALSE;
} }
ptr_LBBR_FLAGS = packet_LBBR_FLAGS; ptr_LBBR_FLAGS = packet_LBBR_FLAGS;
@ -305,6 +330,7 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
ptr_VAD_FLAGS += silk_frame_nb_flags; ptr_VAD_FLAGS += silk_frame_nb_flags;
ptr_LBBR_FLAGS += silk_frame_nb_flags; ptr_LBBR_FLAGS += silk_frame_nb_flags;
} }
/* store the VAD & LBRR flags of all 20 ms silk-frames of the packet; LSB the first frame, MSB: the last */ /* store the VAD & LBRR flags of all 20 ms silk-frames of the packet; LSB the first frame, MSB: the last */
vad_flags_per_silk_frame = 0; vad_flags_per_silk_frame = 0;
fec_flags_per_silk_frame = 0; fec_flags_per_silk_frame = 0;
@ -313,8 +339,10 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: more than %d 20-ms frames in the packet ; only first 15 silk-frames data will be stored (pb silkFastAccelerate)\n", silk_frame_packet); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: more than %d 20-ms frames in the packet ; only first 15 silk-frames data will be stored (pb silkFastAccelerate)\n", silk_frame_packet);
} }
silk_frame_packet = 15; silk_frame_packet = 15;
} }
ptr_LBBR_FLAGS = packet_LBBR_FLAGS; ptr_LBBR_FLAGS = packet_LBBR_FLAGS;
ptr_VAD_FLAGS = packet_VAD_FLAGS; ptr_VAD_FLAGS = packet_VAD_FLAGS;
shift_silk = 0; shift_silk = 0;
@ -324,17 +352,21 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
shift_silk++; shift_silk++;
ptr_LBBR_FLAGS++; ptr_VAD_FLAGS++; ptr_LBBR_FLAGS++; ptr_VAD_FLAGS++;
} }
packet_info->vad_flags_per_silk_frame = vad_flags_per_silk_frame; packet_info->vad_flags_per_silk_frame = vad_flags_per_silk_frame;
packet_info->fec_flags_per_silk_frame = fec_flags_per_silk_frame; packet_info->fec_flags_per_silk_frame = fec_flags_per_silk_frame;
return TRUE;
return SWITCH_TRUE;
} }
if (packet_info->config != 1 && packet_info->config != 5 && packet_info->config != 9) { if (packet_info->config != 1 && packet_info->config != 5 && packet_info->config != 9) {
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: the current parser implementation does not support muliple SILK frames for VAD or FEC detection.\n"); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: the current parser implementation does not support muliple SILK frames for VAD or FEC detection.\n");
} }
return FALSE;
return SWITCH_FALSE;
} }
/* /*
* Parse the VAD and LBRR flags in each Opus frame * Parse the VAD and LBRR flags in each Opus frame
* */ * */
@ -342,16 +374,20 @@ bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes
if (frame_data[f][0] & 0x80) { if (frame_data[f][0] & 0x80) {
packet_info->vad++; packet_info->vad++;
} }
if (frame_data[f][0] & 0x40) { if (frame_data[f][0] & 0x40) {
packet_info->fec++; packet_info->fec++;
} }
if (debug) { if (debug) {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: LP layer opus_frame[%d] VAD[%d] FEC[%d]\n", f+1, (frame_data[f][0]&0x80)>>7, (frame_data[f][0]&0x40)>>6); switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "opus_packet_parse: LP layer opus_frame[%d] VAD[%d] FEC[%d]\n", f+1, (frame_data[f][0]&0x80)>>7, (frame_data[f][0]&0x40)>>6);
} }
} }
packet_info->vad_ms = packet_info->vad * packet_info->ms_per_frame; packet_info->vad_ms = packet_info->vad * packet_info->ms_per_frame;
packet_info->fec_ms = packet_info->fec * packet_info->ms_per_frame; packet_info->fec_ms = packet_info->fec * packet_info->ms_per_frame;
return TRUE;
return SWITCH_TRUE;
} }
/* For Emacs: /* For Emacs:

View File

@ -51,7 +51,7 @@ typedef struct opus_packet_info {
int16_t fec_flags_per_silk_frame; int16_t fec_flags_per_silk_frame;
} opus_packet_info_t; } opus_packet_info_t;
bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes, opus_packet_info_t *packet_info, bool_t debug); switch_bool_t switch_opus_packet_parse(const uint8_t *payload, int payload_length_bytes, opus_packet_info_t *packet_info, switch_bool_t debug);
#endif #endif
/* For Emacs: /* For Emacs: