summaryrefslogtreecommitdiffstats
path: root/src/encoder/encodersndfileflac.cpp
blob: 482df1456542d6b48aa82ae80486ca1f384ccaa6 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
#include "encoder/encodersndfileflac.h"

#include <QtDebug>

#include "encoder/encodercallback.h"
#include "util/sample.h"

namespace {
constexpr SINT kEncBufferSize = 8192; // used inside libsndfile for flac
constexpr CSAMPLE_GAIN kFloatToIntConversionFactor = INT_MIN * -1.0f;

//static
void convertFloat32ToIntFormat(int* pDest, const CSAMPLE* pSrc, SINT numSamples, int format) {
    if (format & SF_FORMAT_PCM_16) {
        // note: LOOP VECTORIZED"
        for (SINT i = 0; i < numSamples; ++i) {
            pDest[i] = static_cast<int>(math_clamp(pSrc[i] * kFloatToIntConversionFactor,
                    static_cast<CSAMPLE>(static_cast<int>(INT_MIN & 0xFFFF0000)),
                    static_cast<CSAMPLE>(static_cast<int>(INT_MAX & 0xFFFF0000))));
        }
    } else if (format & SF_FORMAT_PCM_24) {
        // note: LOOP VECTORIZED"
        for (SINT i = 0; i < numSamples; ++i) {
            pDest[i] = static_cast<int>(math_clamp(pSrc[i] * kFloatToIntConversionFactor,
                    static_cast<CSAMPLE>(static_cast<int>(INT_MIN & 0xFFFFFF00)),
                    static_cast<CSAMPLE>(static_cast<int>(INT_MAX & 0xFFFFFF00))));
        }
    } else {
        DEBUG_ASSERT(!"Not implemented");
    }
}

} // namespace

EncoderSndfileFlac::EncoderSndfileFlac(EncoderCallback* pCallback)
        : EncoderWave(pCallback),
          m_compression(0) {
}

void EncoderSndfileFlac::setEncoderSettings(const EncoderSettings& settings)
{
    m_sfInfo.format = SF_FORMAT_FLAC;

    int radio = settings.getSelectedOption(EncoderFlacSettings::BITS_GROUP);
    switch(radio) {
        case 0:
            m_sfInfo.format |= SF_FORMAT_PCM_16;
            break;
        case 1:
            m_sfInfo.format |= SF_FORMAT_PCM_24;
            break;
        default:
            m_sfInfo.format |= SF_FORMAT_PCM_16;
            qWarning() << " Unexpected radio index on setEncoderSettings: "
                       << radio << ". reverting to Flac 16bits";
            break;
    }

    m_compression = static_cast<double>(settings.getCompression()) / 8.0;
}

void EncoderSndfileFlac::encodeBuffer(const CSAMPLE* pBuffer, const int iBufferSize) {
    if (m_pClampBuffer) {
        SINT numSamplesLeft = iBufferSize;
        while (numSamplesLeft > 0) {
            const SINT numSamplesToWrite = math_min(numSamplesLeft, kEncBufferSize);
            convertFloat32ToIntFormat(m_pClampBuffer.get(),
                    pBuffer,
                    numSamplesToWrite,
                    m_sfInfo.format);
            sf_write_int(m_pSndfile, m_pClampBuffer.get(), numSamplesToWrite);
            pBuffer += numSamplesToWrite;
            numSamplesLeft -= numSamplesToWrite;
        }
    } else {
        sf_write_float(m_pSndfile, pBuffer, iBufferSize);
    }
}

void EncoderSndfileFlac::initStream() {
    EncoderWave::initStream();
#if defined SFC_SUPPORTS_SET_COMPRESSION_LEVEL // Seems that this only exists since version 1.0.26
    // Tell the compression setting to use.
    sf_command(m_pSndfile, SFC_SET_COMPRESSION_LEVEL, &m_compression, sizeof(double));
#endif //SFC_SUPPORTS_SET_COMPRESSION_LEVEL

    // Version 1.0.28 suffers broken clamping https://bugs.launchpad.net/mixxx/+bug/1915298
    // We receive "libsndfile-1.0.28" on Ubuntu Focal 20.04 LTS/Grovy 20.10
    // Older versions are not expected. All newer version have a working internal clamping
    const char* sf_version = sf_version_string();
    if (strstr(sf_version, "-1.0.28") != nullptr) {
        m_pClampBuffer = std::make_unique<int[]>(kEncBufferSize);
        sf_command(m_pSndfile, SFC_SET_CLIPPING, nullptr, SF_FALSE);
    }
}