HART  0.2.0
High level Audio Regression and Testing
Loading...
Searching...
No Matches
hart_rms.hpp
Go to the documentation of this file.
1#pragma once
2
3#include <cmath> // sqrt()
4
8#include "metrics/hart_metric_query.hpp"
9#include "metrics/hart_metrics_common.hpp" // ChannelSubsets
10#include "hart_slice.hpp"
11#include "hart_units.hpp" // Unit
12#include "hart_utils.hpp" // nan(), ratioToDecibels()
13
14namespace hart
15{
16
17/// @brief Calculates root mean square (RMS) of a signal
18/// @details RMS a metric that expresses the average magnitude, or effective
19/// energy level, of an audio signal over time. It is commonly used to estimate
20/// perceived loudness, and to measure overall signal level.
21///
22/// RMS is calculated this way:
23/// @f[
24/// \mathrm{RMS} = \sqrt{\frac{1}{N} \sum_{n=0}^{N-1} x[n]^2}
25/// @f]
26/// (RMS = sqrt((1 / N) * sum(x[n] ** 2))),
27///
28/// where x[n] is audio sample value from one channel, and N
29/// is number of frames in the provided buffer.
30///
31/// Can be expressed as ratio or decibels, supports `Unit:::native`,
32/// `Unit::linear`, `Unit::dB` units. Value in decibels is
33/// calculated as a ratio, not power.
34///
35/// @tparam SampleType
36/// @param buffer Audio buffer to calculate RMS at
37/// @return Chainable `MetricQuery` object, which calculates RMS as linear ratio
38/// or decibels
39template <typename SampleType>
40MetricQuery<double> rms (const AudioBuffer<SampleType>& buffer)
41{
42 typename MetricQuery<double>::SingleChannelMetricEvaluator evaluator =
43 [&buffer]
44 (size_t channel, Slice slice, Unit requestedUnit)
45 -> double
46 {
47 hassert (channel < buffer.getNumChannels());
48
49 if (slice.isEmpty())
50 return hart::nan<double>();
51
52 const auto sliceFrameIndices = buffer.getFrameIndices (slice);
53 const size_t sliceStart = sliceFrameIndices.first;
54 const size_t sliceStop = sliceFrameIndices.second;
55 const size_t numFrames = sliceStop - sliceStart;
56 hassert (numFrames != 0);
57 hassert (sliceStart < sliceStop);
58 hassert (sliceStop <= buffer.getNumFrames());
59
60 const SampleType* channelData = buffer[channel];
61 AccurateSum<SampleType> sumSquares;
62
63 for (size_t frame = sliceStart; frame < sliceStop; ++frame)
64 {
65 const SampleType x = channelData[frame];
66 sumSquares += x * x;
67 }
68
69 const double rmsLinear = std::sqrt (sumSquares.template get<double>() / numFrames);
70
71 switch (requestedUnit)
72 {
73 case Unit::native:
74 case Unit::linear: return rmsLinear;
75
76 case Unit::dB: return hart::ratioToDecibels (rmsLinear);
77
78 default: HART_THROW_OR_RETURN (hart::UnitError, "Unsupported unit", hart::nan<double>());
79 }
80 };
81
82 const size_t numChannels = buffer.getNumChannels();
83 return MetricQuery<double> (
84 std::move (evaluator),
85 numChannels,
87 );
88}
89
90} // namespace hart
Implements Kahan algorithm for floating point accumulations.
Container for audio data.
Manages the metrics calculations.
MetricQuery(SingleChannelMetricEvaluator evaluator, size_t totalNumChannels, std::vector< size_t > &&defaultChannelsToProcess)
Create a metric query object for a metric that operates on one channel at a time.
Thrown when some metric is requested to return a value in an unsupported unit.
#define hassert(condition)
Triggers a HartAssertException if the condition is false
#define HART_THROW_OR_RETURN(ExceptionType, message, returnValue)
Throws an exception if HART_DO_NOT_THROW_EXCEPTIONS is set, prints a message and returns a specified ...
FloatType nan()
Returns a quiet NaN value for the given floating-point type.
static SampleType ratioToDecibels(SampleType valueLinear)
Converts linear value (ratio) to dB.
MetricQuery< double > rms(const AudioBuffer< SampleType > &buffer)
Calculates root mean square (RMS) of a signal.
Definition hart_rms.hpp:40
Unit
Represents a physical unit.
@ dB
Value of something in decibels. Can represent voltage, power, or a domain-specific unit like "LUFS" o...
@ native
Default (native) unit of whatever returns some value.
@ linear
Value of a sample (voltage) in a linear domain.
Helpers to generate common default channel subsets.
static std::vector< size_t > allChannels(size_t numChannels)
Represents a slice of analysis data.
bool isEmpty() const