2015-01-30 20:37:14 +01:00
|
|
|
/*
|
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>
|
2014-08-16 23:23:37 +02:00
|
|
|
| (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"
|
2014-06-16 03:34:39 +02:00
|
|
|
#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>
|
|
|
|
|
2017-09-30 20:08:22 +02:00
|
|
|
# define ORD_KEYWORD(ord) (keyword_t)((ord)->command & 0xFFFF)
|
2017-10-08 15:34:40 +02:00
|
|
|
# define OD_STRING(odata) ((odata) ? (odata)->_str : NULL)
|
2010-08-08 10:06:34 +02:00
|
|
|
|
|
|
|
typedef struct order_data {
|
2014-08-16 23:23:37 +02:00
|
|
|
const char *_str;
|
2014-08-08 01:03:46 +02:00
|
|
|
int _refcount;
|
2010-08-08 10:06:34 +02:00
|
|
|
} order_data;
|
|
|
|
|
2017-10-08 18:41:59 +02:00
|
|
|
#include <critbit.h>
|
2017-10-08 12:39:41 +02:00
|
|
|
|
2017-10-08 18:41:59 +02:00
|
|
|
static critbit_tree cb_orders = { 0 };
|
|
|
|
static int auto_id = 0;
|
|
|
|
|
|
|
|
struct cb_entry {
|
|
|
|
int id;
|
|
|
|
order_data *data;
|
|
|
|
};
|
2017-10-08 12:39:41 +02:00
|
|
|
|
|
|
|
order_data *load_data(int id) {
|
2017-10-08 18:41:59 +02:00
|
|
|
void * match;
|
|
|
|
|
2017-10-08 15:34:40 +02:00
|
|
|
if (id > 0) {
|
2017-10-08 18:41:59 +02:00
|
|
|
if (cb_find_prefix(&cb_orders, &id, sizeof(id), &match, 1, 0) > 0) {
|
|
|
|
struct cb_entry *ent = (struct cb_entry *)match;
|
|
|
|
order_data * od = ent->data;
|
|
|
|
++od->_refcount;
|
|
|
|
return od;
|
|
|
|
}
|
2017-10-08 15:34:40 +02:00
|
|
|
}
|
|
|
|
return NULL;
|
2017-10-08 12:39:41 +02:00
|
|
|
}
|
|
|
|
|
2017-10-08 18:41:59 +02:00
|
|
|
int save_data(order_data *od) {
|
2017-10-08 15:34:40 +02:00
|
|
|
if (od->_str) {
|
2017-10-08 18:41:59 +02:00
|
|
|
struct cb_entry ent;
|
|
|
|
|
2017-10-08 15:34:40 +02:00
|
|
|
++od->_refcount;
|
2017-10-08 18:41:59 +02:00
|
|
|
ent.id = ++auto_id;
|
|
|
|
ent.data = od;
|
|
|
|
cb_insert(&cb_orders, &ent, sizeof(ent));
|
|
|
|
return ent.id;
|
2017-10-08 15:34:40 +02:00
|
|
|
}
|
|
|
|
return 0;
|
2017-10-08 12:39:41 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-08 18:41:59 +02:00
|
|
|
int free_data_cb(const void *match, const void *key, size_t keylen, void *udata) {
|
|
|
|
struct cb_entry * ent = (struct cb_entry *)match;
|
|
|
|
order_data *od = ent->data;
|
2017-10-08 12:39:41 +02:00
|
|
|
if (od->_refcount > 1) {
|
2017-10-08 18:41:59 +02:00
|
|
|
log_error("refcount=%d for order %d, %s", od->_refcount, ent->id, od->_str);
|
2017-10-08 12:39:41 +02:00
|
|
|
}
|
|
|
|
release_data(od);
|
2017-10-08 18:41:59 +02:00
|
|
|
return 0;
|
2017-10-08 12:39:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void free_data(void) {
|
2017-10-08 18:41:59 +02:00
|
|
|
cb_foreach(&cb_orders, NULL, 0, free_data_cb, NULL);
|
|
|
|
cb_clear(&cb_orders);
|
2017-10-08 12:39:41 +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
|
|
|
{
|
2015-05-21 11:52:31 +02:00
|
|
|
assert(src);
|
|
|
|
assert(orig);
|
|
|
|
assert(dlist);
|
2014-08-08 01:03:46 +02:00
|
|
|
while (*dlist != NULL) {
|
|
|
|
order *dst = *dlist;
|
2017-10-08 12:39:41 +02:00
|
|
|
if (dst->id == orig->id) {
|
2014-08-08 01:03:46 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-16 16:19:22 +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.
|
|
|
|
*/
|
2017-10-08 17:07:42 +02:00
|
|
|
char* get_command(const order *ord, const struct locale *lang, char *sbuffer, size_t size) {
|
2014-06-16 03:34:39 +02:00
|
|
|
char *bufp = sbuffer;
|
2017-10-08 12:39:41 +02:00
|
|
|
order_data *od;
|
|
|
|
const char * text;
|
2014-06-16 03:34:39 +02:00
|
|
|
keyword_t kwd = ORD_KEYWORD(ord);
|
|
|
|
int bytes;
|
|
|
|
|
2017-09-30 19:22:24 +02:00
|
|
|
if (ord->command & CMD_QUIET) {
|
2017-03-25 10:37:13 +01:00
|
|
|
if (size > 0) {
|
|
|
|
*bufp++ = '!';
|
|
|
|
--size;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
WARN_STATIC_BUFFER();
|
|
|
|
}
|
|
|
|
}
|
2017-09-30 19:22:24 +02:00
|
|
|
if (ord->command & CMD_PERSIST) {
|
2014-06-16 03:34:39 +02:00
|
|
|
if (size > 0) {
|
|
|
|
*bufp++ = '@';
|
|
|
|
--size;
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
|
|
|
else {
|
2014-06-16 03:34:39 +02:00
|
|
|
WARN_STATIC_BUFFER();
|
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2017-10-08 12:39:41 +02:00
|
|
|
|
|
|
|
od = load_data(ord->id);
|
|
|
|
text = OD_STRING(od);
|
2014-06-16 03:34:39 +02:00
|
|
|
if (kwd != NOKEYWORD) {
|
|
|
|
if (size > 0) {
|
2014-06-17 08:05:39 +02:00
|
|
|
const char *str = (const char *)LOC(lang, keyword(kwd));
|
2014-07-28 14:27:30 +02:00
|
|
|
assert(str);
|
2014-06-16 03:34:39 +02:00
|
|
|
if (text) --size;
|
|
|
|
bytes = (int)strlcpy(bufp, str, size);
|
2015-08-06 16:52:24 +02:00
|
|
|
if (wrptr(&bufp, &size, bytes) != 0) {
|
|
|
|
WARN_STATIC_BUFFER();
|
|
|
|
}
|
2014-06-16 03:34:39 +02:00
|
|
|
if (text) *bufp++ = ' ';
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
|
|
|
else {
|
2014-06-16 03:34:39 +02:00
|
|
|
WARN_STATIC_BUFFER();
|
|
|
|
}
|
2012-05-09 22:06:16 +02:00
|
|
|
}
|
2014-06-16 03:34:39 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
2017-10-08 12:39:41 +02:00
|
|
|
release_data(od);
|
2014-06-16 03:34:39 +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);
|
|
|
|
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;
|
2017-10-08 12:39:41 +02:00
|
|
|
ord->id = src->id;
|
2014-08-08 01:03:46 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2017-10-08 17:07:42 +02:00
|
|
|
static char *mkdata(order_data **pdata, size_t len, const char *str)
|
2014-08-16 23:23:37 +02:00
|
|
|
{
|
|
|
|
order_data *data;
|
|
|
|
char *result;
|
2015-01-30 20:37:14 +01:00
|
|
|
data = malloc(sizeof(order_data) + len + 1);
|
2014-08-16 23:23:37 +02:00
|
|
|
result = (char *)(data + 1);
|
|
|
|
data->_refcount = 0;
|
2017-10-08 15:34:40 +02:00
|
|
|
data->_str = (len > 0) ? result : NULL;
|
2014-08-16 23:23:37 +02:00
|
|
|
if (str) strcpy(result, str);
|
|
|
|
if (pdata) *pdata = data;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-10-08 17:07:42 +02:00
|
|
|
static order_data *create_data(keyword_t kwd, const char *sptr)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2014-08-08 01:03:46 +02:00
|
|
|
const char *s = sptr;
|
|
|
|
order_data *data;
|
|
|
|
|
|
|
|
if (kwd != NOKEYWORD)
|
|
|
|
s = (*sptr) ? sptr : NULL;
|
|
|
|
|
|
|
|
/* orders with no parameter, only one order_data per order required */
|
2017-10-08 15:34:40 +02:00
|
|
|
if (kwd != NOKEYWORD && *sptr == 0) {
|
2017-10-08 17:07:42 +02:00
|
|
|
mkdata(&data, 0, NULL);
|
2017-10-08 15:34:40 +02:00
|
|
|
data->_refcount = 1;
|
2014-08-08 01:03:46 +02:00
|
|
|
return data;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2017-10-08 17:07:42 +02:00
|
|
|
mkdata(&data, s ? strlen(s) : 0, s);
|
2014-08-08 01:03:46 +02:00
|
|
|
data->_refcount = 1;
|
2010-08-08 10:06:34 +02:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2017-09-25 22:15:13 +02:00
|
|
|
static order *create_order_i(order *ord, keyword_t kwd, const char *sptr, bool persistent,
|
2017-10-08 17:07:42 +02:00
|
|
|
bool noerror)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2017-10-08 12:39:41 +02:00
|
|
|
order_data *od;
|
2014-08-08 01:03:46 +02:00
|
|
|
|
2017-09-25 22:15:13 +02:00
|
|
|
assert(ord);
|
2016-11-05 18:20:56 +01:00
|
|
|
if (kwd == NOKEYWORD || keyword_disabled(kwd)) {
|
2014-08-20 23:42:33 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2017-09-25 22:12:49 +02:00
|
|
|
while (isspace(*(unsigned char *)sptr)) ++sptr;
|
2017-10-08 12:39:41 +02:00
|
|
|
|
2017-10-08 17:07:42 +02:00
|
|
|
od = create_data(kwd, sptr);
|
2017-10-08 18:41:59 +02:00
|
|
|
ord->id = save_data(od);
|
2017-10-08 12:39:41 +02:00
|
|
|
release_data(od);
|
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
|
|
|
{
|
2017-09-25 22:15:13 +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;
|
|
|
|
|
2015-08-05 00:03:30 +02:00
|
|
|
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 *);
|
2015-04-20 15:07:10 +02:00
|
|
|
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
|
|
|
}
|
2017-09-25 22:15:13 +02:00
|
|
|
ord = (order *)malloc(sizeof(order));
|
2017-10-08 17:07:42 +02:00
|
|
|
return create_order_i(ord, kwd, zBuffer, false, false);
|
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
|
|
|
{
|
2014-06-14 16:52:32 +02:00
|
|
|
assert(lang);
|
|
|
|
assert(s);
|
|
|
|
if (*s != 0) {
|
2017-09-25 22:08:23 +02:00
|
|
|
keyword_t kwd = NOKEYWORD;
|
|
|
|
const char *sptr = s;
|
2017-03-25 10:37:13 +01:00
|
|
|
bool persistent = false, noerror = false;
|
2014-08-23 09:17:58 +02:00
|
|
|
const char * p;
|
2014-06-14 16:52:32 +02:00
|
|
|
|
2014-12-22 14:21:24 +01:00
|
|
|
p = *sptr ? parse_token_depr(&sptr) : 0;
|
2017-09-25 22:08:23 +02:00
|
|
|
if (p) {
|
|
|
|
while (*p == '!' || *p == '@') {
|
|
|
|
if (*p == '!') noerror = true;
|
|
|
|
else if (*p == '@') persistent = true;
|
|
|
|
++p;
|
|
|
|
}
|
|
|
|
kwd = get_keyword(p, lang);
|
|
|
|
}
|
2014-08-16 11:41:19 +02:00
|
|
|
if (kwd == K_MAKE) {
|
2017-09-25 22:08:23 +02:00
|
|
|
const char *sp = sptr;
|
|
|
|
p = parse_token_depr(&sp);
|
|
|
|
if (p && isparam(p, lang, P_TEMP)) {
|
2014-08-16 11:41:19 +02:00
|
|
|
kwd = K_MAKETEMP;
|
2014-08-17 14:24:19 +02:00
|
|
|
sptr = sp;
|
2014-08-16 11:41:19 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-14 16:52:32 +02:00
|
|
|
if (kwd != NOKEYWORD) {
|
2017-09-25 22:15:13 +02:00
|
|
|
order *ord = (order *)malloc(sizeof(order));
|
2017-10-08 17:07:42 +02:00
|
|
|
return create_order_i(ord, kwd, sptr, persistent, noerror);
|
2014-06-14 16:52:32 +02:00
|
|
|
}
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2014-06-14 16:52:32 +02:00
|
|
|
return NULL;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2010-11-12 05:11:19 +01: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()
|
|
|
|
*/
|
2015-08-04 22:47:55 +02:00
|
|
|
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:
|
2017-06-24 13:36:18 +02:00
|
|
|
case K_GROW:
|
|
|
|
case K_PLANT:
|
2014-08-08 01:03:46 +02:00
|
|
|
case K_PIRACY:
|
2014-08-16 11:41:19 +02:00
|
|
|
case K_MAKE:
|
2014-09-15 15:51:40 +02:00
|
|
|
case K_LOOT:
|
2014-12-23 15:53:28 +01:00
|
|
|
case K_DESTROY:
|
2015-01-08 20:55:29 +01:00
|
|
|
return true;
|
2014-08-08 01:03:46 +02:00
|
|
|
|
|
|
|
default:
|
2015-01-08 20:55:29 +01:00
|
|
|
break;
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
2015-01-08 20:55:29 +01:00
|
|
|
return false;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2010-11-12 05:11:19 +01: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:
|
2017-06-24 13:36:18 +02:00
|
|
|
case K_GROW:
|
|
|
|
case K_PLANT:
|
2014-08-08 01:03:46 +02:00
|
|
|
case K_PIRACY:
|
2014-08-16 11:41:19 +02:00
|
|
|
case K_MAKE:
|
2014-09-15 15:51:40 +02:00
|
|
|
case K_LOOT:
|
2014-12-23 15:53:28 +01:00
|
|
|
case K_DESTROY:
|
2015-01-08 20:55:29 +01:00
|
|
|
return true;
|
2014-08-08 01:03:46 +02:00
|
|
|
|
|
|
|
default:
|
2015-01-08 20:55:29 +01:00
|
|
|
break;
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
2015-01-08 20:55:29 +01:00
|
|
|
return false;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2010-11-12 05:11:19 +01: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()
|
|
|
|
*/
|
2015-08-04 22:47:55 +02:00
|
|
|
bool is_long(keyword_t kwd)
|
2010-11-12 05:11:19 +01:00
|
|
|
{
|
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:
|
2017-06-24 13:36:18 +02:00
|
|
|
case K_GROW:
|
|
|
|
case K_PLANT:
|
2014-08-08 01:03:46 +02:00
|
|
|
case K_PIRACY:
|
2014-08-16 11:41:19 +02:00
|
|
|
case K_MAKE:
|
2014-09-15 15:51:40 +02:00
|
|
|
case K_LOOT:
|
2014-12-23 15:53:28 +01:00
|
|
|
case K_DESTROY:
|
2014-08-08 01:03:46 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
default:
|
2015-01-09 20:41:48 +01:00
|
|
|
break;
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
2015-01-09 20:41:48 +01:00
|
|
|
return false;
|
2010-11-12 05:11:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
}
|
|
|
|
|
2017-10-08 17:07:42 +02:00
|
|
|
char *write_order(const order * ord, const struct locale *lang, 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) {
|
2017-10-08 12:39:41 +02:00
|
|
|
order_data *od = load_data(ord->id);
|
|
|
|
const char *text = OD_STRING(od);
|
2014-08-24 00:29:19 +02:00
|
|
|
if (text) strlcpy(buffer, (const char *)text, size);
|
|
|
|
else buffer[0] = 0;
|
2017-10-08 12:39:41 +02:00
|
|
|
release_data(od);
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
|
|
|
else {
|
2017-10-08 17:07:42 +02:00
|
|
|
get_command(ord, lang, buffer, size);
|
2014-08-08 01:03:46 +02:00
|
|
|
}
|
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-02-22 07:00:03 +01: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
|
|
|
|
2017-10-08 15:34:40 +02:00
|
|
|
static order_data *parser_od;
|
|
|
|
|
2014-08-23 06:45:20 +02:00
|
|
|
keyword_t init_order(const struct order *ord)
|
|
|
|
{
|
2017-10-08 12:39:41 +02:00
|
|
|
if (!ord) {
|
2017-10-08 15:34:40 +02:00
|
|
|
release_data(parser_od);
|
|
|
|
parser_od = NULL;
|
2017-10-08 12:39:41 +02:00
|
|
|
return NOKEYWORD;
|
|
|
|
}
|
|
|
|
else {
|
2017-10-08 15:34:40 +02:00
|
|
|
if (parser_od) {
|
2017-10-08 12:39:41 +02:00
|
|
|
// TODO: warning
|
2017-10-08 15:34:40 +02:00
|
|
|
release_data(parser_od);
|
2017-10-08 12:39:41 +02:00
|
|
|
}
|
2017-10-08 15:34:40 +02:00
|
|
|
parser_od = load_data(ord->id);
|
|
|
|
init_tokens_str(OD_STRING(parser_od));
|
2017-10-08 12:39:41 +02:00
|
|
|
return ORD_KEYWORD(ord);
|
|
|
|
}
|
2014-08-23 06:45:20 +02:00
|
|
|
}
|
2017-10-08 15:34:40 +02:00
|
|
|
|
|
|
|
void close_orders(void) {
|
|
|
|
if (parser_od) {
|
|
|
|
init_order(NULL);
|
|
|
|
}
|
|
|
|
free_data();
|
|
|
|
}
|
|
|
|
|