1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
|
// Copyright 2017 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/audio_system_impl.h"
#include <memory>
#include <string>
#include <utility>
#include "base/memory/ptr_util.h"
#include "base/single_thread_task_runner.h"
#include "base/task_runner_util.h"
#include "media/audio/audio_device_description.h"
#include "media/audio/audio_manager.h"
#include "media/base/bind_to_current_loop.h"
// Using base::Unretained for |audio_manager_| is safe since it is deleted after
// its task runner, and AudioSystemImpl is deleted on the UI thread after the IO
// thread has been stopped and before |audio_manager_| deletion is scheduled.
namespace media {
AudioSystemImpl::AudioSystemImpl(AudioManager* audio_manager)
: audio_manager_(audio_manager) {
DCHECK(audio_manager_);
AudioSystem::SetInstance(this);
}
AudioSystemImpl::~AudioSystemImpl() {
AudioSystem::ClearInstance(this);
}
// static
std::unique_ptr<AudioSystem> AudioSystemImpl::Create(
AudioManager* audio_manager) {
return base::WrapUnique(new AudioSystemImpl(audio_manager));
}
void AudioSystemImpl::GetInputStreamParameters(
const std::string& device_id,
OnAudioParamsCallback on_params_cb) const {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(FROM_HERE,
base::BindOnce(std::move(on_params_cb),
GetInputParametersOnDeviceThread(
audio_manager_, device_id)));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioSystemImpl::GetInputParametersOnDeviceThread,
base::Unretained(audio_manager_), device_id),
std::move(on_params_cb));
}
void AudioSystemImpl::GetOutputStreamParameters(
const std::string& device_id,
OnAudioParamsCallback on_params_cb) const {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(FROM_HERE,
base::BindOnce(std::move(on_params_cb),
GetOutputParametersOnDeviceThread(
audio_manager_, device_id)));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioSystemImpl::GetOutputParametersOnDeviceThread,
base::Unretained(audio_manager_), device_id),
std::move(on_params_cb));
}
void AudioSystemImpl::HasInputDevices(OnBoolCallback on_has_devices_cb) const {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(
FROM_HERE, base::BindOnce(std::move(on_has_devices_cb),
audio_manager_->HasAudioInputDevices()));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioManager::HasAudioInputDevices,
base::Unretained(audio_manager_)),
std::move(on_has_devices_cb));
}
void AudioSystemImpl::HasOutputDevices(OnBoolCallback on_has_devices_cb) const {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(
FROM_HERE, base::BindOnce(std::move(on_has_devices_cb),
audio_manager_->HasAudioOutputDevices()));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioManager::HasAudioOutputDevices,
base::Unretained(audio_manager_)),
std::move(on_has_devices_cb));
}
void AudioSystemImpl::GetDeviceDescriptions(
OnDeviceDescriptionsCallback on_descriptions_cb,
bool for_input) {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(
FROM_HERE,
base::BindOnce(std::move(on_descriptions_cb),
base::Passed(GetDeviceDescriptionsOnDeviceThread(
audio_manager_, for_input))));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioSystemImpl::GetDeviceDescriptionsOnDeviceThread,
base::Unretained(audio_manager_), for_input),
std::move(on_descriptions_cb));
}
void AudioSystemImpl::GetAssociatedOutputDeviceID(
const std::string& input_device_id,
OnDeviceIdCallback on_device_id_cb) {
if (GetTaskRunner()->BelongsToCurrentThread()) {
GetTaskRunner()->PostTask(
FROM_HERE, base::BindOnce(std::move(on_device_id_cb),
audio_manager_->GetAssociatedOutputDeviceID(
input_device_id)));
return;
}
base::PostTaskAndReplyWithResult(
GetTaskRunner(), FROM_HERE,
base::BindOnce(&AudioManager::GetAssociatedOutputDeviceID,
base::Unretained(audio_manager_), input_device_id),
std::move(on_device_id_cb));
}
void AudioSystemImpl::GetInputDeviceInfo(
const std::string& input_device_id,
OnInputDeviceInfoCallback on_input_device_info_cb) {
// No need to bind |on_input_device_info_cb| to the current loop if we are on
// the audio thread. However, the client still expect to receive the reply
// asynchronously, so we always post GetInputDeviceInfoOnDeviceThread(), which
// will syncronously call the (bound to current loop or not) callback.
GetTaskRunner()->PostTask(
FROM_HERE,
base::BindOnce(
&AudioSystemImpl::GetInputDeviceInfoOnDeviceThread,
base::Unretained(audio_manager_), input_device_id,
GetTaskRunner()->BelongsToCurrentThread()
? std::move(on_input_device_info_cb)
: media::BindToCurrentLoop(std::move(on_input_device_info_cb))));
}
base::SingleThreadTaskRunner* AudioSystemImpl::GetTaskRunner() const {
return audio_manager_->GetTaskRunner();
}
// static
AudioParameters AudioSystemImpl::GetInputParametersOnDeviceThread(
AudioManager* audio_manager,
const std::string& device_id) {
DCHECK(audio_manager->GetTaskRunner()->BelongsToCurrentThread());
// TODO(olka): remove this when AudioManager::GetInputStreamParameters()
// returns invalid parameters if the device is not found.
if (!audio_manager->HasAudioInputDevices())
return AudioParameters();
return audio_manager->GetInputStreamParameters(device_id);
}
// static
AudioParameters AudioSystemImpl::GetOutputParametersOnDeviceThread(
AudioManager* audio_manager,
const std::string& device_id) {
DCHECK(audio_manager->GetTaskRunner()->BelongsToCurrentThread());
// TODO(olka): remove this when
// AudioManager::Get[Default]OutputStreamParameters() returns invalid
// parameters if the device is not found.
if (!audio_manager->HasAudioOutputDevices())
return AudioParameters();
return media::AudioDeviceDescription::IsDefaultDevice(device_id)
? audio_manager->GetDefaultOutputStreamParameters()
: audio_manager->GetOutputStreamParameters(device_id);
}
// static
AudioDeviceDescriptions AudioSystemImpl::GetDeviceDescriptionsOnDeviceThread(
AudioManager* audio_manager,
bool for_input) {
DCHECK(audio_manager->GetTaskRunner()->BelongsToCurrentThread());
AudioDeviceDescriptions descriptions;
if (for_input)
audio_manager->GetAudioInputDeviceDescriptions(&descriptions);
else
audio_manager->GetAudioOutputDeviceDescriptions(&descriptions);
return descriptions;
}
// static
void AudioSystemImpl::GetInputDeviceInfoOnDeviceThread(
AudioManager* audio_manager,
const std::string& input_device_id,
AudioSystem::OnInputDeviceInfoCallback on_input_device_info_cb) {
DCHECK(audio_manager->GetTaskRunner()->BelongsToCurrentThread());
const std::string associated_output_device_id =
audio_manager->GetAssociatedOutputDeviceID(input_device_id);
std::move(on_input_device_info_cb)
.Run(GetInputParametersOnDeviceThread(audio_manager, input_device_id),
associated_output_device_id.empty()
? AudioParameters()
: GetOutputParametersOnDeviceThread(audio_manager,
associated_output_device_id),
associated_output_device_id);
}
} // namespace media
|