Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(114)

Side by Side Diff: media/audio/virtual_audio_input_stream.cc

Issue 11298006: Browser-wide audio mirroring for TabCapture API (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Fix more nits, disable for ios Created 8 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "media/audio/virtual_audio_input_stream.h"
6
7 #include <algorithm>
8 #include <utility>
9
10 #include "base/bind.h"
11 #include "base/message_loop.h"
12 #include "media/audio/virtual_audio_output_stream.h"
13
14 namespace media {
15
16 // LoopbackAudioConverter works similar to AudioConverter and converts input
17 // streams to different audio parameters. Then, the LoopbackAudioConverter can
18 // be used as an input to another AudioConverter. This allows us to
19 // use converted audio from AudioOutputStreams as input to an AudioConverter.
20 // For example, this allows converting multiple streams into a common format and
21 // using the converted audio as input to another AudioConverter (i.e. a mixer).
22 class LoopbackAudioConverter : public AudioConverter::InputCallback {
23 public:
24 LoopbackAudioConverter(const AudioParameters& input_params,
25 const AudioParameters& output_params)
26 : audio_converter_(input_params, output_params, false) {}
27
28 virtual ~LoopbackAudioConverter() {}
29
30 void AddInput(AudioConverter::InputCallback* input) {
31 audio_converter_.AddInput(input);
32 }
33
34 void RemoveInput(AudioConverter::InputCallback* input) {
35 audio_converter_.RemoveInput(input);
36 }
37
38 private:
39 virtual double ProvideInput(AudioBus* audio_bus,
40 base::TimeDelta buffer_delay) OVERRIDE {
41 audio_converter_.Convert(audio_bus);
42 return 1.0;
43 }
44
45 AudioConverter audio_converter_;
46
47 DISALLOW_COPY_AND_ASSIGN(LoopbackAudioConverter);
48 };
49
50 VirtualAudioInputStream* VirtualAudioInputStream::MakeStream(
51 AudioManagerBase* manager, const AudioParameters& params) {
52 return new VirtualAudioInputStream(manager, params);
53 }
54
55 VirtualAudioInputStream::VirtualAudioInputStream(AudioManagerBase* manager,
56 const AudioParameters& params)
57 : audio_manager_(manager),
58 callback_(NULL),
59 buffer_duration_ms_(base::TimeDelta::FromMilliseconds(
60 params.frames_per_buffer() * base::Time::kMillisecondsPerSecond /
61 static_cast<float>(params.sample_rate()))),
62 buffer_(new uint8[params.GetBytesPerBuffer()]),
63 params_(params),
64 audio_bus_(AudioBus::Create(params_)),
65 mixer_(params_, params_, false),
66 num_attached_outputs_streams_(0) {
67 }
68
69 VirtualAudioInputStream::~VirtualAudioInputStream() {
70 for (AudioConvertersMap::iterator it = converters_.begin();
71 it != converters_.end(); ++it)
72 delete it->second;
73
74 DCHECK_EQ(0, num_attached_outputs_streams_);
75 }
76
77 bool VirtualAudioInputStream::Open() {
78 memset(buffer_.get(), 0, params_.GetBytesPerBuffer());
79 return true;
80 }
81
82 void VirtualAudioInputStream::Start(AudioInputCallback* callback) {
83 DCHECK(CalledOnAudioThread());
84 callback_ = callback;
85 on_more_data_cb_.Reset(base::Bind(&VirtualAudioInputStream::ReadAudio,
DaleCurtis 2012/12/03 19:27:28 Bad wrapping, all params with 4 space indent or al
justinlin 2012/12/03 21:18:42 Done. OK, I think I finally get this. If it's 4 sp
86 base::Unretained(this)));
87 audio_manager_->GetMessageLoop()->PostTask(FROM_HERE,
DaleCurtis 2012/12/03 19:27:28 Ditto.
justinlin 2012/12/03 21:18:42 Done.
88 on_more_data_cb_.callback());
89 }
90
91 void VirtualAudioInputStream::Stop() {
92 DCHECK(CalledOnAudioThread());
93 on_more_data_cb_.Cancel();
94 }
95
96 void VirtualAudioInputStream::AddOutputStream(
97 VirtualAudioOutputStream* stream, const AudioParameters& output_params) {
98 DCHECK(CalledOnAudioThread());
99
100 AudioConvertersMap::iterator converter = converters_.find(output_params);
101 if (converter == converters_.end()) {
102 std::pair<AudioConvertersMap::iterator, bool> result = converters_.insert(
103 std::make_pair(output_params, new LoopbackAudioConverter(
104 output_params, params_)));
105 converter = result.first;
106
107 // Add to main mixer if we just added a new AudioTransform.
108 mixer_.AddInput(converter->second);
109 }
110 converter->second->AddInput(stream);
111 ++num_attached_outputs_streams_;
112 }
113
114 void VirtualAudioInputStream::RemoveOutputStream(
115 VirtualAudioOutputStream* stream, const AudioParameters& output_params) {
116 DCHECK(CalledOnAudioThread());
117
118 DCHECK(converters_.find(output_params) != converters_.end());
119 converters_[output_params]->RemoveInput(stream);
120
121 --num_attached_outputs_streams_;
122 }
123
124 void VirtualAudioInputStream::ReadAudio() {
125 DCHECK(CalledOnAudioThread());
126 DCHECK(callback_);
127
128 mixer_.Convert(audio_bus_.get());
129 audio_bus_->ToInterleaved(params_.frames_per_buffer(),
130 params_.bits_per_sample() / 8,
131 buffer_.get());
132
133 callback_->OnData(this,
134 buffer_.get(),
135 params_.GetBytesPerBuffer(),
136 params_.GetBytesPerBuffer(),
137 1.0);
138
139 MessageLoop* message_loop = MessageLoop::current();
DaleCurtis 2012/12/03 19:27:28 Yuck. Either use thread checker, or pass the messa
justinlin 2012/12/03 21:18:42 Done. ThreadChecker doesn't exactly work because i
140 if (message_loop) {
141 message_loop->PostDelayedTask(FROM_HERE,
142 on_more_data_cb_.callback(),
143 buffer_duration_ms_);
144 }
145 }
146
147 void VirtualAudioInputStream::Close() {
148 DCHECK(CalledOnAudioThread());
149 if (callback_) {
150 DCHECK(on_more_data_cb_.IsCancelled());
151 callback_->OnClose(this);
152 callback_ = NULL;
153 }
154 audio_manager_->ReleaseInputStream(this);
155 }
156
157 double VirtualAudioInputStream::GetMaxVolume() {
158 return 1.0;
159 }
160
161 void VirtualAudioInputStream::SetVolume(double volume) {}
162
163 double VirtualAudioInputStream::GetVolume() {
164 return 1.0;
165 }
166
167 void VirtualAudioInputStream::SetAutomaticGainControl(bool enabled) {}
168
169 bool VirtualAudioInputStream::GetAutomaticGainControl() {
170 return false;
171 }
172
173 bool VirtualAudioInputStream::CalledOnAudioThread() {
174 return audio_manager_->GetMessageLoop()->BelongsToCurrentThread();
175 }
176
177 } // namespace media
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698