The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
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.

388 lines
14KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library - "Jules' Utility Class Extensions"
  4. Copyright 2004-11 by Raw Material Software Ltd.
  5. ------------------------------------------------------------------------------
  6. JUCE can be redistributed and/or modified under the terms of the GNU General
  7. Public License (Version 2), as published by the Free Software Foundation.
  8. A copy of the license is included in the JUCE distribution, or can be found
  9. online at www.gnu.org/licenses.
  10. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  11. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  12. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  13. ------------------------------------------------------------------------------
  14. To release a closed-source product which uses JUCE, commercial licenses are
  15. available: visit www.rawmaterialsoftware.com/juce for more information.
  16. ==============================================================================
  17. */
  18. #if JUCE_QUICKTIME && ! (JUCE_64BIT || JUCE_IOS)
  19. #if ! JUCE_WINDOWS
  20. #define Point CarbonDummyPointName // (workaround to avoid definition of "Point" by old Carbon headers)
  21. #define Component CarbonDummyCompName
  22. #include <QuickTime/Movies.h>
  23. #include <QuickTime/QTML.h>
  24. #include <QuickTime/QuickTimeComponents.h>
  25. #include <QuickTime/MediaHandlers.h>
  26. #include <QuickTime/ImageCodec.h>
  27. #undef Point
  28. #undef Component
  29. #else
  30. #if JUCE_MSVC
  31. #pragma warning (push)
  32. #pragma warning (disable : 4100)
  33. #endif
  34. /* If you've got an include error here, you probably need to install the QuickTime SDK and
  35. add its header directory to your include path.
  36. Alternatively, if you don't need any QuickTime services, just set the JUCE_QUICKTIME flag to 0.
  37. */
  38. #undef SIZE_MAX
  39. #include <Movies.h>
  40. #include <QTML.h>
  41. #include <QuickTimeComponents.h>
  42. #include <MediaHandlers.h>
  43. #include <ImageCodec.h>
  44. #undef SIZE_MAX
  45. #if JUCE_MSVC
  46. #pragma warning (pop)
  47. #endif
  48. #endif
  49. BEGIN_JUCE_NAMESPACE
  50. bool juce_OpenQuickTimeMovieFromStream (InputStream* input, Movie& movie, Handle& dataHandle);
  51. static const char* const quickTimeFormatName = "QuickTime file";
  52. static const char* const quickTimeExtensions[] = { ".mov", ".mp3", ".mp4", ".m4a", 0 };
  53. //==============================================================================
  54. class QTAudioReader : public AudioFormatReader
  55. {
  56. public:
  57. QTAudioReader (InputStream* const input_, const int trackNum_)
  58. : AudioFormatReader (input_, TRANS (quickTimeFormatName)),
  59. ok (false),
  60. movie (0),
  61. trackNum (trackNum_),
  62. lastSampleRead (0),
  63. lastThreadId (0),
  64. extractor (0),
  65. dataHandle (0)
  66. {
  67. JUCE_AUTORELEASEPOOL
  68. bufferList.calloc (256, 1);
  69. #if JUCE_WINDOWS
  70. if (InitializeQTML (0) != noErr)
  71. return;
  72. #endif
  73. if (EnterMovies() != noErr)
  74. return;
  75. bool opened = juce_OpenQuickTimeMovieFromStream (input_, movie, dataHandle);
  76. if (! opened)
  77. return;
  78. {
  79. const int numTracks = GetMovieTrackCount (movie);
  80. int trackCount = 0;
  81. for (int i = 1; i <= numTracks; ++i)
  82. {
  83. track = GetMovieIndTrack (movie, i);
  84. media = GetTrackMedia (track);
  85. OSType mediaType;
  86. GetMediaHandlerDescription (media, &mediaType, 0, 0);
  87. if (mediaType == SoundMediaType
  88. && trackCount++ == trackNum_)
  89. {
  90. ok = true;
  91. break;
  92. }
  93. }
  94. }
  95. if (! ok)
  96. return;
  97. ok = false;
  98. lengthInSamples = GetMediaDecodeDuration (media);
  99. usesFloatingPointData = false;
  100. samplesPerFrame = (int) (GetMediaDecodeDuration (media) / GetMediaSampleCount (media));
  101. trackUnitsPerFrame = GetMovieTimeScale (movie) * samplesPerFrame
  102. / GetMediaTimeScale (media);
  103. OSStatus err = MovieAudioExtractionBegin (movie, 0, &extractor);
  104. unsigned long output_layout_size;
  105. err = MovieAudioExtractionGetPropertyInfo (extractor,
  106. kQTPropertyClass_MovieAudioExtraction_Audio,
  107. kQTMovieAudioExtractionAudioPropertyID_AudioChannelLayout,
  108. 0, &output_layout_size, 0);
  109. if (err != noErr)
  110. return;
  111. HeapBlock <AudioChannelLayout> qt_audio_channel_layout;
  112. qt_audio_channel_layout.calloc (output_layout_size, 1);
  113. err = MovieAudioExtractionGetProperty (extractor,
  114. kQTPropertyClass_MovieAudioExtraction_Audio,
  115. kQTMovieAudioExtractionAudioPropertyID_AudioChannelLayout,
  116. output_layout_size, qt_audio_channel_layout, 0);
  117. qt_audio_channel_layout[0].mChannelLayoutTag = kAudioChannelLayoutTag_Stereo;
  118. err = MovieAudioExtractionSetProperty (extractor,
  119. kQTPropertyClass_MovieAudioExtraction_Audio,
  120. kQTMovieAudioExtractionAudioPropertyID_AudioChannelLayout,
  121. output_layout_size,
  122. qt_audio_channel_layout);
  123. err = MovieAudioExtractionGetProperty (extractor,
  124. kQTPropertyClass_MovieAudioExtraction_Audio,
  125. kQTMovieAudioExtractionAudioPropertyID_AudioStreamBasicDescription,
  126. sizeof (inputStreamDesc),
  127. &inputStreamDesc, 0);
  128. if (err != noErr)
  129. return;
  130. inputStreamDesc.mFormatFlags = kAudioFormatFlagIsSignedInteger
  131. | kAudioFormatFlagIsPacked
  132. | kAudioFormatFlagsNativeEndian;
  133. inputStreamDesc.mBitsPerChannel = sizeof (SInt16) * 8;
  134. inputStreamDesc.mChannelsPerFrame = jmin ((UInt32) 2, inputStreamDesc.mChannelsPerFrame);
  135. inputStreamDesc.mBytesPerFrame = sizeof (SInt16) * inputStreamDesc.mChannelsPerFrame;
  136. inputStreamDesc.mBytesPerPacket = inputStreamDesc.mBytesPerFrame;
  137. err = MovieAudioExtractionSetProperty (extractor,
  138. kQTPropertyClass_MovieAudioExtraction_Audio,
  139. kQTMovieAudioExtractionAudioPropertyID_AudioStreamBasicDescription,
  140. sizeof (inputStreamDesc),
  141. &inputStreamDesc);
  142. if (err != noErr)
  143. return;
  144. Boolean allChannelsDiscrete = false;
  145. err = MovieAudioExtractionSetProperty (extractor,
  146. kQTPropertyClass_MovieAudioExtraction_Movie,
  147. kQTMovieAudioExtractionMoviePropertyID_AllChannelsDiscrete,
  148. sizeof (allChannelsDiscrete),
  149. &allChannelsDiscrete);
  150. if (err != noErr)
  151. return;
  152. bufferList->mNumberBuffers = 1;
  153. bufferList->mBuffers[0].mNumberChannels = inputStreamDesc.mChannelsPerFrame;
  154. bufferList->mBuffers[0].mDataByteSize = jmax ((UInt32) 4096, (UInt32) (samplesPerFrame * inputStreamDesc.mBytesPerFrame) + 16);
  155. dataBuffer.malloc (bufferList->mBuffers[0].mDataByteSize);
  156. bufferList->mBuffers[0].mData = dataBuffer;
  157. sampleRate = inputStreamDesc.mSampleRate;
  158. bitsPerSample = 16;
  159. numChannels = inputStreamDesc.mChannelsPerFrame;
  160. detachThread();
  161. ok = true;
  162. }
  163. ~QTAudioReader()
  164. {
  165. JUCE_AUTORELEASEPOOL
  166. checkThreadIsAttached();
  167. if (dataHandle != nullptr)
  168. DisposeHandle (dataHandle);
  169. if (extractor != nullptr)
  170. {
  171. MovieAudioExtractionEnd (extractor);
  172. extractor = nullptr;
  173. }
  174. DisposeMovie (movie);
  175. #if JUCE_MAC
  176. ExitMoviesOnThread ();
  177. #endif
  178. }
  179. bool readSamples (int** destSamples, int numDestChannels, int startOffsetInDestBuffer,
  180. int64 startSampleInFile, int numSamples)
  181. {
  182. JUCE_AUTORELEASEPOOL
  183. checkThreadIsAttached();
  184. bool ok = true;
  185. while (numSamples > 0)
  186. {
  187. if (lastSampleRead != startSampleInFile)
  188. {
  189. TimeRecord time;
  190. time.scale = (TimeScale) inputStreamDesc.mSampleRate;
  191. time.base = 0;
  192. time.value.hi = 0;
  193. time.value.lo = (UInt32) startSampleInFile;
  194. OSStatus err = MovieAudioExtractionSetProperty (extractor,
  195. kQTPropertyClass_MovieAudioExtraction_Movie,
  196. kQTMovieAudioExtractionMoviePropertyID_CurrentTime,
  197. sizeof (time), &time);
  198. if (err != noErr)
  199. {
  200. ok = false;
  201. break;
  202. }
  203. }
  204. int framesToDo = jmin (numSamples, (int) (bufferList->mBuffers[0].mDataByteSize / inputStreamDesc.mBytesPerFrame));
  205. bufferList->mBuffers[0].mDataByteSize = inputStreamDesc.mBytesPerFrame * framesToDo;
  206. UInt32 outFlags = 0;
  207. UInt32 actualNumFrames = framesToDo;
  208. OSStatus err = MovieAudioExtractionFillBuffer (extractor, &actualNumFrames, bufferList, &outFlags);
  209. if (err != noErr)
  210. {
  211. ok = false;
  212. break;
  213. }
  214. lastSampleRead = startSampleInFile + actualNumFrames;
  215. const int samplesReceived = actualNumFrames;
  216. for (int j = numDestChannels; --j >= 0;)
  217. {
  218. if (destSamples[j] != nullptr)
  219. {
  220. const short* src = ((const short*) bufferList->mBuffers[0].mData) + j;
  221. for (int i = 0; i < samplesReceived; ++i)
  222. {
  223. destSamples[j][startOffsetInDestBuffer + i] = (*src << 16);
  224. src += numChannels;
  225. }
  226. }
  227. }
  228. startOffsetInDestBuffer += samplesReceived;
  229. startSampleInFile += samplesReceived;
  230. numSamples -= samplesReceived;
  231. if (((outFlags & kQTMovieAudioExtractionComplete) != 0 || samplesReceived == 0) && numSamples > 0)
  232. {
  233. for (int j = numDestChannels; --j >= 0;)
  234. if (destSamples[j] != nullptr)
  235. zeromem (destSamples[j] + startOffsetInDestBuffer, sizeof (int) * numSamples);
  236. break;
  237. }
  238. }
  239. detachThread();
  240. return ok;
  241. }
  242. bool ok;
  243. private:
  244. Movie movie;
  245. Media media;
  246. Track track;
  247. const int trackNum;
  248. double trackUnitsPerFrame;
  249. int samplesPerFrame;
  250. int64 lastSampleRead;
  251. Thread::ThreadID lastThreadId;
  252. MovieAudioExtractionRef extractor;
  253. AudioStreamBasicDescription inputStreamDesc;
  254. HeapBlock <AudioBufferList> bufferList;
  255. HeapBlock <char> dataBuffer;
  256. Handle dataHandle;
  257. //==============================================================================
  258. void checkThreadIsAttached()
  259. {
  260. #if JUCE_MAC
  261. if (Thread::getCurrentThreadId() != lastThreadId)
  262. EnterMoviesOnThread (0);
  263. AttachMovieToCurrentThread (movie);
  264. #endif
  265. }
  266. void detachThread()
  267. {
  268. #if JUCE_MAC
  269. DetachMovieFromCurrentThread (movie);
  270. #endif
  271. }
  272. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (QTAudioReader);
  273. };
  274. //==============================================================================
  275. QuickTimeAudioFormat::QuickTimeAudioFormat()
  276. : AudioFormat (TRANS (quickTimeFormatName), StringArray (quickTimeExtensions))
  277. {
  278. }
  279. QuickTimeAudioFormat::~QuickTimeAudioFormat()
  280. {
  281. }
  282. Array<int> QuickTimeAudioFormat::getPossibleSampleRates() { return Array<int>(); }
  283. Array<int> QuickTimeAudioFormat::getPossibleBitDepths() { return Array<int>(); }
  284. bool QuickTimeAudioFormat::canDoStereo() { return true; }
  285. bool QuickTimeAudioFormat::canDoMono() { return true; }
  286. //==============================================================================
  287. AudioFormatReader* QuickTimeAudioFormat::createReaderFor (InputStream* sourceStream,
  288. const bool deleteStreamIfOpeningFails)
  289. {
  290. ScopedPointer <QTAudioReader> r (new QTAudioReader (sourceStream, 0));
  291. if (r->ok)
  292. return r.release();
  293. if (! deleteStreamIfOpeningFails)
  294. r->input = 0;
  295. return nullptr;
  296. }
  297. AudioFormatWriter* QuickTimeAudioFormat::createWriterFor (OutputStream* /*streamToWriteTo*/,
  298. double /*sampleRateToUse*/,
  299. unsigned int /*numberOfChannels*/,
  300. int /*bitsPerSample*/,
  301. const StringPairArray& /*metadataValues*/,
  302. int /*qualityOptionIndex*/)
  303. {
  304. jassertfalse; // not yet implemented!
  305. return nullptr;
  306. }
  307. END_JUCE_NAMESPACE
  308. #endif