Dither.hxx 2.47 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright 2003-2021 The Music Player Daemon Project
3
 * http://www.musicpd.org
4 5 6 7 8 9 10 11 12 13
 *
 * 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.
14 15 16 17
 *
 * 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.
18 19
 */

Max Kellermann's avatar
Max Kellermann committed
20 21
#ifndef MPD_PCM_DITHER_HXX
#define MPD_PCM_DITHER_HXX
22

23
#include <cstdint>
24

25 26
enum class SampleFormat : uint8_t;

27
class PcmDither {
28 29 30
	int32_t error[3];
	int32_t random;

31
public:
Max Kellermann's avatar
Max Kellermann committed
32
	constexpr PcmDither() noexcept
33 34
		:error{0, 0, 0}, random(0) {}

35 36 37 38
	/**
	 * Shift the given sample by #SBITS-#DBITS to the right, and
	 * apply dithering.
	 *
Max Kellermann's avatar
Max Kellermann committed
39 40 41
	 * @tparam ST the input sample type
	 * @tparam SBITS the input bit width
	 * @tparam DBITS the output bit width
42 43 44
	 * @param sample the input sample value
	 */
	template<typename ST, unsigned SBITS, unsigned DBITS>
Max Kellermann's avatar
Max Kellermann committed
45
	ST DitherShift(ST sample) noexcept;
46

47
	void Dither24To16(int16_t *dest, const int32_t *src,
Max Kellermann's avatar
Max Kellermann committed
48
			  const int32_t *src_end) noexcept;
49

50
	void Dither32To16(int16_t *dest, const int32_t *src,
Max Kellermann's avatar
Max Kellermann committed
51
			  const int32_t *src_end) noexcept;
52

53
private:
54 55 56 57 58 59 60 61 62 63
	/**
	 * Shift the given sample by #scale_bits to the right, and
	 * apply dithering.
	 *
	 * @param T the input sample type
	 * @param MIN the minimum input sample value
	 * @param MAX the maximum input sample value
	 * @param scale_bits the number of bits to be discarded
	 * @param sample the input sample value
	 */
64
	template<typename T, T MIN, T MAX, unsigned scale_bits>
Max Kellermann's avatar
Max Kellermann committed
65
	T Dither(T sample) noexcept;
66

67 68 69 70 71 72 73 74
	/**
	 * Convert the given sample from one sample format to another,
	 * discarding bits.
	 *
	 * @param ST the input #SampleTraits class
	 * @param ST the output #SampleTraits class
	 * @param sample the input sample value
	 */
75
	template<typename ST, typename DT>
Max Kellermann's avatar
Max Kellermann committed
76
	typename DT::value_type DitherConvert(typename ST::value_type sample) noexcept;
77 78

	template<typename ST, typename DT>
79 80 81
	void DitherConvert(typename DT::pointer dest,
			   typename ST::const_pointer src,
			   typename ST::const_pointer src_end) noexcept;
82
};
83

84
#endif