~aleteoryx/muditaos

ref: 25a5d90f4e12ee5d33f4202170f18b59e16a9564 muditaos/module-audio/Audio/Profiles/ProfileConfigUtils.cpp -rw-r--r-- 5.4 KiB
25a5d90f — rrandomsky [CP-2156] Fixed no response when editing a contact to have the same number as another 2 years ago
                                                                                
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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
// Copyright (c) 2017-2023, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md

#include "ProfileConfigUtils.hpp"
#include <fstream>
#include <log/log.hpp>
#include <json11.hpp>
#include <magic_enum.hpp>
namespace audio
{
    namespace strings
    {
        constexpr inline auto samplerate        = "samplerate";
        constexpr inline auto bitWidth          = "bitWidth";
        constexpr inline auto flags             = "flags";
        constexpr inline auto outputVolume      = "outputVolume";
        constexpr inline auto outputPath        = "outputPath";
        constexpr inline auto inputGain         = "inputGain";
        constexpr inline auto inputPath         = "inputPath";
        constexpr inline auto filterParams      = "filterParams";
        constexpr inline auto filterType        = "filterType";
        constexpr inline auto frequency         = "frequency";
        constexpr inline auto Q                 = "Q";
        constexpr inline auto gain              = "gain";
        constexpr inline auto playbackPathGain  = "playbackPathGain";
        constexpr inline auto playbackPathAtten = "playbackPathAtten";
    } // namespace strings

    namespace utils
    {

        template <typename E, typename T>
        constexpr inline typename std::enable_if<std::is_enum<E>::value && std::is_integral<T>::value, E>::type toEnum(
            T value) noexcept
        {
            return static_cast<E>(value);
        }

        equalizer::FilterType toFilterType(const std::string &filterName)
        {
            auto filterType = magic_enum::enum_cast<equalizer::FilterType>(filterName);
            if (filterType.has_value()) {
                return filterType.value();
            }
            else {
                LOG_ERROR("Unknown filter type, using none");
                return equalizer::FilterType::None;
            }
        }

        const std::string readFileToString(std::filesystem::path filePath)
        {
            std::ifstream file;
            std::string configString;
            LOG_DEBUG("Reading profile configuration: %s", filePath.c_str());
            file.open(filePath);
            if (not file.is_open()) {
                LOG_ERROR("Can't open profile configuration file, using defaults!");
                throw std::invalid_argument("Can't open file!");
            }
            while (file) {
                std::string line;
                std::getline(file, line);
                configString += line;
            }
            file.close();
            return configString;
        }
    } // namespace utils

    const audio::codec::Configuration loadConfigurationFromFile(std::filesystem::path filePath)
    {

        auto configString = utils::readFileToString(filePath);
        audio::codec::Configuration config;

        json11::Json configJson;
        std::string err;
        configJson = json11::Json::parse(configString.c_str(), err);
        if (!err.empty()) {
            LOG_ERROR("Failed parsing device string!");
            throw std::invalid_argument("Can't parse the file!");
        }
        config.sampleRate_Hz = configJson[strings::samplerate].int_value();
        config.bitWidth      = configJson[strings::bitWidth].int_value();
        config.flags         = configJson[strings::flags].int_value();
        config.outputVolume  = configJson[strings::outputVolume].number_value();
        config.inputGain     = configJson[strings::inputGain].number_value();
        config.inputPath     = utils::toEnum<codec::InputPath>(configJson[strings::inputPath].int_value());
        config.outputPath    = utils::toEnum<codec::OutputPath>(configJson[strings::outputPath].int_value());

        std::uint8_t playbackPathGain  = configJson[strings::playbackPathGain].int_value();
        std::uint8_t playbackPathAtten = configJson[strings::playbackPathAtten].int_value();

        constexpr std::uint8_t playbackPathGainMax = 3;
        if (playbackPathGain <= playbackPathGainMax) {
            config.playbackPathGain = playbackPathGain;
        }
        else {
            LOG_WARN("PlaybackPathGain value out of range (%u), using fallback value %u!",
                     playbackPathGain,
                     config.playbackPathGain);
        }

        constexpr std::uint8_t playbackPathAttenMax = 15;
        if (playbackPathAtten <= playbackPathAttenMax) {
            config.playbackPathAtten = playbackPathAtten;
        }
        else {
            LOG_WARN("PlaybackPathAtten value out of range (%u), using fallback value %u!",
                     playbackPathAtten,
                     config.playbackPathAtten);
        }

        json11::Json::array paramsArray;
        audio::equalizer::Equalizer filterParams;
        paramsArray = configJson[strings::filterParams].array_items();

        for (size_t i = 0; i < equalizer::bands; i++) {
            auto filterType = utils::toFilterType(paramsArray[i][strings::filterType].string_value());
            auto frequency  = paramsArray[i][strings::frequency].number_value();
            auto samplerate = paramsArray[i][strings::samplerate].int_value();
            auto Q          = paramsArray[i][strings::Q].number_value();
            auto gain       = paramsArray[i][strings::gain].number_value();

            filterParams.at(i) = qfilter_CalculateCoeffs(filterType, frequency, samplerate, Q, gain);
        }

        config.filterCoefficients = filterParams;
        return config;
    }

} // namespace audio