mirror of https://github.com/auygun/kaliber.git
109 lines
2.9 KiB
C++
109 lines
2.9 KiB
C++
#ifndef ENGINE_AUDIO_AUDIO_MIXER_H
|
|
#define ENGINE_AUDIO_AUDIO_MIXER_H
|
|
|
|
#include <atomic>
|
|
#include <list>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <unordered_map>
|
|
|
|
#include "base/closure.h"
|
|
#include "engine/audio/audio_sink.h"
|
|
|
|
namespace base {
|
|
class TaskRunner;
|
|
}
|
|
|
|
namespace eng {
|
|
|
|
class AudioSink;
|
|
class AudioBus;
|
|
|
|
// Mix and render audio with low overhead. A platform specific AudioSink
|
|
// implementation is expected to periodically call RenderAudio() in a background
|
|
// thread.
|
|
class AudioMixer : public AudioSink::Delegate {
|
|
public:
|
|
AudioMixer();
|
|
~AudioMixer();
|
|
|
|
uint64_t CreateResource();
|
|
void DestroyResource(uint64_t resource_id);
|
|
|
|
void Play(uint64_t resource_id,
|
|
std::shared_ptr<AudioBus> sound,
|
|
float amplitude,
|
|
bool reset_pos);
|
|
void Stop(uint64_t resource_id);
|
|
|
|
void SetLoop(uint64_t resource_id, bool loop);
|
|
void SetSimulateStereo(uint64_t resource_id, bool simulate);
|
|
void SetResampleStep(uint64_t resource_id, size_t step);
|
|
void SetMaxAmplitude(uint64_t resource_id, float max_amplitude);
|
|
void SetAmplitudeInc(uint64_t resource_id, float amplitude_inc);
|
|
void SetEndCallback(uint64_t resource_id, base::Closure cb);
|
|
|
|
void SetEnableAudio(bool enable) { audio_enabled_ = enable; }
|
|
|
|
void Suspend();
|
|
void Resume();
|
|
|
|
size_t GetHardwareSampleRate();
|
|
|
|
private:
|
|
enum SampleFlags { kLoop = 1, kStopped = 2, kSimulateStereo = 4 };
|
|
static constexpr int kChannelCount = 2;
|
|
|
|
// An audio resource that gets mixed and rendered to the audio sink.
|
|
struct Resource {
|
|
// Accessed by main thread only.
|
|
bool active = false;
|
|
base::Closure end_cb;
|
|
base::Closure restart_cb;
|
|
|
|
// Initialized by main thread, used by audio thread.
|
|
std::shared_ptr<AudioBus> sound;
|
|
size_t src_index = 0;
|
|
size_t accumulator = 0;
|
|
float amplitude = 1.0f;
|
|
|
|
// Write accessed by main thread, read-only accessed by audio thread.
|
|
std::atomic<unsigned> flags{0};
|
|
std::atomic<size_t> step{100};
|
|
std::atomic<float> amplitude_inc{0};
|
|
std::atomic<float> max_amplitude{1.0f};
|
|
|
|
// Accessed by audio thread and decoder thread.
|
|
std::atomic<bool> streaming_in_progress{false};
|
|
};
|
|
|
|
std::unordered_map<uint64_t, std::shared_ptr<Resource>> resources_;
|
|
uint64_t last_resource_id_ = 0;
|
|
|
|
std::list<std::shared_ptr<Resource>> play_list_[2];
|
|
std::mutex lock_;
|
|
|
|
std::list<std::shared_ptr<Resource>> end_list_;
|
|
|
|
base::TaskRunner* main_thread_task_runner_;
|
|
|
|
std::unique_ptr<AudioSink> audio_sink_;
|
|
|
|
bool audio_enabled_ = true;
|
|
|
|
// AudioSink::Delegate implementation
|
|
int GetChannelCount() final { return kChannelCount; }
|
|
void RenderAudio(float* output_buffer, size_t num_frames) final;
|
|
|
|
void DoStream(std::shared_ptr<Resource> sample, bool loop);
|
|
|
|
void EndCallback(std::shared_ptr<Resource> sample);
|
|
|
|
AudioMixer(const AudioMixer&) = delete;
|
|
AudioMixer& operator=(const AudioMixer&) = delete;
|
|
};
|
|
|
|
} // namespace eng
|
|
|
|
#endif // ENGINE_AUDIO_AUDIO_MIXER_H
|