summaryrefslogtreecommitdiff
path: root/apps/openmw/mwsound/loudness.cpp
blob: ac44d1b40e5b7222eb5a343d2d157c9599606ef0 (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
#include "loudness.hpp"

#include <cstdint>
#include <limits>
#include <algorithm>

#include "soundmanagerimp.hpp"

namespace MWSound
{

void Sound_Loudness::analyzeLoudness(const std::vector< char >& data)
{
    mQueue.insert( mQueue.end(), data.begin(), data.end() );
    if (!mQueue.size())
        return;

    int samplesPerSegment = static_cast<int>(mSampleRate / mSamplesPerSec);
    int numSamples = bytesToFrames(mQueue.size(), mChannelConfig, mSampleType);
    int advance = framesToBytes(1, mChannelConfig, mSampleType);


    int segment=0;
    int sample=0;
    while (segment < numSamples/samplesPerSegment)
    {
        float sum=0;
        int samplesAdded = 0;
        while (sample < numSamples && sample < (segment+1)*samplesPerSegment)
        {
            // get sample on a scale from -1 to 1
            float value = 0;
            if (mSampleType == SampleType_UInt8)
                value = ((char)(mQueue[sample*advance]^0x80))/128.f;
            else if (mSampleType == SampleType_Int16)
            {
                value = *reinterpret_cast<const int16_t*>(&mQueue[sample*advance]);
                value /= float(std::numeric_limits<int16_t>::max());
            }
            else if (mSampleType == SampleType_Float32)
            {
                value = *reinterpret_cast<const float*>(&mQueue[sample*advance]);
                value = std::clamp(value, -1.f, 1.f); // Float samples *should* be scaled to [-1,1] already.
            }

            sum += value*value;
            ++samplesAdded;
            ++sample;
        }

        float rms = 0; // root mean square
        if (samplesAdded > 0)
            rms = std::sqrt(sum / samplesAdded);
        mSamples.push_back(rms);
        ++segment;
    }

    mQueue.erase(mQueue.begin(), mQueue.begin() + sample*advance);
}


float Sound_Loudness::getLoudnessAtTime(float sec) const
{
    if(mSamplesPerSec <= 0.0f || mSamples.empty() || sec < 0.0f)
        return 0.0f;

    size_t index = std::clamp<size_t>(sec * mSamplesPerSec, 0, mSamples.size() - 1);
    return mSamples[index];
}

}