X-Git-Url: https://git.decadent.org.uk/gitweb/?p=nfs-utils.git;a=blobdiff_plain;f=support%2Fnfs%2Fexports.c;h=0dce25ff9291dd28d0c4b608c50fe1cc289f9d62;hp=3b34d4ee55e1f420a6923fada953aa15a30d51b3;hb=de8fd61f8e692d5d8207d7c30e8914da05ca122f;hpb=4b5c4d21704d0767747c71b22f0e91efe59c07d6 diff --git a/support/nfs/exports.c b/support/nfs/exports.c index 3b34d4e..0dce25f 100644 --- a/support/nfs/exports.c +++ b/support/nfs/exports.c @@ -29,8 +29,9 @@ #include "xio.h" #define EXPORT_DEFAULT_FLAGS \ - (NFSEXP_ASYNC|NFSEXP_READONLY|NFSEXP_ROOTSQUASH|NFSEXP_GATHERED_WRITES) + (NFSEXP_READONLY|NFSEXP_ROOTSQUASH|NFSEXP_GATHERED_WRITES) +static char *efname = NULL; static XFILE *efp = NULL; static int first; static int *squids = NULL, nsquids = 0, @@ -55,11 +56,12 @@ setexportent(char *fname, char *type) if (!(efp = xfopen(fname, type))) xlog(L_ERROR, "can't open %s for %sing", fname, strcmp(type, "r")? "writ" : "read"); + efname = strdup(fname); first = 1; } struct exportent * -getexportent(void) +getexportent(int fromkernel) { static struct exportent ee; char exp[512]; @@ -72,6 +74,15 @@ getexportent(void) freesquash(); ee.e_flags = EXPORT_DEFAULT_FLAGS; + /* some kernels assume the default is sync rather than + * async. More recent kernels always report one or other, + * but this test makes sure we assume same as kernel + * Ditto for wgather + */ + if (fromkernel) { + ee.e_flags &= ~NFSEXP_ASYNC; + ee.e_flags &= ~NFSEXP_GATHERED_WRITES; + } ee.e_maptype = CLE_MAP_IDENT; ee.e_anonuid = -2; ee.e_anongid = -2; @@ -98,14 +109,16 @@ getexportent(void) if (ok == 0) exp[0] = '\0'; if ((opt = strchr(exp, '(')) != NULL) { + if (opt == exp) + xlog(L_WARNING, "No host name given with %s %s, suggest *%s to avoid warning", ee.e_path, exp, exp); *opt++ = '\0'; if (!(sp = strchr(opt, ')')) || sp[1] != '\0') { syntaxerr("bad option list"); return NULL; } *sp = '\0'; - if (parseopts(opt, &ee) < 0) - return NULL; + } else { + xlog(L_WARNING, "No options for %s %s: suggest %s(sync) to avoid warning", ee.e_path, exp, exp); } if (strlen(exp) >= sizeof(ee.e_hostname)) { syntaxerr("client name too long"); @@ -114,6 +127,9 @@ getexportent(void) strncpy(ee.e_hostname, exp, sizeof (ee.e_hostname) - 1); ee.e_hostname[sizeof (ee.e_hostname) - 1] = '\0'; + if (parseopts(opt, &ee) < 0) + return NULL; + /* resolve symlinks */ if (realpath(ee.e_path, rpath) != NULL) { rpath[sizeof (rpath) - 1] = '\0'; @@ -131,12 +147,19 @@ putexportent(struct exportent *ep) { FILE *fp; int *id, i; + char *esc=ep->e_path; if (!efp) return; fp = efp->x_fp; - fprintf(fp, "%s\t%s(", ep->e_path, ep->e_hostname); + for (i=0; esc[i]; i++) + if (iscntrl(esc[i]) || esc[i] == '"' || esc[i] == '\\'|| isspace(esc[i])) + fprintf(fp, "\\%03o", esc[i]); + else + fprintf(fp, "%c", esc[i]); + + fprintf(fp, "\t%s(", ep->e_hostname); fprintf(fp, "%s,", (ep->e_flags & NFSEXP_READONLY)? "ro" : "rw"); fprintf(fp, "%ssync,", (ep->e_flags & NFSEXP_ASYNC)? "a" : ""); fprintf(fp, "%swdelay,", (ep->e_flags & NFSEXP_GATHERED_WRITES)? @@ -194,6 +217,9 @@ endexportent(void) if (efp) xfclose(efp); efp = NULL; + if (efname) + free(efname); + efname = NULL; freesquash(); } @@ -241,7 +267,7 @@ mkexportent(char *hname, char *path, char *options) ee.e_path[sizeof (ee.e_path) - 1] = '\0'; strncpy (ee.m_path, ee.e_path, sizeof (ee.m_path) - 1); ee.m_path [sizeof (ee.m_path) - 1] = '\0'; - if (options && parseopts(options, &ee) < 0) + if (parseopts(options, &ee) < 0) return NULL; return ⅇ } @@ -249,7 +275,7 @@ mkexportent(char *hname, char *path, char *options) int updateexportent(struct exportent *eep, char *options) { - if (options && parseopts(options, eep) < 0) + if (parseopts(options, eep) < 0) return 0; return 1; } @@ -260,12 +286,17 @@ updateexportent(struct exportent *eep, char *options) static int parseopts(char *cp, struct exportent *ep) { + int had_sync_opt = 0; squids = ep->e_squids; nsquids = ep->e_nsquids; sqgids = ep->e_sqgids; nsqgids = ep->e_nsqgids; + if (!cp) + goto out; + while (isblank(*cp)) cp++; + while (*cp) { char *opt = strdup(cp); char *optstart = cp; @@ -285,14 +316,20 @@ parseopts(char *cp, struct exportent *ep) ep->e_flags &= ~NFSEXP_INSECURE_PORT; else if (!strcmp(opt, "insecure")) ep->e_flags |= NFSEXP_INSECURE_PORT; - else if (!strcmp(opt, "sync")) + else if (!strcmp(opt, "sync")) { + had_sync_opt = 1; ep->e_flags &= ~NFSEXP_ASYNC; - else if (!strcmp(opt, "async")) + } else if (!strcmp(opt, "async")) { + had_sync_opt = 1; ep->e_flags |= NFSEXP_ASYNC; - else if (!strcmp(opt, "nohide")) + } else if (!strcmp(opt, "nohide")) ep->e_flags |= NFSEXP_CROSSMNT; else if (!strcmp(opt, "hide")) ep->e_flags &= ~NFSEXP_CROSSMNT; + else if (!strcmp(opt, "crossmnt")) /* old style */ + ep->e_flags |= NFSEXP_CROSSMNT; + else if (!strcmp(opt, "nocrossmnt")) /* old style */ + ep->e_flags &= ~NFSEXP_CROSSMNT; else if (!strcmp(opt, "wdelay")) ep->e_flags |= NFSEXP_GATHERED_WRITES; else if (!strcmp(opt, "no_wdelay")) @@ -338,9 +375,8 @@ parseopts(char *cp, struct exportent *ep) return -1; } } else { - xlog(L_ERROR, - "Unknown keyword \"%s\" in export file\n", - opt); + xlog(L_ERROR, "%s:%d: unknown keyword \"%s\"\n", + efname, efp->x_line, opt); ep->e_flags |= NFSEXP_ALLSQUASH | NFSEXP_READONLY; free(opt); return -1; @@ -355,6 +391,13 @@ parseopts(char *cp, struct exportent *ep) ep->e_nsquids = nsquids; ep->e_nsqgids = nsqgids; +out: + if (!had_sync_opt) + xlog(L_WARNING, "No 'sync' or 'async' option specified for export \"%s:%s\".\n" + " Assuming default behaviour ('sync').\n" + " NOTE: this default has changed from previous versions\n", + ep->e_hostname, ep->e_path); + return 1; } @@ -457,14 +500,15 @@ getexport(char *exp, int len) xskip(efp, " \t"); if ((ok = xgettok(efp, 0, exp, len)) < 0) - xlog(L_ERROR, "error parsing export entry"); + xlog(L_ERROR, "%s:%d: syntax error", + efname, efp->x_line); return ok; } static void syntaxerr(char *msg) { - xlog(L_ERROR, "syntax error in exports file (line %d): %s", - efp->x_line, msg); + xlog(L_ERROR, "%s:%d: syntax error: %s", + efname, efp->x_line, msg); }