yolobs-studio/plugins/decklink/audio-repack.c

103 lines
2.5 KiB
C
Raw Normal View History

2017-06-29 19:01:10 +00:00
#include "audio-repack.h"
#include <emmintrin.h>
2019-09-22 21:19:10 +00:00
int check_buffer(struct audio_repack *repack, uint32_t frame_count)
2017-06-29 19:01:10 +00:00
{
2019-09-22 21:19:10 +00:00
const uint32_t new_size =
frame_count * repack->base_dst_size + repack->extra_dst_size;
2017-06-29 19:01:10 +00:00
if (repack->packet_size < new_size) {
2019-09-22 21:19:10 +00:00
repack->packet_buffer =
brealloc(repack->packet_buffer, new_size);
2017-06-29 19:01:10 +00:00
if (!repack->packet_buffer)
return -1;
repack->packet_size = new_size;
}
return 0;
}
/*
2019-07-27 12:47:10 +00:00
* Squash arrays.
* For instance:
2018-02-19 19:54:37 +00:00
* 2.1:
*
* | FL | FR | LFE | emp | emp | emp |emp |emp |
* | | |
* | FL | FR | LFE |
2019-07-27 12:47:10 +00:00
*/
2017-06-29 19:01:10 +00:00
2019-09-22 21:19:10 +00:00
int repack_squash(struct audio_repack *repack, const uint8_t *bsrc,
uint32_t frame_count)
2017-06-29 19:01:10 +00:00
{
if (check_buffer(repack, frame_count) < 0)
return -1;
2018-02-19 19:54:37 +00:00
int squash = repack->extra_dst_size;
2017-06-29 19:01:10 +00:00
const __m128i *src = (__m128i *)bsrc;
const __m128i *esrc = src + frame_count;
2018-02-19 19:54:37 +00:00
uint16_t *dst = (uint16_t *)repack->packet_buffer;
2019-07-27 12:47:10 +00:00
/* Audio needs squashing in order to avoid resampling issues.
* The condition checks for 7.1 audio for which no squash is needed.
2018-02-19 19:54:37 +00:00
*/
2019-07-27 12:47:10 +00:00
if (squash > 0) {
2018-02-19 19:54:37 +00:00
while (src != esrc) {
__m128i target = _mm_load_si128(src++);
_mm_storeu_si128((__m128i *)dst, target);
dst += 8 - squash;
}
2017-06-29 19:01:10 +00:00
}
return 0;
}
2019-09-22 21:19:10 +00:00
int repack_squash_swap(struct audio_repack *repack, const uint8_t *bsrc,
uint32_t frame_count)
2019-07-27 12:47:10 +00:00
{
if (check_buffer(repack, frame_count) < 0)
return -1;
int squash = repack->extra_dst_size;
const __m128i *src = (__m128i *)bsrc;
const __m128i *esrc = src + frame_count;
uint16_t *dst = (uint16_t *)repack->packet_buffer;
while (src != esrc) {
__m128i target = _mm_load_si128(src++);
2019-09-22 21:19:10 +00:00
__m128i buf =
_mm_shufflelo_epi16(target, _MM_SHUFFLE(2, 3, 1, 0));
2019-07-27 12:47:10 +00:00
_mm_storeu_si128((__m128i *)dst, buf);
dst += 8 - squash;
}
return 0;
}
2017-06-29 19:01:10 +00:00
int audio_repack_init(struct audio_repack *repack,
2019-09-22 21:19:10 +00:00
audio_repack_mode_t repack_mode, uint8_t sample_bit)
2017-06-29 19:01:10 +00:00
{
memset(repack, 0, sizeof(*repack));
if (sample_bit != 16)
return -1;
2019-09-22 21:19:10 +00:00
int _audio_repack_ch[8] = {3, 4, 5, 6, 5, 6, 8, 8};
2019-07-27 12:47:10 +00:00
repack->base_src_size = 8 * (16 / 8);
repack->base_dst_size = _audio_repack_ch[repack_mode] * (16 / 8);
repack->extra_dst_size = 8 - _audio_repack_ch[repack_mode];
repack->repack_func = &repack_squash;
if (repack_mode == repack_mode_8to5ch_swap ||
2019-09-22 21:19:10 +00:00
repack_mode == repack_mode_8to6ch_swap ||
repack_mode == repack_mode_8ch_swap)
2018-02-19 19:54:37 +00:00
repack->repack_func = &repack_squash_swap;
2017-06-29 19:01:10 +00:00
return 0;
}
void audio_repack_free(struct audio_repack *repack)
{
if (repack->packet_buffer)
bfree(repack->packet_buffer);
memset(repack, 0, sizeof(*repack));
}