// 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.
//
// Implementation notes:
//
// - It is recommended to first acquire the native sample rate of the default
// output device and then use the same rate when creating this object.
// Use AudioManagerMac::HardwareSampleRate() to retrieve the sample rate.
// - Calling Close() also leads to self destruction.
// - The latency consists of two parts:
// 1) Hardware latency, which includes Audio Unit latency, audio device
// latency;
// 2) The delay between the moment getting the callback and the scheduled time
// stamp that tells when the data is going to be played out.
//
#ifndef MEDIA_AUDIO_APPLE_AUDIO_AUHAL_H_
#define MEDIA_AUDIO_APPLE_AUDIO_AUHAL_H_
#include <AudioUnit/AudioUnit.h>
#include <stddef.h>
#include <stdint.h>
#include <atomic>
#include <memory>
#include "base/cancelable_callback.h"
#include "base/compiler_specific.h"
#include "base/memory/raw_ptr.h"
#include "base/synchronization/lock.h"
#include "base/threading/thread_checker.h"
#include "base/time/time.h"
#include "build/build_config.h"
#include "media/audio/apple/audio_manager_apple.h"
#include "media/audio/audio_io.h"
#include "media/audio/audio_manager.h"
#include "media/audio/apple/scoped_audio_unit.h"
#include "media/audio/system_glitch_reporter.h"
#include "media/base/amplitude_peak_detector.h"
#include "media/base/audio_parameters.h"
#if BUILDFLAG(IS_MAC)
#include <CoreAudio/CoreAudio.h>
#endif
namespace media {
class AudioPullFifo;
// Implementation of AudioOutputStream for Apple using the
// AUHAL Audio Unit present in OS 10.4 and later.
// It is useful for low-latency output.
//
// Overview of operation:
// 1) An object of AUHALStream is created by the AudioManager factory on the
// object's main thread via audio_man->MakeAudioStream(). Calls to the
// control routines (Open/Close/Start/Stop), must be made on this thread.
// 2) Next Open() will be called. At that point the underlying AUHAL Audio Unit
// is created and configured to use the |device|.
// 3) Then Start(source) is called and the device is started which creates its
// own thread (or uses an existing background thread) on which the AUHAL's
// callback will periodically ask for more data as buffers are being
// consumed.
// Note that all AUHAL instances receive callbacks on that very same
// thread, so avoid any contention in the callback as to not cause delays for
// other instances.
// 4) At some point Stop() will be called, which we handle by stopping the
// output Audio Unit.
// 6) Lastly, Close() will be called where we cleanup and notify the audio
// manager, which will delete the object.
// TODO(tommi): Since the callback audio thread is shared for all instances of
// AUHALStream, one stream blocking, can cause others to be delayed. Several
// occurrences of this can cause a buildup of delay which forces the OS
// to skip rendering frames. One known cause of this is the synchronization
// between the browser and render process in AudioSyncReader.
// We need to fix this.
class AUHALStream : public AudioOutputStream {
public:
// |client| creates this object.
// |device| is the CoreAudio device to use for the stream.
// It will often be the default output device.
AUHALStream(AudioManagerApple* manager,
const AudioParameters& params,
AudioDeviceID device,
const AudioManager::LogCallback& log_callback);
AUHALStream(const AUHALStream&) = delete;
AUHALStream& operator=(const AUHALStream&) = delete;
// The dtor is typically called by the AudioManager only and it is usually
// triggered by calling AudioOutputStream::Close().
~AUHALStream() override;
// Implementation of AudioOutputStream.
bool Open() override;
void Close() override;
void Start(AudioSourceCallback* callback) override;
void Stop() override;
void Flush() override;
void SetVolume(double volume) override;
void GetVolume(double* volume) override;
AudioDeviceID device_id() const { return device_; }
size_t requested_buffer_size() const { return params_.frames_per_buffer(); }
AudioUnit audio_unit() const {
return audio_unit_ ? audio_unit_->audio_unit() : nullptr;
}
private:
// AUHAL callback.
static OSStatus InputProc(void* user_data,
AudioUnitRenderActionFlags* flags,
const AudioTimeStamp* time_stamp,
UInt32 bus_number,
UInt32 number_of_frames,
AudioBufferList* io_data);
OSStatus Render(AudioUnitRenderActionFlags* flags,
const AudioTimeStamp* output_time_stamp,
UInt32 bus_number,
UInt32 number_of_frames,
AudioBufferList* io_data);
// Called by either |audio_fifo_| or Render() to provide audio data.
void ProvideInput(int frame_delay, AudioBus* dest);
// Creates the AUHAL, sets its stream format, buffer-size, etc.
bool ConfigureAUHAL();
// Creates the input and output buses.
void CreateIOBusses();
// Returns the playout time for a given AudioTimeStamp.
base::TimeTicks GetPlayoutTime(const AudioTimeStamp* output_time_stamp);
// Updates playout timestamp, current lost frames, and total lost frames and
// glitches.
void UpdatePlayoutTimestamp(const AudioTimeStamp* timestamp);
// Our creator, the audio manager needs to be notified when we close.
const raw_ptr<AudioManagerApple> manager_;
const AudioParameters params_;
// We may get some callbacks after AudioUnitStop() has been called.
base::Lock lock_;
// Pointer to the object that will provide the audio samples.
raw_ptr<AudioSourceCallback> source_ GUARDED_BY(lock_);
// Holds the stream format details such as bitrate.
AudioStreamBasicDescription output_format_;
// The audio device to use with the AUHAL.
// We can potentially handle both input and output with this device.
const AudioDeviceID device_;
// The AUHAL Audio Unit which talks to |device_|.
std::unique_ptr<ScopedAudioUnit> audio_unit_;
// Volume level from 0 to 1.
std::atomic<float> volume_;
// Fixed playout hardware latency.
base::TimeDelta hardware_latency_;
// This flag will be set to false while we're actively receiving callbacks.
bool stopped_;
// Container for retrieving data from AudioSourceCallback::OnMoreData().
std::unique_ptr<AudioBus> output_bus_;
// Dynamically allocated FIFO used when CoreAudio asks for unexpected frame
// sizes.
std::unique_ptr<AudioPullFifo> audio_fifo_ GUARDED_BY(lock_);
// Current playout time. Set by Render().
base::TimeTicks current_playout_time_;
// Stores the timestamp of the previous audio buffer requested by the OS.
// We use this in combination with |last_number_of_frames_| to detect when
// the OS has decided to skip rendering frames (i.e. a glitch).
// This can happen in case of high CPU load or excessive blocking on the
// callback audio thread.
// These variables are only touched on the callback thread and then read
// in the dtor (when no longer receiving callbacks).
// NOTE: Float64 and UInt32 types are used for native API compatibility.
Float64 last_sample_time_ GUARDED_BY(lock_);
UInt32 last_number_of_frames_ GUARDED_BY(lock_);
// Used to aggregate and report glitch metrics to UMA (periodically) and to
// text logs (when a stream ends).
SystemGlitchReporter glitch_reporter_ GUARDED_BY(lock_);
// Used to defer Start() to workaround http://crbug.com/160920.
base::CancelableOnceClosure deferred_start_cb_;
// Callback to send statistics info.
AudioManager::LogCallback log_callback_;
[[maybe_unused]] std::unique_ptr<AmplitudePeakDetector> peak_detector_
GUARDED_BY(lock_);
AudioGlitchInfo::Accumulator glitch_info_accumulator_;
// Used to make sure control functions (Start(), Stop() etc) are called on the
// right thread.
THREAD_CHECKER(thread_checker_);
};
} // namespace media
#endif // MEDIA_AUDIO_APPLE_AUDIO_AUHAL_H_