audio-repack.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. #include "audio-repack.h"
  2. #include <emmintrin.h>
  3. int check_buffer(struct audio_repack *repack,
  4. uint32_t frame_count)
  5. {
  6. const uint32_t new_size = frame_count * repack->base_dst_size
  7. + repack->extra_dst_size;
  8. if (repack->packet_size < new_size) {
  9. repack->packet_buffer = brealloc(
  10. repack->packet_buffer, new_size);
  11. if (!repack->packet_buffer)
  12. return -1;
  13. repack->packet_size = new_size;
  14. }
  15. return 0;
  16. }
  17. /*
  18. Swap channel between LFE and FC, and
  19. squash data array
  20. | FL | FR |LFE | FC | BL | BR |emp |emp |
  21. | | x | |
  22. | FL | FR | FC |LFE | BL | BR |
  23. */
  24. int repack_8to6ch_swap23(struct audio_repack *repack,
  25. const uint8_t *bsrc, uint32_t frame_count)
  26. {
  27. if (check_buffer(repack, frame_count) < 0)
  28. return -1;
  29. const uint32_t size = frame_count * repack->base_src_size;
  30. const __m128i *src = (__m128i *)bsrc;
  31. const __m128i *esrc = src + frame_count;
  32. uint32_t *dst = (uint32_t *)repack->packet_buffer;
  33. while (src != esrc) {
  34. __m128i target = _mm_load_si128(src++);
  35. __m128i buf = _mm_shufflelo_epi16(target, _MM_SHUFFLE(2, 3, 1, 0));
  36. _mm_storeu_si128((__m128i *)dst, buf);
  37. dst += 3;
  38. }
  39. return 0;
  40. }
  41. /*
  42. Swap channel between LFE and FC
  43. | FL | FR |LFE | FC | BL | BR |SBL |SBR |
  44. | | x | | | |
  45. | FL | FR | FC |LFE | BL | BR |SBL |SBR |
  46. */
  47. int repack_8ch_swap23(struct audio_repack *repack,
  48. const uint8_t *bsrc, uint32_t frame_count)
  49. {
  50. if (check_buffer(repack, frame_count) < 0)
  51. return -1;
  52. const uint32_t size = frame_count * repack->base_src_size;
  53. const __m128i *src = (__m128i *)bsrc;
  54. const __m128i *esrc = src + frame_count;
  55. __m128i *dst = (__m128i *)repack->packet_buffer;
  56. while (src != esrc) {
  57. __m128i target = _mm_load_si128(src++);
  58. __m128i buf = _mm_shufflelo_epi16(target, _MM_SHUFFLE(2, 3, 1, 0));
  59. _mm_store_si128(dst++, buf);
  60. }
  61. return 0;
  62. }
  63. int audio_repack_init(struct audio_repack *repack,
  64. audio_repack_mode_t repack_mode, uint8_t sample_bit)
  65. {
  66. memset(repack, 0, sizeof(*repack));
  67. if (sample_bit != 16)
  68. return -1;
  69. switch (repack_mode) {
  70. case repack_mode_8to6ch_swap23:
  71. repack->base_src_size = 8 * (16 / 8);
  72. repack->base_dst_size = 6 * (16 / 8);
  73. repack->extra_dst_size = 2;
  74. repack->repack_func = &repack_8to6ch_swap23;
  75. break;
  76. case repack_mode_8ch_swap23:
  77. repack->base_src_size = 8 * (16 / 8);
  78. repack->base_dst_size = 8 * (16 / 8);
  79. repack->extra_dst_size = 0;
  80. repack->repack_func = &repack_8ch_swap23;
  81. break;
  82. default: return -1;
  83. }
  84. return 0;
  85. }
  86. void audio_repack_free(struct audio_repack *repack)
  87. {
  88. if (repack->packet_buffer)
  89. bfree(repack->packet_buffer);
  90. memset(repack, 0, sizeof(*repack));
  91. }