filter/route: merge the two loops, one-pass parsing
This commit is contained in:
parent
de5be62da5
commit
5e103b5fb7
@ -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 ¶m, 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user