AudioFormat.hxx 6.12 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
 * 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.
 */

#ifndef MPD_AUDIO_FORMAT_HXX
#define MPD_AUDIO_FORMAT_HXX

23
#include "pcm/SampleFormat.hxx" // IWYU pragma: export
Max Kellermann's avatar
Max Kellermann committed
24
#include "pcm/ChannelDefs.hxx" // IWYU pragma: export
25

26
#include <cstddef>
27
#include <cstdint>
28

29
template<std::size_t CAPACITY> class StringBuffer;
30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48

/**
 * 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;

	/**
49 50 51 52 53 54 55 56 57 58 59 60 61
	 * 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
62 63 64
	 */
	uint8_t channels;

65
	AudioFormat() noexcept = default;
66 67

	constexpr AudioFormat(uint32_t _sample_rate,
68
			      SampleFormat _format, uint8_t _channels) noexcept
69 70 71
		:sample_rate(_sample_rate),
		 format(_format), channels(_channels) {}

72
	static constexpr AudioFormat Undefined() noexcept {
73 74 75 76
		return AudioFormat(0, SampleFormat::UNDEFINED,0);
	}

	/**
Max Kellermann's avatar
Max Kellermann committed
77 78
	 * Clears the object, i.e. sets all attributes to an undefined
	 * (invalid) value.
79
	 */
80
	void Clear() noexcept {
81 82 83 84 85 86 87 88
		sample_rate = 0;
		format = SampleFormat::UNDEFINED;
		channels = 0;
	}

	/**
	 * Checks whether the object has a defined value.
	 */
89
	constexpr bool IsDefined() const noexcept {
90 91 92 93 94 95 96 97
		return sample_rate != 0;
	}

	/**
	 * Checks whether the object is full, i.e. all attributes are
	 * defined.  This is more complete than IsDefined(), but
	 * slower.
	 */
98
	constexpr bool IsFullyDefined() const noexcept {
99 100 101 102 103 104 105
		return sample_rate != 0 && format != SampleFormat::UNDEFINED &&
			channels != 0;
	}

	/**
	 * Checks whether the object has at least one defined value.
	 */
106
	constexpr bool IsMaskDefined() const noexcept {
107 108 109 110
		return sample_rate != 0 || format != SampleFormat::UNDEFINED ||
			channels != 0;
	}

111 112
	bool IsValid() const noexcept;
	bool IsMaskValid() const noexcept;
113

114
	constexpr bool operator==(const AudioFormat other) const noexcept {
115 116 117 118 119
		return sample_rate == other.sample_rate &&
			format == other.format &&
			channels == other.channels;
	}

120
	constexpr bool operator!=(const AudioFormat other) const noexcept {
121 122 123
		return !(*this == other);
	}

124
	void ApplyMask(AudioFormat mask) noexcept;
125

126
	[[gnu::pure]]
127
	AudioFormat WithMask(AudioFormat mask) const noexcept {
128 129 130 131 132
		AudioFormat result = *this;
		result.ApplyMask(mask);
		return result;
	}

133
	[[gnu::pure]]
134 135 136 137
	bool MatchMask(AudioFormat mask) const noexcept {
		return WithMask(mask) == *this;
	}

138 139 140
	/**
	 * Returns the size of each (mono) sample in bytes.
	 */
141
	unsigned GetSampleSize() const noexcept;
142 143 144 145

	/**
	 * Returns the size of each full frame in bytes.
	 */
146
	unsigned GetFrameSize() const noexcept;
147

148 149 150 151 152 153 154
	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>
155 156
	constexpr std::size_t TimeToSize(D t) const noexcept {
		return std::size_t(std::size_t(TimeToFrames(t)) * GetFrameSize());
157 158 159 160
	}

	template<typename D>
	constexpr D FramesToTime(std::uintmax_t size) const noexcept {
161
		using Rep = typename D::rep;
162
		using Period = typename D::period;
163
		return D(((Rep(1) * size / Period::num) * Period::den) / sample_rate);
164 165 166 167 168 169
	}

	template<typename D>
	constexpr D SizeToTime(std::uintmax_t size) const noexcept {
		return FramesToTime<D>(size / GetFrameSize());
	}
170 171 172 173 174 175 176
};

/**
 * Checks whether the sample rate is valid.
 *
 * @param sample_rate the sample rate in Hz
 */
177 178
constexpr bool
audio_valid_sample_rate(unsigned sample_rate) noexcept
179 180 181 182 183 184 185 186 187
{
	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
188
AudioFormat::IsValid() const noexcept
189 190 191 192 193 194 195 196 197 198 199
{
	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
200
AudioFormat::IsMaskValid() const noexcept
201 202 203 204 205 206 207 208 209
{
	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
210
AudioFormat::GetSampleSize() const noexcept
211 212 213 214 215
{
	return sample_format_size(format);
}

inline unsigned
216
AudioFormat::GetFrameSize() const noexcept
217 218 219 220 221
{
	return GetSampleSize() * channels;
}

/**
Max Kellermann's avatar
Max Kellermann committed
222
 * Renders the #AudioFormat object into a string, e.g. for printing
223 224
 * it in a log file.
 *
Max Kellermann's avatar
Max Kellermann committed
225
 * @param af the #AudioFormat object
226
 * @return the string buffer
227
 */
228
[[gnu::const]]
229
StringBuffer<24>
230
ToString(AudioFormat af) noexcept;
231 232

#endif