| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- #include "audio-repack.h"
- #include <emmintrin.h>
- int check_buffer(struct audio_repack *repack,
- uint32_t frame_count)
- {
- const uint32_t new_size = frame_count * repack->base_dst_size
- + repack->extra_dst_size;
- if (repack->packet_size < new_size) {
- repack->packet_buffer = brealloc(
- repack->packet_buffer, new_size);
- if (!repack->packet_buffer)
- return -1;
- repack->packet_size = new_size;
- }
- return 0;
- }
- /*
- * Swap channel 3 & 4, 5 & 7, 6 & 8 and squash arrays
- * 2.1:
- *
- * | FL | FR | LFE | emp | emp | emp |emp |emp |
- * | | |
- * | FL | FR | LFE |
- * 4.0 (quad):
- *
- * | FL | FR | BR | BL | emp | emp |emp |emp |
- * | | x |
- * | FL | FR | BL | BC |
- *
- * 4.1:
- *
- * | FL | FR |LFE | FC | BC | emp |emp |emp |
- * | | x | |
- * | FL | FR | FC |LFE | BC |
- *
- * 5.1:
- *
- * | FL | FR |LFE | FC |(emp|emp)|(BL|BR)|
- * | | x x
- * | FL | FR | FC |LFE | BL | BR |
- *
- * 7.1:
- *
- * | FL | FR |LFE | FC |( SL | SR )|(BL |BR )|
- * | | x X
- * | FL | FR | FC |LFE |( BL | BR )|(SL |SR )|
- */
- int repack_squash_swap(struct audio_repack *repack,
- const uint8_t *bsrc, uint32_t frame_count)
- {
- 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;
- /* 2.1 audio does not require re-ordering but still needs squashing
- * in order to avoid sampling issues.
- */
- if (squash == 5) {
- while (src != esrc) {
- __m128i target = _mm_load_si128(src++);
- _mm_storeu_si128((__m128i *)dst, target);
- dst += 8 - squash;
- }
- } else {
- while (src != esrc) {
- __m128i target = _mm_load_si128(src++);
- __m128i buf = _mm_shufflelo_epi16(target, _MM_SHUFFLE(2, 3, 1, 0));
- __m128i buf2 = _mm_shufflehi_epi16(buf, _MM_SHUFFLE(1, 0, 3, 2));
- _mm_storeu_si128((__m128i *)dst, buf2);
- dst += 8 - squash;
- }
- }
- return 0;
- }
- int audio_repack_init(struct audio_repack *repack,
- audio_repack_mode_t repack_mode, uint8_t sample_bit)
- {
- memset(repack, 0, sizeof(*repack));
- if (sample_bit != 16)
- return -1;
- switch (repack_mode) {
- case repack_mode_8to3ch_swap23:
- repack->base_src_size = 8 * (16 / 8);
- repack->base_dst_size = 3 * (16 / 8);
- repack->extra_dst_size = 5;
- repack->repack_func = &repack_squash_swap;
- break;
- case repack_mode_8to4ch_swap23:
- repack->base_src_size = 8 * (16 / 8);
- repack->base_dst_size = 4 * (16 / 8);
- repack->extra_dst_size = 4;
- repack->repack_func = &repack_squash_swap;
- break;
- case repack_mode_8to5ch_swap23:
- repack->base_src_size = 8 * (16 / 8);
- repack->base_dst_size = 5 * (16 / 8);
- repack->extra_dst_size = 3;
- repack->repack_func = &repack_squash_swap;
- break;
- case repack_mode_8to6ch_swap23:
- repack->base_src_size = 8 * (16 / 8);
- repack->base_dst_size = 6 * (16 / 8);
- repack->extra_dst_size = 2;
- repack->repack_func = &repack_squash_swap;
- break;
- case repack_mode_8ch_swap23_swap46_swap57:
- repack->base_src_size = 8 * (16 / 8);
- repack->base_dst_size = 8 * (16 / 8);
- repack->extra_dst_size = 0;
- repack->repack_func = &repack_squash_swap;
- break;
- default: return -1;
- }
- return 0;
- }
- void audio_repack_free(struct audio_repack *repack)
- {
- if (repack->packet_buffer)
- bfree(repack->packet_buffer);
- memset(repack, 0, sizeof(*repack));
- }
|