#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 #include #include #include /** 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; }