2 * OpenAL cross platform audio library
3 * Copyright (C) 1999-2007 by authors.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18 * Or go to http://www.gnu.org/copyleft/lgpl.html
31 #include <system_error>
35 #include "alnumeric.h"
37 #include "althrd_setname.h"
38 #include "core/device.h"
39 #include "core/helpers.h"
40 #include "core/logging.h"
41 #include "ringbuffer.h"
43 #include <sndio.h> /* NOLINT(*-duplicate-include) Not the same header. */
48 using namespace std::string_view_literals
;
50 [[nodiscard
]] constexpr auto GetDefaultName() noexcept
{ return "SndIO Default"sv
; }
52 struct SioPar
: public sio_par
{
53 SioPar() : sio_par
{} { sio_initpar(this); }
55 void clear() { sio_initpar(this); }
58 struct SndioPlayback final
: public BackendBase
{
59 SndioPlayback(DeviceBase
*device
) noexcept
: BackendBase
{device
} { }
60 ~SndioPlayback() override
;
64 void open(std::string_view name
) override
;
65 bool reset() override
;
66 void start() override
;
69 sio_hdl
*mSndHandle
{nullptr};
72 std::vector
<std::byte
> mBuffer
;
74 std::atomic
<bool> mKillNow
{true};
78 SndioPlayback::~SndioPlayback()
81 sio_close(mSndHandle
);
85 int SndioPlayback::mixerProc()
87 const size_t frameStep
{mFrameStep
};
88 const size_t frameSize
{frameStep
* mDevice
->bytesFromFmt()};
91 althrd_setname(GetMixerThreadName());
93 while(!mKillNow
.load(std::memory_order_acquire
)
94 && mDevice
->Connected
.load(std::memory_order_acquire
))
96 al::span
<std::byte
> buffer
{mBuffer
};
98 mDevice
->renderSamples(buffer
.data(), static_cast<uint
>(buffer
.size() / frameSize
),
100 while(!buffer
.empty() && !mKillNow
.load(std::memory_order_acquire
))
102 size_t wrote
{sio_write(mSndHandle
, buffer
.data(), buffer
.size())};
103 if(wrote
> buffer
.size() || wrote
== 0)
105 ERR("sio_write failed: 0x%" PRIx64
"\n", wrote
);
106 mDevice
->handleDisconnect("Failed to write playback samples");
109 buffer
= buffer
.subspan(wrote
);
117 void SndioPlayback::open(std::string_view name
)
120 name
= GetDefaultName();
121 else if(name
!= GetDefaultName())
122 throw al::backend_exception
{al::backend_error::NoDevice
, "Device name \"%.*s\" not found",
123 al::sizei(name
), name
.data()};
125 sio_hdl
*sndHandle
{sio_open(nullptr, SIO_PLAY
, 0)};
127 throw al::backend_exception
{al::backend_error::NoDevice
, "Could not open backend device"};
130 sio_close(mSndHandle
);
131 mSndHandle
= sndHandle
;
133 mDevice
->DeviceName
= name
;
136 bool SndioPlayback::reset()
140 auto tryfmt
= mDevice
->FmtType
;
171 par
.bps
= SIO_BPS(par
.bits
);
172 par
.le
= SIO_LE_NATIVE
;
175 par
.rate
= mDevice
->Frequency
;
176 par
.pchan
= mDevice
->channelsFromFmt();
178 par
.round
= mDevice
->UpdateSize
;
179 par
.appbufsz
= mDevice
->BufferSize
- mDevice
->UpdateSize
;
180 if(!par
.appbufsz
) par
.appbufsz
= mDevice
->UpdateSize
;
183 if(!sio_setpar(mSndHandle
, &par
))
184 throw al::backend_exception
{al::backend_error::DeviceError
,
185 "Failed to set device parameters"};
188 if(!sio_getpar(mSndHandle
, &par
))
189 throw al::backend_exception
{al::backend_error::DeviceError
,
190 "Failed to get device parameters"};
192 if(par
.bps
> 1 && par
.le
!= SIO_LE_NATIVE
)
193 throw al::backend_exception
{al::backend_error::DeviceError
,
194 "%s-endian samples not supported", par
.le
? "Little" : "Big"};
195 if(par
.bits
< par
.bps
*8 && !par
.msb
)
196 throw al::backend_exception
{al::backend_error::DeviceError
,
197 "MSB-padded samples not supported (%u of %u bits)", par
.bits
, par
.bps
*8};
199 throw al::backend_exception
{al::backend_error::DeviceError
,
200 "No playback channels on device"};
204 catch(al::backend_exception
&e
) {
205 if(tryfmt
== DevFmtShort
)
208 tryfmt
= DevFmtShort
;
213 mDevice
->FmtType
= (par
.sig
==1) ? DevFmtByte
: DevFmtUByte
;
214 else if(par
.bps
== 2)
215 mDevice
->FmtType
= (par
.sig
==1) ? DevFmtShort
: DevFmtUShort
;
216 else if(par
.bps
== 4)
217 mDevice
->FmtType
= (par
.sig
==1) ? DevFmtInt
: DevFmtUInt
;
219 throw al::backend_exception
{al::backend_error::DeviceError
,
220 "Unhandled sample format: %s %u-bit", (par
.sig
?"signed":"unsigned"), par
.bps
*8};
222 mFrameStep
= par
.pchan
;
223 if(par
.pchan
!= mDevice
->channelsFromFmt())
225 WARN("Got %u channel%s for %s\n", par
.pchan
, (par
.pchan
==1)?"":"s",
226 DevFmtChannelsString(mDevice
->FmtChans
));
227 if(par
.pchan
< 2) mDevice
->FmtChans
= DevFmtMono
;
228 else mDevice
->FmtChans
= DevFmtStereo
;
230 mDevice
->Frequency
= par
.rate
;
232 setDefaultChannelOrder();
234 mDevice
->UpdateSize
= par
.round
;
235 mDevice
->BufferSize
= par
.bufsz
+ par
.round
;
237 mBuffer
.resize(size_t{mDevice
->UpdateSize
} * par
.pchan
*par
.bps
);
239 std::fill(mBuffer
.begin(), mBuffer
.end(), std::byte
{});
240 else if(par
.bits
== 8)
241 std::fill_n(mBuffer
.data(), mBuffer
.size(), std::byte(0x80));
242 else if(par
.bits
== 16)
243 std::fill_n(reinterpret_cast<uint16_t*>(mBuffer
.data()), mBuffer
.size()/2, 0x8000);
244 else if(par
.bits
== 32)
245 std::fill_n(reinterpret_cast<uint32_t*>(mBuffer
.data()), mBuffer
.size()/4, 0x80000000u
);
250 void SndioPlayback::start()
252 if(!sio_start(mSndHandle
))
253 throw al::backend_exception
{al::backend_error::DeviceError
, "Error starting playback"};
256 mKillNow
.store(false, std::memory_order_release
);
257 mThread
= std::thread
{std::mem_fn(&SndioPlayback::mixerProc
), this};
259 catch(std::exception
& e
) {
260 sio_stop(mSndHandle
);
261 throw al::backend_exception
{al::backend_error::DeviceError
,
262 "Failed to start mixing thread: %s", e
.what()};
266 void SndioPlayback::stop()
268 if(mKillNow
.exchange(true, std::memory_order_acq_rel
) || !mThread
.joinable())
272 if(!sio_stop(mSndHandle
))
273 ERR("Error stopping device\n");
277 /* TODO: This could be improved by avoiding the ring buffer and record thread,
278 * counting the available samples with the sio_onmove callback and reading
279 * directly from the device. However, this depends on reasonable support for
280 * capture buffer sizes apps may request.
282 struct SndioCapture final
: public BackendBase
{
283 SndioCapture(DeviceBase
*device
) noexcept
: BackendBase
{device
} { }
284 ~SndioCapture() override
;
288 void open(std::string_view name
) override
;
289 void start() override
;
290 void stop() override
;
291 void captureSamples(std::byte
*buffer
, uint samples
) override
;
292 uint
availableSamples() override
;
294 sio_hdl
*mSndHandle
{nullptr};
298 std::atomic
<bool> mKillNow
{true};
302 SndioCapture::~SndioCapture()
305 sio_close(mSndHandle
);
306 mSndHandle
= nullptr;
309 int SndioCapture::recordProc()
312 althrd_setname(GetRecordThreadName());
314 const uint frameSize
{mDevice
->frameSizeFromFmt()};
316 int nfds_pre
{sio_nfds(mSndHandle
)};
319 mDevice
->handleDisconnect("Incorrect return value from sio_nfds(): %d", nfds_pre
);
323 auto fds
= std::vector
<pollfd
>(static_cast<uint
>(nfds_pre
));
325 while(!mKillNow
.load(std::memory_order_acquire
)
326 && mDevice
->Connected
.load(std::memory_order_acquire
))
328 /* Wait until there's some samples to read. */
329 const int nfds
{sio_pollfd(mSndHandle
, fds
.data(), POLLIN
)};
332 mDevice
->handleDisconnect("Failed to get polling fds: %d", nfds
);
335 int pollres
{::poll(fds
.data(), fds
.size(), 2000)};
338 if(errno
== EINTR
) continue;
339 mDevice
->handleDisconnect("Poll error: %s",
340 std::generic_category().message(errno
).c_str());
346 const int revents
{sio_revents(mSndHandle
, fds
.data())};
347 if((revents
&POLLHUP
))
349 mDevice
->handleDisconnect("Got POLLHUP from poll events");
352 if(!(revents
&POLLIN
))
355 auto data
= mRing
->getWriteVector();
356 al::span
<std::byte
> buffer
{data
.first
.buf
, data
.first
.len
*frameSize
};
357 while(!buffer
.empty())
359 size_t got
{sio_read(mSndHandle
, buffer
.data(), buffer
.size())};
362 if(got
> buffer
.size())
364 ERR("sio_read failed: 0x%" PRIx64
"\n", got
);
365 mDevice
->handleDisconnect("sio_read failed: 0x%" PRIx64
, got
);
369 mRing
->writeAdvance(got
/ frameSize
);
370 buffer
= buffer
.subspan(got
);
373 data
= mRing
->getWriteVector();
374 buffer
= {data
.first
.buf
, data
.first
.len
*frameSize
};
379 /* Got samples to read, but no place to store it. Drop it. */
380 static std::array
<char,4096> junk
;
381 sio_read(mSndHandle
, junk
.data(), junk
.size() - (junk
.size()%frameSize
));
389 void SndioCapture::open(std::string_view name
)
392 name
= GetDefaultName();
393 else if(name
!= GetDefaultName())
394 throw al::backend_exception
{al::backend_error::NoDevice
, "Device name \"%.*s\" not found",
395 al::sizei(name
), name
.data()};
397 mSndHandle
= sio_open(nullptr, SIO_REC
, true);
398 if(mSndHandle
== nullptr)
399 throw al::backend_exception
{al::backend_error::NoDevice
, "Could not open backend device"};
402 switch(mDevice
->FmtType
)
429 throw al::backend_exception
{al::backend_error::DeviceError
,
430 "%s capture samples not supported", DevFmtTypeString(mDevice
->FmtType
)};
432 par
.bps
= SIO_BPS(par
.bits
);
433 par
.le
= SIO_LE_NATIVE
;
435 par
.rchan
= mDevice
->channelsFromFmt();
436 par
.rate
= mDevice
->Frequency
;
438 par
.appbufsz
= std::max(mDevice
->BufferSize
, mDevice
->Frequency
/10u);
439 par
.round
= std::min(par
.appbufsz
/2u, mDevice
->Frequency
/40u);
441 if(!sio_setpar(mSndHandle
, &par
) || !sio_getpar(mSndHandle
, &par
))
442 throw al::backend_exception
{al::backend_error::DeviceError
,
443 "Failed to set device parameters"};
445 if(par
.bps
> 1 && par
.le
!= SIO_LE_NATIVE
)
446 throw al::backend_exception
{al::backend_error::DeviceError
,
447 "%s-endian samples not supported", par
.le
? "Little" : "Big"};
448 if(par
.bits
< par
.bps
*8 && !par
.msb
)
449 throw al::backend_exception
{al::backend_error::DeviceError
,
450 "Padded samples not supported (got %u of %u bits)", par
.bits
, par
.bps
*8};
452 auto match_fmt
= [](DevFmtType fmttype
, const sio_par
&p
) -> bool
454 return (fmttype
== DevFmtByte
&& p
.bps
== 1 && p
.sig
!= 0)
455 || (fmttype
== DevFmtUByte
&& p
.bps
== 1 && p
.sig
== 0)
456 || (fmttype
== DevFmtShort
&& p
.bps
== 2 && p
.sig
!= 0)
457 || (fmttype
== DevFmtUShort
&& p
.bps
== 2 && p
.sig
== 0)
458 || (fmttype
== DevFmtInt
&& p
.bps
== 4 && p
.sig
!= 0)
459 || (fmttype
== DevFmtUInt
&& p
.bps
== 4 && p
.sig
== 0);
461 if(!match_fmt(mDevice
->FmtType
, par
) || mDevice
->channelsFromFmt() != par
.rchan
462 || mDevice
->Frequency
!= par
.rate
)
463 throw al::backend_exception
{al::backend_error::DeviceError
,
464 "Failed to set format %s %s %uhz, got %c%u %u-channel %uhz instead",
465 DevFmtTypeString(mDevice
->FmtType
), DevFmtChannelsString(mDevice
->FmtChans
),
466 mDevice
->Frequency
, par
.sig
?'s':'u', par
.bps
*8, par
.rchan
, par
.rate
};
468 mRing
= RingBuffer::Create(mDevice
->BufferSize
, size_t{par
.bps
}*par
.rchan
, false);
469 mDevice
->BufferSize
= static_cast<uint
>(mRing
->writeSpace());
470 mDevice
->UpdateSize
= par
.round
;
472 setDefaultChannelOrder();
474 mDevice
->DeviceName
= name
;
477 void SndioCapture::start()
479 if(!sio_start(mSndHandle
))
480 throw al::backend_exception
{al::backend_error::DeviceError
, "Error starting capture"};
483 mKillNow
.store(false, std::memory_order_release
);
484 mThread
= std::thread
{std::mem_fn(&SndioCapture::recordProc
), this};
486 catch(std::exception
& e
) {
487 sio_stop(mSndHandle
);
488 throw al::backend_exception
{al::backend_error::DeviceError
,
489 "Failed to start capture thread: %s", e
.what()};
493 void SndioCapture::stop()
495 if(mKillNow
.exchange(true, std::memory_order_acq_rel
) || !mThread
.joinable())
499 if(!sio_stop(mSndHandle
))
500 ERR("Error stopping device\n");
503 void SndioCapture::captureSamples(std::byte
*buffer
, uint samples
)
504 { std::ignore
= mRing
->read(buffer
, samples
); }
506 uint
SndioCapture::availableSamples()
507 { return static_cast<uint
>(mRing
->readSpace()); }
511 BackendFactory
&SndIOBackendFactory::getFactory()
513 static SndIOBackendFactory factory
{};
517 bool SndIOBackendFactory::init()
520 bool SndIOBackendFactory::querySupport(BackendType type
)
521 { return (type
== BackendType::Playback
|| type
== BackendType::Capture
); }
523 auto SndIOBackendFactory::enumerate(BackendType type
) -> std::vector
<std::string
>
527 case BackendType::Playback
:
528 case BackendType::Capture
:
529 return std::vector
{std::string
{GetDefaultName()}};
534 BackendPtr
SndIOBackendFactory::createBackend(DeviceBase
*device
, BackendType type
)
536 if(type
== BackendType::Playback
)
537 return BackendPtr
{new SndioPlayback
{device
}};
538 if(type
== BackendType::Capture
)
539 return BackendPtr
{new SndioCapture
{device
}};