audio-repack.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  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 3 & 4, 5 & 7, 6 & 8 and squash arrays
  19. * 4.0 (quad):
  20. *
  21. * | FL | FR | BR | BL | emp | emp |emp |emp |
  22. * | | x |
  23. * | FL | FR | BL | BC |
  24. *
  25. * 4.1:
  26. *
  27. * | FL | FR |LFE | FC | BC | emp |emp |emp |
  28. * | | x | |
  29. * | FL | FR | FC |LFE | BC |
  30. *
  31. * 5.1:
  32. *
  33. * | FL | FR |LFE | FC |(emp|emp)|(BL|BR)|
  34. * | | x x
  35. * | FL | FR | FC |LFE | BL | BR |
  36. *
  37. * 7.1:
  38. *
  39. * | FL | FR |LFE | FC |( SL | SR )|(BL |BR )|
  40. * | | x X
  41. * | FL | FR | FC |LFE |( BL | BR )|(SL |SR )|
  42. */
  43. int repack_squash_swap(struct audio_repack *repack,
  44. const uint8_t *bsrc, uint32_t frame_count)
  45. {
  46. if (check_buffer(repack, frame_count) < 0)
  47. return -1;
  48. int squash = repack->extra_dst_size;
  49. const __m128i *src = (__m128i *)bsrc;
  50. const __m128i *esrc = src + frame_count;
  51. uint16_t *dst = (uint16_t *)repack->packet_buffer;
  52. while (src != esrc) {
  53. __m128i target = _mm_load_si128(src++);
  54. __m128i buf = _mm_shufflelo_epi16(target,_MM_SHUFFLE(2, 3, 1, 0));
  55. __m128i buf2 = _mm_shufflehi_epi16(buf, _MM_SHUFFLE(1, 0, 3, 2));
  56. _mm_storeu_si128((__m128i *)dst, buf2);
  57. dst += 8 - squash;
  58. }
  59. return 0;
  60. }
  61. int audio_repack_init(struct audio_repack *repack,
  62. audio_repack_mode_t repack_mode, uint8_t sample_bit)
  63. {
  64. memset(repack, 0, sizeof(*repack));
  65. if (sample_bit != 16)
  66. return -1;
  67. switch (repack_mode) {
  68. case repack_mode_8to4ch_swap23:
  69. repack->base_src_size = 8 * (16 / 8);
  70. repack->base_dst_size = 4 * (16 / 8);
  71. repack->extra_dst_size = 4;
  72. repack->repack_func = &repack_squash_swap;
  73. break;
  74. case repack_mode_8to5ch_swap23:
  75. repack->base_src_size = 8 * (16 / 8);
  76. repack->base_dst_size = 5 * (16 / 8);
  77. repack->extra_dst_size = 3;
  78. repack->repack_func = &repack_squash_swap;
  79. break;
  80. case repack_mode_8to6ch_swap23:
  81. repack->base_src_size = 8 * (16 / 8);
  82. repack->base_dst_size = 6 * (16 / 8);
  83. repack->extra_dst_size = 2;
  84. repack->repack_func = &repack_squash_swap;
  85. break;
  86. case repack_mode_8ch_swap23_swap46_swap57:
  87. repack->base_src_size = 8 * (16 / 8);
  88. repack->base_dst_size = 8 * (16 / 8);
  89. repack->extra_dst_size = 0;
  90. repack->repack_func = &repack_squash_swap;
  91. break;
  92. default: return -1;
  93. }
  94. return 0;
  95. }
  96. void audio_repack_free(struct audio_repack *repack)
  97. {
  98. if (repack->packet_buffer)
  99. bfree(repack->packet_buffer);
  100. memset(repack, 0, sizeof(*repack));
  101. }