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
media / audio / cras / cras_unified.h [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// Creates a unified stream based on the cras (ChromeOS audio server) interface.
//
// CrasUnifiedStream object is *not* thread-safe and should only be used
// from the audio thread.
#ifndef MEDIA_AUDIO_CRAS_CRAS_UNIFIED_H_
#define MEDIA_AUDIO_CRAS_CRAS_UNIFIED_H_
#include <cras_client.h>
#include <stddef.h>
#include <stdint.h>
#include <memory>
#include "base/compiler_specific.h"
#include "media/audio/audio_io.h"
#include "media/audio/cras/audio_manager_cras_base.h"
#include "media/audio/system_glitch_reporter.h"
#include "media/base/amplitude_peak_detector.h"
#include "media/base/audio_parameters.h"
namespace media {
// Implementation of AudioOuputStream for Chrome OS using the Chrome OS audio
// server.
// TODO(dgreid): This class is used for only output, either remove all the
// relevant input code and change the class to CrasOutputStream or merge
// cras_input.cc into this unified implementation.
class MEDIA_EXPORT CrasUnifiedStream : public AudioOutputStream {
public:
// The ctor takes all the usual parameters, plus |manager| which is the
// audio manager who is creating this object.
CrasUnifiedStream(const AudioParameters& params,
AudioManagerCrasBase* manager,
const std::string& device_id,
const AudioManager::LogCallback& log_callback);
CrasUnifiedStream(const CrasUnifiedStream&) = delete;
CrasUnifiedStream& operator=(const CrasUnifiedStream&) = delete;
// The dtor is typically called by the AudioManager only and it is usually
// triggered by calling AudioUnifiedStream::Close().
~CrasUnifiedStream() override;
// Implementation of AudioOutputStream.
bool Open() override;
void Close() override;
void Flush() override;
void Start(AudioSourceCallback* callback) override;
void Stop() override;
void SetVolume(double volume) override;
void GetVolume(double* volume) override;
private:
// Handles captured audio and fills the output with audio to be played.
static int UnifiedCallback(struct libcras_stream_cb_data* data);
// Handles notification that there was an error with the playback stream.
static int StreamError(cras_client* client,
cras_stream_id_t stream_id,
int err,
void* arg);
// Writes audio for a playback stream.
uint32_t WriteAudio(size_t frames,
uint8_t* buffer,
const timespec* latency_ts);
// Deals with an error that occurred in the stream. Called from
// StreamError().
void NotifyStreamError(int err);
// There is 3 main reasons for output audio glitches.
// 1. Client is too slow to reply and audio data is not delivered to the
// buffer on time.
// 2. Audio thread woke up late.
// 3. Issues with the driver
// All 3 issues result in there not being enough valid playback data and zero
// frames are filled to avoid playing noise. The duration of the zero frames
// filled are all calculated using |underrun_duration|.
void CalculateAudioGlitches(base::TimeDelta underrun_duration);
// Called from the dtor and when the stream is reset.
void ReportAndResetStats();
// The client used to communicate with the audio server.
struct libcras_client* client_ = NULL;
// ID of the playing stream.
cras_stream_id_t stream_id_ = 0;
// PCM parameters for the stream.
AudioParameters params_;
// True if stream is playing.
bool is_playing_ = 0;
// Volume level from 0.0 to 1.0.
float volume_ = 1.0;
// Audio manager that created us. Used to report that we've been closed.
AudioManagerCrasBase* const manager_;
// Callback to get audio samples.
AudioSourceCallback* source_callback_ = NULL;
// Container for exchanging data with AudioSourceCallback::OnMoreData().
const std::unique_ptr<AudioBus> output_bus_;
// Direciton of the stream.
const CRAS_STREAM_DIRECTION stream_direction_ = CRAS_STREAM_OUTPUT;
// Index of the CRAS device to stream output to.
const int pin_device_;
// Used to aggregate and report glitch metrics to UMA (periodically) and to
// text logs (when a stream ends).
SystemGlitchReporter glitch_reporter_;
// Callback to send statistics info.
const AudioManager::LogCallback log_callback_;
// Contains the duration of the underrun frames passed in from the previous
// callback.
// Underrun occurs when the there isn't enough valid playback data. When this
// happens, zero frames have to be filled in. The duration of the underrun is
// the duration of the zero frames filled in.
base::TimeDelta last_underrun_duration_;
// Used to accumulate glitch info to report to |source_callback_|
AudioGlitchInfo::Accumulator glitch_info_accumulator_;
std::unique_ptr<AmplitudePeakDetector> peak_detector_;
};
} // namespace media
#endif // MEDIA_AUDIO_CRAS_CRAS_UNIFIED_H_