123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- #include <pulse/pulseaudio.h>
- struct AudioPulseAudio : AudioDriver {
- AudioPulseAudio& self = *this;
- AudioPulseAudio(Audio& super) : AudioDriver(super) {}
- ~AudioPulseAudio() { terminate(); }
- auto create() -> bool override {
- super.setChannels(2);
- super.setFrequency(48000);
- super.setLatency(40);
- return initialize();
- }
- auto driver() -> string override { return "PulseAudio"; }
- auto ready() -> bool override { return _ready; }
- auto hasBlocking() -> bool override { return true; }
- auto hasDynamic() -> bool override { return true; }
- auto hasFrequencies() -> vector<uint> override {
- return {44100, 48000, 96000};
- }
- auto hasLatencies() -> vector<uint> override {
- return {20, 40, 60, 80, 100};
- }
- auto setBlocking(bool blocking) -> bool override { return true; }
- auto setFrequency(uint frequency) -> bool override { return initialize(); }
- auto setLatency(uint latency) -> bool override { return initialize(); }
- auto level() -> double override {
- pa_mainloop_iterate(_mainLoop, 0, nullptr);
- auto length = pa_stream_writable_size(_stream);
- return (double)(_bufferSize - length) / _bufferSize;
- }
- auto output(const double samples[]) -> void override {
- pa_stream_begin_write(_stream, (void**)&_buffer, &_period);
- _buffer[_offset] = (uint16_t)sclamp<16>(samples[0] * 32767.0) << 0;
- _buffer[_offset] |= (uint16_t)sclamp<16>(samples[1] * 32767.0) << 16;
- if((++_offset + 1) * pa_frame_size(&_specification) <= _period) return;
- while(true) {
- pa_mainloop_iterate(_mainLoop, 0, nullptr);
- auto length = pa_stream_writable_size(_stream);
- if(length >= _offset * pa_frame_size(&_specification)) break;
- if(!self.blocking) {
- _offset = 0;
- return;
- }
- }
- pa_stream_write(_stream, (const void*)_buffer, _offset * pa_frame_size(&_specification), nullptr, 0LL, PA_SEEK_RELATIVE);
- _buffer = nullptr;
- _offset = 0;
- }
- private:
- auto initialize() -> bool {
- terminate();
- _mainLoop = pa_mainloop_new();
- _context = pa_context_new(pa_mainloop_get_api(_mainLoop), "ruby::pulseAudio");
- pa_context_connect(_context, nullptr, PA_CONTEXT_NOFLAGS, nullptr);
- pa_context_state_t contextState;
- do {
- pa_mainloop_iterate(_mainLoop, 1, nullptr);
- contextState = pa_context_get_state(_context);
- if(!PA_CONTEXT_IS_GOOD(contextState)) return false;
- } while(contextState != PA_CONTEXT_READY);
- _specification.format = PA_SAMPLE_S16LE;
- _specification.channels = 2;
- _specification.rate = self.frequency;
- _stream = pa_stream_new(_context, "audio", &_specification, nullptr);
- pa_buffer_attr bufferAttributes;
- bufferAttributes.maxlength = -1;
- bufferAttributes.tlength = pa_usec_to_bytes(self.latency * PA_USEC_PER_MSEC, &_specification);
- bufferAttributes.prebuf = -1;
- bufferAttributes.minreq = -1;
- bufferAttributes.fragsize = -1;
- pa_stream_flags_t flags = (pa_stream_flags_t)(PA_STREAM_ADJUST_LATENCY | PA_STREAM_VARIABLE_RATE);
- pa_stream_connect_playback(_stream, nullptr, &bufferAttributes, flags, nullptr, nullptr);
- pa_stream_state_t streamState;
- do {
- pa_mainloop_iterate(_mainLoop, 1, nullptr);
- streamState = pa_stream_get_state(_stream);
- if(!PA_STREAM_IS_GOOD(streamState)) return false;
- } while(streamState != PA_STREAM_READY);
- const pa_buffer_attr* attributes = pa_stream_get_buffer_attr(_stream);
- _period = attributes->minreq;
- _bufferSize = attributes->tlength;
- _offset = 0;
- return _ready = true;
- }
- auto terminate() -> void {
- _ready = false;
- if(_buffer) {
- pa_stream_cancel_write(_stream);
- _buffer = nullptr;
- }
- if(_stream) {
- pa_stream_disconnect(_stream);
- pa_stream_unref(_stream);
- _stream = nullptr;
- }
- if(_context) {
- pa_context_disconnect(_context);
- pa_context_unref(_context);
- _context = nullptr;
- }
- if(_mainLoop) {
- pa_mainloop_free(_mainLoop);
- _mainLoop = nullptr;
- }
- }
- bool _ready = false;
- uint32_t* _buffer = nullptr;
- size_t _period = 0;
- size_t _bufferSize = 0;
- uint _offset = 0;
- pa_mainloop* _mainLoop = nullptr;
- pa_context* _context = nullptr;
- pa_stream* _stream = nullptr;
- pa_sample_spec _specification;
- };
|