summaryrefslogtreecommitdiff
path: root/src/codecs/sample.cpp
diff options
context:
space:
mode:
authorjacqueline <me@jacqueline.id.au>2023-12-20 10:57:24 +1100
committerjacqueline <me@jacqueline.id.au>2023-12-20 10:57:24 +1100
commitd4a0085753b9397ac4a1452520decbeb994bb30c (patch)
treec0c48257b827cf42323cfc47e63f1a9f9eef82a7 /src/codecs/sample.cpp
parent2ccaaf5724fe08e63e06b677a42326d4f8e0550e (diff)
downloadtangara-fw-d4a0085753b9397ac4a1452520decbeb994bb30c.tar.gz
Fix some dither clipping issues
Diffstat (limited to 'src/codecs/sample.cpp')
-rw-r--r--src/codecs/sample.cpp16
1 files changed, 11 insertions, 5 deletions
diff --git a/src/codecs/sample.cpp b/src/codecs/sample.cpp
index c5c96fb9..d4860b94 100644
--- a/src/codecs/sample.cpp
+++ b/src/codecs/sample.cpp
@@ -5,6 +5,7 @@
*/
#include "sample.hpp"
+#include <stdint.h>
#include <cstdint>
@@ -15,11 +16,16 @@ namespace sample {
static uint64_t sSeed1{0};
static uint64_t sSeed2{0};
-auto applyDither(int64_t src, uint_fast8_t bits) -> int32_t {
- uint64_t mask = 0xFFFFFFFF; // 32 ones
- mask >>= 32 - bits; // `bits` ones
- uint64_t noise = komirand(&sSeed1, &sSeed2) & mask; // `bits` random noise
- return std::clamp<int64_t>(src + noise, INT32_MIN, INT32_MAX);
+auto shiftWithDither(int64_t src, uint_fast8_t bits) -> Sample {
+ // Generate `bits` random bits
+ uint64_t mask = 0xFFFFFFFF;
+ mask >>= 32 - bits;
+ int64_t noise = static_cast<int32_t>(komirand(&sSeed1, &sSeed2) & mask);
+ // Centre the noise around 0.
+ noise -= (mask >> 1);
+ // Apply to the sample, then clip and shift to 16 bit.
+ Sample clipped = Clip((src + noise) >> bits);
+ return clipped;
}
} // namespace sample