diff --git a/alc/backends/pipewire.cpp b/alc/backends/pipewire.cpp index 0e99fdc4..68e6f527 100644 --- a/alc/backends/pipewire.cpp +++ b/alc/backends/pipewire.cpp @@ -579,6 +579,23 @@ uint32_t get_param_range(const spa_pod *value, const al::span,3> vals) return 0; } +template +std::vector> get_param_enum(const spa_pod *value) +{ + std::vector> vals; + + uint32_t nvals{}, choice{}; + value = spa_pod_get_values(value, &nvals, &choice); + + if(get_pod_type(value) == T && nvals > 0 && choice == SPA_CHOICE_Enum) + { + vals.resize(nvals); + std::copy_n(get_pod_body>(value), vals.size(), vals.begin()); + } + + return vals; +} + template uint32_t get_param_array(const spa_pod *value, const al::span,N> vals) { @@ -596,20 +613,43 @@ al::optional> get_param(const spa_pod *value) void parse_srate(DeviceNode *node, const spa_pod *value) { - /* TODO: Can this be anything else? An "enum" choice? Floats? Or will the - * sample rate always be a range choice between ints? + /* TODO: Can this be anything else? Floats? Will the sample rate always be + * a range or enum choice of ints? */ if(get_pod_type(value) == SPA_TYPE_Choice) { int32_t srate[3]{}; - if(get_param_range(value, al::span{srate}) < 1) + if(get_param_range(value, al::span{srate}) > 0) + { + /* [0] is the default, [1] is the min, and [2] is the max. */ + TRACE("Device ID %u sample rate: %d (range: %d -> %d)\n", node->mId, srate[0], srate[1], + srate[2]); + srate[0] = clampi(srate[0], MIN_OUTPUT_RATE, MAX_OUTPUT_RATE); + node->mSampleRate = static_cast(srate[0]); return; + } - /* [0] is the default, [1] is the min, and [2] is the max. */ - TRACE("Device ID %u sample rate: %d (range: %d -> %d)\n", node->mId, srate[0], srate[1], - srate[2]); - srate[0] = clampi(srate[0], MIN_OUTPUT_RATE, MAX_OUTPUT_RATE); - node->mSampleRate = static_cast(srate[0]); + auto srates = get_param_enum(value); + if(!srates.empty()) + { + /* [0] is the default, [1...size()-1] are available selections. */ + std::string others{(srates.size() > 1) ? std::to_string(srates[1]) : std::string{}}; + for(size_t i{2};i < srates.size();++i) + { + others += ", "; + others += std::to_string(srates[i]); + } + TRACE("Device ID %u sample rate: %d (%s)\n", node->mId, srates[0], others.c_str()); + for(const auto &rate : srates) + { + if(rate >= MIN_OUTPUT_RATE && rate <= MAX_OUTPUT_RATE) + { + node->mSampleRate = static_cast(rate); + break; + } + } + return; + } } }