pcm/Dsd{16,32}: stash odd frames away for the next call

Similar to commit 32380d1db0, these are
the final parts for really fixing
https://github.com/MusicPlayerDaemon/MPD/issues/469
This commit is contained in:
Max Kellermann 2019-06-18 11:19:27 +02:00
parent 0cc94fe30c
commit bf26adf555
5 changed files with 66 additions and 21 deletions

View File

@ -20,6 +20,8 @@
#include "Dsd16.hxx" #include "Dsd16.hxx"
#include "util/ConstBuffer.hxx" #include "util/ConstBuffer.hxx"
#include <functional>
/** /**
* Construct a 16 bit integer from two bytes. * Construct a 16 bit integer from two bytes.
*/ */
@ -54,18 +56,14 @@ void
Dsd16Converter::Open(unsigned _channels) noexcept Dsd16Converter::Open(unsigned _channels) noexcept
{ {
channels = _channels; channels = _channels;
rest_buffer.Open(channels);
} }
ConstBuffer<uint16_t> ConstBuffer<uint16_t>
Dsd16Converter::Convert(ConstBuffer<uint8_t> _src) noexcept Dsd16Converter::Convert(ConstBuffer<uint8_t> src) noexcept
{ {
const size_t in_frames = _src.size / channels; using namespace std::placeholders;
const size_t out_frames = in_frames / 2; return rest_buffer.Process<uint16_t>(buffer, src, channels,
const size_t out_samples = out_frames * channels; std::bind(Dsd8To16, _1, _2, _3, channels));
const uint8_t *src = _src.data;
const auto dest = buffer.GetT<uint16_t>(out_samples);
Dsd8To16(dest, src, out_frames, channels);
return {dest, out_samples};
} }

View File

@ -21,6 +21,7 @@
#define MPD_PCM_DSD_16_HXX #define MPD_PCM_DSD_16_HXX
#include "Buffer.hxx" #include "Buffer.hxx"
#include "RestBuffer.hxx"
#include <stdint.h> #include <stdint.h>
@ -34,10 +35,13 @@ class Dsd16Converter {
PcmBuffer buffer; PcmBuffer buffer;
PcmRestBuffer<uint8_t, 2> rest_buffer;
public: public:
void Open(unsigned _channels) noexcept; void Open(unsigned _channels) noexcept;
void Reset() noexcept { void Reset() noexcept {
rest_buffer.Reset();
} }
ConstBuffer<uint16_t> Convert(ConstBuffer<uint8_t> src) noexcept; ConstBuffer<uint16_t> Convert(ConstBuffer<uint8_t> src) noexcept;

View File

@ -18,9 +18,10 @@
*/ */
#include "Dsd32.hxx" #include "Dsd32.hxx"
#include "Buffer.hxx"
#include "util/ConstBuffer.hxx" #include "util/ConstBuffer.hxx"
#include <functional>
/** /**
* Construct a 32 bit integer from four bytes. * Construct a 32 bit integer from four bytes.
*/ */
@ -57,18 +58,14 @@ void
Dsd32Converter::Open(unsigned _channels) noexcept Dsd32Converter::Open(unsigned _channels) noexcept
{ {
channels = _channels; channels = _channels;
rest_buffer.Open(channels);
} }
ConstBuffer<uint32_t> ConstBuffer<uint32_t>
Dsd32Converter::Convert(ConstBuffer<uint8_t> _src) noexcept Dsd32Converter::Convert(ConstBuffer<uint8_t> src) noexcept
{ {
const size_t in_frames = _src.size / channels; using namespace std::placeholders;
const size_t out_frames = in_frames / 4; return rest_buffer.Process<uint32_t>(buffer, src, channels,
const size_t out_samples = out_frames * channels; std::bind(Dsd8To32, _1, _2, _3, channels));
const uint8_t *src = _src.data;
const auto dest = buffer.GetT<uint32_t>(out_samples);
Dsd8To32(dest, src, out_frames, channels);
return {dest, out_samples};
} }

View File

@ -21,6 +21,7 @@
#define MPD_PCM_DSD_32_HXX #define MPD_PCM_DSD_32_HXX
#include "Buffer.hxx" #include "Buffer.hxx"
#include "RestBuffer.hxx"
#include <stdint.h> #include <stdint.h>
@ -34,10 +35,13 @@ class Dsd32Converter {
PcmBuffer buffer; PcmBuffer buffer;
PcmRestBuffer<uint8_t, 4> rest_buffer;
public: public:
void Open(unsigned _channels) noexcept; void Open(unsigned _channels) noexcept;
void Reset() noexcept { void Reset() noexcept {
rest_buffer.Reset();
} }
ConstBuffer<uint32_t> Convert(ConstBuffer<uint8_t> src) noexcept; ConstBuffer<uint32_t> Convert(ConstBuffer<uint8_t> src) noexcept;

View File

@ -152,6 +152,27 @@ TEST(PcmTest, ExportDsdU16)
auto dest = e.Export({src, sizeof(src)}); auto dest = e.Export({src, sizeof(src)});
EXPECT_EQ(sizeof(expected), dest.size); EXPECT_EQ(sizeof(expected), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected, dest.size) == 0); EXPECT_TRUE(memcmp(dest.data, expected, dest.size) == 0);
/* no output, 2/4 remains */
static constexpr uint8_t src2[] = { 0x11, 0x22 };
static constexpr uint16_t expected2[] = {};
dest = e.Export({src2, sizeof(src2)});
EXPECT_EQ(sizeof(expected2), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected2, dest.size) == 0);
/* one full frame and 2/4 remains */
static constexpr uint8_t src3[] = { 0x33, 0x44, 0x55, 0x66 };
static constexpr uint16_t expected3[] = { 0x1133, 0x2244 };
dest = e.Export({src3, sizeof(src3)});
EXPECT_EQ(sizeof(expected3), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected3, dest.size) == 0);
/* two full frames and 2/4 remains again */
static constexpr uint8_t src4[] = { 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee };
static constexpr uint16_t expected4[] = { 0x5577, 0x6688, 0x99bb, 0xaacc };
dest = e.Export({src4, sizeof(src4)});
EXPECT_EQ(sizeof(expected4), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected4, dest.size) == 0);
} }
TEST(PcmTest, ExportDsdU32) TEST(PcmTest, ExportDsdU32)
@ -182,6 +203,27 @@ TEST(PcmTest, ExportDsdU32)
auto dest = e.Export({src, sizeof(src)}); auto dest = e.Export({src, sizeof(src)});
EXPECT_EQ(sizeof(expected), dest.size); EXPECT_EQ(sizeof(expected), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected, dest.size) == 0); EXPECT_TRUE(memcmp(dest.data, expected, dest.size) == 0);
/* no output, 4/8 remains */
static constexpr uint8_t src2[] = { 0x11, 0x22, 0x33, 0x44 };
static constexpr uint32_t expected2[] = {};
dest = e.Export({src2, sizeof(src2)});
EXPECT_EQ(sizeof(expected2), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected2, dest.size) == 0);
/* one full frame and 4/8 remains */
static constexpr uint8_t src3[] = { 0x55, 0x66, 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc };
static constexpr uint32_t expected3[] = { 0x11335577, 0x22446688 };
dest = e.Export({src3, sizeof(src3)});
EXPECT_EQ(sizeof(expected3), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected3, dest.size) == 0);
/* two full frames and 2/4 remains again */
static constexpr uint8_t src4[] = { 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08 };
static constexpr uint32_t expected4[] = { 0x99bb0103, 0xaacc0204 };
dest = e.Export({src4, sizeof(src4)});
EXPECT_EQ(sizeof(expected4), dest.size);
EXPECT_TRUE(memcmp(dest.data, expected4, dest.size) == 0);
} }
TEST(PcmTest, ExportDop) TEST(PcmTest, ExportDop)