Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

720 lines
41KB

  1. /*
  2. ==============================================================================
  3. This file is part of the Water library.
  4. Copyright (c) 2016 ROLI Ltd.
  5. Copyright (C) 2018 Filipe Coelho <falktx@falktx.com>
  6. Permission is granted to use this software under the terms of the ISC license
  7. http://www.isc.org/downloads/software-support-policy/isc-license/
  8. Permission to use, copy, modify, and/or distribute this software for any
  9. purpose with or without fee is hereby granted, provided that the above
  10. copyright notice and this permission notice appear in all copies.
  11. THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH REGARD
  12. TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
  13. FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT,
  14. OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF
  15. USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  16. TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
  17. OF THIS SOFTWARE.
  18. ==============================================================================
  19. */
  20. #ifndef WATER_AUDIODATACONVERTERS_H_INCLUDED
  21. #define WATER_AUDIODATACONVERTERS_H_INCLUDED
  22. #include "../memory/ByteOrder.h"
  23. namespace water {
  24. //==============================================================================
  25. /**
  26. This class a container which holds all the classes pertaining to the AudioData::Pointer
  27. audio sample format class.
  28. @see AudioData::Pointer.
  29. */
  30. class AudioData
  31. {
  32. public:
  33. //==============================================================================
  34. // These types can be used as the SampleFormat template parameter for the AudioData::Pointer class.
  35. class Int8; /**< Used as a template parameter for AudioData::Pointer. Indicates an 8-bit integer packed data format. */
  36. class UInt8; /**< Used as a template parameter for AudioData::Pointer. Indicates an 8-bit unsigned integer packed data format. */
  37. class Int16; /**< Used as a template parameter for AudioData::Pointer. Indicates an 16-bit integer packed data format. */
  38. class Int24; /**< Used as a template parameter for AudioData::Pointer. Indicates an 24-bit integer packed data format. */
  39. class Int32; /**< Used as a template parameter for AudioData::Pointer. Indicates an 32-bit integer packed data format. */
  40. class Float32; /**< Used as a template parameter for AudioData::Pointer. Indicates an 32-bit float data format. */
  41. //==============================================================================
  42. // These types can be used as the Endianness template parameter for the AudioData::Pointer class.
  43. class BigEndian; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples are stored in big-endian order. */
  44. class LittleEndian; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples are stored in little-endian order. */
  45. class NativeEndian; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples are stored in the CPU's native endianness. */
  46. //==============================================================================
  47. // These types can be used as the InterleavingType template parameter for the AudioData::Pointer class.
  48. class NonInterleaved; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples are stored contiguously. */
  49. class Interleaved; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples are interleaved with a number of other channels. */
  50. //==============================================================================
  51. // These types can be used as the Constness template parameter for the AudioData::Pointer class.
  52. class NonConst; /**< Used as a template parameter for AudioData::Pointer. Indicates that the pointer can be used for non-const data. */
  53. class Const; /**< Used as a template parameter for AudioData::Pointer. Indicates that the samples can only be used for const data.. */
  54. #ifndef DOXYGEN
  55. //==============================================================================
  56. class BigEndian
  57. {
  58. public:
  59. template <class SampleFormatType> static inline float getAsFloat (SampleFormatType& s) noexcept { return s.getAsFloatBE(); }
  60. template <class SampleFormatType> static inline void setAsFloat (SampleFormatType& s, float newValue) noexcept { s.setAsFloatBE (newValue); }
  61. template <class SampleFormatType> static inline int32 getAsInt32 (SampleFormatType& s) noexcept { return s.getAsInt32BE(); }
  62. template <class SampleFormatType> static inline void setAsInt32 (SampleFormatType& s, int32 newValue) noexcept { s.setAsInt32BE (newValue); }
  63. template <class SourceType, class DestType> static inline void copyFrom (DestType& dest, SourceType& source) noexcept { dest.copyFromBE (source); }
  64. enum { isBigEndian = 1 };
  65. };
  66. class LittleEndian
  67. {
  68. public:
  69. template <class SampleFormatType> static inline float getAsFloat (SampleFormatType& s) noexcept { return s.getAsFloatLE(); }
  70. template <class SampleFormatType> static inline void setAsFloat (SampleFormatType& s, float newValue) noexcept { s.setAsFloatLE (newValue); }
  71. template <class SampleFormatType> static inline int32 getAsInt32 (SampleFormatType& s) noexcept { return s.getAsInt32LE(); }
  72. template <class SampleFormatType> static inline void setAsInt32 (SampleFormatType& s, int32 newValue) noexcept { s.setAsInt32LE (newValue); }
  73. template <class SourceType, class DestType> static inline void copyFrom (DestType& dest, SourceType& source) noexcept { dest.copyFromLE (source); }
  74. enum { isBigEndian = 0 };
  75. };
  76. #if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__
  77. class NativeEndian : public BigEndian {};
  78. #else
  79. class NativeEndian : public LittleEndian {};
  80. #endif
  81. //==============================================================================
  82. class Int8
  83. {
  84. public:
  85. inline Int8 (void* d) noexcept : data (static_cast<int8*> (d)) {}
  86. inline void advance() noexcept { ++data; }
  87. inline void skip (int numSamples) noexcept { data += numSamples; }
  88. inline float getAsFloatLE() const noexcept { return (float) (*data * (1.0 / (1.0 + maxValue))); }
  89. inline float getAsFloatBE() const noexcept { return getAsFloatLE(); }
  90. inline void setAsFloatLE (float newValue) noexcept { *data = (int8) jlimit ((int) -maxValue, (int) maxValue, roundToInt (newValue * (1.0 + maxValue))); }
  91. inline void setAsFloatBE (float newValue) noexcept { setAsFloatLE (newValue); }
  92. inline int32 getAsInt32LE() const noexcept { return (int) (*data << 24); }
  93. inline int32 getAsInt32BE() const noexcept { return getAsInt32LE(); }
  94. inline void setAsInt32LE (int newValue) noexcept { *data = (int8) (newValue >> 24); }
  95. inline void setAsInt32BE (int newValue) noexcept { setAsInt32LE (newValue); }
  96. inline void clear() noexcept { *data = 0; }
  97. inline void clearMultiple (int num) noexcept { zeromem (data, (size_t) (num * bytesPerSample)) ;}
  98. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  99. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  100. inline void copyFromSameType (Int8& source) noexcept { *data = *source.data; }
  101. int8* data;
  102. enum { bytesPerSample = 1, maxValue = 0x7f, resolution = (1 << 24), isFloat = 0 };
  103. };
  104. class UInt8
  105. {
  106. public:
  107. inline UInt8 (void* d) noexcept : data (static_cast<uint8*> (d)) {}
  108. inline void advance() noexcept { ++data; }
  109. inline void skip (int numSamples) noexcept { data += numSamples; }
  110. inline float getAsFloatLE() const noexcept { return (float) ((*data - 128) * (1.0 / (1.0 + maxValue))); }
  111. inline float getAsFloatBE() const noexcept { return getAsFloatLE(); }
  112. inline void setAsFloatLE (float newValue) noexcept { *data = (uint8) jlimit (0, 255, 128 + roundToInt (newValue * (1.0 + maxValue))); }
  113. inline void setAsFloatBE (float newValue) noexcept { setAsFloatLE (newValue); }
  114. inline int32 getAsInt32LE() const noexcept { return (int) ((*data - 128) << 24); }
  115. inline int32 getAsInt32BE() const noexcept { return getAsInt32LE(); }
  116. inline void setAsInt32LE (int newValue) noexcept { *data = (uint8) (128 + (newValue >> 24)); }
  117. inline void setAsInt32BE (int newValue) noexcept { setAsInt32LE (newValue); }
  118. inline void clear() noexcept { *data = 128; }
  119. inline void clearMultiple (int num) noexcept { memset (data, 128, (size_t) num) ;}
  120. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  121. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  122. inline void copyFromSameType (UInt8& source) noexcept { *data = *source.data; }
  123. uint8* data;
  124. enum { bytesPerSample = 1, maxValue = 0x7f, resolution = (1 << 24), isFloat = 0 };
  125. };
  126. class Int16
  127. {
  128. public:
  129. inline Int16 (void* d) noexcept : data (static_cast<uint16*> (d)) {}
  130. inline void advance() noexcept { ++data; }
  131. inline void skip (int numSamples) noexcept { data += numSamples; }
  132. inline float getAsFloatLE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int16) ByteOrder::swapIfBigEndian (*data)); }
  133. inline float getAsFloatBE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int16) ByteOrder::swapIfLittleEndian (*data)); }
  134. inline void setAsFloatLE (float newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint16) jlimit ((int) -maxValue, (int) maxValue, roundToInt (newValue * (1.0 + maxValue)))); }
  135. inline void setAsFloatBE (float newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint16) jlimit ((int) -maxValue, (int) maxValue, roundToInt (newValue * (1.0 + maxValue)))); }
  136. inline int32 getAsInt32LE() const noexcept { return (int32) (ByteOrder::swapIfBigEndian ((uint16) *data) << 16); }
  137. inline int32 getAsInt32BE() const noexcept { return (int32) (ByteOrder::swapIfLittleEndian ((uint16) *data) << 16); }
  138. inline void setAsInt32LE (int32 newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint16) (newValue >> 16)); }
  139. inline void setAsInt32BE (int32 newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint16) (newValue >> 16)); }
  140. inline void clear() noexcept { *data = 0; }
  141. inline void clearMultiple (int num) noexcept { zeromem (data, (size_t) (num * bytesPerSample)) ;}
  142. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  143. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  144. inline void copyFromSameType (Int16& source) noexcept { *data = *source.data; }
  145. uint16* data;
  146. enum { bytesPerSample = 2, maxValue = 0x7fff, resolution = (1 << 16), isFloat = 0 };
  147. };
  148. class Int24
  149. {
  150. public:
  151. inline Int24 (void* d) noexcept : data (static_cast<char*> (d)) {}
  152. inline void advance() noexcept { data += 3; }
  153. inline void skip (int numSamples) noexcept { data += 3 * numSamples; }
  154. inline float getAsFloatLE() const noexcept { return (float) (ByteOrder::littleEndian24Bit (data) * (1.0 / (1.0 + maxValue))); }
  155. inline float getAsFloatBE() const noexcept { return (float) (ByteOrder::bigEndian24Bit (data) * (1.0 / (1.0 + maxValue))); }
  156. inline void setAsFloatLE (float newValue) noexcept { ByteOrder::littleEndian24BitToChars (jlimit ((int) -maxValue, (int) maxValue, roundToInt (newValue * (1.0 + maxValue))), data); }
  157. inline void setAsFloatBE (float newValue) noexcept { ByteOrder::bigEndian24BitToChars (jlimit ((int) -maxValue, (int) maxValue, roundToInt (newValue * (1.0 + maxValue))), data); }
  158. inline int32 getAsInt32LE() const noexcept { return (int32) ByteOrder::littleEndian24Bit (data) << 8; }
  159. inline int32 getAsInt32BE() const noexcept { return (int32) ByteOrder::bigEndian24Bit (data) << 8; }
  160. inline void setAsInt32LE (int32 newValue) noexcept { ByteOrder::littleEndian24BitToChars (newValue >> 8, data); }
  161. inline void setAsInt32BE (int32 newValue) noexcept { ByteOrder::bigEndian24BitToChars (newValue >> 8, data); }
  162. inline void clear() noexcept { data[0] = 0; data[1] = 0; data[2] = 0; }
  163. inline void clearMultiple (int num) noexcept { zeromem (data, (size_t) (num * bytesPerSample)) ;}
  164. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  165. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  166. inline void copyFromSameType (Int24& source) noexcept { data[0] = source.data[0]; data[1] = source.data[1]; data[2] = source.data[2]; }
  167. char* data;
  168. enum { bytesPerSample = 3, maxValue = 0x7fffff, resolution = (1 << 8), isFloat = 0 };
  169. };
  170. class Int32
  171. {
  172. public:
  173. inline Int32 (void* d) noexcept : data (static_cast<uint32*> (d)) {}
  174. inline void advance() noexcept { ++data; }
  175. inline void skip (int numSamples) noexcept { data += numSamples; }
  176. inline float getAsFloatLE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int32) ByteOrder::swapIfBigEndian (*data)); }
  177. inline float getAsFloatBE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int32) ByteOrder::swapIfLittleEndian (*data)); }
  178. inline void setAsFloatLE (float newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint32) (maxValue * jlimit (-1.0, 1.0, (double) newValue))); }
  179. inline void setAsFloatBE (float newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint32) (maxValue * jlimit (-1.0, 1.0, (double) newValue))); }
  180. inline int32 getAsInt32LE() const noexcept { return (int32) ByteOrder::swapIfBigEndian (*data); }
  181. inline int32 getAsInt32BE() const noexcept { return (int32) ByteOrder::swapIfLittleEndian (*data); }
  182. inline void setAsInt32LE (int32 newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint32) newValue); }
  183. inline void setAsInt32BE (int32 newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint32) newValue); }
  184. inline void clear() noexcept { *data = 0; }
  185. inline void clearMultiple (int num) noexcept { zeromem (data, (size_t) (num * bytesPerSample)) ;}
  186. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  187. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  188. inline void copyFromSameType (Int32& source) noexcept { *data = *source.data; }
  189. uint32* data;
  190. enum { bytesPerSample = 4, maxValue = 0x7fffffff, resolution = 1, isFloat = 0 };
  191. };
  192. /** A 32-bit integer type, of which only the bottom 24 bits are used. */
  193. class Int24in32 : public Int32
  194. {
  195. public:
  196. inline Int24in32 (void* d) noexcept : Int32 (d) {}
  197. inline float getAsFloatLE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int32) ByteOrder::swapIfBigEndian (*data)); }
  198. inline float getAsFloatBE() const noexcept { return (float) ((1.0 / (1.0 + maxValue)) * (int32) ByteOrder::swapIfLittleEndian (*data)); }
  199. inline void setAsFloatLE (float newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint32) (maxValue * jlimit (-1.0, 1.0, (double) newValue))); }
  200. inline void setAsFloatBE (float newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint32) (maxValue * jlimit (-1.0, 1.0, (double) newValue))); }
  201. inline int32 getAsInt32LE() const noexcept { return (int32) ByteOrder::swapIfBigEndian (*data) << 8; }
  202. inline int32 getAsInt32BE() const noexcept { return (int32) ByteOrder::swapIfLittleEndian (*data) << 8; }
  203. inline void setAsInt32LE (int32 newValue) noexcept { *data = ByteOrder::swapIfBigEndian ((uint32) newValue >> 8); }
  204. inline void setAsInt32BE (int32 newValue) noexcept { *data = ByteOrder::swapIfLittleEndian ((uint32) newValue >> 8); }
  205. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsInt32LE (source.getAsInt32()); }
  206. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsInt32BE (source.getAsInt32()); }
  207. inline void copyFromSameType (Int24in32& source) noexcept { *data = *source.data; }
  208. enum { bytesPerSample = 4, maxValue = 0x7fffff, resolution = (1 << 8), isFloat = 0 };
  209. };
  210. class Float32
  211. {
  212. public:
  213. inline Float32 (void* d) noexcept : data (static_cast<float*> (d)) {}
  214. inline void advance() noexcept { ++data; }
  215. inline void skip (int numSamples) noexcept { data += numSamples; }
  216. #if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__
  217. inline float getAsFloatBE() const noexcept { return *data; }
  218. inline void setAsFloatBE (float newValue) noexcept { *data = newValue; }
  219. inline float getAsFloatLE() const noexcept { union { uint32 asInt; float asFloat; } n; n.asInt = ByteOrder::swap (*(uint32*) data); return n.asFloat; }
  220. inline void setAsFloatLE (float newValue) noexcept { union { uint32 asInt; float asFloat; } n; n.asFloat = newValue; *(uint32*) data = ByteOrder::swap (n.asInt); }
  221. #else
  222. inline float getAsFloatLE() const noexcept { return *data; }
  223. inline void setAsFloatLE (float newValue) noexcept { *data = newValue; }
  224. inline float getAsFloatBE() const noexcept { union { uint32 asInt; float asFloat; } n; n.asInt = ByteOrder::swap (*(uint32*) data); return n.asFloat; }
  225. inline void setAsFloatBE (float newValue) noexcept { union { uint32 asInt; float asFloat; } n; n.asFloat = newValue; *(uint32*) data = ByteOrder::swap (n.asInt); }
  226. #endif
  227. inline int32 getAsInt32LE() const noexcept { return (int32) roundToInt (jlimit (-1.0, 1.0, (double) getAsFloatLE()) * (double) maxValue); }
  228. inline int32 getAsInt32BE() const noexcept { return (int32) roundToInt (jlimit (-1.0, 1.0, (double) getAsFloatBE()) * (double) maxValue); }
  229. inline void setAsInt32LE (int32 newValue) noexcept { setAsFloatLE ((float) (newValue * (1.0 / (1.0 + maxValue)))); }
  230. inline void setAsInt32BE (int32 newValue) noexcept { setAsFloatBE ((float) (newValue * (1.0 / (1.0 + maxValue)))); }
  231. inline void clear() noexcept { *data = 0; }
  232. inline void clearMultiple (int num) noexcept { zeromem (data, (size_t) (num * bytesPerSample)) ;}
  233. template <class SourceType> inline void copyFromLE (SourceType& source) noexcept { setAsFloatLE (source.getAsFloat()); }
  234. template <class SourceType> inline void copyFromBE (SourceType& source) noexcept { setAsFloatBE (source.getAsFloat()); }
  235. inline void copyFromSameType (Float32& source) noexcept { *data = *source.data; }
  236. float* data;
  237. enum { bytesPerSample = 4, maxValue = 0x7fffffff, resolution = (1 << 8), isFloat = 1 };
  238. };
  239. //==============================================================================
  240. class NonInterleaved
  241. {
  242. public:
  243. inline NonInterleaved() noexcept {}
  244. inline NonInterleaved (const NonInterleaved&) noexcept {}
  245. inline NonInterleaved (const int) noexcept {}
  246. inline void copyFrom (const NonInterleaved&) noexcept {}
  247. template <class SampleFormatType> inline void advanceData (SampleFormatType& s) noexcept { s.advance(); }
  248. template <class SampleFormatType> inline void advanceDataBy (SampleFormatType& s, int numSamples) noexcept { s.skip (numSamples); }
  249. template <class SampleFormatType> inline void clear (SampleFormatType& s, int numSamples) noexcept { s.clearMultiple (numSamples); }
  250. template <class SampleFormatType> inline static int getNumBytesBetweenSamples (const SampleFormatType&) noexcept { return SampleFormatType::bytesPerSample; }
  251. enum { isInterleavedType = 0, numInterleavedChannels = 1 };
  252. };
  253. class Interleaved
  254. {
  255. public:
  256. inline Interleaved() noexcept : numInterleavedChannels (1) {}
  257. inline Interleaved (const Interleaved& other) noexcept : numInterleavedChannels (other.numInterleavedChannels) {}
  258. inline Interleaved (const int numInterleavedChans) noexcept : numInterleavedChannels (numInterleavedChans) {}
  259. inline void copyFrom (const Interleaved& other) noexcept { numInterleavedChannels = other.numInterleavedChannels; }
  260. template <class SampleFormatType> inline void advanceData (SampleFormatType& s) noexcept { s.skip (numInterleavedChannels); }
  261. template <class SampleFormatType> inline void advanceDataBy (SampleFormatType& s, int numSamples) noexcept { s.skip (numInterleavedChannels * numSamples); }
  262. template <class SampleFormatType> inline void clear (SampleFormatType& s, int numSamples) noexcept { while (--numSamples >= 0) { s.clear(); s.skip (numInterleavedChannels); } }
  263. template <class SampleFormatType> inline int getNumBytesBetweenSamples (const SampleFormatType&) const noexcept { return numInterleavedChannels * SampleFormatType::bytesPerSample; }
  264. int numInterleavedChannels;
  265. enum { isInterleavedType = 1 };
  266. };
  267. //==============================================================================
  268. class NonConst
  269. {
  270. public:
  271. typedef void VoidType;
  272. static inline void* toVoidPtr (VoidType* v) noexcept { return v; }
  273. enum { isConst = 0 };
  274. };
  275. class Const
  276. {
  277. public:
  278. typedef const void VoidType;
  279. static inline void* toVoidPtr (VoidType* v) noexcept { return const_cast<void*> (v); }
  280. enum { isConst = 1 };
  281. };
  282. #endif
  283. //==============================================================================
  284. /**
  285. A pointer to a block of audio data with a particular encoding.
  286. This object can be used to read and write from blocks of encoded audio samples. To create one, you specify
  287. the audio format as a series of template parameters, e.g.
  288. @code
  289. // this creates a pointer for reading from a const array of 16-bit little-endian packed samples.
  290. AudioData::Pointer <AudioData::Int16,
  291. AudioData::LittleEndian,
  292. AudioData::NonInterleaved,
  293. AudioData::Const> pointer (someRawAudioData);
  294. // These methods read the sample that is being pointed to
  295. float firstSampleAsFloat = pointer.getAsFloat();
  296. int32 firstSampleAsInt = pointer.getAsInt32();
  297. ++pointer; // moves the pointer to the next sample.
  298. pointer += 3; // skips the next 3 samples.
  299. @endcode
  300. The convertSamples() method lets you copy a range of samples from one format to another, automatically
  301. converting its format.
  302. @see AudioData::Converter
  303. */
  304. template <typename SampleFormat,
  305. typename Endianness,
  306. typename InterleavingType,
  307. typename Constness>
  308. class Pointer : private InterleavingType // (inherited for EBCO)
  309. {
  310. public:
  311. //==============================================================================
  312. /** Creates a non-interleaved pointer from some raw data in the appropriate format.
  313. This constructor is only used if you've specified the AudioData::NonInterleaved option -
  314. for interleaved formats, use the constructor that also takes a number of channels.
  315. */
  316. Pointer (typename Constness::VoidType* sourceData) noexcept
  317. : data (Constness::toVoidPtr (sourceData))
  318. {
  319. // If you're using interleaved data, call the other constructor! If you're using non-interleaved data,
  320. // you should pass NonInterleaved as the template parameter for the interleaving type!
  321. static_jassert (InterleavingType::isInterleavedType == 0);
  322. }
  323. /** Creates a pointer from some raw data in the appropriate format with the specified number of interleaved channels.
  324. For non-interleaved data, use the other constructor.
  325. */
  326. Pointer (typename Constness::VoidType* sourceData, int numInterleaved) noexcept
  327. : InterleavingType (numInterleaved), data (Constness::toVoidPtr (sourceData))
  328. {
  329. }
  330. /** Creates a copy of another pointer. */
  331. Pointer (const Pointer& other) noexcept
  332. : InterleavingType (other), data (other.data)
  333. {
  334. }
  335. Pointer& operator= (const Pointer& other) noexcept
  336. {
  337. InterleavingType::operator= (other);
  338. data = other.data;
  339. return *this;
  340. }
  341. //==============================================================================
  342. /** Returns the value of the first sample as a floating point value.
  343. The value will be in the range -1.0 to 1.0 for integer formats. For floating point
  344. formats, the value could be outside that range, although -1 to 1 is the standard range.
  345. */
  346. inline float getAsFloat() const noexcept { return Endianness::getAsFloat (data); }
  347. /** Sets the value of the first sample as a floating point value.
  348. (This method can only be used if the AudioData::NonConst option was used).
  349. The value should be in the range -1.0 to 1.0 - for integer formats, values outside that
  350. range will be clipped. For floating point formats, any value passed in here will be
  351. written directly, although -1 to 1 is the standard range.
  352. */
  353. inline void setAsFloat (float newValue) noexcept
  354. {
  355. static_jassert (Constness::isConst == 0); // trying to write to a const pointer! For a writeable one, use AudioData::NonConst instead!
  356. Endianness::setAsFloat (data, newValue);
  357. }
  358. /** Returns the value of the first sample as a 32-bit integer.
  359. The value returned will be in the range 0x80000000 to 0x7fffffff, and shorter values will be
  360. shifted to fill this range (e.g. if you're reading from 24-bit data, the values will be shifted up
  361. by 8 bits when returned here). If the source data is floating point, values beyond -1.0 to 1.0 will
  362. be clipped so that -1.0 maps onto -0x7fffffff and 1.0 maps to 0x7fffffff.
  363. */
  364. inline int32 getAsInt32() const noexcept { return Endianness::getAsInt32 (data); }
  365. /** Sets the value of the first sample as a 32-bit integer.
  366. This will be mapped to the range of the format that is being written - see getAsInt32().
  367. */
  368. inline void setAsInt32 (int32 newValue) noexcept
  369. {
  370. static_jassert (Constness::isConst == 0); // trying to write to a const pointer! For a writeable one, use AudioData::NonConst instead!
  371. Endianness::setAsInt32 (data, newValue);
  372. }
  373. /** Moves the pointer along to the next sample. */
  374. inline Pointer& operator++() noexcept { advance(); return *this; }
  375. /** Moves the pointer back to the previous sample. */
  376. inline Pointer& operator--() noexcept { this->advanceDataBy (data, -1); return *this; }
  377. /** Adds a number of samples to the pointer's position. */
  378. Pointer& operator+= (int samplesToJump) noexcept { this->advanceDataBy (data, samplesToJump); return *this; }
  379. /** Writes a stream of samples into this pointer from another pointer.
  380. This will copy the specified number of samples, converting between formats appropriately.
  381. */
  382. void convertSamples (Pointer source, int numSamples) const noexcept
  383. {
  384. static_jassert (Constness::isConst == 0); // trying to write to a const pointer! For a writeable one, use AudioData::NonConst instead!
  385. for (Pointer dest (*this); --numSamples >= 0;)
  386. {
  387. dest.data.copyFromSameType (source.data);
  388. dest.advance();
  389. source.advance();
  390. }
  391. }
  392. /** Writes a stream of samples into this pointer from another pointer.
  393. This will copy the specified number of samples, converting between formats appropriately.
  394. */
  395. template <class OtherPointerType>
  396. void convertSamples (OtherPointerType source, int numSamples) const noexcept
  397. {
  398. static_jassert (Constness::isConst == 0); // trying to write to a const pointer! For a writeable one, use AudioData::NonConst instead!
  399. Pointer dest (*this);
  400. if (source.getRawData() != getRawData() || source.getNumBytesBetweenSamples() >= getNumBytesBetweenSamples())
  401. {
  402. while (--numSamples >= 0)
  403. {
  404. Endianness::copyFrom (dest.data, source);
  405. dest.advance();
  406. ++source;
  407. }
  408. }
  409. else // copy backwards if we're increasing the sample width..
  410. {
  411. dest += numSamples;
  412. source += numSamples;
  413. while (--numSamples >= 0)
  414. Endianness::copyFrom ((--dest).data, --source);
  415. }
  416. }
  417. /** Sets a number of samples to zero. */
  418. void clearSamples (int numSamples) const noexcept
  419. {
  420. Pointer dest (*this);
  421. dest.clear (dest.data, numSamples);
  422. }
  423. #if 0
  424. /** Scans a block of data, returning the lowest and highest levels as floats */
  425. Range<float> findMinAndMax (size_t numSamples) const noexcept
  426. {
  427. if (numSamples == 0)
  428. return Range<float>();
  429. Pointer dest (*this);
  430. if (isFloatingPoint())
  431. {
  432. float mn = dest.getAsFloat();
  433. dest.advance();
  434. float mx = mn;
  435. while (--numSamples > 0)
  436. {
  437. const float v = dest.getAsFloat();
  438. dest.advance();
  439. if (mx < v) mx = v;
  440. if (v < mn) mn = v;
  441. }
  442. return Range<float> (mn, mx);
  443. }
  444. int32 mn = dest.getAsInt32();
  445. dest.advance();
  446. int32 mx = mn;
  447. while (--numSamples > 0)
  448. {
  449. const int v = dest.getAsInt32();
  450. dest.advance();
  451. if (mx < v) mx = v;
  452. if (v < mn) mn = v;
  453. }
  454. return Range<float> (mn * (float) (1.0 / (1.0 + Int32::maxValue)),
  455. mx * (float) (1.0 / (1.0 + Int32::maxValue)));
  456. }
  457. /** Scans a block of data, returning the lowest and highest levels as floats */
  458. void findMinAndMax (size_t numSamples, float& minValue, float& maxValue) const noexcept
  459. {
  460. Range<float> r (findMinAndMax (numSamples));
  461. minValue = r.getStart();
  462. maxValue = r.getEnd();
  463. }
  464. #endif
  465. /** Returns true if the pointer is using a floating-point format. */
  466. static bool isFloatingPoint() noexcept { return (bool) SampleFormat::isFloat; }
  467. /** Returns true if the format is big-endian. */
  468. static bool isBigEndian() noexcept { return (bool) Endianness::isBigEndian; }
  469. /** Returns the number of bytes in each sample (ignoring the number of interleaved channels). */
  470. static int getBytesPerSample() noexcept { return (int) SampleFormat::bytesPerSample; }
  471. /** Returns the number of interleaved channels in the format. */
  472. int getNumInterleavedChannels() const noexcept { return (int) this->numInterleavedChannels; }
  473. /** Returns the number of bytes between the start address of each sample. */
  474. int getNumBytesBetweenSamples() const noexcept { return InterleavingType::getNumBytesBetweenSamples (data); }
  475. /** Returns the accuracy of this format when represented as a 32-bit integer.
  476. This is the smallest number above 0 that can be represented in the sample format, converted to
  477. a 32-bit range. E,g. if the format is 8-bit, its resolution is 0x01000000; if the format is 24-bit,
  478. its resolution is 0x100.
  479. */
  480. static int get32BitResolution() noexcept { return (int) SampleFormat::resolution; }
  481. /** Returns a pointer to the underlying data. */
  482. const void* getRawData() const noexcept { return data.data; }
  483. private:
  484. //==============================================================================
  485. SampleFormat data;
  486. inline void advance() noexcept { this->advanceData (data); }
  487. Pointer operator++ (int); // private to force you to use the more efficient pre-increment!
  488. Pointer operator-- (int);
  489. };
  490. //==============================================================================
  491. /** A base class for objects that are used to convert between two different sample formats.
  492. The AudioData::ConverterInstance implements this base class and can be templated, so
  493. you can create an instance that converts between two particular formats, and then
  494. store this in the abstract base class.
  495. @see AudioData::ConverterInstance
  496. */
  497. class Converter
  498. {
  499. public:
  500. virtual ~Converter() {}
  501. /** Converts a sequence of samples from the converter's source format into the dest format. */
  502. virtual void convertSamples (void* destSamples, const void* sourceSamples, int numSamples) const = 0;
  503. /** Converts a sequence of samples from the converter's source format into the dest format.
  504. This method takes sub-channel indexes, which can be used with interleaved formats in order to choose a
  505. particular sub-channel of the data to be used.
  506. */
  507. virtual void convertSamples (void* destSamples, int destSubChannel,
  508. const void* sourceSamples, int sourceSubChannel, int numSamples) const = 0;
  509. };
  510. //==============================================================================
  511. /**
  512. A class that converts between two templated AudioData::Pointer types, and which
  513. implements the AudioData::Converter interface.
  514. This can be used as a concrete instance of the AudioData::Converter abstract class.
  515. @see AudioData::Converter
  516. */
  517. template <class SourceSampleType, class DestSampleType>
  518. class ConverterInstance : public Converter
  519. {
  520. public:
  521. ConverterInstance (int numSourceChannels = 1, int numDestChannels = 1)
  522. : sourceChannels (numSourceChannels), destChannels (numDestChannels)
  523. {}
  524. void convertSamples (void* dest, const void* source, int numSamples) const override
  525. {
  526. SourceSampleType s (source, sourceChannels);
  527. DestSampleType d (dest, destChannels);
  528. d.convertSamples (s, numSamples);
  529. }
  530. void convertSamples (void* dest, int destSubChannel,
  531. const void* source, int sourceSubChannel, int numSamples) const override
  532. {
  533. jassert (destSubChannel < destChannels && sourceSubChannel < sourceChannels);
  534. SourceSampleType s (addBytesToPointer (source, sourceSubChannel * SourceSampleType::getBytesPerSample()), sourceChannels);
  535. DestSampleType d (addBytesToPointer (dest, destSubChannel * DestSampleType::getBytesPerSample()), destChannels);
  536. d.convertSamples (s, numSamples);
  537. }
  538. private:
  539. const int sourceChannels, destChannels;
  540. CARLA_DECLARE_NON_COPY_CLASS (ConverterInstance)
  541. };
  542. };
  543. //==============================================================================
  544. /**
  545. A set of routines to convert buffers of 32-bit floating point data to and from
  546. various integer formats.
  547. Note that these functions are deprecated - the AudioData class provides a much more
  548. flexible set of conversion classes now.
  549. */
  550. class AudioDataConverters
  551. {
  552. public:
  553. //==============================================================================
  554. static void convertFloatToInt16LE (const float* source, void* dest, int numSamples, int destBytesPerSample = 2);
  555. static void convertFloatToInt16BE (const float* source, void* dest, int numSamples, int destBytesPerSample = 2);
  556. static void convertFloatToInt24LE (const float* source, void* dest, int numSamples, int destBytesPerSample = 3);
  557. static void convertFloatToInt24BE (const float* source, void* dest, int numSamples, int destBytesPerSample = 3);
  558. static void convertFloatToInt32LE (const float* source, void* dest, int numSamples, int destBytesPerSample = 4);
  559. static void convertFloatToInt32BE (const float* source, void* dest, int numSamples, int destBytesPerSample = 4);
  560. static void convertFloatToFloat32LE (const float* source, void* dest, int numSamples, int destBytesPerSample = 4);
  561. static void convertFloatToFloat32BE (const float* source, void* dest, int numSamples, int destBytesPerSample = 4);
  562. //==============================================================================
  563. static void convertInt16LEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 2);
  564. static void convertInt16BEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 2);
  565. static void convertInt24LEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 3);
  566. static void convertInt24BEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 3);
  567. static void convertInt32LEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 4);
  568. static void convertInt32BEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 4);
  569. static void convertFloat32LEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 4);
  570. static void convertFloat32BEToFloat (const void* source, float* dest, int numSamples, int srcBytesPerSample = 4);
  571. //==============================================================================
  572. enum DataFormat
  573. {
  574. int16LE,
  575. int16BE,
  576. int24LE,
  577. int24BE,
  578. int32LE,
  579. int32BE,
  580. float32LE,
  581. float32BE,
  582. };
  583. static void convertFloatToFormat (DataFormat destFormat,
  584. const float* source, void* dest, int numSamples);
  585. static void convertFormatToFloat (DataFormat sourceFormat,
  586. const void* source, float* dest, int numSamples);
  587. //==============================================================================
  588. static void interleaveSamples (const float** source, float* dest,
  589. int numSamples, int numChannels);
  590. static void deinterleaveSamples (const float* source, float** dest,
  591. int numSamples, int numChannels);
  592. private:
  593. AudioDataConverters();
  594. CARLA_DECLARE_NON_COPY_CLASS (AudioDataConverters)
  595. };
  596. }
  597. #endif // WATER_AUDIODATACONVERTERS_H_INCLUDED