sfxSndStream.cpp 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2012 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. #include "sfx/media/sfxSndStream.h"
  23. #include "core/stream/fileStream.h"
  24. #include "core/stream/stream.h"
  25. #include "console/console.h"
  26. #define MAX_BUFFER 4096
  27. bool SFXSndStream::_readHeader()
  28. {
  29. mCurPos = 0;
  30. mBytesRead = 0;
  31. dMemset(&sfinfo, 0, sizeof(SF_INFO));
  32. vio.get_filelen = sndFileLen;
  33. vio.seek = sndSeek;
  34. vio.read = sndRead;
  35. vio.write = sndWrite;
  36. vio.tell = sndTell;
  37. vio_data.length = 0;
  38. vio_data.offset = 0;
  39. vio_data.data = mStream;
  40. if ((sndFile = sf_open_virtual(&vio, SFM_READ, &sfinfo, &vio_data)) == NULL)
  41. {
  42. Con::printf("SFXSndStream - _readHeader failed: %s", sf_strerror(sndFile));
  43. return false;
  44. }
  45. S32 bitsPerSample = 0;
  46. switch ((sfinfo.format & SF_FORMAT_SUBMASK))
  47. {
  48. case SF_FORMAT_PCM_S8:
  49. case SF_FORMAT_PCM_U8:
  50. bitsPerSample = 8;
  51. break;
  52. case SF_FORMAT_PCM_16:
  53. bitsPerSample = 16;
  54. break;
  55. case SF_FORMAT_VORBIS:
  56. bitsPerSample = 16;
  57. sf_command(sndFile, SFC_SET_SCALE_FLOAT_INT_READ, NULL, SF_TRUE);
  58. break;
  59. case SF_FORMAT_PCM_24:
  60. bitsPerSample = 24;
  61. break;
  62. case SF_FORMAT_PCM_32:
  63. case SF_FORMAT_FLOAT:
  64. bitsPerSample = 32;
  65. break;
  66. default:
  67. // missed, set it to 16 anyway.
  68. bitsPerSample = 16;
  69. break;
  70. }
  71. mFormat.set(sfinfo.channels, bitsPerSample * sfinfo.channels, sfinfo.samplerate);
  72. mSamples = sfinfo.frames;
  73. return true;
  74. }
  75. void SFXSndStream::_close()
  76. {
  77. if (!sndFile)
  78. return;
  79. sf_close(sndFile);
  80. }
  81. SFXSndStream* SFXSndStream::create(Stream* stream)
  82. {
  83. SFXSndStream* sfxStream = new SFXSndStream();
  84. if (sfxStream->open(stream, true))
  85. return sfxStream;
  86. delete sfxStream;
  87. return nullptr;
  88. }
  89. void SFXSndStream::reset()
  90. {
  91. vio_data.offset = 0;
  92. }
  93. U32 SFXSndStream::read(U8* buffer, U32 length)
  94. {
  95. if (!sndFile)
  96. {
  97. Con::errorf("SFXSndStream - read: Called on uninitialized stream.");
  98. return 0;
  99. }
  100. U32 framesToRead = length / mFormat.getBytesPerSample();
  101. U32 framesRead = 0;
  102. switch (sfinfo.format & SF_FORMAT_SUBMASK)
  103. {
  104. case SF_FORMAT_PCM_S8:
  105. case SF_FORMAT_PCM_U8:
  106. framesRead = sf_readf_int(sndFile, reinterpret_cast<int*>(buffer), framesToRead);
  107. break;
  108. case SF_FORMAT_PCM_16:
  109. case SF_FORMAT_VORBIS:
  110. framesRead = sf_readf_short(sndFile, reinterpret_cast<short*>(buffer), framesToRead);
  111. break;
  112. case SF_FORMAT_PCM_24:
  113. framesRead = sf_readf_int(sndFile, reinterpret_cast<int*>(buffer), framesToRead); // 24-bit usually stored in 32-bit containers
  114. break;
  115. case SF_FORMAT_PCM_32:
  116. case SF_FORMAT_FLOAT:
  117. framesRead = sf_readf_float(sndFile, reinterpret_cast<float*>(buffer), framesToRead);
  118. break;
  119. default:
  120. Con::errorf("SFXSndStream - read: Unsupported format.");
  121. return 0;
  122. }
  123. if (framesRead != sfinfo.frames)
  124. {
  125. Con::errorf("SFXSndStream - read: %s", sf_strerror(sndFile));
  126. }
  127. // reset stream
  128. setPosition(0);
  129. return framesRead * mFormat.getBytesPerSample();
  130. }
  131. bool SFXSndStream::isEOS() const
  132. {
  133. return (Parent::isEOS() || (mStream && vio_data.length == vio_data.offset));
  134. }
  135. U32 SFXSndStream::getPosition() const
  136. {
  137. return vio_data.offset;
  138. }
  139. void SFXSndStream::setPosition(U32 offset)
  140. {
  141. sf_seek(sndFile, offset / mFormat.getBytesPerSample(), SEEK_SET);
  142. }
  143. sf_count_t SFXSndStream::sndSeek(sf_count_t offset, int whence, void* user_data)
  144. {
  145. VIO_DATA* vf = (VIO_DATA*)user_data;
  146. Stream* stream = reinterpret_cast<Stream*>(vf->data);
  147. switch (whence)
  148. {
  149. case SEEK_SET:
  150. vf->offset = offset;
  151. break;
  152. case SEEK_CUR:
  153. vf->offset = vf->offset + offset;
  154. break;
  155. case SEEK_END:
  156. vf->offset = vf->length - offset;
  157. break;
  158. default:
  159. break;
  160. };
  161. return stream->setPosition(vf->offset) ? 0 : -1;
  162. }
  163. sf_count_t SFXSndStream::sndRead(void* ptr, sf_count_t count, void* user_data)
  164. {
  165. VIO_DATA* vf = (VIO_DATA*)user_data;
  166. Stream* stream = reinterpret_cast<Stream*>(vf->data);
  167. if (vf->offset + count > vf->length)
  168. count = vf->length - vf->offset;
  169. stream->read((U32)(count), ptr);
  170. vf->offset += count;
  171. return count;
  172. }
  173. sf_count_t SFXSndStream::sndWrite(const void* ptr, sf_count_t count, void* user_data)
  174. {
  175. return sf_count_t();
  176. }
  177. sf_count_t SFXSndStream::sndTell(void* user_data)
  178. {
  179. VIO_DATA* vf = (VIO_DATA*)user_data;
  180. return vf->offset;
  181. }
  182. sf_count_t SFXSndStream::sndFileLen(void* user_data)
  183. {
  184. VIO_DATA* vf = (VIO_DATA*)user_data;
  185. Stream* stream = reinterpret_cast<Stream*>(vf->data);
  186. vf->length = stream->getStreamSize();
  187. return vf->length;
  188. }