blob: 16b5457f5161a2d76808d057e9d519903945a492 [file] [log] [blame]
// 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/fake_audio_input_stream.h"
#include "base/atomicops.h"
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/command_line.h"
#include "base/files/file_path.h"
#include "base/memory/ptr_util.h"
#include "base/single_thread_task_runner.h"
#include "base/strings/string_split.h"
#include "base/time/time.h"
#include "media/audio/audio_manager_base.h"
#include "media/audio/simple_sources.h"
#include "media/base/audio_bus.h"
#include "media/base/media_switches.h"
namespace media {
namespace {
base::subtle::AtomicWord g_fake_input_streams_are_muted = 0;
}
AudioInputStream* FakeAudioInputStream::MakeFakeStream(
AudioManagerBase* manager,
const AudioParameters& params) {
return new FakeAudioInputStream(manager, params);
}
FakeAudioInputStream::FakeAudioInputStream(AudioManagerBase* manager,
const AudioParameters& params)
: audio_manager_(manager),
callback_(NULL),
fake_audio_worker_(manager->GetWorkerTaskRunner(), params),
params_(params),
audio_bus_(AudioBus::Create(params)) {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
}
FakeAudioInputStream::~FakeAudioInputStream() {
DCHECK(!callback_);
}
bool FakeAudioInputStream::Open() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
audio_bus_->Zero();
return true;
}
void FakeAudioInputStream::Start(AudioInputCallback* callback) {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
callback_ = callback;
fake_audio_worker_.Start(base::Bind(
&FakeAudioInputStream::ReadAudioFromSource, base::Unretained(this)));
}
void FakeAudioInputStream::Stop() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
fake_audio_worker_.Stop();
callback_ = NULL;
}
void FakeAudioInputStream::Close() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
DCHECK(!callback_);
audio_manager_->ReleaseInputStream(this);
}
double FakeAudioInputStream::GetMaxVolume() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
return 1.0;
}
void FakeAudioInputStream::SetVolume(double volume) {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
}
double FakeAudioInputStream::GetVolume() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
return 1.0;
}
bool FakeAudioInputStream::IsMuted() {
DCHECK(audio_manager_->GetTaskRunner()->BelongsToCurrentThread());
return base::subtle::NoBarrier_Load(&g_fake_input_streams_are_muted) != 0;
}
bool FakeAudioInputStream::SetAutomaticGainControl(bool enabled) {
return false;
}
bool FakeAudioInputStream::GetAutomaticGainControl() {
return false;
}
void FakeAudioInputStream::SetOutputDeviceForAec(
const std::string& output_device_id) {
// Not supported. Do nothing.
}
void FakeAudioInputStream::ReadAudioFromSource() {
DCHECK(audio_manager_->GetWorkerTaskRunner()->BelongsToCurrentThread());
DCHECK(callback_);
if (!audio_source_)
audio_source_ = ChooseSource();
audio_source_->OnMoreData(base::TimeDelta(), base::TimeTicks::Now(), 0,
audio_bus_.get());
callback_->OnData(audio_bus_.get(), base::TimeTicks::Now(), 1.0);
}
using AudioSourceCallback = AudioOutputStream::AudioSourceCallback;
std::unique_ptr<AudioSourceCallback> FakeAudioInputStream::ChooseSource() {
DCHECK(audio_manager_->GetWorkerTaskRunner()->BelongsToCurrentThread());
if (base::CommandLine::ForCurrentProcess()->HasSwitch(
switches::kUseFileForFakeAudioCapture)) {
base::CommandLine::StringType switch_value =
base::CommandLine::ForCurrentProcess()->GetSwitchValueNative(
switches::kUseFileForFakeAudioCapture);
base::CommandLine::StringVector parameters =
base::SplitString(switch_value, FILE_PATH_LITERAL("%"),
base::TRIM_WHITESPACE, base::SPLIT_WANT_NONEMPTY);
CHECK(parameters.size() > 0) << "You must pass <file>[%noloop] to --"
<< switches::kUseFileForFakeAudioCapture
<< ".";
base::FilePath path_to_wav_file = base::FilePath(parameters[0]);
bool looping = true;
if (parameters.size() == 2) {
CHECK(parameters[1] == FILE_PATH_LITERAL("noloop"))
<< "Unknown parameter " << parameters[1] << " to "
<< switches::kUseFileForFakeAudioCapture << ".";
looping = false;
}
return std::make_unique<FileSource>(params_, path_to_wav_file, looping);
}
return std::make_unique<BeepingSource>(params_);
}
void FakeAudioInputStream::BeepOnce() {
BeepingSource::BeepOnce();
}
void FakeAudioInputStream::SetGlobalMutedState(bool is_muted) {
base::subtle::NoBarrier_Store(&g_fake_input_streams_are_muted,
(is_muted ? 1 : 0));
}
} // namespace media