server/src/kernel/order.c

602 lines
15 KiB
C
Raw Normal View History

/*
2014-08-08 01:03:46 +02:00
+-------------------+
2010-08-08 10:06:34 +02:00
| | Christian Schlittchen <corwin@amber.kn-bremen.de>
| Eressea PBEM host | Enno Rehling <enno@eressea.de>
| (c) 1998 - 2014 | Katja Zedel <katze@felidae.kn-bremen.de>
2010-08-08 10:06:34 +02:00
| |
2014-08-08 01:03:46 +02:00
+-------------------+
2010-08-08 10:06:34 +02:00
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
2014-08-08 01:03:46 +02:00
*/
2010-08-08 10:06:34 +02:00
#include <platform.h>
#include <kernel/config.h>
#include "order.h"
#include "skill.h"
#include "keyword.h"
2010-08-08 10:06:34 +02:00
#include <util/base36.h>
#include <util/bsdstring.h>
#include <util/language.h>
#include <util/log.h>
#include <util/parser.h>
/* libc includes */
#include <assert.h>
#include <ctype.h>
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
# define ORD_KEYWORD(ord) (keyword_t)((ord)->command & 0xFFFF)
2010-08-08 10:06:34 +02:00
# define ORD_LOCALE(ord) locale_array[(ord)->data->_lindex]->lang
# define ORD_STRING(ord) (ord)->data->_str
typedef struct locale_data {
2014-08-08 01:03:46 +02:00
struct order_data *short_orders[MAXKEYWORDS];
struct order_data *study_orders[MAXSKILLS];
const struct locale *lang;
2010-08-08 10:06:34 +02:00
} locale_data;
static struct locale_data *locale_array[MAXLOCALES];
2010-08-08 10:06:34 +02:00
typedef struct order_data {
const char *_str;
2014-08-08 01:03:46 +02:00
int _refcount;
int _lindex;
2010-08-08 10:06:34 +02:00
} order_data;
2011-03-07 08:02:35 +01:00
static void release_data(order_data * data)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (data) {
if (--data->_refcount == 0) {
free(data);
}
2010-08-08 10:06:34 +02:00
}
}
2011-03-07 08:02:35 +01:00
void replace_order(order ** dlist, order * orig, const order * src)
2010-08-08 10:06:34 +02:00
{
assert(src);
assert(orig);
assert(dlist);
2014-08-08 01:03:46 +02:00
while (*dlist != NULL) {
order *dst = *dlist;
if (dst->data == orig->data) {
order *cpy = copy_order(src);
*dlist = cpy;
cpy->next = dst->next;
dst->next = 0;
free_order(dst);
}
dlist = &(*dlist)->next;
2010-08-08 10:06:34 +02:00
}
}
keyword_t getkeyword(const order * ord)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (ord == NULL) {
return NOKEYWORD;
}
return ORD_KEYWORD(ord);
2010-08-08 10:06:34 +02:00
}
/** returns a plain-text representation of the order.
* This is the inverse function to the parse_order command. Note that
* keywords are expanded to their full length.
*/
char* get_command(const order *ord, char *sbuffer, size_t size) {
char *bufp = sbuffer;
const char *text = ORD_STRING(ord);
keyword_t kwd = ORD_KEYWORD(ord);
int bytes;
2017-09-30 19:22:24 +02:00
if (ord->command & CMD_QUIET) {
if (size > 0) {
*bufp++ = '!';
--size;
}
else {
WARN_STATIC_BUFFER();
}
}
2017-09-30 19:22:24 +02:00
if (ord->command & CMD_PERSIST) {
if (size > 0) {
*bufp++ = '@';
--size;
2014-08-08 01:03:46 +02:00
}
else {
WARN_STATIC_BUFFER();
}
2010-08-08 10:06:34 +02:00
}
if (kwd != NOKEYWORD) {
const struct locale *lang = ORD_LOCALE(ord);
if (size > 0) {
2014-06-17 08:05:39 +02:00
const char *str = (const char *)LOC(lang, keyword(kwd));
assert(str);
if (text) --size;
bytes = (int)strlcpy(bufp, str, size);
if (wrptr(&bufp, &size, bytes) != 0) {
WARN_STATIC_BUFFER();
}
if (text) *bufp++ = ' ';
2014-08-08 01:03:46 +02:00
}
else {
WARN_STATIC_BUFFER();
}
}
if (text) {
bytes = (int)strlcpy(bufp, (const char *)text, size);
if (wrptr(&bufp, &size, bytes) != 0) {
WARN_STATIC_BUFFER();
if (bufp - sbuffer >= 6) {
bufp -= 6;
while (bufp > sbuffer && (*bufp & 0x80) != 0) {
++size;
--bufp;
}
memcpy(bufp, "[...]", 6); /* TODO: make sure this only happens in eval_command */
bufp += 6;
}
2010-08-08 10:06:34 +02:00
}
}
if (size > 0) *bufp = 0;
return sbuffer;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void free_order(order * ord)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (ord != NULL) {
assert(ord->next == 0);
2010-08-08 10:06:34 +02:00
2014-08-08 01:03:46 +02:00
release_data(ord->data);
free(ord);
}
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
order *copy_order(const order * src)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (src != NULL) {
order *ord = (order *)malloc(sizeof(order));
ord->next = NULL;
2017-09-30 19:22:24 +02:00
ord->command = src->command;
2014-08-08 01:03:46 +02:00
ord->data = src->data;
++ord->data->_refcount;
return ord;
}
return NULL;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void set_order(struct order **destp, struct order *src)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (*destp == src)
return;
free_order(*destp);
*destp = src;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void free_orders(order ** olist)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
while (*olist) {
order *ord = *olist;
*olist = ord->next;
ord->next = NULL;
free_order(ord);
}
2010-08-08 10:06:34 +02:00
}
static char *mkdata(order_data **pdata, size_t len, int lindex, const char *str)
{
order_data *data;
char *result;
data = malloc(sizeof(order_data) + len + 1);
result = (char *)(data + 1);
data->_lindex = lindex;
data->_refcount = 0;
data->_str = 0;
data->_str = (len > 0) ? result : 0;
if (str) strcpy(result, str);
if (pdata) *pdata = data;
return result;
}
2011-03-07 08:02:35 +01:00
static order_data *create_data(keyword_t kwd, const char *sptr, int lindex)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
const char *s = sptr;
order_data *data;
const struct locale *lang = locale_array[lindex]->lang;
if (kwd != NOKEYWORD)
s = (*sptr) ? sptr : NULL;
/* learning, only one order_data per skill required */
if (kwd == K_STUDY) {
skill_t sk = get_skill(parse_token_depr(&sptr), lang);
2014-08-08 01:03:46 +02:00
switch (sk) {
case NOSKILL: /* fehler */
break;
case SK_MAGIC: /* kann parameter haben */
if (*sptr != 0)
break;
default: /* nur skill als Parameter, keine extras */
data = locale_array[lindex]->study_orders[sk];
if (data == NULL) {
const char *skname = skillname(sk, lang);
const char *spc = strchr(skname, ' ');
size_t len = strlen(skname);
char *dst = mkdata(&data, len + (spc ? 3 : 0), lindex, spc ? 0 : skname);
2014-08-08 01:03:46 +02:00
locale_array[lindex]->study_orders[sk] = data;
if (spc) {
dst[0] = '\"';
memcpy(dst + 1, skname, len);
dst[len + 1] = '\"';
dst[len + 2] = '\0';
2014-08-08 01:03:46 +02:00
}
data->_refcount = 1;
}
++data->_refcount;
return data;
2010-08-08 10:06:34 +02:00
}
}
2014-08-08 01:03:46 +02:00
/* orders with no parameter, only one order_data per order required */
else if (kwd != NOKEYWORD && *sptr == 0) {
data = locale_array[lindex]->short_orders[kwd];
if (data == NULL) {
mkdata(&data, 0, lindex, 0);
2014-08-08 01:03:46 +02:00
data->_refcount = 1;
locale_array[lindex]->short_orders[kwd] = data;
2014-08-08 01:03:46 +02:00
}
++data->_refcount;
return data;
2010-08-08 10:06:34 +02:00
}
mkdata(&data, s ? strlen(s) : 0, lindex, s);
2014-08-08 01:03:46 +02:00
data->_refcount = 1;
2010-08-08 10:06:34 +02:00
return data;
}
2015-10-14 22:06:40 +02:00
static void clear_localedata(int lindex) {
int i;
for (i = 0; i != MAXKEYWORDS; ++i) {
release_data(locale_array[lindex]->short_orders[i]);
2015-10-14 22:06:40 +02:00
locale_array[lindex]->short_orders[i] = 0;
}
for (i = 0; i != MAXSKILLS; ++i) {
release_data(locale_array[lindex]->study_orders[i]);
2015-10-14 22:06:40 +02:00
locale_array[lindex]->study_orders[i] = 0;
}
2015-10-14 22:06:40 +02:00
locale_array[lindex]->lang = 0;
}
void close_orders(void) {
int i;
for (i = 0; i != MAXLOCALES; ++i) {
2015-10-14 22:06:40 +02:00
if (locale_array[i]){
clear_localedata(i);
free(locale_array[i]);
locale_array[i] = 0;
}
}
}
static order *create_order_i(order *ord, keyword_t kwd, const char *sptr, bool persistent,
bool noerror, const struct locale *lang)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
int lindex;
assert(ord);
if (kwd == NOKEYWORD || keyword_disabled(kwd)) {
log_error("trying to create an order for disabled keyword %s.", keyword(kwd));
return NULL;
}
2014-08-08 01:03:46 +02:00
/* if this is just nonsense, then we skip it. */
if (lomem) {
switch (kwd) {
case K_KOMMENTAR:
case NOKEYWORD:
return NULL;
default:
break;
}
2010-08-08 10:06:34 +02:00
}
lindex = locale_index(lang);
assert(lindex < MAXLOCALES);
if (!locale_array[lindex]) {
locale_array[lindex] = (locale_data *)calloc(1, sizeof(locale_data));
}
else if (locale_array[lindex]->lang != lang) {
2015-10-14 22:06:40 +02:00
clear_localedata(lindex);
}
locale_array[lindex]->lang = lang;
2010-08-08 10:06:34 +02:00
2017-09-30 19:22:24 +02:00
ord->command = (int)kwd;
if (persistent) ord->command |= CMD_PERSIST;
if (noerror) ord->command |= CMD_QUIET;
2014-08-08 01:03:46 +02:00
ord->next = NULL;
2010-08-08 10:06:34 +02:00
while (isspace(*(unsigned char *)sptr)) ++sptr;
2014-08-08 01:03:46 +02:00
ord->data = create_data(kwd, sptr, lindex);
2010-08-08 10:06:34 +02:00
2014-08-08 01:03:46 +02:00
return ord;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
order *create_order(keyword_t kwd, const struct locale * lang,
2014-08-08 01:03:46 +02:00
const char *params, ...)
2010-08-08 10:06:34 +02:00
{
order *ord;
2014-08-08 01:03:46 +02:00
char zBuffer[DISPLAYSIZE];
if (params) {
char *bufp = zBuffer;
int bytes;
size_t size = sizeof(zBuffer) - 1;
va_list marker;
assert(lang);
2014-08-08 01:03:46 +02:00
va_start(marker, params);
while (*params) {
if (*params == '%') {
int i;
const char *s;
++params;
switch (*params) {
case 's':
s = va_arg(marker, const char *);
assert(s);
2014-08-08 01:03:46 +02:00
bytes = (int)strlcpy(bufp, s, size);
if (wrptr(&bufp, &size, bytes) != 0)
WARN_STATIC_BUFFER();
break;
case 'd':
i = va_arg(marker, int);
bytes = (int)strlcpy(bufp, itoa10(i), size);
if (wrptr(&bufp, &size, bytes) != 0)
WARN_STATIC_BUFFER();
break;
case 'i':
i = va_arg(marker, int);
bytes = (int)strlcpy(bufp, itoa36(i), size);
if (wrptr(&bufp, &size, bytes) != 0)
WARN_STATIC_BUFFER();
break;
default:
assert(!"unknown format-character in create_order");
}
}
else if (size > 0) {
*bufp++ = *params;
--size;
}
++params;
2010-08-08 10:06:34 +02:00
}
2014-08-08 01:03:46 +02:00
va_end(marker);
*bufp = 0;
}
else {
zBuffer[0] = 0;
2010-08-08 10:06:34 +02:00
}
ord = (order *)malloc(sizeof(order));
return create_order_i(ord, kwd, zBuffer, false, false, lang);
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
order *parse_order(const char *s, const struct locale * lang)
2010-08-08 10:06:34 +02:00
{
assert(lang);
assert(s);
if (*s != 0) {
keyword_t kwd = NOKEYWORD;
const char *sptr = s;
bool persistent = false, noerror = false;
const char * p;
p = *sptr ? parse_token_depr(&sptr) : 0;
if (p) {
while (*p == '!' || *p == '@') {
if (*p == '!') noerror = true;
else if (*p == '@') persistent = true;
++p;
}
kwd = get_keyword(p, lang);
}
if (kwd == K_MAKE) {
const char *sp = sptr;
p = parse_token_depr(&sp);
if (p && isparam(p, lang, P_TEMP)) {
kwd = K_MAKETEMP;
sptr = sp;
}
}
if (kwd != NOKEYWORD) {
order *ord = (order *)malloc(sizeof(order));
return create_order_i(ord, kwd, sptr, persistent, noerror, lang);
}
2010-08-08 10:06:34 +02:00
}
return NULL;
2010-08-08 10:06:34 +02:00
}
/**
* Returns true if the order qualifies as "repeated". An order is repeated if it will overwrite the
* old default order. K_BUY is in this category, but not K_MOVE.
*
* \param ord An order.
* \return true if the order is long
* \sa is_exclusive(), is_repeated(), is_persistent()
*/
bool is_repeated(keyword_t kwd)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
switch (kwd) {
case K_CAST:
case K_BUY:
case K_SELL:
case K_ROUTE:
case K_DRIVE:
case K_WORK:
case K_BESIEGE:
case K_ENTERTAIN:
case K_TAX:
case K_RESEARCH:
case K_SPY:
case K_STEAL:
case K_SABOTAGE:
case K_STUDY:
case K_TEACH:
case K_GROW:
case K_PLANT:
2014-08-08 01:03:46 +02:00
case K_PIRACY:
case K_MAKE:
case K_LOOT:
case K_DESTROY:
return true;
2014-08-08 01:03:46 +02:00
default:
break;
2014-08-08 01:03:46 +02:00
}
return false;
2010-08-08 10:06:34 +02:00
}
/**
* Returns true if the order qualifies as "exclusive". An order is exclusive if it makes all other
* long orders illegal. K_MOVE is in this category, but not K_BUY.
*
* \param ord An order.
* \return true if the order is long
* \sa is_exclusive(), is_repeated(), is_persistent()
*/
2012-06-24 08:04:12 +02:00
bool is_exclusive(const order * ord)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
keyword_t kwd = ORD_KEYWORD(ord);
switch (kwd) {
case K_MOVE:
case K_ROUTE:
case K_DRIVE:
case K_WORK:
case K_BESIEGE:
case K_ENTERTAIN:
case K_TAX:
case K_RESEARCH:
case K_SPY:
case K_STEAL:
case K_SABOTAGE:
case K_STUDY:
case K_TEACH:
case K_GROW:
case K_PLANT:
2014-08-08 01:03:46 +02:00
case K_PIRACY:
case K_MAKE:
case K_LOOT:
case K_DESTROY:
return true;
2014-08-08 01:03:46 +02:00
default:
break;
2014-08-08 01:03:46 +02:00
}
return false;
2010-08-08 10:06:34 +02:00
}
/**
* Returns true if the order qualifies as "long". An order is long if it excludes most other long
* orders.
*
* \param ord An order.
* \return true if the order is long
* \sa is_exclusive(), is_repeated(), is_persistent()
*/
bool is_long(keyword_t kwd)
{
2014-08-08 01:03:46 +02:00
switch (kwd) {
case K_CAST:
case K_BUY:
case K_SELL:
case K_MOVE:
case K_ROUTE:
case K_DRIVE:
case K_WORK:
case K_BESIEGE:
case K_ENTERTAIN:
case K_TAX:
case K_RESEARCH:
case K_SPY:
case K_STEAL:
case K_SABOTAGE:
case K_STUDY:
case K_TEACH:
case K_GROW:
case K_PLANT:
2014-08-08 01:03:46 +02:00
case K_PIRACY:
case K_MAKE:
case K_LOOT:
case K_DESTROY:
2014-08-08 01:03:46 +02:00
return true;
default:
break;
2014-08-08 01:03:46 +02:00
}
return false;
}
/**
* Returns true if the order qualifies as "persistent". An order is persistent if it will be
* included in the template orders. @-orders, comments and most long orders are in this category,
* but not K_MOVE.
*
* \param ord An order.
* \return true if the order is persistent
* \sa is_exclusive(), is_repeated(), is_persistent()
*/
2012-06-24 08:04:12 +02:00
bool is_persistent(const order * ord)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
keyword_t kwd = ORD_KEYWORD(ord);
switch (kwd) {
case K_MOVE:
case NOKEYWORD:
/* lang, aber niemals persistent! */
return false;
case K_KOMMENTAR:
return true;
default:
2017-09-30 19:22:24 +02:00
return (ord->command & CMD_PERSIST) || is_repeated(kwd);
2014-08-08 01:03:46 +02:00
}
2010-08-08 10:06:34 +02:00
}
2017-09-30 19:44:39 +02:00
bool is_silent(const order * ord)
{
return (ord->command & CMD_QUIET) != 0;
}
2011-03-07 08:02:35 +01:00
char *write_order(const order * ord, char *buffer, size_t size)
2010-08-08 10:06:34 +02:00
{
2014-08-08 01:03:46 +02:00
if (ord == 0) {
buffer[0] = 0;
}
else {
keyword_t kwd = ORD_KEYWORD(ord);
if (kwd == NOKEYWORD) {
const char *text = ORD_STRING(ord);
2014-08-24 00:29:19 +02:00
if (text) strlcpy(buffer, (const char *)text, size);
else buffer[0] = 0;
2014-08-08 01:03:46 +02:00
}
else {
get_command(ord, buffer, size);
}
2010-08-08 10:06:34 +02:00
}
2014-08-08 01:03:46 +02:00
return buffer;
2010-08-08 10:06:34 +02:00
}
2011-03-07 08:02:35 +01:00
void push_order(order ** ordp, order * ord)
{
2014-08-08 01:03:46 +02:00
while (*ordp)
ordp = &(*ordp)->next;
*ordp = ord;
2011-03-07 08:02:35 +01:00
}
2012-06-24 07:36:17 +02:00
keyword_t init_order(const struct order *ord)
{
assert(ord && ord->data);
2014-12-30 00:00:57 +01:00
init_tokens_str(ord->data->_str);
return ORD_KEYWORD(ord);
}