filter/route: merge the two loops, one-pass parsing

This commit is contained in:
Max Kellermann 2013-10-18 12:15:28 +02:00
parent de5be62da5
commit 5e103b5fb7

View File

@ -48,10 +48,9 @@
#include "FilterInternal.hxx" #include "FilterInternal.hxx"
#include "FilterRegistry.hxx" #include "FilterRegistry.hxx"
#include "pcm/PcmBuffer.hxx" #include "pcm/PcmBuffer.hxx"
#include "util/StringUtil.hxx"
#include "util/Error.hxx" #include "util/Error.hxx"
#include <glib.h>
#include <algorithm> #include <algorithm>
#include <assert.h> #include <assert.h>
@ -135,89 +134,71 @@ RouteFilter::Configure(const config_param &param, Error &error) {
* dynamic realloc() instead of one count run and one malloc(). * dynamic realloc() instead of one count run and one malloc().
*/ */
gchar **tokens;
int number_of_copies;
// A cowardly default, just passthrough stereo
const char *const routes = param.GetBlockValue("routes", "0>0, 1>1");
std::fill_n(sources, MAX_CHANNELS, -1); std::fill_n(sources, MAX_CHANNELS, -1);
min_input_channels = 0; min_input_channels = 0;
min_output_channels = 0; min_output_channels = 0;
tokens = g_strsplit(routes, ",", 255); // A cowardly default, just passthrough stereo
number_of_copies = g_strv_length(tokens); const char *routes = param.GetBlockValue("routes", "0>0, 1>1");
while (true) {
routes = strchug_fast(routes);
// Start by figuring out a few basic things about the routing set char *endptr;
for (int c=0; c<number_of_copies; ++c) { const unsigned source = strtoul(routes, &endptr, 10);
endptr = strchug_fast(endptr);
// String and int representations of the source/destination if (endptr == routes || *endptr != '>') {
gchar **sd; error.Set(config_domain,
"Malformed 'routes' specification");
// Squeeze whitespace
g_strstrip(tokens[c]);
// Split the a>b string into source and destination
sd = g_strsplit(tokens[c], ">", 2);
if (g_strv_length(sd) != 2) {
error.Format(config_domain,
"Invalid copy around %d in routes spec: %s",
param.line, tokens[c]);
g_strfreev(sd);
g_strfreev(tokens);
return false; return false;
} }
unsigned source = strtoul(sd[0], NULL, 10); if (source >= MAX_CHANNELS) {
unsigned dest = strtoul(sd[1], NULL, 10); error.Format(config_domain,
"Invalid source channel number: %u",
source);
return false;
}
// Keep track of the highest channel numbers seen
// as either in- or outputs
if (source >= min_input_channels) if (source >= min_input_channels)
min_input_channels = source + 1; min_input_channels = source + 1;
routes = strchug_fast(endptr + 1);
unsigned dest = strtoul(routes, &endptr, 10);
endptr = strchug_fast(endptr);
if (endptr == routes) {
error.Set(config_domain,
"Malformed 'routes' specification");
return false;
}
if (dest >= MAX_CHANNELS) {
error.Format(config_domain,
"Invalid destination channel number: %u",
dest);
return false;
}
if (dest >= min_output_channels) if (dest >= min_output_channels)
min_output_channels = dest + 1; min_output_channels = dest + 1;
g_strfreev(sd); sources[dest] = source;
}
if (!audio_valid_channel_count(min_output_channels)) { routes = endptr;
g_strfreev(tokens);
error.Format(config_domain,
"Invalid number of output channels requested: %d",
min_output_channels);
return false;
}
// Run through the spec again, and save the if (*routes == 0)
// actual mapping output <- input break;
for (int c=0; c<number_of_copies; ++c) {
// String and int representations of the source/destination if (*routes != ',') {
gchar **sd; error.Set(config_domain,
"Malformed 'routes' specification");
// Split the a>b string into source and destination
sd = g_strsplit(tokens[c], ">", 2);
if (g_strv_length(sd) != 2) {
error.Format(config_domain,
"Invalid copy around %d in routes spec: %s",
param.line, tokens[c]);
g_strfreev(sd);
g_strfreev(tokens);
return false; return false;
} }
unsigned source = strtoul(sd[0], NULL, 10); ++routes;
unsigned dest = strtoul(sd[1], NULL, 10);
sources[dest] = source;
g_strfreev(sd);
} }
g_strfreev(tokens);
return true; return true;
} }