2023-03-06 14:42:04 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
// Copyright The Music Player Daemon Project
|
2013-08-03 21:00:50 +02:00
|
|
|
|
|
|
|
#ifndef MPD_AUDIO_FORMAT_HXX
|
|
|
|
#define MPD_AUDIO_FORMAT_HXX
|
|
|
|
|
2020-06-10 21:10:28 +02:00
|
|
|
#include "pcm/SampleFormat.hxx" // IWYU pragma: export
|
2020-06-10 22:58:14 +02:00
|
|
|
#include "pcm/ChannelDefs.hxx" // IWYU pragma: export
|
2013-08-03 21:00:50 +02:00
|
|
|
|
2020-03-13 00:46:28 +01:00
|
|
|
#include <cstddef>
|
2019-07-05 09:59:00 +02:00
|
|
|
#include <cstdint>
|
2018-09-21 19:32:35 +02:00
|
|
|
|
2021-08-26 13:40:04 +02:00
|
|
|
template<std::size_t CAPACITY> class StringBuffer;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This structure describes the format of a raw PCM stream.
|
|
|
|
*/
|
|
|
|
struct AudioFormat {
|
|
|
|
/**
|
|
|
|
* The sample rate in Hz. A better name for this attribute is
|
|
|
|
* "frame rate", because technically, you have two samples per
|
|
|
|
* frame in stereo sound.
|
|
|
|
*/
|
|
|
|
uint32_t sample_rate;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The format samples are stored in. See the #sample_format
|
|
|
|
* enum for valid values.
|
|
|
|
*/
|
|
|
|
SampleFormat format;
|
|
|
|
|
|
|
|
/**
|
2015-10-26 23:55:17 +01:00
|
|
|
* The number of channels.
|
|
|
|
*
|
|
|
|
* Channel order follows the FLAC convention
|
|
|
|
* (https://xiph.org/flac/format.html):
|
|
|
|
*
|
|
|
|
* - 1 channel: mono
|
|
|
|
* - 2 channels: left, right
|
|
|
|
* - 3 channels: left, right, center
|
|
|
|
* - 4 channels: front left, front right, back left, back right
|
|
|
|
* - 5 channels: front left, front right, front center, back/surround left, back/surround right
|
|
|
|
* - 6 channels: front left, front right, front center, LFE, back/surround left, back/surround right
|
|
|
|
* - 7 channels: front left, front right, front center, LFE, back center, side left, side right
|
|
|
|
* - 8 channels: front left, front right, front center, LFE, back left, back right, side left, side right
|
2013-08-03 21:00:50 +02:00
|
|
|
*/
|
|
|
|
uint8_t channels;
|
|
|
|
|
2021-08-26 13:36:25 +02:00
|
|
|
AudioFormat() noexcept = default;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
|
|
|
constexpr AudioFormat(uint32_t _sample_rate,
|
2021-08-26 13:36:25 +02:00
|
|
|
SampleFormat _format, uint8_t _channels) noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
:sample_rate(_sample_rate),
|
|
|
|
format(_format), channels(_channels) {}
|
|
|
|
|
2021-08-26 13:36:25 +02:00
|
|
|
static constexpr AudioFormat Undefined() noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return AudioFormat(0, SampleFormat::UNDEFINED,0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-17 11:21:29 +01:00
|
|
|
* Clears the object, i.e. sets all attributes to an undefined
|
|
|
|
* (invalid) value.
|
2013-08-03 21:00:50 +02:00
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
void Clear() noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
sample_rate = 0;
|
|
|
|
format = SampleFormat::UNDEFINED;
|
|
|
|
channels = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks whether the object has a defined value.
|
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
constexpr bool IsDefined() const noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return sample_rate != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks whether the object is full, i.e. all attributes are
|
|
|
|
* defined. This is more complete than IsDefined(), but
|
|
|
|
* slower.
|
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
constexpr bool IsFullyDefined() const noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return sample_rate != 0 && format != SampleFormat::UNDEFINED &&
|
|
|
|
channels != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks whether the object has at least one defined value.
|
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
constexpr bool IsMaskDefined() const noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return sample_rate != 0 || format != SampleFormat::UNDEFINED ||
|
|
|
|
channels != 0;
|
|
|
|
}
|
|
|
|
|
2021-08-26 13:36:25 +02:00
|
|
|
bool IsValid() const noexcept;
|
|
|
|
bool IsMaskValid() const noexcept;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
2021-08-26 13:36:25 +02:00
|
|
|
constexpr bool operator==(const AudioFormat other) const noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return sample_rate == other.sample_rate &&
|
|
|
|
format == other.format &&
|
|
|
|
channels == other.channels;
|
|
|
|
}
|
|
|
|
|
2021-08-26 13:36:25 +02:00
|
|
|
constexpr bool operator!=(const AudioFormat other) const noexcept {
|
2013-08-03 21:00:50 +02:00
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
2017-05-08 14:44:49 +02:00
|
|
|
void ApplyMask(AudioFormat mask) noexcept;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
2021-08-26 13:39:12 +02:00
|
|
|
[[gnu::pure]]
|
2017-06-03 21:33:44 +02:00
|
|
|
AudioFormat WithMask(AudioFormat mask) const noexcept {
|
2016-12-13 20:55:48 +01:00
|
|
|
AudioFormat result = *this;
|
|
|
|
result.ApplyMask(mask);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-08-26 13:39:12 +02:00
|
|
|
[[gnu::pure]]
|
2017-11-10 23:05:48 +01:00
|
|
|
bool MatchMask(AudioFormat mask) const noexcept {
|
|
|
|
return WithMask(mask) == *this;
|
|
|
|
}
|
|
|
|
|
2013-08-03 21:00:50 +02:00
|
|
|
/**
|
|
|
|
* Returns the size of each (mono) sample in bytes.
|
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
unsigned GetSampleSize() const noexcept;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the size of each full frame in bytes.
|
|
|
|
*/
|
2021-08-26 13:36:25 +02:00
|
|
|
unsigned GetFrameSize() const noexcept;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
2018-09-21 19:32:35 +02:00
|
|
|
template<typename D>
|
|
|
|
constexpr auto TimeToFrames(D t) const noexcept {
|
|
|
|
using Period = typename D::period;
|
|
|
|
return ((t.count() * sample_rate) / Period::den) * Period::num;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename D>
|
2021-08-26 13:40:04 +02:00
|
|
|
constexpr std::size_t TimeToSize(D t) const noexcept {
|
|
|
|
return std::size_t(std::size_t(TimeToFrames(t)) * GetFrameSize());
|
2018-09-21 19:32:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename D>
|
|
|
|
constexpr D FramesToTime(std::uintmax_t size) const noexcept {
|
2018-09-23 15:14:36 +02:00
|
|
|
using Rep = typename D::rep;
|
2018-09-21 19:32:35 +02:00
|
|
|
using Period = typename D::period;
|
2018-09-23 15:14:36 +02:00
|
|
|
return D(((Rep(1) * size / Period::num) * Period::den) / sample_rate);
|
2018-09-21 19:32:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename D>
|
|
|
|
constexpr D SizeToTime(std::uintmax_t size) const noexcept {
|
|
|
|
return FramesToTime<D>(size / GetFrameSize());
|
|
|
|
}
|
2013-08-03 21:00:50 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks whether the sample rate is valid.
|
|
|
|
*
|
|
|
|
* @param sample_rate the sample rate in Hz
|
|
|
|
*/
|
2019-03-08 10:29:03 +01:00
|
|
|
constexpr bool
|
|
|
|
audio_valid_sample_rate(unsigned sample_rate) noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
{
|
|
|
|
return sample_rate > 0 && sample_rate < (1 << 30);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns false if the format is not valid for playback with MPD.
|
|
|
|
* This function performs some basic validity checks.
|
|
|
|
*/
|
|
|
|
inline bool
|
2021-08-26 13:36:25 +02:00
|
|
|
AudioFormat::IsValid() const noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
{
|
|
|
|
return audio_valid_sample_rate(sample_rate) &&
|
|
|
|
audio_valid_sample_format(format) &&
|
|
|
|
audio_valid_channel_count(channels);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns false if the format mask is not valid for playback with
|
|
|
|
* MPD. This function performs some basic validity checks.
|
|
|
|
*/
|
|
|
|
inline bool
|
2021-08-26 13:36:25 +02:00
|
|
|
AudioFormat::IsMaskValid() const noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
{
|
|
|
|
return (sample_rate == 0 ||
|
|
|
|
audio_valid_sample_rate(sample_rate)) &&
|
|
|
|
(format == SampleFormat::UNDEFINED ||
|
|
|
|
audio_valid_sample_format(format)) &&
|
|
|
|
(channels == 0 || audio_valid_channel_count(channels));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
2021-08-26 13:36:25 +02:00
|
|
|
AudioFormat::GetSampleSize() const noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
{
|
|
|
|
return sample_format_size(format);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
2021-08-26 13:36:25 +02:00
|
|
|
AudioFormat::GetFrameSize() const noexcept
|
2013-08-03 21:00:50 +02:00
|
|
|
{
|
|
|
|
return GetSampleSize() * channels;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-17 11:21:29 +01:00
|
|
|
* Renders the #AudioFormat object into a string, e.g. for printing
|
2013-08-03 21:00:50 +02:00
|
|
|
* it in a log file.
|
|
|
|
*
|
2015-03-17 11:21:29 +01:00
|
|
|
* @param af the #AudioFormat object
|
2017-01-17 22:48:34 +01:00
|
|
|
* @return the string buffer
|
2013-08-03 21:00:50 +02:00
|
|
|
*/
|
2021-08-26 13:39:12 +02:00
|
|
|
[[gnu::const]]
|
2017-01-17 22:04:31 +01:00
|
|
|
StringBuffer<24>
|
2017-05-08 14:44:49 +02:00
|
|
|
ToString(AudioFormat af) noexcept;
|
2013-08-03 21:00:50 +02:00
|
|
|
|
|
|
|
#endif
|