forked from github/server
157 lines
3.8 KiB
C
157 lines
3.8 KiB
C
#ifdef _MSC_VER
|
|
#define _CRT_SECURE_NO_WARNINGS
|
|
#endif
|
|
#include "crmessage.h"
|
|
|
|
#include "macros.h"
|
|
#include "message.h"
|
|
#include "strings.h"
|
|
#include "log.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
|
|
/** type to string conversion **/
|
|
typedef struct tsf_list {
|
|
struct tsf_list *next;
|
|
const char *name;
|
|
tostring_f fun;
|
|
} tsf_list;
|
|
|
|
static tsf_list *tostringfs;
|
|
|
|
void crmessage_done(void) {
|
|
tsf_list **tsp = &tostringfs;
|
|
while (*tsp) {
|
|
tsf_list *ts = *tsp;
|
|
*tsp = ts->next;
|
|
free(ts);
|
|
}
|
|
}
|
|
|
|
static tostring_f tsf_find(const char *name)
|
|
{
|
|
if (name != NULL) {
|
|
tsf_list *tsf;
|
|
for (tsf = tostringfs; tsf; tsf = tsf->next) {
|
|
if (!strcmp(tsf->name, name))
|
|
return tsf->fun;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void tsf_register(const char *name, tostring_f fun)
|
|
{
|
|
tsf_list *tsf;
|
|
for (tsf = tostringfs; tsf; tsf = tsf->next) {
|
|
if (!strcmp(tsf->name, name))
|
|
break;
|
|
}
|
|
if (tsf == NULL) {
|
|
tsf = malloc(sizeof(tsf_list));
|
|
if (!tsf) abort();
|
|
tsf->fun = fun;
|
|
tsf->name = name;
|
|
tsf->next = tostringfs;
|
|
tostringfs = tsf;
|
|
}
|
|
}
|
|
|
|
/** crmesssage **/
|
|
typedef struct crmessage_type {
|
|
const struct message_type *mtype;
|
|
tostring_f *renderers;
|
|
struct crmessage_type *next;
|
|
} crmessage_type;
|
|
|
|
#define CRMAXHASH 63
|
|
static crmessage_type *crtypes[CRMAXHASH];
|
|
|
|
static crmessage_type *crt_find(const struct message_type *mtype)
|
|
{
|
|
unsigned int hash = mtype->key % CRMAXHASH;
|
|
crmessage_type *found = NULL;
|
|
crmessage_type *type = crtypes[hash];
|
|
while (type) {
|
|
if (type->mtype == mtype)
|
|
found = type;
|
|
type = type->next;
|
|
}
|
|
return found;
|
|
}
|
|
|
|
void crt_register(const struct message_type *mtype)
|
|
{
|
|
unsigned int hash = mtype->key % CRMAXHASH;
|
|
crmessage_type *crt = crtypes[hash];
|
|
while (crt && crt->mtype != mtype) {
|
|
crt = crt->next;
|
|
}
|
|
if (!crt) {
|
|
crt = malloc(sizeof(crmessage_type));
|
|
if (!crt) abort();
|
|
crt->mtype = mtype;
|
|
crt->next = crtypes[hash];
|
|
crtypes[hash] = crt;
|
|
if (mtype->nparameters > 0) {
|
|
int i;
|
|
crt->renderers = malloc(sizeof(tostring_f) * mtype->nparameters);
|
|
if (!crt->renderers) abort();
|
|
/* can be scrapped for memory vs. speed */
|
|
for (i = 0; i != mtype->nparameters; ++i) {
|
|
crt->renderers[i] = tsf_find(mtype->types[i]->name);
|
|
}
|
|
}
|
|
else {
|
|
crt->renderers = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
int cr_render(const message * msg, char *buffer, const void *userdata)
|
|
{
|
|
int i;
|
|
char *c = buffer;
|
|
struct crmessage_type *crt = crt_find(msg->type);
|
|
|
|
if (crt == NULL)
|
|
return -1;
|
|
for (i = 0; i != msg->type->nparameters; ++i) {
|
|
if (crt->renderers[i] == NULL) {
|
|
log_error("No renderer for argument %s:%s of \"%s\"\n", msg->type->pnames[i], msg->type->types[i]->name, msg->type->name);
|
|
continue; /* strcpy(c, (const char*)msg->locale_string(u->faction->locale, parameters[i])); */
|
|
}
|
|
else {
|
|
if (crt->renderers[i](msg->parameters[i], msg->type->pnames[i], c, userdata) != 0)
|
|
continue;
|
|
}
|
|
c += strlen(c);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int cr_string(variant var, const char *name, char *buffer, const void *userdata)
|
|
{
|
|
sprintf(buffer, "\"%s\";%s\n", (const char *)var.v, name);
|
|
UNUSED_ARG(userdata);
|
|
return 0;
|
|
}
|
|
|
|
int cr_int(variant var, const char *name, char *buffer, const void *userdata)
|
|
{
|
|
sprintf(buffer, "%d;%s\n", var.i, name);
|
|
UNUSED_ARG(userdata);
|
|
return 0;
|
|
}
|
|
|
|
int cr_ignore(variant var, const char *name, char *buffer, const void *userdata)
|
|
{
|
|
UNUSED_ARG(var);
|
|
UNUSED_ARG(name);
|
|
UNUSED_ARG(buffer);
|
|
UNUSED_ARG(userdata);
|
|
return -1;
|
|
}
|