lib/alsa/Error: a std::system_error category for libasound errors
This commit is contained in:
parent
5ff0bbd0f8
commit
5d0941476a
@ -26,6 +26,7 @@
|
|||||||
|
|
||||||
#include "AlsaInputPlugin.hxx"
|
#include "AlsaInputPlugin.hxx"
|
||||||
#include "lib/alsa/NonBlock.hxx"
|
#include "lib/alsa/NonBlock.hxx"
|
||||||
|
#include "lib/alsa/Error.hxx"
|
||||||
#include "lib/alsa/Format.hxx"
|
#include "lib/alsa/Format.hxx"
|
||||||
#include "../InputPlugin.hxx"
|
#include "../InputPlugin.hxx"
|
||||||
#include "../AsyncInputStream.hxx"
|
#include "../AsyncInputStream.hxx"
|
||||||
@ -332,28 +333,23 @@ AlsaInputStream::ConfigureCapture(AudioFormat audio_format)
|
|||||||
snd_pcm_hw_params_alloca(&hw_params);
|
snd_pcm_hw_params_alloca(&hw_params);
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params_any(capture_handle, hw_params)) < 0)
|
if ((err = snd_pcm_hw_params_any(capture_handle, hw_params)) < 0)
|
||||||
throw FormatRuntimeError("Cannot initialize hardware parameter structure (%s)",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_any() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params_set_access(capture_handle, hw_params,
|
if ((err = snd_pcm_hw_params_set_access(capture_handle, hw_params,
|
||||||
SND_PCM_ACCESS_RW_INTERLEAVED)) < 0)
|
SND_PCM_ACCESS_RW_INTERLEAVED)) < 0)
|
||||||
throw FormatRuntimeError("Cannot set access type (%s)",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_set_access() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params_set_format(capture_handle, hw_params,
|
if ((err = snd_pcm_hw_params_set_format(capture_handle, hw_params,
|
||||||
ToAlsaPcmFormat(audio_format.format))) < 0)
|
ToAlsaPcmFormat(audio_format.format))) < 0)
|
||||||
throw FormatRuntimeError("Cannot set sample format (%s)",
|
throw Alsa::MakeError(err, "Cannot set sample format");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params_set_channels(capture_handle,
|
if ((err = snd_pcm_hw_params_set_channels(capture_handle,
|
||||||
hw_params, audio_format.channels)) < 0)
|
hw_params, audio_format.channels)) < 0)
|
||||||
throw FormatRuntimeError("Cannot set channels (%s)",
|
throw Alsa::MakeError(err, "Cannot set channels");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params_set_rate(capture_handle,
|
if ((err = snd_pcm_hw_params_set_rate(capture_handle,
|
||||||
hw_params, audio_format.sample_rate, 0)) < 0)
|
hw_params, audio_format.sample_rate, 0)) < 0)
|
||||||
throw FormatRuntimeError("Cannot set sample rate (%s)",
|
throw Alsa::MakeError(err, "Cannot set sample rate");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
snd_pcm_uframes_t buffer_size_min, buffer_size_max;
|
snd_pcm_uframes_t buffer_size_min, buffer_size_max;
|
||||||
snd_pcm_hw_params_get_buffer_size_min(hw_params, &buffer_size_min);
|
snd_pcm_hw_params_get_buffer_size_min(hw_params, &buffer_size_min);
|
||||||
@ -388,26 +384,22 @@ AlsaInputStream::ConfigureCapture(AudioFormat audio_format)
|
|||||||
int direction = -1;
|
int direction = -1;
|
||||||
if ((err = snd_pcm_hw_params_set_period_size_near(capture_handle,
|
if ((err = snd_pcm_hw_params_set_period_size_near(capture_handle,
|
||||||
hw_params, &period_size, &direction)) < 0)
|
hw_params, &period_size, &direction)) < 0)
|
||||||
throw FormatRuntimeError("Cannot set period size (%s)",
|
throw Alsa::MakeError(err, "Cannot set period size");
|
||||||
snd_strerror(err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((err = snd_pcm_hw_params(capture_handle, hw_params)) < 0)
|
if ((err = snd_pcm_hw_params(capture_handle, hw_params)) < 0)
|
||||||
throw FormatRuntimeError("Cannot set parameters (%s)",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
snd_pcm_uframes_t alsa_buffer_size;
|
snd_pcm_uframes_t alsa_buffer_size;
|
||||||
err = snd_pcm_hw_params_get_buffer_size(hw_params, &alsa_buffer_size);
|
err = snd_pcm_hw_params_get_buffer_size(hw_params, &alsa_buffer_size);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_get_buffer_size() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_get_buffer_size() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
snd_pcm_uframes_t alsa_period_size;
|
snd_pcm_uframes_t alsa_period_size;
|
||||||
err = snd_pcm_hw_params_get_period_size(hw_params, &alsa_period_size,
|
err = snd_pcm_hw_params_get_period_size(hw_params, &alsa_period_size,
|
||||||
nullptr);
|
nullptr);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_get_period_size() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_get_period_size() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
FmtDebug(alsa_input_domain, "buffer_size={} period_size={}",
|
FmtDebug(alsa_input_domain, "buffer_size={} period_size={}",
|
||||||
alsa_buffer_size, alsa_period_size);
|
alsa_buffer_size, alsa_period_size);
|
||||||
@ -418,8 +410,7 @@ AlsaInputStream::ConfigureCapture(AudioFormat audio_format)
|
|||||||
snd_pcm_sw_params_current(capture_handle, sw_params);
|
snd_pcm_sw_params_current(capture_handle, sw_params);
|
||||||
|
|
||||||
if ((err = snd_pcm_sw_params(capture_handle, sw_params)) < 0)
|
if ((err = snd_pcm_sw_params(capture_handle, sw_params)) < 0)
|
||||||
throw FormatRuntimeError("unable to install sw params (%s)",
|
throw Alsa::MakeError(err, "snd_pcm_sw_params() failed");
|
||||||
snd_strerror(err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
@ -430,8 +421,9 @@ AlsaInputStream::OpenDevice(const SourceSpec &spec)
|
|||||||
if ((err = snd_pcm_open(&capture_handle, spec.GetDeviceName(),
|
if ((err = snd_pcm_open(&capture_handle, spec.GetDeviceName(),
|
||||||
SND_PCM_STREAM_CAPTURE,
|
SND_PCM_STREAM_CAPTURE,
|
||||||
SND_PCM_NONBLOCK | global_config.mode)) < 0)
|
SND_PCM_NONBLOCK | global_config.mode)) < 0)
|
||||||
throw FormatRuntimeError("Failed to open device: %s (%s)",
|
throw Alsa::MakeError(err,
|
||||||
spec.GetDeviceName(), snd_strerror(err));
|
fmt::format("Failed to open device {}",
|
||||||
|
spec.GetDeviceName()).c_str());
|
||||||
|
|
||||||
try {
|
try {
|
||||||
ConfigureCapture(spec.GetAudioFormat());
|
ConfigureCapture(spec.GetAudioFormat());
|
||||||
|
44
src/lib/alsa/Error.cxx
Normal file
44
src/lib/alsa/Error.cxx
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Max Kellermann <max.kellermann@gmail.com>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||||||
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
||||||
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "Error.hxx"
|
||||||
|
|
||||||
|
#include <alsa/error.h>
|
||||||
|
|
||||||
|
namespace Alsa {
|
||||||
|
|
||||||
|
ErrorCategory error_category;
|
||||||
|
|
||||||
|
std::string
|
||||||
|
ErrorCategory::message(int condition) const
|
||||||
|
{
|
||||||
|
return snd_strerror(condition);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Avahi
|
53
src/lib/alsa/Error.hxx
Normal file
53
src/lib/alsa/Error.hxx
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2021 Max Kellermann <max.kellermann@gmail.com>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* - Redistributions in binary form must reproduce the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer in the
|
||||||
|
* documentation and/or other materials provided with the
|
||||||
|
* distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||||||
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||||||
|
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||||
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||||||
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
||||||
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <system_error>
|
||||||
|
|
||||||
|
namespace Alsa {
|
||||||
|
|
||||||
|
class ErrorCategory final : public std::error_category {
|
||||||
|
public:
|
||||||
|
const char *name() const noexcept override {
|
||||||
|
return "libasound";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string message(int condition) const override;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern ErrorCategory error_category;
|
||||||
|
|
||||||
|
inline std::system_error
|
||||||
|
MakeError(int error, const char *msg) noexcept
|
||||||
|
{
|
||||||
|
return std::system_error(error, error_category, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Avahi
|
@ -18,6 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "HwSetup.hxx"
|
#include "HwSetup.hxx"
|
||||||
|
#include "Error.hxx"
|
||||||
#include "Format.hxx"
|
#include "Format.hxx"
|
||||||
#include "util/ByteOrder.hxx"
|
#include "util/ByteOrder.hxx"
|
||||||
#include "util/Domain.hxx"
|
#include "util/Domain.hxx"
|
||||||
@ -185,29 +186,27 @@ SetupHw(snd_pcm_t *pcm,
|
|||||||
/* configure HW params */
|
/* configure HW params */
|
||||||
err = snd_pcm_hw_params_any(pcm, hwparams);
|
err = snd_pcm_hw_params_any(pcm, hwparams);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_any() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_any() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_hw_params_set_access(pcm, hwparams,
|
err = snd_pcm_hw_params_set_access(pcm, hwparams,
|
||||||
SND_PCM_ACCESS_RW_INTERLEAVED);
|
SND_PCM_ACCESS_RW_INTERLEAVED);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_set_access() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_set_access() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = SetupSampleFormat(pcm, hwparams,
|
err = SetupSampleFormat(pcm, hwparams,
|
||||||
audio_format.format, params);
|
audio_format.format, params);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("Failed to configure format %s: %s",
|
throw Alsa::MakeError(err,
|
||||||
sample_format_to_string(audio_format.format),
|
fmt::format("Failed to configure format {}",
|
||||||
snd_strerror(-err));
|
audio_format.format).c_str());
|
||||||
|
|
||||||
unsigned int channels = audio_format.channels;
|
unsigned int channels = audio_format.channels;
|
||||||
err = snd_pcm_hw_params_set_channels_near(pcm, hwparams,
|
err = snd_pcm_hw_params_set_channels_near(pcm, hwparams,
|
||||||
&channels);
|
&channels);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("Failed to configure %i channels: %s",
|
throw Alsa::MakeError(err,
|
||||||
(int)audio_format.channels,
|
fmt::format("Failed to configure {} channels",
|
||||||
snd_strerror(-err));
|
audio_format.channels).c_str());
|
||||||
|
|
||||||
audio_format.channels = (int8_t)channels;
|
audio_format.channels = (int8_t)channels;
|
||||||
|
|
||||||
@ -218,9 +217,9 @@ SetupHw(snd_pcm_t *pcm,
|
|||||||
err = snd_pcm_hw_params_set_rate_near(pcm, hwparams,
|
err = snd_pcm_hw_params_set_rate_near(pcm, hwparams,
|
||||||
&output_sample_rate, nullptr);
|
&output_sample_rate, nullptr);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("Failed to configure sample rate %u Hz: %s",
|
throw Alsa::MakeError(err,
|
||||||
requested_sample_rate,
|
fmt::format("Failed to configure sample rate {} Hz",
|
||||||
snd_strerror(-err));
|
requested_sample_rate).c_str());
|
||||||
|
|
||||||
if (output_sample_rate == 0)
|
if (output_sample_rate == 0)
|
||||||
throw FormatRuntimeError("Failed to configure sample rate %u Hz",
|
throw FormatRuntimeError("Failed to configure sample rate %u Hz",
|
||||||
@ -253,8 +252,7 @@ SetupHw(snd_pcm_t *pcm,
|
|||||||
err = snd_pcm_hw_params_set_buffer_time_near(pcm, hwparams,
|
err = snd_pcm_hw_params_set_buffer_time_near(pcm, hwparams,
|
||||||
&buffer_time, nullptr);
|
&buffer_time, nullptr);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_set_buffer_time_near() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_set_buffer_time_near() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
} else {
|
} else {
|
||||||
err = snd_pcm_hw_params_get_buffer_time(hwparams, &buffer_time,
|
err = snd_pcm_hw_params_get_buffer_time(hwparams, &buffer_time,
|
||||||
nullptr);
|
nullptr);
|
||||||
@ -275,32 +273,27 @@ SetupHw(snd_pcm_t *pcm,
|
|||||||
err = snd_pcm_hw_params_set_period_time_near(pcm, hwparams,
|
err = snd_pcm_hw_params_set_period_time_near(pcm, hwparams,
|
||||||
&period_time, nullptr);
|
&period_time, nullptr);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_set_period_time_near() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_set_period_time_near() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
err = snd_pcm_hw_params(pcm, hwparams);
|
err = snd_pcm_hw_params(pcm, hwparams);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
HwResult result;
|
HwResult result;
|
||||||
|
|
||||||
err = snd_pcm_hw_params_get_format(hwparams, &result.format);
|
err = snd_pcm_hw_params_get_format(hwparams, &result.format);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_get_format() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_get_format() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_hw_params_get_buffer_size(hwparams, &result.buffer_size);
|
err = snd_pcm_hw_params_get_buffer_size(hwparams, &result.buffer_size);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_get_buffer_size() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_get_buffer_size() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_hw_params_get_period_size(hwparams, &result.period_size,
|
err = snd_pcm_hw_params_get_period_size(hwparams, &result.period_size,
|
||||||
nullptr);
|
nullptr);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_hw_params_get_period_size() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_hw_params_get_period_size() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "NonBlock.hxx"
|
#include "NonBlock.hxx"
|
||||||
|
#include "Error.hxx"
|
||||||
#include "event/MultiSocketMonitor.hxx"
|
#include "event/MultiSocketMonitor.hxx"
|
||||||
#include "util/RuntimeError.hxx"
|
#include "util/RuntimeError.hxx"
|
||||||
|
|
||||||
@ -29,8 +30,7 @@ AlsaNonBlockPcm::PrepareSockets(MultiSocketMonitor &m, snd_pcm_t *pcm)
|
|||||||
if (count == 0)
|
if (count == 0)
|
||||||
throw std::runtime_error("snd_pcm_poll_descriptors_count() failed");
|
throw std::runtime_error("snd_pcm_poll_descriptors_count() failed");
|
||||||
else
|
else
|
||||||
throw FormatRuntimeError("snd_pcm_poll_descriptors_count() failed: %s",
|
throw Alsa::MakeError(count, "snd_pcm_poll_descriptors_count() failed");
|
||||||
snd_strerror(-count));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct pollfd *pfds = pfd_buffer.Get(count);
|
struct pollfd *pfds = pfd_buffer.Get(count);
|
||||||
@ -40,8 +40,7 @@ AlsaNonBlockPcm::PrepareSockets(MultiSocketMonitor &m, snd_pcm_t *pcm)
|
|||||||
if (count == 0)
|
if (count == 0)
|
||||||
throw std::runtime_error("snd_pcm_poll_descriptors() failed");
|
throw std::runtime_error("snd_pcm_poll_descriptors() failed");
|
||||||
else
|
else
|
||||||
throw FormatRuntimeError("snd_pcm_poll_descriptors() failed: %s",
|
throw Alsa::MakeError(count, "snd_pcm_poll_descriptors() failed");
|
||||||
snd_strerror(-count));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
m.ReplaceSocketList(pfds, count);
|
m.ReplaceSocketList(pfds, count);
|
||||||
@ -71,8 +70,7 @@ AlsaNonBlockPcm::DispatchSockets(MultiSocketMonitor &m,
|
|||||||
unsigned short dummy;
|
unsigned short dummy;
|
||||||
int err = snd_pcm_poll_descriptors_revents(pcm, pfds, i - pfds, &dummy);
|
int err = snd_pcm_poll_descriptors_revents(pcm, pfds, i - pfds, &dummy);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_poll_descriptors_revents() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_poll_descriptors_revents() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Event::Duration
|
Event::Duration
|
||||||
|
@ -14,6 +14,7 @@ conf.set('ENABLE_ALSA', true)
|
|||||||
alsa = static_library(
|
alsa = static_library(
|
||||||
'alsa',
|
'alsa',
|
||||||
'Version.cxx',
|
'Version.cxx',
|
||||||
|
'Error.cxx',
|
||||||
'AllowedFormat.cxx',
|
'AllowedFormat.cxx',
|
||||||
'HwSetup.cxx',
|
'HwSetup.cxx',
|
||||||
'NonBlock.cxx',
|
'NonBlock.cxx',
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "lib/alsa/NonBlock.hxx"
|
#include "lib/alsa/NonBlock.hxx"
|
||||||
|
#include "lib/alsa/Error.hxx"
|
||||||
#include "mixer/MixerInternal.hxx"
|
#include "mixer/MixerInternal.hxx"
|
||||||
#include "mixer/Listener.hxx"
|
#include "mixer/Listener.hxx"
|
||||||
#include "output/OutputAPI.hxx"
|
#include "output/OutputAPI.hxx"
|
||||||
@ -264,16 +265,15 @@ AlsaMixer::Setup()
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
if ((err = snd_mixer_attach(handle, device)) < 0)
|
if ((err = snd_mixer_attach(handle, device)) < 0)
|
||||||
throw FormatRuntimeError("failed to attach to %s: %s",
|
throw Alsa::MakeError(err,
|
||||||
device, snd_strerror(err));
|
fmt::format("failed to attach to {}",
|
||||||
|
device).c_str());
|
||||||
|
|
||||||
if ((err = snd_mixer_selem_register(handle, nullptr, nullptr)) < 0)
|
if ((err = snd_mixer_selem_register(handle, nullptr, nullptr)) < 0)
|
||||||
throw FormatRuntimeError("snd_mixer_selem_register() failed: %s",
|
throw Alsa::MakeError(err, "snd_mixer_selem_register() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
if ((err = snd_mixer_load(handle)) < 0)
|
if ((err = snd_mixer_load(handle)) < 0)
|
||||||
throw FormatRuntimeError("snd_mixer_load() failed: %s\n",
|
throw Alsa::MakeError(err, "snd_mixer_load() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
elem = alsa_mixer_lookup_elem(handle, control, index);
|
elem = alsa_mixer_lookup_elem(handle, control, index);
|
||||||
if (elem == nullptr)
|
if (elem == nullptr)
|
||||||
@ -294,8 +294,7 @@ AlsaMixer::Open()
|
|||||||
|
|
||||||
err = snd_mixer_open(&handle, 0);
|
err = snd_mixer_open(&handle, 0);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_mixer_open() failed: %s",
|
throw Alsa::MakeError(err, "snd_mixer_open() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Setup();
|
Setup();
|
||||||
@ -325,8 +324,7 @@ AlsaMixer::GetVolume()
|
|||||||
|
|
||||||
err = snd_mixer_handle_events(handle);
|
err = snd_mixer_handle_events(handle);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_mixer_handle_events() failed: %s",
|
throw Alsa::MakeError(err, "snd_mixer_handle_events() failed");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
int volume = GetPercentVolume();
|
int volume = GetPercentVolume();
|
||||||
if (resulting_volume >= 0 && volume == resulting_volume)
|
if (resulting_volume >= 0 && volume == resulting_volume)
|
||||||
@ -343,8 +341,7 @@ AlsaMixer::SetVolume(unsigned volume)
|
|||||||
|
|
||||||
int err = set_normalized_playback_volume(elem, 0.01*volume, 1);
|
int err = set_normalized_playback_volume(elem, 0.01*volume, 1);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("failed to set ALSA volume: %s",
|
throw Alsa::MakeError(err, "failed to set ALSA volume");
|
||||||
snd_strerror(err));
|
|
||||||
|
|
||||||
desired_volume = volume;
|
desired_volume = volume;
|
||||||
resulting_volume = GetPercentVolume();
|
resulting_volume = GetPercentVolume();
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "AlsaOutputPlugin.hxx"
|
#include "AlsaOutputPlugin.hxx"
|
||||||
#include "lib/alsa/AllowedFormat.hxx"
|
#include "lib/alsa/AllowedFormat.hxx"
|
||||||
|
#include "lib/alsa/Error.hxx"
|
||||||
#include "lib/alsa/HwSetup.hxx"
|
#include "lib/alsa/HwSetup.hxx"
|
||||||
#include "lib/alsa/NonBlock.hxx"
|
#include "lib/alsa/NonBlock.hxx"
|
||||||
#include "lib/alsa/PeriodBuffer.hxx"
|
#include "lib/alsa/PeriodBuffer.hxx"
|
||||||
@ -519,24 +520,20 @@ AlsaSetupSw(snd_pcm_t *pcm, snd_pcm_uframes_t start_threshold,
|
|||||||
|
|
||||||
int err = snd_pcm_sw_params_current(pcm, swparams);
|
int err = snd_pcm_sw_params_current(pcm, swparams);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_sw_params_current() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_sw_params_current() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_sw_params_set_start_threshold(pcm, swparams,
|
err = snd_pcm_sw_params_set_start_threshold(pcm, swparams,
|
||||||
start_threshold);
|
start_threshold);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_sw_params_set_start_threshold() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_sw_params_set_start_threshold() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_sw_params_set_avail_min(pcm, swparams, avail_min);
|
err = snd_pcm_sw_params_set_avail_min(pcm, swparams, avail_min);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_sw_params_set_avail_min() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_sw_params_set_avail_min() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
|
|
||||||
err = snd_pcm_sw_params(pcm, swparams);
|
err = snd_pcm_sw_params(pcm, swparams);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_sw_params() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_sw_params() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void
|
inline void
|
||||||
@ -704,8 +701,9 @@ AlsaOutput::Open(AudioFormat &audio_format)
|
|||||||
int err = snd_pcm_open(&pcm, GetDevice(),
|
int err = snd_pcm_open(&pcm, GetDevice(),
|
||||||
SND_PCM_STREAM_PLAYBACK, mode);
|
SND_PCM_STREAM_PLAYBACK, mode);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("Failed to open ALSA device \"%s\": %s",
|
throw Alsa::MakeError(err,
|
||||||
GetDevice(), snd_strerror(err));
|
fmt::format("Failed to open ALSA device \"{}\"",
|
||||||
|
GetDevice()).c_str());
|
||||||
|
|
||||||
FmtDebug(alsa_output_domain, "opened {} type={}",
|
FmtDebug(alsa_output_domain, "opened {} type={}",
|
||||||
snd_pcm_name(pcm),
|
snd_pcm_name(pcm),
|
||||||
@ -897,8 +895,8 @@ AlsaOutput::DrainInternal()
|
|||||||
if (frames_written == -EAGAIN)
|
if (frames_written == -EAGAIN)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
throw FormatRuntimeError("snd_pcm_writei() failed: %s",
|
throw Alsa::MakeError(frames_written,
|
||||||
snd_strerror(-frames_written));
|
"snd_pcm_writei() failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* need to call CopyRingToPeriodBuffer() and
|
/* need to call CopyRingToPeriodBuffer() and
|
||||||
@ -947,8 +945,7 @@ AlsaOutput::DrainInternal()
|
|||||||
else if (result == -EAGAIN)
|
else if (result == -EAGAIN)
|
||||||
return false;
|
return false;
|
||||||
else
|
else
|
||||||
throw FormatRuntimeError("snd_pcm_drain() failed: %s",
|
throw Alsa::MakeError(result, "snd_pcm_drain() failed");
|
||||||
snd_strerror(-result));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -1147,8 +1144,7 @@ try {
|
|||||||
|
|
||||||
int err = snd_pcm_prepare(pcm);
|
int err = snd_pcm_prepare(pcm);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_prepare() failed: %s",
|
throw Alsa::MakeError(err, "snd_pcm_prepare() failed");
|
||||||
snd_strerror(-err));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
@ -1236,8 +1232,8 @@ try {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
if (Recover(frames_written) < 0)
|
if (Recover(frames_written) < 0)
|
||||||
throw FormatRuntimeError("snd_pcm_writei() failed: %s",
|
throw Alsa::MakeError(frames_written,
|
||||||
snd_strerror(-frames_written));
|
"snd_pcm_writei() failed");
|
||||||
|
|
||||||
/* recovered; try again in the next DispatchSockets()
|
/* recovered; try again in the next DispatchSockets()
|
||||||
call */
|
call */
|
||||||
|
Loading…
Reference in New Issue
Block a user