2019-11-05 13:02:41 +01:00
|
|
|
/*
|
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
*
|
|
|
|
|
* Support for the verb/device/modifier core logic and API,
|
|
|
|
|
* command line tool and file parser was kindly sponsored by
|
|
|
|
|
* Texas Instruments Inc.
|
|
|
|
|
* Support for multiple active modifiers and devices,
|
|
|
|
|
* transition sequences, multiple client access and user defined use
|
|
|
|
|
* cases was kindly sponsored by Wolfson Microelectronics PLC.
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2019 Red Hat Inc.
|
|
|
|
|
* Authors: Jaroslav Kysela <perex@perex.cz>
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "ucm_local.h"
|
2019-11-06 10:44:29 +01:00
|
|
|
#include <regex.h>
|
2019-11-05 13:02:41 +01:00
|
|
|
|
|
|
|
|
static int get_string(snd_config_t *compound, const char *key, const char **str)
|
|
|
|
|
{
|
|
|
|
|
snd_config_t *node;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = snd_config_search(compound, key, &node);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
return snd_config_get_string(node, str);
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-06 09:51:47 +01:00
|
|
|
static int if_eval_string(snd_use_case_mgr_t *uc_mgr, snd_config_t *eval)
|
|
|
|
|
{
|
|
|
|
|
const char *string1 = NULL, *string2 = NULL;
|
|
|
|
|
char *s1, *s2;
|
|
|
|
|
int err;
|
|
|
|
|
|
2020-05-26 15:32:36 +02:00
|
|
|
if (uc_mgr->conf_format >= 3) {
|
|
|
|
|
err = get_string(eval, "Empty", &string1);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("String error (If.Condition.Empty)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (string1) {
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s1, string1);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
err = s1 == NULL || s1[0] == '\0';
|
|
|
|
|
free(s1);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-06 09:51:47 +01:00
|
|
|
err = get_string(eval, "String1", &string1);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("String error (If.Condition.String1)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "String2", &string2);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("String error (If.Condition.String2)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (string1 || string2) {
|
|
|
|
|
if (string1 == NULL) {
|
|
|
|
|
uc_error("If.Condition.String1 not defined");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (string2 == NULL) {
|
|
|
|
|
uc_error("If.Condition.String2 not defined");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s1, string1);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s2, string2);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
free(s1);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-11-28 19:25:03 +01:00
|
|
|
err = strcasecmp(s1, s2) == 0;
|
2019-11-06 09:51:47 +01:00
|
|
|
free(s2);
|
|
|
|
|
free(s1);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Haystack", &string1);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("String error (If.Condition.Haystack)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Needle", &string2);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("String error (If.Condition.Needle)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (string1 || string2) {
|
|
|
|
|
if (string1 == NULL) {
|
|
|
|
|
uc_error("If.Condition.Haystack not defined");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
if (string2 == NULL) {
|
|
|
|
|
uc_error("If.Condition.Needle not defined");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s1, string1);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s2, string2);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
free(s1);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2019-11-28 19:25:03 +01:00
|
|
|
err = strstr(s1, s2) != NULL;
|
2019-11-06 09:51:47 +01:00
|
|
|
free(s2);
|
|
|
|
|
free(s1);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uc_error("Unknown String condition arguments");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-06 10:44:29 +01:00
|
|
|
static int if_eval_regex_match(snd_use_case_mgr_t *uc_mgr, snd_config_t *eval)
|
|
|
|
|
{
|
|
|
|
|
const char *string, *regex_string;
|
|
|
|
|
char *s;
|
|
|
|
|
regex_t re;
|
|
|
|
|
int options = REG_EXTENDED | REG_ICASE;
|
|
|
|
|
regmatch_t match[1];
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "String", &string);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
uc_error("RegexMatch error (If.Condition.String)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Regex", ®ex_string);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
uc_error("RegexMatch error (If.Condition.Regex)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s, regex_string);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
err = regcomp(&re, s, options);
|
|
|
|
|
free(s);
|
|
|
|
|
if (err) {
|
|
|
|
|
uc_error("Regex '%s' compilation failed (code %d)", err);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s, string);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
regfree(&re);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
err = regexec(&re, s, ARRAY_SIZE(match), match, 0);
|
|
|
|
|
free(s);
|
|
|
|
|
regfree(&re);
|
|
|
|
|
return err == 0;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
static int if_eval_control_exists(snd_use_case_mgr_t *uc_mgr, snd_config_t *eval)
|
|
|
|
|
{
|
|
|
|
|
snd_ctl_t *ctl;
|
2020-06-08 09:03:56 +02:00
|
|
|
struct ctl_list *ctl_list;
|
2019-11-29 09:33:50 +01:00
|
|
|
const char *device = NULL, *ctldef, *enumval = NULL, *name;
|
2019-11-05 13:02:41 +01:00
|
|
|
snd_ctl_elem_id_t *elem_id;
|
|
|
|
|
snd_ctl_elem_info_t *elem_info;
|
2019-11-29 09:33:50 +01:00
|
|
|
snd_ctl_elem_type_t type;
|
2019-11-05 13:02:41 +01:00
|
|
|
char *s;
|
2019-11-29 09:33:50 +01:00
|
|
|
int err, i, items;
|
2019-11-06 09:51:47 +01:00
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
snd_ctl_elem_id_alloca(&elem_id);
|
|
|
|
|
snd_ctl_elem_info_alloca(&elem_info);
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Device", &device);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
2019-11-06 09:51:47 +01:00
|
|
|
uc_error("ControlExists error (If.Condition.Device)");
|
2019-11-05 13:02:41 +01:00
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Control", &ctldef);
|
|
|
|
|
if (err < 0) {
|
2019-11-06 09:51:47 +01:00
|
|
|
uc_error("ControlExists error (If.Condition.Control)");
|
2019-11-05 13:02:41 +01:00
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-29 09:33:50 +01:00
|
|
|
err = get_string(eval, "ControlEnum", &enumval);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("ControlExists error (If.Condition.ControlEnum)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s, ctldef);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
err = snd_ctl_ascii_elem_id_parse(elem_id, s);
|
|
|
|
|
free(s);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
uc_error("unable to parse element identificator (%s)", ctldef);
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (device == NULL) {
|
|
|
|
|
ctl = uc_mgr_get_ctl(uc_mgr);
|
|
|
|
|
if (ctl == NULL) {
|
|
|
|
|
uc_error("cannot determine control device");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s, device);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
2020-06-08 09:03:56 +02:00
|
|
|
err = uc_mgr_open_ctl(uc_mgr, &ctl_list, s, 1);
|
2019-11-05 13:02:41 +01:00
|
|
|
free(s);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
2020-06-08 09:03:56 +02:00
|
|
|
ctl = ctl_list->ctl;
|
2019-11-05 13:02:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
snd_ctl_elem_info_set_id(elem_info, elem_id);
|
|
|
|
|
err = snd_ctl_elem_info(ctl, elem_info);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
2019-11-29 09:33:50 +01:00
|
|
|
if (enumval) {
|
|
|
|
|
type = snd_ctl_elem_info_get_type(elem_info);
|
|
|
|
|
if (type != SND_CTL_ELEM_TYPE_ENUMERATED)
|
|
|
|
|
return 0;
|
|
|
|
|
err = uc_mgr_get_substituted_value(uc_mgr, &s, enumval);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
items = snd_ctl_elem_info_get_items(elem_info);
|
|
|
|
|
for (i = 0; i < items; i++) {
|
|
|
|
|
snd_ctl_elem_info_set_item(elem_info, i);
|
|
|
|
|
err = snd_ctl_elem_info(ctl, elem_info);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
free(s);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
name = snd_ctl_elem_info_get_item_name(elem_info);
|
|
|
|
|
if (strcasecmp(name, s) == 0) {
|
|
|
|
|
free(s);
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
free(s);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int if_eval(snd_use_case_mgr_t *uc_mgr, snd_config_t *eval)
|
|
|
|
|
{
|
|
|
|
|
const char *type;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
if (snd_config_get_type(eval) != SND_CONFIG_TYPE_COMPOUND) {
|
|
|
|
|
uc_error("compound type expected for If.Condition");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = get_string(eval, "Type", &type);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
uc_error("type block error (If.Condition)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-24 19:17:49 +02:00
|
|
|
if (strcmp(type, "AlwaysTrue") == 0)
|
|
|
|
|
return 1;
|
2019-11-05 13:02:41 +01:00
|
|
|
|
2019-11-06 09:51:47 +01:00
|
|
|
if (strcmp(type, "String") == 0)
|
|
|
|
|
return if_eval_string(uc_mgr, eval);
|
|
|
|
|
|
2020-06-24 19:17:49 +02:00
|
|
|
if (strcmp(type, "ControlExists") == 0)
|
|
|
|
|
return if_eval_control_exists(uc_mgr, eval);
|
|
|
|
|
|
2019-11-06 10:44:29 +01:00
|
|
|
if (strcmp(type, "RegexMatch") == 0)
|
|
|
|
|
return if_eval_regex_match(uc_mgr, eval);
|
|
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
uc_error("unknown If.Condition.Type");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int if_eval_one(snd_use_case_mgr_t *uc_mgr,
|
|
|
|
|
snd_config_t *cond,
|
2019-11-14 16:57:22 +01:00
|
|
|
snd_config_t **result,
|
|
|
|
|
snd_config_t **before,
|
|
|
|
|
snd_config_t **after)
|
2019-11-05 13:02:41 +01:00
|
|
|
{
|
|
|
|
|
snd_config_t *expr, *_true = NULL, *_false = NULL;
|
|
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
*result = NULL;
|
|
|
|
|
|
|
|
|
|
if (snd_config_get_type(cond) != SND_CONFIG_TYPE_COMPOUND) {
|
|
|
|
|
uc_error("compound type expected for If.1");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (snd_config_search(cond, "Condition", &expr) < 0) {
|
|
|
|
|
uc_error("condition block expected (If)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = snd_config_search(cond, "True", &_true);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("true block error (If)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = snd_config_search(cond, "False", &_false);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("false block error (If)");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-14 16:57:22 +01:00
|
|
|
err = snd_config_search(cond, "Before", before);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("before block identifier error");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = snd_config_search(cond, "After", after);
|
|
|
|
|
if (err < 0 && err != -ENOENT) {
|
|
|
|
|
uc_error("before block identifier error");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-05 13:02:41 +01:00
|
|
|
err = if_eval(uc_mgr, expr);
|
|
|
|
|
if (err > 0) {
|
|
|
|
|
*result = _true;
|
|
|
|
|
return 0;
|
|
|
|
|
} else if (err == 0) {
|
|
|
|
|
*result = _false;
|
|
|
|
|
return 0;
|
|
|
|
|
} else {
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
static void config_dump(snd_config_t *cfg)
|
|
|
|
|
{
|
|
|
|
|
snd_output_t *out;
|
|
|
|
|
snd_output_stdio_attach(&out, stderr, 0);
|
|
|
|
|
snd_output_printf(out, "-----\n");
|
|
|
|
|
snd_config_save(cfg, out);
|
|
|
|
|
snd_output_close(out);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* put back the result from all conditions to the parent
|
|
|
|
|
*/
|
|
|
|
|
int uc_mgr_evaluate_condition(snd_use_case_mgr_t *uc_mgr,
|
|
|
|
|
snd_config_t *parent,
|
|
|
|
|
snd_config_t *cond)
|
|
|
|
|
{
|
2020-05-16 15:47:19 +02:00
|
|
|
snd_config_iterator_t i, next;
|
|
|
|
|
snd_config_t *a, *n, *before, *after;
|
2019-11-05 13:02:41 +01:00
|
|
|
int err;
|
|
|
|
|
|
|
|
|
|
if (uc_mgr->conf_format < 2) {
|
|
|
|
|
uc_error("conditions are not supported for v1 syntax");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (snd_config_get_type(cond) != SND_CONFIG_TYPE_COMPOUND) {
|
|
|
|
|
uc_error("compound type expected for If");
|
|
|
|
|
return -EINVAL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
snd_config_for_each(i, next, cond) {
|
|
|
|
|
n = snd_config_iterator_entry(i);
|
2019-11-14 16:57:22 +01:00
|
|
|
before = after = NULL;
|
|
|
|
|
err = if_eval_one(uc_mgr, n, &a, &before, &after);
|
2019-11-05 13:02:41 +01:00
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
2019-11-14 16:57:22 +01:00
|
|
|
if (a == NULL)
|
|
|
|
|
continue;
|
2020-05-16 15:47:19 +02:00
|
|
|
err = uc_mgr_evaluate_inplace(uc_mgr, a);
|
|
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
2020-07-06 16:34:33 +02:00
|
|
|
err = uc_mgr_config_tree_merge(uc_mgr, parent, a, before, after);
|
2020-05-16 15:47:19 +02:00
|
|
|
if (err < 0)
|
|
|
|
|
return err;
|
|
|
|
|
snd_config_delete(a);
|
2019-11-05 13:02:41 +01:00
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|