/* Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include <ctype.h>
#include <stdlib.h>
#include <syslog.h>
#include "array.h"
#include "cras_expr.h"
static const char *copy_str(const char *begin, const char *end)
{
char *s = malloc(end - begin + 1);
memcpy(s, begin, end - begin);
s[end - begin] = '\0';
return s;
}
static void value_set_boolean(struct cras_expr_value *value, char boolean)
{
cras_expr_value_free(value);
value->type = CRAS_EXPR_VALUE_TYPE_BOOLEAN;
value->u.boolean = !!boolean;
}
static void value_set_integer(struct cras_expr_value *value, int integer)
{
cras_expr_value_free(value);
value->type = CRAS_EXPR_VALUE_TYPE_INT;
value->u.integer = integer;
}
static void value_set_string2(struct cras_expr_value *value, const char *begin,
const char *end)
{
cras_expr_value_free(value);
value->type = CRAS_EXPR_VALUE_TYPE_STRING;
value->u.string = copy_str(begin, end);
}
static void value_set_string(struct cras_expr_value *value, const char *str)
{
value_set_string2(value, str, str + strlen(str));
}
static void cras_expr_value_set_function(struct cras_expr_value *value,
cras_expr_function_type function)
{
cras_expr_value_free(value);
value->type = CRAS_EXPR_VALUE_TYPE_FUNCTION;
value->u.function = function;
}
static void copy_value(struct cras_expr_value *value,
struct cras_expr_value *original)
{
cras_expr_value_free(value); /* free the original value first */
value->type = original->type;
switch (value->type) {
case CRAS_EXPR_VALUE_TYPE_NONE:
break;
case CRAS_EXPR_VALUE_TYPE_BOOLEAN:
value->u.boolean = original->u.boolean;
break;
case CRAS_EXPR_VALUE_TYPE_INT:
value->u.integer = original->u.integer;
break;
case CRAS_EXPR_VALUE_TYPE_STRING:
value->u.string = strdup(original->u.string);
break;
case CRAS_EXPR_VALUE_TYPE_FUNCTION:
value->u.function = original->u.function;
break;
}
}
void cras_expr_value_free(struct cras_expr_value *value)
{
switch (value->type) {
case CRAS_EXPR_VALUE_TYPE_STRING:
free((char *)value->u.string);
value->u.string = NULL;
break;
case CRAS_EXPR_VALUE_TYPE_NONE:
case CRAS_EXPR_VALUE_TYPE_BOOLEAN:
case CRAS_EXPR_VALUE_TYPE_INT:
case CRAS_EXPR_VALUE_TYPE_FUNCTION:
break;
}
value->type = CRAS_EXPR_VALUE_TYPE_NONE;
}
void cras_expr_value_dump(struct dumper *d, const struct cras_expr_value *value)
{
switch (value->type) {
case CRAS_EXPR_VALUE_TYPE_NONE:
dumpf(d, "none");
break;
case CRAS_EXPR_VALUE_TYPE_BOOLEAN:
dumpf(d, "boolean(%s)",
value->u.boolean ? "true" : "false");
break;
case CRAS_EXPR_VALUE_TYPE_INT:
dumpf(d, "integer(%d)", value->u.integer);
break;
case CRAS_EXPR_VALUE_TYPE_STRING:
dumpf(d, "string(%s)", value->u.string);
break;
case CRAS_EXPR_VALUE_TYPE_FUNCTION:
dumpf(d, "function(%p)", value->u.function);
break;
}
}
static struct cras_expr_value *find_value(struct cras_expr_env *env,
const char *name)
{
int i;
const char **key;
FOR_ARRAY_ELEMENT(&env->keys, i, key) {
if (strcmp(*key, name) == 0)
return ARRAY_ELEMENT(&env->values, i);
}
return NULL;
}
/* Insert a (key, value) pair to the environment. The value is
* initialized to zero. Return the pointer to value so it can be set
* to the proper value. */
static struct cras_expr_value *insert_value(struct cras_expr_env *env,
const char *key)
{
*ARRAY_APPEND_ZERO(&env->keys) = strdup(key);
return ARRAY_APPEND_ZERO(&env->values);
}
static struct cras_expr_value *find_or_insert_value(struct cras_expr_env *env,
const char *key)
{
struct cras_expr_value *value = find_value(env, key);
if (!value)
value = insert_value(env, key);
return value;
}
static void function_not(cras_expr_value_array *operands,
struct cras_expr_value *result)
{
struct cras_expr_value *value;
int is_false;
if (ARRAY_COUNT(operands) != 2) {
cras_expr_value_free(result);
syslog(LOG_ERR, "not takes one argument");
return;
}
value = ARRAY_ELEMENT(operands, 1);
is_false = (value->type == CRAS_EXPR_VALUE_TYPE_BOOLEAN &&
!value->u.boolean);
value_set_boolean(result, is_false);
}
static void function_and(cras_expr_value_array *operands,
struct cras_expr_value *result)
{
int i;
struct cras_expr_value *value;
int n = ARRAY_COUNT(operands);
/* no operands -- return #t */
if (n <= 1) {
value_set_boolean(result, 1);
return;
}
/* if there is any #f, return it */
FOR_ARRAY_ELEMENT(operands, i, value) {
if (i == 0)
continue; /* ignore "and" itself */
if (value->type == CRAS_EXPR_VALUE_TYPE_BOOLEAN &&
!value->u.boolean) {
value_set_boolean(result, 0);
return;
}
}
/* otherwise return the last element */
copy_value(result, ARRAY_ELEMENT(operands, n - 1));
}
static void function_or(cras_expr_value_array *operands,
struct cras_expr_value *result)
{
int i;
struct cras_expr_value *value;
FOR_ARRAY_ELEMENT(operands, i, value) {
if (i == 0)
continue; /* ignore "or" itself */
if (value->type != CRAS_EXPR_VALUE_TYPE_BOOLEAN ||
value->u.boolean) {
copy_value(result, value);
return;
}
}
value_set_boolean(result, 0);
}
static char function_equal_real(cras_expr_value_array *operands)
{
int i;
struct cras_expr_value *value, *prev;
FOR_ARRAY_ELEMENT(operands, i, value) {
if (i <= 1)
continue; /* ignore equal? and first operand */
/* compare with the previous operand */
prev = ARRAY_ELEMENT(operands, i - 1);
if (prev->type != value->type)
return 0;
switch (prev->type) {
case CRAS_EXPR_VALUE_TYPE_NONE:
break;
case CRAS_EXPR_VALUE_TYPE_BOOLEAN:
if (prev->u.boolean != value->u.boolean)
return 0;
break;
case CRAS_EXPR_VALUE_TYPE_INT:
if (prev->u.integer != value->u.integer)
return 0;
break;
case CRAS_EXPR_VALUE_TYPE_STRING:
if (strcmp(prev->u.string, value->u.string) != 0)
return 0;
break;
case CRAS_EXPR_VALUE_TYPE_FUNCTION:
if (prev->u.function != value->u.function)
return 0;
break;
}
}
return 1;
}
static void function_equal(cras_expr_value_array *operands,
struct cras_expr_value *result)
{
value_set_boolean(result, function_equal_real(operands));
}
static void env_set_variable(struct cras_expr_env *env, const char *name,
struct cras_expr_value *new_value)
{
struct cras_expr_value *value = find_or_insert_value(env, name);
copy_value(value, new_value);
}
void cras_expr_env_install_builtins(struct cras_expr_env *env)
{
struct cras_expr_value value = CRAS_EXPR_VALUE_INIT;
/* initialize env with builtin functions */
cras_expr_value_set_function(&value, &function_not);
env_set_variable(env, "not", &value);
cras_expr_value_set_function(&value, &function_and);
env_set_variable(env, "and", &value);
cras_expr_value_set_function(&value, &function_or);
env_set_variable(env, "or", &value);
cras_expr_value_set_function(&value, &function_equal);
env_set_variable(env, "equal?", &value);
cras_expr_value_free(&value);
}
void cras_expr_env_set_variable_boolean(struct cras_expr_env *env,
const char *name, char boolean)
{
struct cras_expr_value *value = find_or_insert_value(env, name);
value_set_boolean(value, boolean);
}
void cras_expr_env_set_variable_integer(struct cras_expr_env *env,
const char *name, int integer)
{
struct cras_expr_value *value = find_or_insert_value(env, name);
value_set_integer(value, integer);
}
void cras_expr_env_set_variable_string(struct cras_expr_env *env,
const char *name, const char *str)
{
struct cras_expr_value *value = find_or_insert_value(env, name);
value_set_string(value, str);
}
void cras_expr_env_free(struct cras_expr_env *env)
{
int i;
const char **key;
struct cras_expr_value *value;
FOR_ARRAY_ELEMENT(&env->keys, i, key) {
free((char *)*key);
}
FOR_ARRAY_ELEMENT(&env->values, i, value) {
cras_expr_value_free(value);
}
ARRAY_FREE(&env->keys);
ARRAY_FREE(&env->values);
}
void cras_expr_env_dump(struct dumper *d, const struct cras_expr_env *env)
{
int i;
const char **key;
struct cras_expr_value *value;
dumpf(d, "--- environment ---\n");
FOR_ARRAY_ELEMENT(&env->keys, i, key) {
dumpf(d, " key=%s,", *key);
dumpf(d, " value=");
value = ARRAY_ELEMENT(&env->values, i);
cras_expr_value_dump(d, value);
dumpf(d, "\n");
}
}
static struct cras_expr_expression *new_boolean_literal(char boolean)
{
struct cras_expr_expression *expr;
expr = calloc(1, sizeof(struct cras_expr_expression));
expr->type = EXPR_TYPE_LITERAL;
value_set_boolean(&expr->u.literal, boolean);
return expr;
}
static struct cras_expr_expression *new_integer_literal(int integer)
{
struct cras_expr_expression *expr;
expr = calloc(1, sizeof(struct cras_expr_expression));
expr->type = EXPR_TYPE_LITERAL;
value_set_integer(&expr->u.literal, integer);
return expr;
}
static struct cras_expr_expression *new_string_literal(const char *begin,
const char *end)
{
struct cras_expr_expression *expr;
expr = calloc(1, sizeof(struct cras_expr_expression));
expr->type = EXPR_TYPE_LITERAL;
value_set_string2(&expr->u.literal, begin, end);
return expr;
}
static struct cras_expr_expression *new_variable(const char *begin,
const char *end)
{
struct cras_expr_expression *expr;
expr = calloc(1, sizeof(struct cras_expr_expression));
expr->type = EXPR_TYPE_VARIABLE;
expr->u.variable = copy_str(begin, end);
return expr;
}
static struct cras_expr_expression *new_compound_expression()
{
struct cras_expr_expression *expr;
expr = calloc(1, sizeof(struct cras_expr_expression));
expr->type = EXPR_TYPE_COMPOUND;
return expr;
}
static void add_sub_expression(struct cras_expr_expression *expr,
struct cras_expr_expression *sub)
{
ARRAY_APPEND(&expr->u.children, sub);
}
static int is_identifier_char(char c)
{
if (isspace(c))
return 0;
if (c == '\0')
return 0;
if (isalpha(c))
return 1;
if (c == '_' || c == '-' || c == '?')
return 1;
return 0;
}
static struct cras_expr_expression *parse_one_expr(const char **str)
{
/* skip whitespace */
while (isspace(**str))
(*str)++;
if (**str == '\0')
return NULL;
/* boolean literal: #t, #f */
if (**str == '#') {
(*str)++;
char c = **str;
if (c == 't' || c == 'f') {
(*str)++;
return new_boolean_literal(c == 't');
} else {
syslog(LOG_ERR, "unexpected char after #: '%c'",
c);
}
return NULL;
}
/* integer literal: (-)[0-9]+ */
if (isdigit(**str) || (**str == '-' && isdigit((*str)[1])))
return new_integer_literal(strtol(*str, (char **)str, 10));
/* string literal: "..." */
if (**str == '"') {
const char *begin = *str + 1;
const char *end = strchr(begin, '"');
if (end == NULL) {
syslog(LOG_ERR, "no matching \"");
end = begin;
*str = begin;
} else {
*str = end + 1;
}
return new_string_literal(begin, end);
}
/* compound expression: (expr1 expr2 ...) */
if (**str == '(') {
(*str)++;
struct cras_expr_expression *expr = new_compound_expression();
while (1) {
struct cras_expr_expression *next = parse_one_expr(str);
if (next == NULL)
break;
add_sub_expression(expr, next);
}
if (**str != ')') {
syslog(LOG_ERR, "no matching ): found '%c'", **str);
cras_expr_expression_free(expr);
return NULL;
} else {
(*str)++;
}
return expr;
}
/* variable name */
if (is_identifier_char(**str)) {
const char *begin = *str;
while (is_identifier_char(**str))
(*str)++;
return new_variable(begin, *str);
}
return NULL;
}
struct cras_expr_expression *cras_expr_expression_parse(const char *str)
{
if (!str)
return NULL;
return parse_one_expr(&str);
}
static void dump_value(struct dumper *d, const struct cras_expr_value *value,
int indent)
{
switch (value->type) {
case CRAS_EXPR_VALUE_TYPE_NONE:
dumpf(d, "%*s(none)", indent, "");
break;
case CRAS_EXPR_VALUE_TYPE_BOOLEAN:
dumpf(d, "%*s%s", indent, "",
value->u.boolean ? "#t" : "#f");
break;
case CRAS_EXPR_VALUE_TYPE_INT:
dumpf(d, "%*s%d", indent, "", value->u.integer);
break;
case CRAS_EXPR_VALUE_TYPE_STRING:
dumpf(d, "%*s%s", indent, "", value->u.string);
break;
case CRAS_EXPR_VALUE_TYPE_FUNCTION:
dumpf(d, "%*s%p", indent, "", value->u.function);
break;
}
}
static void dump_one_expression(struct dumper *d,
const struct cras_expr_expression *expr,
int indent)
{
int i;
struct cras_expr_expression **sub;
switch (expr->type) {
case EXPR_TYPE_NONE:
dumpf(d, "%*snone", indent, "");
break;
case EXPR_TYPE_LITERAL:
dumpf(d, "%*sliteral", indent, "");
dump_value(d, &expr->u.literal, indent + 2);
break;
case EXPR_TYPE_VARIABLE:
dumpf(d, "%*svariable (%s)", indent, "",
expr->u.variable);
break;
case EXPR_TYPE_COMPOUND:
dumpf(d, "%*scompound", indent, "");
FOR_ARRAY_ELEMENT(&expr->u.children, i, sub) {
dump_one_expression(d, *sub, indent + 2);
}
break;
}
}
void cras_expr_expression_dump(struct dumper *d,
const struct cras_expr_expression *expr)
{
dump_one_expression(d, expr, 0);
}
void cras_expr_expression_free(struct cras_expr_expression *expr)
{
if (!expr)
return;
switch (expr->type) {
case EXPR_TYPE_NONE:
break;
case EXPR_TYPE_LITERAL:
cras_expr_value_free(&expr->u.literal);
break;
case EXPR_TYPE_VARIABLE:
free((char *)expr->u.variable);
break;
case EXPR_TYPE_COMPOUND:
{
int i;
struct cras_expr_expression **psub;
FOR_ARRAY_ELEMENT(&expr->u.children, i, psub) {
cras_expr_expression_free(*psub);
}
ARRAY_FREE(&expr->u.children);
break;
}
}
free(expr);
}
void cras_expr_expression_eval(struct cras_expr_expression *expr,
struct cras_expr_env *env,
struct cras_expr_value *result)
{
cras_expr_value_free(result);
switch (expr->type) {
case EXPR_TYPE_NONE:
break;
case EXPR_TYPE_LITERAL:
copy_value(result, &expr->u.literal);
break;
case EXPR_TYPE_VARIABLE:
{
struct cras_expr_value *value = find_value(env,
expr->u.variable);
if (value == NULL) {
syslog(LOG_ERR, "cannot find value for %s",
expr->u.variable);
} else {
copy_value(result, value);
}
break;
}
case EXPR_TYPE_COMPOUND:
{
int i;
struct cras_expr_expression **psub;
cras_expr_value_array values = ARRAY_INIT;
struct cras_expr_value *value;
FOR_ARRAY_ELEMENT(&expr->u.children, i, psub) {
value = ARRAY_APPEND_ZERO(&values);
cras_expr_expression_eval(*psub, env, value);
}
if (ARRAY_COUNT(&values) > 0) {
struct cras_expr_value *f = ARRAY_ELEMENT(&values, 0);
if (f->type == CRAS_EXPR_VALUE_TYPE_FUNCTION)
f->u.function(&values, result);
else
syslog(LOG_ERR,
"first element is not a function");
} else {
syslog(LOG_ERR, "empty compound expression?");
}
FOR_ARRAY_ELEMENT(&values, i, value) {
cras_expr_value_free(value);
}
ARRAY_FREE(&values);
break;
}
}
}
int cras_expr_expression_eval_int(struct cras_expr_expression *expr,
struct cras_expr_env *env,
int *integer)
{
int rc = 0;
struct cras_expr_value value = CRAS_EXPR_VALUE_INIT;
cras_expr_expression_eval(expr, env, &value);
if (value.type == CRAS_EXPR_VALUE_TYPE_INT) {
*integer = value.u.integer;
} else {
syslog(LOG_ERR, "value type is not integer (%d)", value.type);
rc = -1;
}
cras_expr_value_free(&value);
return rc;
}
int cras_expr_expression_eval_boolean(struct cras_expr_expression *expr,
struct cras_expr_env *env,
char *boolean)
{
int rc = 0;
struct cras_expr_value value = CRAS_EXPR_VALUE_INIT;
cras_expr_expression_eval(expr, env, &value);
if (value.type == CRAS_EXPR_VALUE_TYPE_BOOLEAN) {
*boolean = value.u.boolean;
} else {
syslog(LOG_ERR, "value type is not boolean (%d)", value.type);
rc = -1;
}
cras_expr_value_free(&value);
return rc;
}