2010-08-08 10:06:34 +02:00
|
|
|
/* vi: set ts=2:
|
|
|
|
+-------------------+ Enno Rehling <enno@eressea.de>
|
|
|
|
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
| (c) 1998 - 2008 | Katja Zedel <katze@felidae.kn-bremen.de>
|
|
|
|
+-------------------+
|
|
|
|
This program may not be used, modified or distributed
|
|
|
|
without prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
#include <platform.h>
|
|
|
|
#include "config.h"
|
|
|
|
#include "textstore.h"
|
|
|
|
|
|
|
|
#include "save.h"
|
|
|
|
#include "version.h"
|
|
|
|
#include <util/base36.h>
|
|
|
|
#include <util/log.h>
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <libxml/encoding.h>
|
|
|
|
|
|
|
|
#define NULL_TOKEN '@'
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_brk(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
putc('\n', (FILE *) store->userdata);
|
2010-08-08 10:06:34 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_id(struct storage *store, int arg)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
return fprintf((FILE *) store->userdata, "%s ", itoa36(arg));
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_r_id(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
char id[8];
|
2011-03-07 08:02:35 +01:00
|
|
|
fscanf((FILE *) store->userdata, "%7s", id);
|
2010-08-08 10:06:34 +02:00
|
|
|
return atoi36(id);
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_int(struct storage *store, int arg)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
return fprintf((FILE *) store->userdata, "%d ", arg);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_r_int(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
int result;
|
2011-03-07 08:02:35 +01:00
|
|
|
fscanf((FILE *) store->userdata, "%d", &result);
|
2010-08-08 10:06:34 +02:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_flt(struct storage *store, float arg)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
return fprintf((FILE *) store->userdata, "%f ", arg);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static float txt_r_flt(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
double result;
|
2011-03-08 08:44:20 +01:00
|
|
|
fscanf((FILE *) store->userdata, "%lf", &result);
|
2010-08-08 10:06:34 +02:00
|
|
|
return (float)result;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_tok(struct storage *store, const char *tok)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
int result;
|
2011-03-07 08:02:35 +01:00
|
|
|
if (tok == NULL || tok[0] == 0) {
|
|
|
|
result = fputc(NULL_TOKEN, (FILE *) store->userdata);
|
2010-08-08 10:06:34 +02:00
|
|
|
} else {
|
|
|
|
#ifndef NDEBUG
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *find = strchr(tok, ' ');
|
|
|
|
if (!find)
|
|
|
|
find = strchr(tok, NULL_TOKEN);
|
2010-08-08 10:06:34 +02:00
|
|
|
assert(!find || !"reserved character in token");
|
|
|
|
#endif
|
2011-03-07 08:02:35 +01:00
|
|
|
assert(tok[0] != ' ');
|
|
|
|
result = fputs(tok, (FILE *) store->userdata);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
fputc(' ', (FILE *) store->userdata);
|
2010-08-08 10:06:34 +02:00
|
|
|
return result;
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
static char *txt_r_tok(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
char result[256];
|
2011-03-07 08:02:35 +01:00
|
|
|
fscanf((FILE *) store->userdata, "%256s", result);
|
|
|
|
if (result[0] == NULL_TOKEN || result[0] == 0) {
|
2010-08-08 10:06:34 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return strdup(result);
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
|
|
|
|
static void txt_r_tok_buf(struct storage *store, char *result, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
char format[16];
|
2011-03-07 08:02:35 +01:00
|
|
|
if (result && size > 0) {
|
|
|
|
format[0] = '%';
|
|
|
|
sprintf(format + 1, "%us", size);
|
|
|
|
fscanf((FILE *) store->userdata, format, result);
|
|
|
|
if (result[0] == NULL_TOKEN) {
|
2010-08-08 10:06:34 +02:00
|
|
|
result[0] = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* trick to skip when no result expected */
|
2011-03-07 08:02:35 +01:00
|
|
|
fscanf((FILE *) store->userdata, "%*s");
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_str(struct storage *store, const char *str)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
int result = fwritestr((FILE *) store->userdata, str);
|
|
|
|
fputc(' ', (FILE *) store->userdata);
|
|
|
|
return result + 1;
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static char *txt_r_str(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
char buffer[DISPLAYSIZE];
|
|
|
|
/* you should not use this */
|
2011-03-07 08:02:35 +01:00
|
|
|
freadstr((FILE *) store->userdata, store->encoding, buffer, sizeof(buffer));
|
2010-08-08 10:06:34 +02:00
|
|
|
return strdup(buffer);
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void txt_r_str_buf(struct storage *store, char *result, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
freadstr((FILE *) store->userdata, store->encoding, result, size);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_open(struct storage *store, const char *filename, int mode)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
const char *modes[] = { 0, "rt", "wt", "at" };
|
|
|
|
FILE *F = fopen(filename, modes[mode]);
|
2010-08-08 10:06:34 +02:00
|
|
|
store->userdata = F;
|
|
|
|
if (F) {
|
2011-03-08 07:31:06 +01:00
|
|
|
const unsigned char utf8_bom[4] = { 0xef, 0xbb, 0xbf };
|
2011-03-07 08:02:35 +01:00
|
|
|
if (mode == IO_READ) {
|
2010-08-08 10:06:34 +02:00
|
|
|
char token[8];
|
|
|
|
/* recognize UTF8 BOM */
|
|
|
|
store->r_tok_buf(store, token, sizeof(token));
|
2011-03-07 08:02:35 +01:00
|
|
|
if (memcmp(token, utf8_bom, 3) == 0) {
|
|
|
|
if (enc_gamedata != XML_CHAR_ENCODING_UTF8) {
|
2010-08-08 10:06:34 +02:00
|
|
|
log_warning(("Found UTF-8 BOM, assuming unicode game data.\n"));
|
|
|
|
store->encoding = XML_CHAR_ENCODING_UTF8;
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
store->version = atoi(token + 3);
|
2010-08-08 10:06:34 +02:00
|
|
|
} else {
|
2011-03-07 08:02:35 +01:00
|
|
|
if (store->encoding == XML_CHAR_ENCODING_NONE) {
|
2010-08-08 10:06:34 +02:00
|
|
|
store->encoding = XML_CHAR_ENCODING_8859_1;
|
|
|
|
log_warning(("No BOM, assuming 8859-1 game data.\n"));
|
|
|
|
}
|
|
|
|
store->version = atoi(token);
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
} else if (store->encoding == XML_CHAR_ENCODING_UTF8) {
|
2011-03-08 08:44:20 +01:00
|
|
|
fputs((const char *)utf8_bom, F);
|
2010-08-08 10:06:34 +02:00
|
|
|
fprintf(F, "%d\n", RELEASE_VERSION);
|
|
|
|
}
|
|
|
|
}
|
2011-03-07 08:02:35 +01:00
|
|
|
return (F == NULL);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_w_bin(struct storage *store, void *arg, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
assert(!"not implemented!");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static void txt_r_bin(struct storage *store, void *result, size_t size)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
|
|
|
assert(!"not implemented!");
|
|
|
|
}
|
|
|
|
|
2011-03-07 08:02:35 +01:00
|
|
|
static int txt_close(struct storage *store)
|
2010-08-08 10:06:34 +02:00
|
|
|
{
|
2011-03-07 08:02:35 +01:00
|
|
|
return fclose((FILE *) store->userdata);
|
2010-08-08 10:06:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const storage text_store = {
|
|
|
|
txt_w_brk,
|
|
|
|
txt_w_int, txt_r_int,
|
|
|
|
txt_w_flt, txt_r_flt,
|
|
|
|
txt_w_id, txt_r_id,
|
|
|
|
txt_w_tok, txt_r_tok, txt_r_tok_buf,
|
|
|
|
txt_w_str, txt_r_str, txt_r_str_buf,
|
2011-03-07 08:02:35 +01:00
|
|
|
txt_w_bin, txt_r_bin,
|
2010-08-08 10:06:34 +02:00
|
|
|
txt_open, txt_close,
|
|
|
|
0, 0, NULL
|
|
|
|
};
|