server/src/util/translation.c

479 lines
10 KiB
C
Raw Normal View History

2010-08-08 10:06:34 +02:00
/* vi: set ts=2:
+-------------------+ Christian Schlittchen <corwin@amber.kn-bremen.de>
| | Enno Rehling <enno@eressea.de>
| Eressea PBEM host | Katja Zedel <katze@felidae.kn-bremen.de>
| (c) 1998 - 2003 | Henning Peters <faroul@beyond.kn-bremen.de>
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
+-------------------+ Stefan Reich <reich@halbling.de>
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
*/
#include <platform.h>
#include "translation.h"
2010-08-08 10:06:34 +02:00
#include "bsdstring.h"
#include "critbit.h"
#include "log.h"
2010-08-08 10:06:34 +02:00
/* libc includes */
#include <assert.h>
#include <ctype.h>
#include <string.h>
#include <stdlib.h>
#include <stdarg.h>
/**
** simple operand stack
**/
typedef struct opstack {
2011-03-07 08:02:35 +01:00
variant *begin;
variant *top;
2010-08-08 10:06:34 +02:00
int size;
} opstack;
2011-03-07 08:02:35 +01:00
variant opstack_pop(opstack ** stackp)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
opstack *stack = *stackp;
2010-08-08 10:06:34 +02:00
assert(stack);
2011-03-07 08:02:35 +01:00
assert(stack->top > stack->begin);
2010-08-08 10:06:34 +02:00
return *(--stack->top);
}
2011-03-07 08:02:35 +01:00
void opstack_push(opstack ** stackp, variant data)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
opstack *stack = *stackp;
if (stack == NULL) {
stack = (opstack *) malloc(sizeof(opstack));
2010-08-08 10:06:34 +02:00
stack->size = 1;
stack->begin = malloc(sizeof(variant) * stack->size);
stack->top = stack->begin;
*stackp = stack;
}
if (stack->top - stack->begin == stack->size) {
size_t pos = stack->top - stack->begin;
stack->size += stack->size;
stack->begin = realloc(stack->begin, sizeof(variant) * stack->size);
stack->top = stack->begin + pos;
2011-03-07 08:02:35 +01:00
}
2010-08-08 10:06:34 +02:00
*stack->top++ = data;
}
/**
** static buffer malloc
**/
#define BBUFSIZE 128*1024
static struct {
2011-03-07 08:02:35 +01:00
char *begin;
char *end;
char *last;
char *current;
2010-08-08 10:06:34 +02:00
} buffer;
2011-03-07 08:02:35 +01:00
char *balloc(size_t size)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
static int init = 0; /* STATIC_XCALL: used across calls */
if (!init) {
init = 1;
buffer.current = buffer.begin = malloc(BBUFSIZE);
buffer.end = buffer.begin + BBUFSIZE;
}
if (buffer.current + size > buffer.end) {
/* out of memory! */
return NULL;
}
buffer.last = buffer.current;
buffer.current += size;
return buffer.last;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void bfree(char *c)
2010-08-08 10:06:34 +02:00
/* only release this memory if it was part of the last allocation
* that's a joke, but who cares.
*/
{
2011-03-07 08:02:35 +01:00
if (c >= buffer.last && c < buffer.current)
buffer.current = c;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void brelease(void)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
buffer.last = buffer.current = buffer.begin;
2010-08-08 10:06:34 +02:00
}
/**
** constant values
**/
typedef struct variable {
2011-03-07 08:02:35 +01:00
struct variable *next;
const char *symbol;
variant value;
2010-08-08 10:06:34 +02:00
} variable;
2011-03-07 08:02:35 +01:00
static variable *variables;
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
static void free_variables(void)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
variables = NULL;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static void add_variable(const char *symbol, variant value)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
variable *var = (variable *) balloc(sizeof(variable));
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
var->value = value;
var->symbol = symbol;
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
var->next = variables;
variables = var;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static variable *find_variable(const char *symbol)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
variable *var = variables;
while (var) {
if (!strcmp(var->symbol, symbol))
break;
var = var->next;
}
return var;
2010-08-08 10:06:34 +02:00
}
/**
** constant values
**/
static struct critbit_tree functions = { 0 };
2010-08-08 10:06:34 +02:00
2011-03-07 08:02:35 +01:00
static void free_functions(void)
2010-08-08 10:06:34 +02:00
{
cb_clear(&functions);
functions.root = 0;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void add_function(const char *symbol, evalfun parse)
2010-08-08 10:06:34 +02:00
{
char buffer[64];
size_t len = strlen(symbol);
assert(len+1+sizeof(parse)<=sizeof(buffer));
len = cb_new_kv(symbol, len, &parse, sizeof(parse), buffer);
cb_insert(&functions, buffer, len);
2010-08-08 10:06:34 +02:00
}
static evalfun find_function(const char *symbol)
2010-08-08 10:06:34 +02:00
{
const void * matches;
if (cb_find_prefix(&functions, symbol, strlen(symbol)+1, &matches, 1, 0)) {
evalfun result;
cb_get_kv(matches, &result, sizeof(result));
return result;
2011-03-07 08:02:35 +01:00
}
return 0;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static const char *parse(opstack **, const char *in, const void *);
2010-08-08 10:06:34 +02:00
/* static const char * sample = "\"enno and $bool($if($eq($i,0),\"noone else\",\"$i other people\"))\""; */
2011-03-07 08:02:35 +01:00
static const char *parse_symbol(opstack ** stack, const char *in,
const void *userdata)
2010-08-08 10:06:34 +02:00
/* in is the symbol name and following text, starting after the $
* result goes on the stack
*/
{
bool braces = false;
2010-08-08 10:06:34 +02:00
char symbol[32];
2011-03-07 08:02:35 +01:00
char *cp = symbol; /* current position */
if (*in == '{') {
2010-08-08 10:06:34 +02:00
braces = true;
++in;
}
2011-03-07 08:02:35 +01:00
while (isalnum(*in) || *in == '.')
*cp++ = *in++;
2010-08-08 10:06:34 +02:00
*cp = '\0';
/* symbol will now contain the symbol name */
2011-03-07 08:02:35 +01:00
if (*in == '(') {
2010-08-08 10:06:34 +02:00
/* it's a function we need to parse, start by reading the parameters */
evalfun foo;
2010-08-08 10:06:34 +02:00
while (*in != ')') {
2011-03-07 08:02:35 +01:00
in = parse(stack, ++in, userdata); /* will push the result on the stack */
if (in == NULL)
return NULL;
2010-08-08 10:06:34 +02:00
}
++in;
foo = find_function(symbol);
2011-03-07 08:02:35 +01:00
if (foo == NULL) {
log_error("parser does not know about \"%s\" function.\n", symbol);
2010-08-08 10:06:34 +02:00
return NULL;
}
foo(stack, userdata); /* will pop parameters from stack (reverse order!) and push the result */
2010-08-08 10:06:34 +02:00
} else {
2011-03-07 08:02:35 +01:00
variable *var = find_variable(symbol);
if (braces && *in == '}') {
2010-08-08 10:06:34 +02:00
++in;
}
/* it's a constant (variable is a misnomer, but heck, const was taken;)) */
2011-03-07 08:02:35 +01:00
if (var == NULL) {
log_error("parser does not know about \"%s\" variable.\n", symbol);
2010-08-08 10:06:34 +02:00
return NULL;
}
opush(stack, var->value);
}
return in;
}
#define TOKENSIZE 4096
2011-03-07 08:02:35 +01:00
static const char *parse_string(opstack ** stack, const char *in,
const void *userdata)
{ /* (char*) -> char* */
char *c;
char *buffer = balloc(TOKENSIZE);
2010-08-08 10:06:34 +02:00
size_t size = TOKENSIZE - 1;
2011-03-07 08:02:35 +01:00
const char *ic = in;
char *oc = buffer;
2010-08-08 10:06:34 +02:00
/* mode flags */
bool f_escape = false;
bool bDone = false;
2010-08-08 10:06:34 +02:00
variant var;
while (*ic && !bDone) {
if (f_escape) {
f_escape = false;
switch (*ic) {
case 'n':
2011-03-07 08:02:35 +01:00
if (size > 0) {
*oc++ = '\n';
--size;
}
2010-08-08 10:06:34 +02:00
break;
case 't':
2011-03-07 08:02:35 +01:00
if (size > 0) {
*oc++ = '\t';
--size;
}
2010-08-08 10:06:34 +02:00
break;
default:
2011-03-07 08:02:35 +01:00
if (size > 0) {
*oc++ = *ic++;
--size;
}
2010-08-08 10:06:34 +02:00
}
} else {
int ch = (unsigned char)(*ic);
int bytes;
switch (ch) {
case '\\':
f_escape = true;
++ic;
break;
case '"':
bDone = true;
++ic;
break;
case '$':
ic = parse_symbol(stack, ++ic, userdata);
2011-03-07 08:02:35 +01:00
if (ic == NULL)
return NULL;
c = (char *)opop_v(stack);
2010-08-08 10:06:34 +02:00
bytes = (int)strlcpy(oc, c, size);
2011-03-07 08:02:35 +01:00
if (bytes < (int)size)
oc += bytes;
else
oc += size;
2010-08-08 10:06:34 +02:00
bfree(c);
break;
default:
2011-03-07 08:02:35 +01:00
if (size > 0) {
*oc++ = *ic++;
--size;
} else
++ic;
2010-08-08 10:06:34 +02:00
}
}
}
*oc++ = '\0';
bfree(oc);
var.v = buffer;
opush(stack, var);
return ic;
}
2011-03-07 08:02:35 +01:00
static const char *parse_int(opstack ** stack, const char *in)
2010-08-08 10:06:34 +02:00
{
int k = 0;
int vz = 1;
bool ok = false;
2010-08-08 10:06:34 +02:00
variant var;
do {
switch (*in) {
case '+':
++in;
break;
case '-':
++in;
2011-03-07 08:02:35 +01:00
vz = vz * -1;
2010-08-08 10:06:34 +02:00
break;
default:
ok = true;
}
} while (!ok);
2011-03-07 08:02:35 +01:00
while (isdigit(*(unsigned char *)in)) {
k = k * 10 + (*in++) - '0';
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
var.i = k * vz;
2010-08-08 10:06:34 +02:00
opush(stack, var);
return in;
}
2011-03-07 08:02:35 +01:00
static const char *parse(opstack ** stack, const char *inn,
const void *userdata)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
const char *b = inn;
2010-08-08 10:06:34 +02:00
while (*b) {
switch (*b) {
case '"':
return parse_string(stack, ++b, userdata);
break;
case '$':
return parse_symbol(stack, ++b, userdata);
break;
default:
2011-03-07 08:02:35 +01:00
if (isdigit(*(unsigned char *)b) || *b == '-' || *b == '+') {
2010-08-08 10:06:34 +02:00
return parse_int(stack, b);
2011-03-07 08:02:35 +01:00
} else
++b;
2010-08-08 10:06:34 +02:00
}
}
log_error("could not parse \"%s\". Probably invalid message syntax.", inn);
2010-08-08 10:06:34 +02:00
return NULL;
}
2011-03-07 08:02:35 +01:00
const char *translate(const char *format, const void *userdata,
const char *vars, variant args[])
2010-08-08 10:06:34 +02:00
{
int i = 0;
const char *ic = vars;
char symbol[32];
char *oc = symbol;
2011-03-07 08:02:35 +01:00
opstack *stack = NULL;
const char *rv;
2010-08-08 10:06:34 +02:00
brelease();
free_variables();
assert(format);
assert(*ic == 0 || isalnum(*ic));
while (*ic) {
*oc++ = *ic++;
if (!isalnum(*ic)) {
variant x = args[i++];
*oc = '\0';
oc = symbol;
2011-03-07 08:02:35 +01:00
add_variable(strcpy(balloc(strlen(symbol) + 1), symbol), x);
while (*ic && !isalnum(*ic))
++ic;
2010-08-08 10:06:34 +02:00
}
}
2011-03-07 08:02:35 +01:00
if (format[0] == '"') {
2010-08-08 10:06:34 +02:00
rv = parse(&stack, format, userdata);
2011-03-07 08:02:35 +01:00
} else {
2010-08-08 10:06:34 +02:00
rv = parse_string(&stack, format, userdata);
}
2011-03-07 08:02:35 +01:00
if (rv != NULL) {
2010-08-08 10:06:34 +02:00
if (rv[0]) {
log_error("residual data after parsing: %s\n", rv);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
rv = (const char *)opop(&stack).v;
2010-08-08 10:06:34 +02:00
free(stack->begin);
free(stack);
}
return rv;
}
2011-03-07 08:02:35 +01:00
static void eval_lt(opstack ** stack, const void *userdata)
{ /* (int, int) -> int */
2010-08-08 10:06:34 +02:00
int a = opop_i(stack);
int b = opop_i(stack);
2011-03-07 08:02:35 +01:00
int rval = (b < a) ? 1 : 0;
2010-08-08 10:06:34 +02:00
opush_i(stack, rval);
unused(userdata);
}
2011-03-07 08:02:35 +01:00
static void eval_eq(opstack ** stack, const void *userdata)
{ /* (int, int) -> int */
2010-08-08 10:06:34 +02:00
int a = opop_i(stack);
int b = opop_i(stack);
2011-03-07 08:02:35 +01:00
int rval = (a == b) ? 1 : 0;
2010-08-08 10:06:34 +02:00
opush_i(stack, rval);
unused(userdata);
}
2011-03-07 08:02:35 +01:00
static void eval_add(opstack ** stack, const void *userdata)
{ /* (int, int) -> int */
int a = opop_i(stack);
int b = opop_i(stack);
opush_i(stack, a + b);
unused(userdata);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static void eval_isnull(opstack ** stack, const void *userdata)
{ /* (int, int) -> int */
void *a = opop_v(stack);
opush_i(stack, (a == NULL) ? 1 : 0);
unused(userdata);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static void eval_if(opstack ** stack, const void *userdata)
{ /* (int, int) -> int */
void *a = opop_v(stack);
void *b = opop_v(stack);
int cond = opop_i(stack);
opush_v(stack, cond ? b : a);
unused(userdata);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
static void eval_strlen(opstack ** stack, const void *userdata)
{ /* string -> int */
const char *c = (const char *)opop_v(stack);
opush_i(stack, c ? (int)strlen(c) : 0);
unused(userdata);
2010-08-08 10:06:34 +02:00
}
#include "base36.h"
2011-03-07 08:02:35 +01:00
static void eval_int(opstack ** stack, const void *userdata)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
int i = opop_i(stack);
const char *c = itoa10(i);
2010-08-08 10:06:34 +02:00
size_t len = strlen(c);
variant var;
2011-03-07 08:02:35 +01:00
var.v = strcpy(balloc(len + 1), c);
2010-08-08 10:06:34 +02:00
opush(stack, var);
}
2011-03-07 08:02:35 +01:00
void translation_init(void)
2010-08-08 10:06:34 +02:00
{
add_function("lt", &eval_lt);
add_function("eq", &eval_eq);
add_function("int", &eval_int);
add_function("add", &eval_add);
add_function("strlen", &eval_strlen);
add_function("if", &eval_if);
add_function("isnull", &eval_isnull);
}
2011-03-07 08:02:35 +01:00
void translation_done(void)
2010-08-08 10:06:34 +02:00
{
2011-03-07 08:02:35 +01:00
free_functions();
free(buffer.begin);
2010-08-08 10:06:34 +02:00
}