Implement human-readable dialect for xdump
This is for the forthcoming empdump utility program. The xdump
command still does not support the human-readable dialect.
xundump has supported the human-readable dialect for a long time
(commit 4871a10a
).
This commit is contained in:
parent
f9e28d0491
commit
ed0c98d3c8
3 changed files with 149 additions and 28 deletions
|
@ -42,11 +42,13 @@
|
||||||
struct xdstr {
|
struct xdstr {
|
||||||
natid cnum; /* dump for this country */
|
natid cnum; /* dump for this country */
|
||||||
int divine; /* is this a deity dump? */
|
int divine; /* is this a deity dump? */
|
||||||
|
int human; /* dump human-readable format */
|
||||||
void (*pr)(char *fmt, ...); /* callback for printing dump */
|
void (*pr)(char *fmt, ...); /* callback for printing dump */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct xdstr *xdinit(struct xdstr *, natid, void (*)(char *, ...));
|
struct xdstr *xdinit(struct xdstr *, natid, int, void (*)(char *, ...));
|
||||||
extern void xdhdr(struct xdstr *, char *, int);
|
extern void xdhdr(struct xdstr *, char *, int);
|
||||||
|
extern void xdcolhdr(struct xdstr *, struct castr[]);
|
||||||
extern void xdflds(struct xdstr *, struct castr[], void *);
|
extern void xdflds(struct xdstr *, struct castr[], void *);
|
||||||
extern struct valstr *xdeval(struct valstr *, struct xdstr *, struct castr *, void *, int);
|
extern struct valstr *xdeval(struct valstr *, struct xdstr *, struct castr *, void *, int);
|
||||||
extern char *xdprval(struct xdstr *, struct valstr *, char *);
|
extern char *xdprval(struct xdstr *, struct valstr *, char *);
|
||||||
|
|
|
@ -167,7 +167,7 @@ xdump(void)
|
||||||
if (!p || !*p)
|
if (!p || !*p)
|
||||||
return RET_SYN;
|
return RET_SYN;
|
||||||
|
|
||||||
xdinit(&xd, player->cnum, pr);
|
xdinit(&xd, player->cnum, 0, pr);
|
||||||
natp = getnatp(player->cnum);
|
natp = getnatp(player->cnum);
|
||||||
type = isdigit(p[0]) ? atoi(p) : ef_byname(p);
|
type = isdigit(p[0]) ? atoi(p) : ef_byname(p);
|
||||||
if (type < 0 || type >= EF_MAX)
|
if (type < 0 || type >= EF_MAX)
|
||||||
|
|
|
@ -76,20 +76,23 @@
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
|
#include <limits.h>
|
||||||
#include "file.h"
|
#include "file.h"
|
||||||
#include "nat.h"
|
#include "nat.h"
|
||||||
#include "xdump.h"
|
#include "xdump.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialize XD to dump for country CNUM.
|
* Initialize XD to dump for country CNUM.
|
||||||
|
* If HUMAN, dump in human-readable format.
|
||||||
* Dump is to be delivered through callback PR.
|
* Dump is to be delivered through callback PR.
|
||||||
* Return XD.
|
* Return XD.
|
||||||
*/
|
*/
|
||||||
struct xdstr *
|
struct xdstr *
|
||||||
xdinit(struct xdstr *xd, natid cnum, void (*pr)(char *fmt, ...))
|
xdinit(struct xdstr *xd, natid cnum, int human, void (*pr)(char *fmt, ...))
|
||||||
{
|
{
|
||||||
xd->cnum = cnum;
|
xd->cnum = cnum;
|
||||||
xd->divine = getnatp(cnum)->nat_stat == STAT_GOD;
|
xd->divine = getnatp(cnum)->nat_stat == STAT_GOD;
|
||||||
|
xd->human = human;
|
||||||
xd->pr = pr;
|
xd->pr = pr;
|
||||||
return xd;
|
return xd;
|
||||||
}
|
}
|
||||||
|
@ -110,14 +113,38 @@ xdeval(struct valstr *val, struct xdstr *xd,
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Dump VAL prefixed with SEP, return " ".
|
* Dump string STR to XD with funny characters escaped.
|
||||||
* VAL must be evaluated.
|
* Dump at most N characters.
|
||||||
*/
|
*/
|
||||||
char *
|
static void
|
||||||
xdprval(struct xdstr *xd, struct valstr *val, char *sep)
|
xdpresc(struct xdstr *xd, char *str, size_t n)
|
||||||
{
|
{
|
||||||
unsigned char *s, *e, *l;
|
unsigned char *s, *e, *l;
|
||||||
|
|
||||||
|
s = (unsigned char *)str;
|
||||||
|
l = s + n;
|
||||||
|
for (;;) {
|
||||||
|
for (e = s;
|
||||||
|
e < l && *e != '"' && *e != '\\' && isgraph(*e);
|
||||||
|
++e)
|
||||||
|
;
|
||||||
|
xd->pr("%.*s", (int)(e-s), s);
|
||||||
|
if (e < l && *e)
|
||||||
|
xd->pr("\\%03o", *e++);
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
s = e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dump VAL prefixed with SEP to XD, in machine readable format.
|
||||||
|
* VAL must be evaluated.
|
||||||
|
* Return " ".
|
||||||
|
*/
|
||||||
|
static char *
|
||||||
|
xdprval_nosym(struct xdstr *xd, struct valstr *val, char *sep)
|
||||||
|
{
|
||||||
if (CANT_HAPPEN(val->val_cat != NSC_VAL)) {
|
if (CANT_HAPPEN(val->val_cat != NSC_VAL)) {
|
||||||
xd->pr("%snil", sep);
|
xd->pr("%snil", sep);
|
||||||
return " ";
|
return " ";
|
||||||
|
@ -131,23 +158,9 @@ xdprval(struct xdstr *xd, struct valstr *val, char *sep)
|
||||||
xd->pr("%s%#g", sep, val->val_as.dbl);
|
xd->pr("%s%#g", sep, val->val_as.dbl);
|
||||||
break;
|
break;
|
||||||
case NSC_STRING:
|
case NSC_STRING:
|
||||||
s = (unsigned char *)val->val_as.str.base;
|
if (val->val_as.str.base) {
|
||||||
if (s) {
|
|
||||||
xd->pr("%s\"", sep);
|
xd->pr("%s\"", sep);
|
||||||
l = s + val->val_as.str.maxsz;
|
xdpresc(xd, val->val_as.str.base, val->val_as.str.maxsz);
|
||||||
/* FIXME maxsz == INT_MAX ! */
|
|
||||||
for (;;) {
|
|
||||||
for (e = s;
|
|
||||||
e < l && *e != '"' && *e != '\\' && isgraph(*e);
|
|
||||||
++e)
|
|
||||||
;
|
|
||||||
xd->pr("%.*s", (int)(e-s), s);
|
|
||||||
if (e < l && *e)
|
|
||||||
xd->pr("\\%03o", *e++);
|
|
||||||
else
|
|
||||||
break;
|
|
||||||
s = e;
|
|
||||||
}
|
|
||||||
xd->pr("\"");
|
xd->pr("\"");
|
||||||
} else
|
} else
|
||||||
xd->pr("%snil", sep);
|
xd->pr("%snil", sep);
|
||||||
|
@ -159,6 +172,70 @@ xdprval(struct xdstr *xd, struct valstr *val, char *sep)
|
||||||
return " ";
|
return " ";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dump symbol with value KEY from symbol table TYPE to XD.
|
||||||
|
* Prefix with SEP, return " ".
|
||||||
|
*/
|
||||||
|
static char *
|
||||||
|
xdprsym(struct xdstr *xd, int key, int type, char *sep)
|
||||||
|
{
|
||||||
|
char *sym = symbol_by_value(key, ef_ptr(type, 0));
|
||||||
|
|
||||||
|
if (CANT_HAPPEN(!sym))
|
||||||
|
xd->pr("%s%ld", sep, key);
|
||||||
|
else {
|
||||||
|
xd->pr("%s", sep);
|
||||||
|
xdpresc(xd, sym, INT_MAX);
|
||||||
|
}
|
||||||
|
return " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dump VAL prefixed with SEP to XD, return " ".
|
||||||
|
* VAL must be evaluated.
|
||||||
|
* CA describes the field from which the value was fetched.
|
||||||
|
*/
|
||||||
|
static char *
|
||||||
|
xdprval_sym(struct xdstr *xd, struct valstr *val, struct castr *ca, char *sep)
|
||||||
|
{
|
||||||
|
unsigned long bit;
|
||||||
|
struct castr *ca_sym = ef_cadef(ca->ca_table);
|
||||||
|
|
||||||
|
if (CANT_HAPPEN(val->val_cat != NSC_VAL)) {
|
||||||
|
xd->pr("%snil", sep);
|
||||||
|
return " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!xd->human || val->val_type != NSC_LONG
|
||||||
|
|| ca->ca_table == EF_BAD || ca_sym != symbol_ca)
|
||||||
|
return xdprval_nosym(xd, val, sep);
|
||||||
|
|
||||||
|
if (ca->ca_flags & NSC_BITS) {
|
||||||
|
xd->pr("%s(", sep);
|
||||||
|
sep = "";
|
||||||
|
for (bit = 1; bit; bit <<= 1) {
|
||||||
|
if (bit & val->val_as.lng)
|
||||||
|
sep = xdprsym(xd, bit, ca->ca_table, sep);
|
||||||
|
}
|
||||||
|
xd->pr(")");
|
||||||
|
return " ";
|
||||||
|
}
|
||||||
|
|
||||||
|
return xdprsym(xd, val->val_as.lng, ca->ca_table, sep);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dump VAL prefixed with SEP to XD, return " ".
|
||||||
|
* XD must not be human-readable.
|
||||||
|
* VAL must be evaluated.
|
||||||
|
*/
|
||||||
|
char *
|
||||||
|
xdprval(struct xdstr *xd, struct valstr *val, char *sep)
|
||||||
|
{
|
||||||
|
CANT_HAPPEN(xd->human);
|
||||||
|
return xdprval_nosym(xd, val, sep);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Dump field values of a context object to XD.
|
* Dump field values of a context object to XD.
|
||||||
* CA[] describes fields.
|
* CA[] describes fields.
|
||||||
|
@ -180,25 +257,66 @@ xdflds(struct xdstr *xd, struct castr ca[], void *ptr)
|
||||||
j = 0;
|
j = 0;
|
||||||
do {
|
do {
|
||||||
xdeval(&val, xd, &ca[i], ptr, j);
|
xdeval(&val, xd, &ca[i], ptr, j);
|
||||||
sep = xdprval(xd, &val, sep);
|
sep = xdprval_sym(xd, &val, &ca[i], sep);
|
||||||
} while (++j < n);
|
} while (++j < n);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Dump header for dump NAME.
|
* Dump header for dump NAME to XD.
|
||||||
* If META, it's for the meta-data dump rather than the data dump.
|
* If META, it's for the meta-data dump rather than the data dump.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
xdhdr(struct xdstr *xd, char *name, int meta)
|
xdhdr(struct xdstr *xd, char *name, int meta)
|
||||||
{
|
{
|
||||||
xd->pr("XDUMP %s%s %ld\n", meta ? "meta " : "", name, (long)time(NULL));
|
if (xd->human) {
|
||||||
|
xd->pr("config %s\n", name);
|
||||||
|
} else
|
||||||
|
xd->pr("XDUMP %s%s %ld\n",
|
||||||
|
meta ? "meta " : "",
|
||||||
|
name, (long)time(NULL));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Dump footer for a dump that dumped N objects. */
|
/*
|
||||||
|
* Dump column header to XD.
|
||||||
|
* CA[] describes fields.
|
||||||
|
* Does nothing unless XD is human-readable.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
xdcolhdr(struct xdstr *xd, struct castr ca[])
|
||||||
|
{
|
||||||
|
int i, j, n;
|
||||||
|
char *sep = "";
|
||||||
|
|
||||||
|
if (!xd->human)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (i = 0; ca[i].ca_name; ++i) {
|
||||||
|
if (ca[i].ca_flags & NSC_DEITY && !xd->divine)
|
||||||
|
continue;
|
||||||
|
if (ca[i].ca_flags & NSC_EXTRA)
|
||||||
|
continue;
|
||||||
|
n = ca[i].ca_type != NSC_STRINGY ? ca[i].ca_len : 0;
|
||||||
|
if (n) {
|
||||||
|
for (j = 0; j < n; j++) {
|
||||||
|
xd->pr("%s%s(%d)",sep, ca[i].ca_name, j);
|
||||||
|
sep = " ";
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
xd->pr("%s%s", sep, ca[i].ca_name);
|
||||||
|
sep = " ";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
xd->pr("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Dump footer for a dump that dumped N objects to XD. */
|
||||||
void
|
void
|
||||||
xdftr(struct xdstr *xd, int n)
|
xdftr(struct xdstr *xd, int n)
|
||||||
{
|
{
|
||||||
|
if (xd->human)
|
||||||
|
xd->pr("/config\n");
|
||||||
|
else
|
||||||
xd->pr("/%d\n", n);
|
xd->pr("/%d\n", n);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,6 +335,7 @@ xdmeta(struct xdstr *xd, int type)
|
||||||
return RET_SYN;
|
return RET_SYN;
|
||||||
|
|
||||||
xdhdr(xd, ef_nameof(type), 1);
|
xdhdr(xd, ef_nameof(type), 1);
|
||||||
|
xdcolhdr(xd, ca);
|
||||||
|
|
||||||
for (i = 0; ca[i].ca_name; i++) {
|
for (i = 0; ca[i].ca_name; i++) {
|
||||||
if (ca[i].ca_flags & NSC_DEITY && !xd->divine)
|
if (ca[i].ca_flags & NSC_DEITY && !xd->divine)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue