// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "media/audio/mac/audio_input_mac.h" #include <CoreServices/CoreServices.h> #include "base/basictypes.h" #include "base/logging.h" #include "base/mac/mac_logging.h" #include "media/audio/audio_manager_base.h" namespace media { PCMQueueInAudioInputStream::PCMQueueInAudioInputStream( AudioManagerBase* manager, const AudioParameters& params) : manager_(manager), callback_(NULL), audio_queue_(NULL), buffer_size_bytes_(0), started_(false) { // We must have a manager. DCHECK(manager_); // A frame is one sample across all channels. In interleaved audio the per // frame fields identify the set of n |channels|. In uncompressed audio, a // packet is always one frame. format_.mSampleRate = params.sample_rate(); format_.mFormatID = kAudioFormatLinearPCM; format_.mFormatFlags = kLinearPCMFormatFlagIsPacked | kLinearPCMFormatFlagIsSignedInteger; format_.mBitsPerChannel = params.bits_per_sample(); format_.mChannelsPerFrame = params.channels(); format_.mFramesPerPacket = 1; format_.mBytesPerPacket = (params.bits_per_sample() * params.channels()) / 8; format_.mBytesPerFrame = format_.mBytesPerPacket; format_.mReserved = 0; buffer_size_bytes_ = params.GetBytesPerBuffer(); } PCMQueueInAudioInputStream::~PCMQueueInAudioInputStream() { DCHECK(!callback_); DCHECK(!audio_queue_); } bool PCMQueueInAudioInputStream::Open() { OSStatus err = AudioQueueNewInput(&format_, &HandleInputBufferStatic, this, NULL, // Use OS CFRunLoop for |callback| kCFRunLoopCommonModes, 0, // Reserved &audio_queue_); if (err != noErr) { HandleError(err); return false; } return SetupBuffers(); } void PCMQueueInAudioInputStream::Start(AudioInputCallback* callback) { DCHECK(callback); DLOG_IF(ERROR, !audio_queue_) << "Open() has not been called successfully"; if (callback_ || !audio_queue_) return; callback_ = callback; OSStatus err = AudioQueueStart(audio_queue_, NULL); if (err != noErr) { HandleError(err); } else { started_ = true; } } void PCMQueueInAudioInputStream::Stop() { if (!audio_queue_ || !started_) return; // We request a synchronous stop, so the next call can take some time. In // the windows implementation we block here as well. OSStatus err = AudioQueueStop(audio_queue_, true); if (err != noErr) HandleError(err); started_ = false; } void PCMQueueInAudioInputStream::Close() { // It is valid to call Close() before calling Open() or Start(), thus // |audio_queue_| and |callback_| might be NULL. if (audio_queue_) { OSStatus err = AudioQueueDispose(audio_queue_, true); audio_queue_ = NULL; if (err != noErr) HandleError(err); } if (callback_) { callback_->OnClose(this); callback_ = NULL; } manager_->ReleaseInputStream(this); // CARE: This object may now be destroyed. } double PCMQueueInAudioInputStream::GetMaxVolume() { NOTREACHED() << "Only supported for low-latency mode."; return 0.0; } void PCMQueueInAudioInputStream::SetVolume(double volume) { NOTREACHED() << "Only supported for low-latency mode."; } double PCMQueueInAudioInputStream::GetVolume() { NOTREACHED() << "Only supported for low-latency mode."; return 0.0; } void PCMQueueInAudioInputStream::SetAutomaticGainControl(bool enabled) { NOTREACHED() << "Only supported for low-latency mode."; } bool PCMQueueInAudioInputStream::GetAutomaticGainControl() { NOTREACHED() << "Only supported for low-latency mode."; return false; } void PCMQueueInAudioInputStream::HandleError(OSStatus err) { if (callback_) callback_->OnError(this); // This point should never be reached. OSSTATUS_DCHECK(0, err); } bool PCMQueueInAudioInputStream::SetupBuffers() { DCHECK(buffer_size_bytes_); for (int i = 0; i < kNumberBuffers; ++i) { AudioQueueBufferRef buffer; OSStatus err = AudioQueueAllocateBuffer(audio_queue_, buffer_size_bytes_, &buffer); if (err == noErr) err = QueueNextBuffer(buffer); if (err != noErr) { HandleError(err); return false; } // |buffer| will automatically be freed when |audio_queue_| is released. } return true; } OSStatus PCMQueueInAudioInputStream::QueueNextBuffer( AudioQueueBufferRef audio_buffer) { // Only the first 2 params are needed for recording. return AudioQueueEnqueueBuffer(audio_queue_, audio_buffer, 0, NULL); } // static void PCMQueueInAudioInputStream::HandleInputBufferStatic( void* data, AudioQueueRef audio_queue, AudioQueueBufferRef audio_buffer, const AudioTimeStamp* start_time, UInt32 num_packets, const AudioStreamPacketDescription* desc) { reinterpret_cast<PCMQueueInAudioInputStream*>(data)-> HandleInputBuffer(audio_queue, audio_buffer, start_time, num_packets, desc); } void PCMQueueInAudioInputStream::HandleInputBuffer( AudioQueueRef audio_queue, AudioQueueBufferRef audio_buffer, const AudioTimeStamp* start_time, UInt32 num_packets, const AudioStreamPacketDescription* packet_desc) { DCHECK_EQ(audio_queue_, audio_queue); DCHECK(audio_buffer->mAudioData); if (!callback_) { // This can happen if Stop() was called without start. DCHECK_EQ(0U, audio_buffer->mAudioDataByteSize); return; } if (audio_buffer->mAudioDataByteSize) { // The AudioQueue API may use a large internal buffer and repeatedly call us // back to back once that internal buffer is filled. When this happens the // renderer client does not have enough time to read data back from the // shared memory before the next write comes along. If HandleInputBuffer() // is called too frequently, Sleep() at least 5ms to ensure the shared // memory doesn't get trampled. // TODO(dalecurtis): This is a HACK. Long term the AudioQueue path is going // away in favor of the AudioUnit based AUAudioInputStream(). Tracked by // http://crbug.com/161383. base::TimeDelta elapsed = base::TimeTicks::Now() - last_fill_; const base::TimeDelta kMinDelay = base::TimeDelta::FromMilliseconds(5); if (elapsed < kMinDelay) base::PlatformThread::Sleep(kMinDelay - elapsed); callback_->OnData(this, reinterpret_cast<const uint8*>(audio_buffer->mAudioData), audio_buffer->mAudioDataByteSize, audio_buffer->mAudioDataByteSize, 0.0); last_fill_ = base::TimeTicks::Now(); } // Recycle the buffer. OSStatus err = QueueNextBuffer(audio_buffer); if (err != noErr) { if (err == kAudioQueueErr_EnqueueDuringReset) { // This is the error you get if you try to enqueue a buffer and the // queue has been closed. Not really a problem if indeed the queue // has been closed. // TODO(joth): PCMQueueOutAudioOutputStream uses callback_ to provide an // extra guard for this situation, but it seems to introduce more // complications than it solves (memory barrier issues accessing it from // multiple threads, looses the means to indicate OnClosed to client). // Should determine if we need to do something equivalent here. return; } HandleError(err); } } } // namespace media