/* * Copyright 2003-2020 The Music Player Daemon Project * http://www.musicpd.org * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "SolarisOutputPlugin.hxx" #include "../OutputAPI.hxx" #include "io/FileDescriptor.hxx" #include "system/Error.hxx" #include #include #include #include #include #include #if defined(__sun) #include #include #elif defined(__NetBSD__) #include #else /* some fake declarations that allow build this plugin on systems other than Solaris, just to see if it compiles */ #ifndef I_FLUSH #define I_FLUSH 0 #endif #define AUDIO_INITINFO(v) #define AUDIO_GETINFO 0 #define AUDIO_SETINFO 0 #define AUDIO_ENCODING_LINEAR 0 struct audio_info { struct { unsigned sample_rate, channels, precision, encoding; } play; }; #endif class SolarisOutput final : AudioOutput { /* configuration */ const char *const device; FileDescriptor fd; explicit SolarisOutput(const ConfigBlock &block) :AudioOutput(0), device(block.GetBlockValue("device", "/dev/audio")) {} public: static AudioOutput *Create(EventLoop &, const ConfigBlock &block) { return new SolarisOutput(block); } private: void Open(AudioFormat &audio_format) override; void Close() noexcept override; size_t Play(const void *chunk, size_t size) override; void Cancel() noexcept override; }; static bool solaris_output_test_default_device(void) { struct stat st; return stat("/dev/audio", &st) == 0 && S_ISCHR(st.st_mode) && access("/dev/audio", W_OK) == 0; } void SolarisOutput::Open(AudioFormat &audio_format) { struct audio_info info; int ret; AUDIO_INITINFO(&info); /* open the device in non-blocking mode */ if (!fd.Open(device, O_WRONLY|O_NONBLOCK)) throw FormatErrno("Failed to open %s", device); /* restore blocking mode */ fd.SetBlocking(); /* configure the audio device */ info.play.sample_rate = audio_format.sample_rate; info.play.channels = audio_format.channels; info.play.encoding = AUDIO_ENCODING_LINEAR; switch (audio_format.format) { case SampleFormat::S8: info.play.precision = 8; break; case SampleFormat::S16: info.play.precision = 16; break; default: info.play.precision = 32; audio_format.format = SampleFormat::S32; break; } ret = ioctl(fd.Get(), AUDIO_SETINFO, &info); if (ret < 0) { const int e = errno; fd.Close(); throw MakeErrno(e, "AUDIO_SETINFO failed"); } } void SolarisOutput::Close() noexcept { fd.Close(); } size_t SolarisOutput::Play(const void *chunk, size_t size) { ssize_t nbytes = fd.Write(chunk, size); if (nbytes <= 0) throw MakeErrno("Write failed"); return nbytes; } void SolarisOutput::Cancel() noexcept { #if defined(AUDIO_FLUSH) ioctl(fd.Get(), AUDIO_FLUSH); #elif defined(I_FLUSH) ioctl(fd.Get(), I_FLUSH); #endif } const struct AudioOutputPlugin solaris_output_plugin = { "solaris", solaris_output_test_default_device, &SolarisOutput::Create, nullptr, };