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
media / audio / aecdump_recording_manager.cc [blame]
// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/audio/aecdump_recording_manager.h"
#include "base/functional/bind.h"
#include "base/functional/callback_helpers.h"
#include "base/task/single_thread_task_runner.h"
#include "base/task/thread_pool.h"
#include "media/audio/audio_manager.h"
namespace media {
namespace {
void CloseFileWithoutBlocking(base::File file) {
// Post as a low-priority task to a thread pool to avoid blocking the
// current thread.
base::ThreadPool::PostTask(FROM_HERE,
{base::TaskPriority::LOWEST, base::MayBlock()},
base::DoNothingWithBoundArgs(std::move(file)));
}
} // namespace
AecdumpRecordingManager::AecdumpRecordingManager(
scoped_refptr<base::SingleThreadTaskRunner> task_runner)
: task_runner_(std::move(task_runner)) {
DCHECK(task_runner_->BelongsToCurrentThread());
}
AecdumpRecordingManager::~AecdumpRecordingManager() {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK_EQ(aecdump_recording_sources_.size(), 0u);
DCHECK(!IsDebugRecordingEnabled());
}
void AecdumpRecordingManager::EnableDebugRecording(
CreateFileCallback create_file_callback) {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK(create_file_callback);
DCHECK(!create_file_callback_);
create_file_callback_ = std::move(create_file_callback);
for (const auto& it : aecdump_recording_sources_) {
AecdumpRecordingSource* source = it.first;
uint32_t id = it.second;
create_file_callback_.Run(
id, /*reply_callback=*/base::BindOnce(
&StartRecordingIfValidPointer, weak_factory_.GetWeakPtr(), source));
}
}
void AecdumpRecordingManager::StartRecording(AecdumpRecordingSource* source,
base::File file) {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK(IsDebugRecordingEnabled());
if (aecdump_recording_sources_.find(source) !=
aecdump_recording_sources_.end()) {
source->StartAecdump(std::move(file));
return;
}
// The source is deregistered and we are responsible for closing the file.
CloseFileWithoutBlocking(std::move(file));
}
void AecdumpRecordingManager::DisableDebugRecording() {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK(create_file_callback_);
for (const auto& it : aecdump_recording_sources_) {
AecdumpRecordingSource* source = it.first;
source->StopAecdump();
}
create_file_callback_.Reset();
// By invalidating weak pointers, we cancel any recordings in the process of
// being started (file creation).
weak_factory_.InvalidateWeakPtrs();
}
void AecdumpRecordingManager::RegisterAecdumpSource(
AecdumpRecordingSource* source) {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK(aecdump_recording_sources_.find(source) ==
aecdump_recording_sources_.end());
const uint32_t id = recording_id_counter_++;
if (IsDebugRecordingEnabled()) {
create_file_callback_.Run(id, /*reply_callback=*/base::BindOnce(
&AecdumpRecordingManager::StartRecording,
weak_factory_.GetWeakPtr(), source));
}
aecdump_recording_sources_[source] = id;
}
void AecdumpRecordingManager::DeregisterAecdumpSource(
AecdumpRecordingSource* source) {
DCHECK(task_runner_->BelongsToCurrentThread());
DCHECK(aecdump_recording_sources_.find(source) !=
aecdump_recording_sources_.end());
if (IsDebugRecordingEnabled()) {
source->StopAecdump();
}
aecdump_recording_sources_.erase(source);
}
bool AecdumpRecordingManager::IsDebugRecordingEnabled() const {
DCHECK(task_runner_->BelongsToCurrentThread());
return !create_file_callback_.is_null();
}
// static
void AecdumpRecordingManager::StartRecordingIfValidPointer(
base::WeakPtr<AecdumpRecordingManager> manager,
AecdumpRecordingSource* source,
base::File file) {
if (manager) {
manager->StartRecording(source, std::move(file));
return;
}
// Recording has been stopped and we are responsible for closing the file.
CloseFileWithoutBlocking(std::move(file));
}
} // namespace media