PcmHelper.cs 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. using System;
  2. using System.Numerics;
  3. using System.Runtime.CompilerServices;
  4. namespace Ryujinx.Audio.Renderer.Dsp
  5. {
  6. public static class PcmHelper
  7. {
  8. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  9. public static int GetCountToDecode(int startSampleOffset, int endSampleOffset, int offset, int count)
  10. {
  11. return Math.Min(count, endSampleOffset - startSampleOffset - offset);
  12. }
  13. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  14. public static ulong GetBufferOffset<T>(int startSampleOffset, int offset, int channelCount) where T : unmanaged
  15. {
  16. return (ulong)(Unsafe.SizeOf<T>() * channelCount * (startSampleOffset + offset));
  17. }
  18. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  19. public static int GetBufferSize<T>(int startSampleOffset, int endSampleOffset, int offset, int count) where T : unmanaged
  20. {
  21. return GetCountToDecode(startSampleOffset, endSampleOffset, offset, count) * Unsafe.SizeOf<T>();
  22. }
  23. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  24. public static float ConvertSampleToPcmFloat(short sample)
  25. {
  26. return (float)sample / short.MaxValue;
  27. }
  28. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  29. public static short ConvertSampleToPcmInt16(float sample)
  30. {
  31. return Saturate(sample * short.MaxValue);
  32. }
  33. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  34. public static void ConvertSampleToPcm8(Span<sbyte> output, ReadOnlySpan<short> input)
  35. {
  36. for (int i = 0; i < input.Length; i++)
  37. {
  38. // Output most significant byte
  39. output[i] = (sbyte)(input[i] >> 8);
  40. }
  41. }
  42. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  43. public static void ConvertSampleToPcm24(Span<byte> output, ReadOnlySpan<short> input)
  44. {
  45. for (int i = 0; i < input.Length; i++)
  46. {
  47. output[i * 3 + 2] = (byte)(input[i] >> 8);
  48. output[i * 3 + 1] = (byte)(input[i] & 0xff);
  49. output[i * 3 + 0] = 0;
  50. }
  51. }
  52. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  53. public static void ConvertSampleToPcm32(Span<int> output, ReadOnlySpan<short> input)
  54. {
  55. for (int i = 0; i < input.Length; i++)
  56. {
  57. output[i] = ((int)input[i]) << 16;
  58. }
  59. }
  60. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  61. public static void ConvertSampleToPcmFloat(Span<float> output, ReadOnlySpan<short> input)
  62. {
  63. for (int i = 0; i < input.Length; i++)
  64. {
  65. output[i] = ConvertSampleToPcmFloat(input[i]);
  66. }
  67. }
  68. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  69. public static int Decode(Span<short> output, ReadOnlySpan<short> input, int startSampleOffset, int endSampleOffset, int channelIndex, int channelCount)
  70. {
  71. if (input.IsEmpty || endSampleOffset < startSampleOffset)
  72. {
  73. return 0;
  74. }
  75. int decodedCount = input.Length / channelCount;
  76. for (int i = 0; i < decodedCount; i++)
  77. {
  78. output[i] = input[i * channelCount + channelIndex];
  79. }
  80. return decodedCount;
  81. }
  82. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  83. public static int Decode(Span<short> output, ReadOnlySpan<float> input, int startSampleOffset, int endSampleOffset, int channelIndex, int channelCount)
  84. {
  85. if (input.IsEmpty || endSampleOffset < startSampleOffset)
  86. {
  87. return 0;
  88. }
  89. int decodedCount = input.Length / channelCount;
  90. for (int i = 0; i < decodedCount; i++)
  91. {
  92. output[i] = ConvertSampleToPcmInt16(input[i * channelCount + channelIndex]);
  93. }
  94. return decodedCount;
  95. }
  96. [MethodImpl(MethodImplOptions.AggressiveInlining)]
  97. public static short Saturate(float value)
  98. {
  99. if (value > short.MaxValue)
  100. {
  101. return short.MaxValue;
  102. }
  103. if (value < short.MinValue)
  104. {
  105. return short.MinValue;
  106. }
  107. return (short)value;
  108. }
  109. }
  110. }