server/src/common/kernel/message.c

413 lines
9.9 KiB
C
Raw Normal View History

2001-01-25 10:37:55 +01:00
/* vi: set ts=2:
*
* Eressea PB(E)M host Copyright (C) 1998-2003
2001-01-25 10:37:55 +01:00
* Christian Schlittchen (corwin@amber.kn-bremen.de)
* Katja Zedel (katze@felidae.kn-bremen.de)
* Henning Peters (faroul@beyond.kn-bremen.de)
* Enno Rehling (enno@eressea-pbem.de)
* Ingo Wilken (Ingo.Wilken@informatik.uni-oldenburg.de)
*
* based on:
*
* Atlantis v1.0 13 September 1993 Copyright 1993 by Russell Wallace
* Atlantis v1.7 Copyright 1996 by Alex Schr<EFBFBD>der
*
* This program may not be used, modified or distributed without
* prior permission by the authors of Eressea.
* This program may not be sold or used commercially without prior written
* permission from the authors.
*/
#include <config.h>
#include "eressea.h"
#include "message.h"
/* kernel includes */
2004-06-21 18:45:27 +02:00
#include "building.h"
2001-01-25 10:37:55 +01:00
#include "faction.h"
#include "item.h"
2004-06-21 18:45:27 +02:00
#include "order.h"
#include "plane.h"
#include "region.h"
#include "unit.h"
2001-01-25 10:37:55 +01:00
/* util includes */
2004-06-21 18:45:27 +02:00
#include <util/base36.h>
#include <util/goodies.h>
#include <util/message.h>
#include <util/nrmessage.h>
#include <util/crmessage.h>
#include <util/log.h>
2001-02-11 21:56:20 +01:00
/* libc includes */
2001-01-25 10:37:55 +01:00
#include <stddef.h>
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <assert.h>
typedef struct msg_setting {
struct msg_setting *next;
const struct message_type *type;
2001-01-25 10:37:55 +01:00
int level;
} msg_setting;
/************ Compatibility function *************/
#define MAXSTRLEN (4*DISPLAYSIZE+3)
#include "region.h"
#include "eressea.h"
messageclass * msgclasses;
const messageclass *
mc_find(const char * name)
{
messageclass ** mcp = &msgclasses;
if (name==NULL) return NULL;
for (;*mcp;mcp=&(*mcp)->next) {
messageclass * mc = *mcp;
if (!strcmp(mc->name, name)) break;
}
return *mcp;
}
const messageclass *
mc_add(const char * name)
{
messageclass ** mcp = &msgclasses;
if (name==NULL) return NULL;
for (;*mcp;mcp=&(*mcp)->next) {
messageclass * mc = *mcp;
if (!strcmp(mc->name, name)) break;
}
if (!*mcp) {
messageclass * mc = calloc(sizeof(messageclass), 1);
mc->name = strdup(name);
*mcp = mc;
}
return *mcp;
}
static void
arg_set(variant args[], const message_type * mtype, const char * buffer, variant v)
{
int i;
for (i=0;i!=mtype->nparameters;++i) {
if (!strcmp(buffer, mtype->pnames[i])) break;
}
2004-06-21 18:45:27 +02:00
if (i!=mtype->nparameters) {
args[i] = v;
} else {
fprintf(stderr, "invalid parameter %s for message type %s\n", buffer, mtype->name);
assert(!"program aborted.");
}
}
struct message *
2004-06-21 18:45:27 +02:00
msg_feedback(const struct unit * u, struct order * ord, const char * name, const char* sig, ...)
{
va_list marker;
const message_type * mtype = mt_find(name);
char paramname[64];
const char *ic = sig;
variant args[16];
variant var;
memset(args, 0, sizeof(args));
2004-06-21 18:45:27 +02:00
if (ord==NULL) ord = u->thisorder;
if (!mtype) {
log_error(("trying to create message of unknown type \"%s\"\n", name));
return NULL;
}
var.v = (void*)u;
arg_set(args, mtype, "unit", var);
var.v = (void*)u->region;
arg_set(args, mtype, "region", var);
var.v = (void*)ord;
arg_set(args, mtype, "command", var);
va_start(marker, sig);
while (*ic && !isalnum(*ic)) ic++;
while (*ic) {
char * oc = paramname;
int i;
while (isalnum(*ic)) *oc++ = *ic++;
*oc = '\0';
for (i=0;i!=mtype->nparameters;++i) {
if (!strcmp(paramname, mtype->pnames[i])) break;
}
if (i!=mtype->nparameters) {
if (mtype->types[i]->vtype==VAR_VOIDPTR) {
args[i].v = va_arg(marker, void*);
} else if (mtype->types[i]->vtype==VAR_INT) {
args[i].i = va_arg(marker, int);
} else {
assert(!"unknown variant type");
}
} else {
log_error(("invalid parameter %s for message type %s\n", paramname, mtype->name));
assert(!"program aborted.");
}
while (*ic && !isalnum(*ic)) ic++;
}
va_end(marker);
return msg_create(mtype, args);
}
message *
msg_message(const char * name, const char* sig, ...)
/* msg_message("oops_error", "unit region command", u, r, cmd) */
{
va_list marker;
const message_type * mtype = mt_find(name);
char paramname[64];
const char *ic = sig;
variant args[16];
memset(args, 0, sizeof(args));
if (!mtype) {
log_warning(("trying to create message of unknown type \"%s\"\n", name));
return NULL;
}
va_start(marker, sig);
while (*ic && !isalnum(*ic)) ic++;
while (*ic) {
char * oc = paramname;
int i;
while (isalnum(*ic)) *oc++ = *ic++;
*oc = '\0';
for (i=0;i!=mtype->nparameters;++i) {
if (!strcmp(paramname, mtype->pnames[i])) break;
}
if (i!=mtype->nparameters) {
if (mtype->types[i]->vtype==VAR_VOIDPTR) {
args[i].v = va_arg(marker, void*);
} else if (mtype->types[i]->vtype==VAR_INT) {
args[i].i = va_arg(marker, int);
} else {
assert(!"unknown variant type");
}
} else {
log_error(("invalid parameter %s for message type %s\n", paramname, mtype->name));
assert(!"program aborted.");
}
while (*ic && !isalnum(*ic)) ic++;
}
va_end(marker);
return msg_create(mtype, args);
}
message *
new_message(struct faction * receiver, const char* sig, ...)
/* compatibility function, to keep the old function calls valid *
* all old messagetypes are converted into a message with ONLY string parameters,
* this function will convert given parameters to a string representation
* based on the signature - like render() once did */
{
const message_type * mtype;
va_list marker;
const char * signature = strchr(sig, '%');
char buffer[128];
int i=0;
const char * c = sig;
variant args[16];
2001-03-01 02:38:12 +01:00
memset(args, 0, sizeof(args));
assert(signature-sig < sizeof(buffer));
2002-04-21 12:52:22 +02:00
strnzcpy(buffer, sig, signature-sig);
mtype = mt_find(buffer);
if (!mtype) {
log_error(("trying to create message of unknown type \"%s\"\n", buffer));
return NULL;
}
while(*c!='%') buffer[i++] = *(c++);
buffer[i] = 0;
va_start(marker, sig);
while (*c) {
char type;
char *p = buffer;
assert(*c=='%');
type = *(++c);
/*
case 'f': (*ep)->type = IT_FACTION; break;
case 'u': (*ep)->type = IT_UNIT; break;
case 'r': (*ep)->type = IT_REGION; break;
case 'h': (*ep)->type = IT_SHIP; break;
case 'b': (*ep)->type = IT_BUILDING; break;
case 'X': (*ep)->type = IT_RESOURCETYPE; break;
case 'x': (*ep)->type = IT_RESOURCE; break;
case 't': (*ep)->type = IT_SKILL; break;
case 's': (*ep)->type = IT_STRING; break;
case 'i': (*ep)->type = IT_INT; break;
case 'd': (*ep)->type = IT_DIRECTION; break;
case 'S': (*ep)->type = IT_FSPECIAL; break;
*/
c+=2;
while (*c && isalnum(*(unsigned char*)c)) *(p++) = *(c++);
*p = '\0';
for (i=0;i!=mtype->nparameters;++i) {
if (!strcmp(buffer, mtype->pnames[i])) break;
}
2001-03-01 02:38:12 +01:00
if (i==mtype->nparameters) {
log_error(("unknown message-parameter for %s: %s (%p)\n", mtype->name, buffer, va_arg(marker, void*)));
continue;
}
switch(type) {
case 's':
args[i].v = (void*)va_arg(marker, const char *);
break;
case 'f':
args[i].v = (void*)va_arg(marker, const struct faction*);
break;
case 'u':
args[i].v = (void*)va_arg(marker, const struct unit*);
break;
case 'r':
args[i].v = (void*)va_arg(marker, const struct region*);
break;
case 'h':
args[i].v = (void*)va_arg(marker, const struct ship*);
break;
case 'b':
args[i].v = (void*)va_arg(marker, const struct building*);
break;
case 'X':
args[i].v = (void*)va_arg(marker, const resource_type *);
break;
case 't':
case 'i':
args[i].i = va_arg(marker, int);
break;
case 'd':
args[i].i = i;
break;
case 'S':
default:
args[i].v = NULL;
}
}
va_end(marker);
return msg_create(mtype, args);
}
static void
caddmessage(region * r, faction * f, const char *s, msg_t mtype, int level)
2001-01-25 10:37:55 +01:00
{
message * m = NULL;
unused(level);
2001-01-25 10:37:55 +01:00
switch (mtype) {
case MSG_INCOME:
assert(f);
m = add_message(&f->msgs, msg_message("msg_economy", "string", s));
2001-01-25 10:37:55 +01:00
break;
case MSG_BATTLE:
assert(0 || !"battle-meldungen nicht <20>ber addmessage machen");
break;
case MSG_MOVE:
assert(f);
m = add_message(&f->msgs, msg_message("msg_movement", "string", s));
2001-01-25 10:37:55 +01:00
break;
case MSG_COMMERCE:
assert(f);
m = add_message(&f->msgs, msg_message("msg_economy", "string", s));
2001-01-25 10:37:55 +01:00
break;
case MSG_PRODUCE:
assert(f);
m = add_message(&f->msgs, msg_message("msg_production", "string", s));
2001-01-25 10:37:55 +01:00
break;
case MSG_MAGIC:
case MSG_COMMENT:
case MSG_MESSAGE:
case MSG_ORCVERMEHRUNG:
case MSG_EVENT:
/* Botschaften an REGION oder einzelne PARTEI */
m = msg_message("msg_event", "string", s);
if (!r) {
assert(f);
m = add_message(&f->msgs, m);
} else {
2001-05-27 16:02:12 +02:00
if (f==NULL) add_message(&r->msgs, m);
else r_addmessage(r, f, m);
}
2001-01-25 10:37:55 +01:00
break;
default:
assert(!"Ung<EFBFBD>ltige Msg-Klasse!");
2001-01-25 10:37:55 +01:00
}
if (m) msg_release(m);
2001-01-25 10:37:55 +01:00
}
void
addmessage(region * r, faction * f, const char *s, msg_t mtype, int level)
{
caddmessage(r, f, s, mtype, level);
}
2004-06-21 18:45:27 +02:00
void
mistake(const unit * u, struct order * ord, const char *comment, int mtype)
2001-01-25 10:37:55 +01:00
{
2004-06-21 18:45:27 +02:00
if (u->faction->no != MONSTER_FACTION) {
ADDMSG(&u->faction->msgs, msg_message("mistake",
2006-01-02 23:41:36 +01:00
"command error unit region", ord, comment, u, u->region));
2004-06-21 18:45:27 +02:00
}
2001-01-25 10:37:55 +01:00
}
void
2004-06-21 18:45:27 +02:00
cmistake(const unit * u, struct order *ord, int mno, int mtype)
2001-01-25 10:37:55 +01:00
{
static char ebuf[20];
unused(mtype);
2001-01-25 10:37:55 +01:00
if (u->faction->no == MONSTER_FACTION) return;
sprintf(ebuf, "error%d", mno);
2004-06-21 18:45:27 +02:00
ADDMSG(&u->faction->msgs, msg_message(ebuf,
2005-06-11 10:09:55 +02:00
"command unit region", ord, u, u->region));
2001-01-25 10:37:55 +01:00
}
extern unsigned int new_hashstring(const char* s);
void
free_messagelist(message_list * msgs)
{
struct mlist ** mlistptr = &msgs->begin;
while (*mlistptr) {
struct mlist * ml = *mlistptr;
*mlistptr = ml->next;
msg_release(ml->msg);
free(ml);
}
free(msgs);
}
message *
add_message(message_list** pm, message * m)
2001-01-25 10:37:55 +01:00
{
if (!lomem && m!=NULL) {
struct mlist * mnew = malloc(sizeof(struct mlist));
if (*pm==NULL) {
*pm = malloc(sizeof(message_list));
(*pm)->end=&(*pm)->begin;
}
mnew->msg = msg_addref(m);
mnew->next = NULL;
*((*pm)->end) = mnew;
(*pm)->end=&mnew->next;
}
return m;
2001-01-25 10:37:55 +01:00
}