#ifndef lint static const char RCSid[] = "$Id: loadvars.c,v 2.20 2023/06/10 15:49:55 greg Exp $"; #endif /* * Routines for loading and checking variables from file. */ #include "copyright.h" #include #include #include #include "standard.h" #include "vars.h" #define NOCHAR 127 /* constant for character to delete */ extern char *fgetline(); void loadvars( /* load variables into vv from file */ const char *rfname ) { FILE *fp; char buf[512]; char *cp; if (rfname == NULL) fp = stdin; else if ((fp = fopen(rfname, "r")) == NULL) { perror(rfname); quit(1); } while (fgetline(buf, sizeof(buf), fp) != NULL) { for (cp = buf; *cp; cp++) { switch (*cp) { case '\\': *cp++ = NOCHAR; continue; case '#': *cp = '\0'; break; default: continue; } break; } if (setvariable(buf, matchvar) < 0) { fprintf(stderr, "%s: unknown variable: %s\n", rfname, buf); quit(1); } } if (fp != stdin) fclose(fp); } int setvariable( /* assign variable according to string */ const char *ass, VARIABLE *(*mv)(const char*) ) { int quote = '\0'; char varname[32]; int n; char *cp; VARIABLE *vp; int i; while (isspace(*ass)) /* skip leading space */ ass++; cp = varname; /* extract name */ while (cp < varname+sizeof(varname)-1 && *ass && !isspace(*ass) && *ass != '=') *cp++ = *ass++; *cp = '\0'; if (!varname[0]) return(0); /* no variable name! */ /* trim value */ while (isspace(*ass) || *ass == '=') ass++; for (n = strlen(ass); n > 0; n--) if (!isspace(ass[n-1])) break; if (!n) return(0); /* no assignment */ /* match variable from list */ vp = (*mv)(varname); if (vp == NULL) return(-1); /* assign new value */ if ( (i = vp->nass) ) { cp = vp->value; while (i--) while (*cp++) ; i = cp - vp->value; vp->value = (char *)realloc((void *)vp->value, i+n+1); } else vp->value = (char *)malloc(n+1); if (vp->value == NULL) { perror(progname); quit(1); } cp = vp->value+i; /* copy value */ *cp = *ass; for (i = 1; i <= n; i++) { if (ass[i] == NOCHAR) continue; if (quote) { /* don't change quoted parts */ quote *= (ass[i] != quote); } else { /* otherwise, squeeze spaces */ if (isspace(*cp)) while (isspace(ass[i])) i++; if ((ass[i] == '"') | (ass[i] == '\'')) quote = ass[i]; } *++cp = ass[i]; } if (isspace(*cp)) /* remove trailing space */ *cp = '\0'; return(++vp->nass); } VARIABLE * matchvar( /* match a variable by its name */ const char *nam ) { int n = strlen(nam); int i; for (i = 0; i < NVARS; i++) if (n >= vv[i].nick && !strncmp(nam, vv[i].name, n)) return(vv+i); return(NULL); } char * nvalue( /* return nth variable value */ int vn, int n ) { char *cp; if ((vval(vn) == NULL) | (n < 0) | (n >= vdef(vn))) return(NULL); cp = vval(vn); while (n--) while (*cp++) ; return(cp); } void checkvalues(void) /* check assignments */ { int i; for (i = 0; i < NVARS; i++) if (vv[i].fixval != NULL) (*vv[i].fixval)(vv+i); } void onevalue( /* only one assignment for this variable */ VARIABLE *vp ) { if (vp->nass < 2) return; if (!nowarn) fprintf(stderr, "%s: warning - multiple assignment of variable '%s'\n", progname, vp->name); do vp->value += strlen(vp->value)+1; while (--vp->nass > 1); } void catvalues( /* concatenate variable values */ VARIABLE *vp ) { char *cp; if (vp->nass < 2) return; for (cp = vp->value; vp->nass > 1; vp->nass--) { while (*cp) cp++; *cp++ = ' '; } } int badmatch( /* case insensitive truncated comparison */ char *tv, char *cv ) { if (!*tv) return(1); /* null string cannot match */ do if (UPPER(*tv) != *cv++) return(1); while (*++tv); return(0); /* OK */ } void boolvalue( /* check boolean for legal values */ VARIABLE *vp ) { if (!vp->nass) return; onevalue(vp); switch (UPPER(vp->value[0])) { case 'T': if (badmatch(vp->value, "TRUE")) break; return; case 'F': if (badmatch(vp->value, "FALSE")) break; return; } fprintf(stderr, "%s: illegal value for boolean variable '%s'\n", progname, vp->name); quit(1); } void qualvalue( /* check qualitative var. for legal values */ VARIABLE *vp ) { if (!vp->nass) return; onevalue(vp); switch (UPPER(vp->value[0])) { case 'L': if (badmatch(vp->value, "LOW")) break; return; case 'M': if (badmatch(vp->value, "MEDIUM")) break; return; case 'H': if (badmatch(vp->value, "HIGH")) break; return; } fprintf(stderr, "%s: illegal value for qualitative variable '%s'\n", progname, vp->name); quit(1); } void strvalue( /* check for single (quoted) string value */ VARIABLE *vp ) { if (!vp->nass) return; onevalue(vp); if ((vp->value[0] == '"') | (vp->value[0] == '\'')) { char *cp = vp->value + strlen(vp->value+1); if ((cp != vp->value) & (*cp == vp->value[0])) { vp->value++; /* elide quotation marks */ *cp = '\0'; } } } void intvalue( /* check integer variable for legal values */ VARIABLE *vp ) { if (!vp->nass) return; onevalue(vp); if (isint(vp->value)) return; fprintf(stderr, "%s: illegal value for integer variable '%s'\n", progname, vp->name); quit(1); } void fltvalue( /* check float variable for legal values */ VARIABLE *vp ) { if (!vp->nass) return; onevalue(vp); if (isflt(vp->value)) return; fprintf(stderr, "%s: illegal value for real variable '%s'\n", progname, vp->name); quit(1); } void printvars( /* print variable values */ FILE *fp ) { int i, j, k, clipline; char *cp; for (i = 0; i < NVARS; i++) /* print each variable */ for (j = 0; j < vdef(i); j++) { /* print each assignment */ fputs(vnam(i), fp); fputs("= ", fp); k = clipline = ( vv[i].fixval == catvalues ? 64 : 236 ) - strlen(vnam(i)) ; cp = nvalue(i, j); while (*cp) { putc(*cp++, fp); if (--k <= 0) { /* line too long */ while (*cp && !isspace(*cp)) fputc(*cp++, fp); /* finish this word */ if (*cp) { /* start new line */ if (vv[i].fixval == catvalues) { fputc('\n', fp); fputs(vnam(i), fp); fputc('=', fp); } else fputs(" \\\n", fp); k = clipline; } } } fputc('\n', fp); } fflush(fp); }