| Index: content/renderer/media/audio_input_device.cc
|
| ===================================================================
|
| --- content/renderer/media/audio_input_device.cc (revision 94305)
|
| +++ content/renderer/media/audio_input_device.cc (working copy)
|
| @@ -22,8 +22,13 @@
|
| callback_(callback),
|
| audio_delay_milliseconds_(0),
|
| volume_(1.0),
|
| + capture_thread_("AudioInputDevice"),
|
| + delegate_event_(false, false),
|
| + stop_event_(false, false),
|
| stream_id_(0) {
|
| filter_ = RenderThread::current()->audio_input_message_filter();
|
| + capture_thread_.Start();
|
| + capture_message_loop_proxy_ = capture_thread_.message_loop_proxy();
|
| audio_data_.reserve(channels);
|
| for (int i = 0; i < channels; ++i) {
|
| float* channel_data = new float[buffer_size];
|
| @@ -32,47 +37,22 @@
|
| }
|
|
|
| AudioInputDevice::~AudioInputDevice() {
|
| - // Make sure we have been shut down.
|
| - DCHECK_EQ(0, stream_id_);
|
| Stop();
|
| + capture_thread_.Stop();
|
| for (int i = 0; i < channels_; ++i)
|
| delete [] audio_data_[i];
|
| }
|
|
|
| bool AudioInputDevice::Start() {
|
| - // Make sure we don't call Start() more than once.
|
| - DCHECK_EQ(0, stream_id_);
|
| - if (stream_id_)
|
| - return false;
|
| -
|
| - AudioParameters params;
|
| - // TODO(henrika): add support for low-latency mode?
|
| - params.format = AudioParameters::AUDIO_PCM_LINEAR;
|
| - params.channels = channels_;
|
| - params.sample_rate = static_cast<int>(sample_rate_);
|
| - params.bits_per_sample = bits_per_sample_;
|
| - params.samples_per_packet = buffer_size_;
|
| -
|
| - ChildProcess::current()->io_message_loop()->PostTask(
|
| - FROM_HERE,
|
| - NewRunnableMethod(this, &AudioInputDevice::InitializeOnIOThread, params));
|
| -
|
| + capture_message_loop_proxy_->PostTask(FROM_HERE,
|
| + NewRunnableMethod(this, &AudioInputDevice::StartOnCaptureThread));
|
| return true;
|
| }
|
|
|
| bool AudioInputDevice::Stop() {
|
| - if (!stream_id_)
|
| - return false;
|
| -
|
| - ChildProcess::current()->io_message_loop()->PostTask(
|
| - FROM_HERE,
|
| - NewRunnableMethod(this, &AudioInputDevice::ShutDownOnIOThread));
|
| -
|
| - if (audio_thread_.get()) {
|
| - socket_->Close();
|
| - audio_thread_->Join();
|
| - }
|
| -
|
| + capture_message_loop_proxy_->PostTask(FROM_HERE,
|
| + NewRunnableMethod(this, &AudioInputDevice::ShutDownOnCaptureThread));
|
| + stop_event_.Wait();
|
| return true;
|
| }
|
|
|
| @@ -86,36 +66,30 @@
|
| return false;
|
| }
|
|
|
| -void AudioInputDevice::InitializeOnIOThread(const AudioParameters& params) {
|
| - stream_id_ = filter_->AddDelegate(this);
|
| - Send(new AudioInputHostMsg_CreateStream(stream_id_, params, true));
|
| -}
|
| -
|
| -void AudioInputDevice::StartOnIOThread() {
|
| +void AudioInputDevice::StartOnCaptureThread() {
|
| + // Make sure we don't call Start() more than once.
|
| if (stream_id_)
|
| - Send(new AudioInputHostMsg_RecordStream(stream_id_));
|
| -}
|
| -
|
| -void AudioInputDevice::ShutDownOnIOThread() {
|
| - // Make sure we don't call shutdown more than once.
|
| - if (!stream_id_)
|
| return;
|
|
|
| - filter_->RemoveDelegate(stream_id_);
|
| - Send(new AudioInputHostMsg_CloseStream(stream_id_));
|
| - stream_id_ = 0;
|
| -}
|
| + AudioParameters params;
|
| + // TODO(henrika): add support for low-latency mode?
|
| + params.format = AudioParameters::AUDIO_PCM_LINEAR;
|
| + params.channels = channels_;
|
| + params.sample_rate = static_cast<int>(sample_rate_);
|
| + params.bits_per_sample = bits_per_sample_;
|
| + params.samples_per_packet = buffer_size_;
|
|
|
| -void AudioInputDevice::SetVolumeOnIOThread(double volume) {
|
| - if (stream_id_)
|
| - Send(new AudioInputHostMsg_SetVolume(stream_id_, volume));
|
| + ChildProcess::current()->io_message_loop_proxy()->PostTask(FROM_HERE,
|
| + NewRunnableMethod(this, &AudioInputDevice::AddDelegateOnIOThread));
|
| + delegate_event_.Wait();
|
| +
|
| + SendOnIOThread(new AudioInputHostMsg_CreateStream(stream_id_, params, true));
|
| }
|
|
|
| -void AudioInputDevice::OnLowLatencyCreated(
|
| +void AudioInputDevice::StartRecordingOnCaptureThread(
|
| base::SharedMemoryHandle handle,
|
| base::SyncSocket::Handle socket_handle,
|
| uint32 length) {
|
| - DCHECK(MessageLoop::current() == ChildProcess::current()->io_message_loop());
|
| #if defined(OS_WIN)
|
| DCHECK(handle);
|
| DCHECK(socket_handle);
|
| @@ -125,26 +99,70 @@
|
| #endif
|
| DCHECK(length);
|
|
|
| + if (!stream_id_) {
|
| + base::SharedMemory::CloseHandle(handle);
|
| + base::SyncSocket socket(socket_handle);
|
| + return;
|
| + }
|
| +
|
| shared_memory_.reset(new base::SharedMemory(handle, false));
|
| shared_memory_->Map(length);
|
|
|
| socket_.reset(new base::SyncSocket(socket_handle));
|
|
|
| audio_thread_.reset(
|
| - new base::DelegateSimpleThread(this, "renderer_audio_input_thread"));
|
| + new base::DelegateSimpleThread(this, "RendererAudioInputThread"));
|
| audio_thread_->Start();
|
|
|
| - MessageLoop::current()->PostTask(
|
| - FROM_HERE,
|
| - NewRunnableMethod(this, &AudioInputDevice::StartOnIOThread));
|
| + SendOnIOThread(new AudioInputHostMsg_RecordStream(stream_id_));
|
| }
|
|
|
| +void AudioInputDevice::ShutDownOnCaptureThread() {
|
| + // Make sure we don't call shutdown more than once.
|
| + if (!stream_id_) {
|
| + stop_event_.Signal();
|
| + return;
|
| + }
|
| +
|
| + ChildProcess::current()->io_message_loop_proxy()->PostTask(FROM_HERE,
|
| + NewRunnableMethod(this, &AudioInputDevice::RemoveDelegateOnIOThread,
|
| + stream_id_));
|
| + SendOnIOThread(new AudioInputHostMsg_CloseStream(stream_id_));
|
| + stream_id_ = 0;
|
| +
|
| + if (audio_thread_.get()) {
|
| + socket_->Close();
|
| + audio_thread_->Join();
|
| + audio_thread_.reset(NULL);
|
| + }
|
| +
|
| + delegate_event_.Wait();
|
| + stop_event_.Signal();
|
| +}
|
| +
|
| +void AudioInputDevice::SetVolumeOnCaptureThread(double volume) {
|
| + if (stream_id_)
|
| + SendOnIOThread(new AudioInputHostMsg_SetVolume(stream_id_, volume));
|
| +}
|
| +
|
| +void AudioInputDevice::OnLowLatencyCreated(
|
| + base::SharedMemoryHandle handle,
|
| + base::SyncSocket::Handle socket_handle,
|
| + uint32 length) {
|
| + DCHECK(MessageLoop::current() == ChildProcess::current()->io_message_loop());
|
| + capture_message_loop_proxy_->PostTask(FROM_HERE,
|
| + NewRunnableMethod(this, &AudioInputDevice::StartRecordingOnCaptureThread,
|
| + handle, socket_handle, length));
|
| +}
|
| +
|
| void AudioInputDevice::OnVolume(double volume) {
|
| NOTIMPLEMENTED();
|
| }
|
|
|
| -void AudioInputDevice::Send(IPC::Message* message) {
|
| - filter_->Send(message);
|
| +void AudioInputDevice::SendOnIOThread(IPC::Message* message) {
|
| + ChildProcess::current()->io_message_loop_proxy()->PostTask(FROM_HERE,
|
| + NewRunnableMethod(filter_.get(),
|
| + &AudioInputMessageFilter::Send, message));
|
| }
|
|
|
| // Our audio thread runs here. We receive captured audio samples on
|
| @@ -197,3 +215,15 @@
|
| number_of_frames,
|
| audio_delay_milliseconds_);
|
| }
|
| +
|
| +void AudioInputDevice::AddDelegateOnIOThread() {
|
| + stream_id_ = filter_->AddDelegate(this);
|
| + VLOG(1) << "add stream_id = " << stream_id_;
|
| + delegate_event_.Signal();
|
| +}
|
| +
|
| +void AudioInputDevice::RemoveDelegateOnIOThread(int32 stream_id) {
|
| + filter_->RemoveDelegate(stream_id);
|
| + VLOG(1) << "remove stream_id = " << stream_id;
|
| + delegate_event_.Signal();
|
| +}
|
|
|