Comments.
This commit is contained in:
parent
c73f7c207f
commit
fe602d00d6
1 changed files with 150 additions and 11 deletions
|
@ -33,13 +33,19 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
* See doc/xdump! And keep it up-to-date.
|
||||||
|
*
|
||||||
|
* Parsing of machine-readable xdump is not precise: it recognizes
|
||||||
|
* comments, accepts whitespace in place of single space, and accepts
|
||||||
|
* the full human-readable field syntax instead of its machine-
|
||||||
|
* readable subset.
|
||||||
|
*
|
||||||
* FIXME:
|
* FIXME:
|
||||||
* - Normalize terminology: table/rows/columns or file/records/fields
|
* - Normalize terminology: table/rows/columns or file/records/fields
|
||||||
* - Loading tables with NSC_STRING elements more than once leaks memory
|
* - Loading tables with NSC_STRING elements more than once leaks memory
|
||||||
* TODO:
|
* TODO:
|
||||||
* - Check each partial table supplies the same rows
|
* - Check each partial table supplies the same rows
|
||||||
* - Check EFF_CFG tables are dense
|
* - Check EFF_CFG tables are dense
|
||||||
* - Symbolic references to non-symbol tables
|
|
||||||
* - Symbolic array indexes
|
* - Symbolic array indexes
|
||||||
* TODO, but hardly worth the effort:
|
* TODO, but hardly worth the effort:
|
||||||
* - Permit reordering of array elements
|
* - Permit reordering of array elements
|
||||||
|
@ -57,16 +63,18 @@
|
||||||
#include "optlist.h"
|
#include "optlist.h"
|
||||||
#include "prototypes.h"
|
#include "prototypes.h"
|
||||||
|
|
||||||
static char *fname;
|
static char *fname; /* Name of file being read */
|
||||||
static int lineno;
|
static int lineno; /* Current line number */
|
||||||
static int human;
|
static int human; /* Reading human-readable syntax? */
|
||||||
static int ellipsis, is_partial;
|
static int ellipsis; /* Header ended with ...? */
|
||||||
static int cur_type, cur_id;
|
static int is_partial; /* Is input split into parts? */
|
||||||
static void *cur_obj;
|
static int cur_type; /* Current table's file type */
|
||||||
|
static void *cur_obj; /* The object being read into */
|
||||||
|
static int cur_id; /* and its index in the table */
|
||||||
static int cur_obj_is_blank;
|
static int cur_obj_is_blank;
|
||||||
static int nflds;
|
static int nflds; /* #fields in input records */
|
||||||
static struct castr **fldca;
|
static struct castr **fldca; /* Map field number to selector */
|
||||||
static int *fldidx;
|
static int *fldidx; /* Map field number to index */
|
||||||
static int *caflds; /* Map selector number to #fields seen */
|
static int *caflds; /* Map selector number to #fields seen */
|
||||||
static int *cafldspp; /* ditto, in previous parts */
|
static int *cafldspp; /* ditto, in previous parts */
|
||||||
|
|
||||||
|
@ -83,6 +91,9 @@ static int add2symset(int, long *, char *);
|
||||||
static int xubody(FILE *);
|
static int xubody(FILE *);
|
||||||
static int xutail(FILE *, struct castr *);
|
static int xutail(FILE *, struct castr *);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Gripe about the current line to stderr, return -1.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
gripe(char *fmt, ...)
|
gripe(char *fmt, ...)
|
||||||
{
|
{
|
||||||
|
@ -97,6 +108,10 @@ gripe(char *fmt, ...)
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read and ignore field separators from FP.
|
||||||
|
* Return first character that is not a field separator.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
skipfs(FILE *fp)
|
skipfs(FILE *fp)
|
||||||
{
|
{
|
||||||
|
@ -115,6 +130,11 @@ skipfs(FILE *fp)
|
||||||
return ch;
|
return ch;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read an identifier from FP into BUF.
|
||||||
|
* BUF must have space for 1024 characters.
|
||||||
|
* Return number of characters read on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
getid(FILE *fp, char *buf)
|
getid(FILE *fp, char *buf)
|
||||||
{
|
{
|
||||||
|
@ -125,6 +145,10 @@ getid(FILE *fp, char *buf)
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode escape sequences in BUF.
|
||||||
|
* Return BUF on success, null pointer on failure.
|
||||||
|
*/
|
||||||
static char *
|
static char *
|
||||||
xuesc(char *buf)
|
xuesc(char *buf)
|
||||||
{
|
{
|
||||||
|
@ -146,6 +170,13 @@ xuesc(char *buf)
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Try to read a field name from FP.
|
||||||
|
* I is the field number, counting from zero.
|
||||||
|
* If a name is read, set fldca[I] and fldidx[I] for it, and update
|
||||||
|
* caflds[].
|
||||||
|
* Return 1 if a name or ... was read, 0 on end of line, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xufldname(FILE *fp, int i)
|
xufldname(FILE *fp, int i)
|
||||||
{
|
{
|
||||||
|
@ -202,6 +233,11 @@ xufldname(FILE *fp, int i)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Try to read a field value from FP.
|
||||||
|
* I is the field number, counting from zero.
|
||||||
|
* Return 1 if a value was read, 0 on end of line, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xufld(FILE *fp, int i)
|
xufld(FILE *fp, int i)
|
||||||
{
|
{
|
||||||
|
@ -271,6 +307,11 @@ xufld(FILE *fp, int i)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read fields from FP.
|
||||||
|
* Use PARSE() to read each field.
|
||||||
|
* Return number of fields read on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xuflds(FILE *fp, int (*parse)(FILE *, int))
|
xuflds(FILE *fp, int (*parse)(FILE *, int))
|
||||||
{
|
{
|
||||||
|
@ -290,6 +331,13 @@ xuflds(FILE *fp, int (*parse)(FILE *, int))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Define the FLDNO-th field.
|
||||||
|
* If IDX is negative, define as selector NAME, else as NAME(IDX).
|
||||||
|
* Set fldca[FLDNO] and fldidx[FLDNO] accordingly.
|
||||||
|
* Update caflds[].
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
deffld(int fldno, char *name, int idx)
|
deffld(int fldno, char *name, int idx)
|
||||||
{
|
{
|
||||||
|
@ -329,6 +377,11 @@ deffld(int fldno, char *name, int idx)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Record that header ends with ...
|
||||||
|
* Set ellipsis and is_partial.
|
||||||
|
* Return 0 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
defellipsis(void)
|
defellipsis(void)
|
||||||
{
|
{
|
||||||
|
@ -340,6 +393,10 @@ defellipsis(void)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check fields in xdump are sane.
|
||||||
|
* Return 0 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
chkflds(void)
|
chkflds(void)
|
||||||
{
|
{
|
||||||
|
@ -373,6 +430,11 @@ chkflds(void)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get selector for field FLDNO.
|
||||||
|
* Assign the field's selector index to *IDX, unless it is null.
|
||||||
|
* Return the selector on success, null pointer on error.
|
||||||
|
*/
|
||||||
static struct castr *
|
static struct castr *
|
||||||
getfld(int fldno, int *idx)
|
getfld(int fldno, int *idx)
|
||||||
{
|
{
|
||||||
|
@ -387,6 +449,9 @@ getfld(int fldno, int *idx)
|
||||||
return fldca[fldno];
|
return fldca[fldno];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Is a new value for field FLDNO required to match the old one?
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
fldval_must_match(int fldno)
|
fldval_must_match(int fldno)
|
||||||
{
|
{
|
||||||
|
@ -402,6 +467,11 @@ fldval_must_match(int fldno)
|
||||||
|| fldidx[fldno] < cafldspp[i];
|
|| fldidx[fldno] < cafldspp[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get the current object.
|
||||||
|
* Store it in cur_obj, and set cur_obj_is_blank accordingly.
|
||||||
|
* Return cur_obj, which is null on error.
|
||||||
|
*/
|
||||||
static void *
|
static void *
|
||||||
getobj(void)
|
getobj(void)
|
||||||
{
|
{
|
||||||
|
@ -411,7 +481,7 @@ getobj(void)
|
||||||
if (!cur_obj) {
|
if (!cur_obj) {
|
||||||
cur_obj_is_blank = cur_id >= ep->fids;
|
cur_obj_is_blank = cur_id >= ep->fids;
|
||||||
if (cur_obj_is_blank) {
|
if (cur_obj_is_blank) {
|
||||||
/* TODO grow cache (and posssibly file) unless EFF_STATIC */
|
/* TODO grow cache (and possibly file) unless EFF_STATIC */
|
||||||
if (cur_id < ep->csize - !!need_sentinel)
|
if (cur_id < ep->csize - !!need_sentinel)
|
||||||
ep->cids = ep->fids = cur_id + 1;
|
ep->cids = ep->fids = cur_id + 1;
|
||||||
/* else: ef_ptr() will fail */
|
/* else: ef_ptr() will fail */
|
||||||
|
@ -425,6 +495,10 @@ getobj(void)
|
||||||
return cur_obj;
|
return cur_obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set value of field FLDNO in current object to DBL.
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
setnum(int fldno, double dbl)
|
setnum(int fldno, double dbl)
|
||||||
{
|
{
|
||||||
|
@ -511,6 +585,10 @@ setnum(int fldno, double dbl)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set value of field FLDNO in current object to STR.
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
setstr(int fldno, char *str)
|
setstr(int fldno, char *str)
|
||||||
{
|
{
|
||||||
|
@ -564,6 +642,11 @@ setstr(int fldno, char *str)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Resolve symbol name ID in table referred to by CA.
|
||||||
|
* Use field number N for error messages.
|
||||||
|
* Return index in referred table on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xunsymbol(char *id, struct castr *ca, int n)
|
xunsymbol(char *id, struct castr *ca, int n)
|
||||||
{
|
{
|
||||||
|
@ -575,6 +658,10 @@ xunsymbol(char *id, struct castr *ca, int n)
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Map symbol index to symbol value.
|
||||||
|
* CA is the table, and I is the index in it.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
symval(struct castr *ca, int i)
|
symval(struct castr *ca, int i)
|
||||||
{
|
{
|
||||||
|
@ -589,6 +676,10 @@ symval(struct castr *ca, int i)
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set value of field FLDNO in current object to value of symbol SYM.
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
setsym(int fldno, char *sym)
|
setsym(int fldno, char *sym)
|
||||||
{
|
{
|
||||||
|
@ -608,6 +699,10 @@ setsym(int fldno, char *sym)
|
||||||
return setnum(fldno, symval(ca, i));
|
return setnum(fldno, symval(ca, i));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Create an empty symbol set for field FLDNO in *SET.
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
mtsymset(int fldno, long *set)
|
mtsymset(int fldno, long *set)
|
||||||
{
|
{
|
||||||
|
@ -624,6 +719,11 @@ mtsymset(int fldno, long *set)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Add a symbol to a symbol set for field FLDNO in *SET.
|
||||||
|
* SYM is the name of the symbol to add.
|
||||||
|
* Return 1 on success, -1 on error.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
add2symset(int fldno, long *set, char *sym)
|
add2symset(int fldno, long *set, char *sym)
|
||||||
{
|
{
|
||||||
|
@ -641,6 +741,13 @@ add2symset(int fldno, long *set, char *sym)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read an xdump table header line from FP.
|
||||||
|
* Expect header for EXPECTED_TABLE, unless it is EF_BAD.
|
||||||
|
* Recognize header for machine- and human-readable syntax, and set
|
||||||
|
* human accordingly.
|
||||||
|
* Return table type on success, -2 on EOF before header, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xuheader(FILE *fp, int expected_table)
|
xuheader(FILE *fp, int expected_table)
|
||||||
{
|
{
|
||||||
|
@ -681,6 +788,13 @@ xuheader(FILE *fp, int expected_table)
|
||||||
return type;
|
return type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Find fields in this xdump.
|
||||||
|
* If reading human-readable syntax, read a field header line from FP.
|
||||||
|
* Else take fields from the table's selectors in CA[].
|
||||||
|
* Set ellipsis, nflds, fldca[], fldidx[] and caflds[] accordingly.
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xufldhdr(FILE *fp, struct castr ca[])
|
xufldhdr(FILE *fp, struct castr ca[])
|
||||||
{
|
{
|
||||||
|
@ -721,6 +835,13 @@ xufldhdr(FILE *fp, struct castr ca[])
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read xdump footer from FP.
|
||||||
|
* CA[] contains the table's selectors.
|
||||||
|
* The body had RECS records.
|
||||||
|
* Update cafldspp[] from caflds[].
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xufooter(FILE *fp, struct castr ca[], int recs)
|
xufooter(FILE *fp, struct castr ca[], int recs)
|
||||||
{
|
{
|
||||||
|
@ -749,6 +870,15 @@ xufooter(FILE *fp, struct castr ca[], int recs)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read an xdump table from FP.
|
||||||
|
* Both machine- and human-readable xdump syntax are recognized.
|
||||||
|
* Expect table EXPECTED_TABLE, unless it is EF_BAD.
|
||||||
|
* Report errors to stderr.
|
||||||
|
* Messages assume FP starts in the file FILE at line *PLNO.
|
||||||
|
* Update *PLNO to reflect lines read from FP.
|
||||||
|
* Return table type on success, -2 on EOF before header, -1 on failure.
|
||||||
|
*/
|
||||||
int
|
int
|
||||||
xundump(FILE *fp, char *file, int *plno, int expected_table)
|
xundump(FILE *fp, char *file, int *plno, int expected_table)
|
||||||
{
|
{
|
||||||
|
@ -794,6 +924,11 @@ xundump(FILE *fp, char *file, int *plno, int expected_table)
|
||||||
return type;
|
return type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read the remainder of an xdump after the table header line from FP.
|
||||||
|
* CA[] contains the table's selectors.
|
||||||
|
* Return 0 on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xutail(FILE *fp, struct castr *ca)
|
xutail(FILE *fp, struct castr *ca)
|
||||||
{
|
{
|
||||||
|
@ -814,6 +949,10 @@ xutail(FILE *fp, struct castr *ca)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Read the body of an xdump table from FP.
|
||||||
|
* Return number of rows read on success, -1 on failure.
|
||||||
|
*/
|
||||||
static int
|
static int
|
||||||
xubody(FILE *fp)
|
xubody(FILE *fp)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue