blob: 25e5944a0ad76d802f9962521a998b1042db6541 [file] [log] [blame]
/* Private version of the libc *mntent() routines. */
/* Note slightly different prototypes. */
/* 1999-02-22 Arkadiusz Miskiewicz <misiek@pld.ORG.PL>
* - added Native Language Support
*
* 2006-06-08 Amit Gud <agud@redhat.com>
* - Moved to nfs-utils/support/nfs from util-linux/mount
*/
#include <stdio.h>
#include <string.h> /* for strchr */
#include <ctype.h> /* for isdigit */
#include <sys/stat.h> /* for umask */
#include <unistd.h> /* for ftruncate */
#include <errno.h> /* for errno */
#include "nfs_mntent.h"
#include "nls.h"
#include "xcommon.h"
/* Unfortunately the classical Unix /etc/mtab and /etc/fstab
do not handle directory names containing spaces.
Here we mangle them, replacing a space by \040.
What do other Unices do? */
static unsigned char need_escaping[] = { ' ', '\t', '\n', '\\' };
static char *
mangle(const char *arg) {
const unsigned char *s = (const unsigned char *)arg;
char *ss, *sp;
unsigned int n;
n = strlen(arg);
ss = sp = xmalloc(4*n+1);
while(1) {
for (n = 0; n < sizeof(need_escaping); n++) {
if (*s == need_escaping[n]) {
*sp++ = '\\';
*sp++ = '0' + ((*s & 0300) >> 6);
*sp++ = '0' + ((*s & 070) >> 3);
*sp++ = '0' + (*s & 07);
goto next;
}
}
*sp++ = *s;
if (*s == 0)
break;
next:
s++;
}
return ss;
}
static int
is_space_or_tab (char c) {
return (c == ' ' || c == '\t');
}
static char *
skip_spaces(char *s) {
while (is_space_or_tab(*s))
s++;
return s;
}
static char *
skip_nonspaces(char *s) {
while (*s && !is_space_or_tab(*s))
s++;
return s;
}
#define isoctal(a) (((a) & ~7) == '0')
/* returns malloced pointer - no more strdup required */
static char *
unmangle(char *s) {
char *ret, *ss, *sp;
ss = skip_nonspaces(s);
ret = sp = xmalloc(ss-s+1);
while(s != ss) {
if (*s == '\\' && isoctal(s[1]) && isoctal(s[2]) && isoctal(s[3])) {
*sp++ = 64*(s[1] & 7) + 8*(s[2] & 7) + (s[3] & 7);
s += 4;
} else
*sp++ = *s++;
}
*sp = 0;
return ret;
}
/*
* fstat'ing the file and allocating a buffer holding all of it
* may be a bad idea: if the file is /proc/mounts, the stat
* returns 0.
* (On the other hand, mangling and unmangling is meaningless
* for /proc/mounts.)
*/
mntFILE *
nfs_setmntent (const char *file, char *mode) {
mntFILE *mfp = xmalloc(sizeof(*mfp));
mode_t old_umask = umask(077);
mfp->mntent_fp = fopen(file, mode);
umask(old_umask);
mfp->mntent_file = xstrdup(file);
mfp->mntent_errs = (mfp->mntent_fp == NULL);
mfp->mntent_softerrs = 0;
mfp->mntent_lineno = 0;
return mfp;
}
void
nfs_endmntent (mntFILE *mfp) {
if (mfp) {
if (mfp->mntent_fp)
fclose(mfp->mntent_fp);
if (mfp->mntent_file)
free(mfp->mntent_file);
free(mfp);
}
}
int
nfs_addmntent (mntFILE *mfp, struct mntent *mnt) {
char *m1, *m2, *m3, *m4;
int res;
off_t length;
if (fseek (mfp->mntent_fp, 0, SEEK_END))
return 1; /* failure */
length = ftell(mfp->mntent_fp);
m1 = mangle(mnt->mnt_fsname);
m2 = mangle(mnt->mnt_dir);
m3 = mangle(mnt->mnt_type);
m4 = mangle(mnt->mnt_opts);
res = fprintf (mfp->mntent_fp, "%s %s %s %s %d %d\n",
m1, m2, m3, m4, mnt->mnt_freq, mnt->mnt_passno);
free(m1);
free(m2);
free(m3);
free(m4);
if (res >= 0) {
res = fflush(mfp->mntent_fp);
if (res < 0) {
nfs_error("Cant't flush out mtab: %s", strerror(errno));
/* Avoid leaving a corrupt mtab file */
if (ftruncate(fileno(mfp->mntent_fp), length))
{/* Ignore this failure; Why confuse things */}
}
}
return (res < 0) ? 1 : 0;
}
/* Read the next entry from the file fp. Stop reading at an incorrect entry. */
struct mntent *
nfs_getmntent (mntFILE *mfp) {
static char buf[4096];
static struct mntent me;
char *s;
again:
if (mfp->mntent_errs || mfp->mntent_softerrs >= ERR_MAX)
return NULL;
/* read the next non-blank non-comment line */
do {
if (fgets (buf, sizeof(buf), mfp->mntent_fp) == NULL)
return NULL;
mfp->mntent_lineno++;
s = strchr (buf, '\n');
if (s == NULL) {
/* Missing final newline? Otherwise extremely */
/* long line - assume file was corrupted */
if (feof(mfp->mntent_fp)) {
fprintf(stderr, _("[mntent]: warning: no final "
"newline at the end of %s\n"),
mfp->mntent_file);
s = strchr (buf, 0);
} else {
mfp->mntent_errs = 1;
goto err;
}
}
*s = 0;
if (--s >= buf && *s == '\r')
*s = 0;
s = skip_spaces(buf);
} while (*s == '\0' || *s == '#');
me.mnt_fsname = unmangle(s);
s = skip_nonspaces(s);
s = skip_spaces(s);
me.mnt_dir = unmangle(s);
s = skip_nonspaces(s);
s = skip_spaces(s);
me.mnt_type = unmangle(s);
s = skip_nonspaces(s);
s = skip_spaces(s);
me.mnt_opts = unmangle(s);
s = skip_nonspaces(s);
s = skip_spaces(s);
if (isdigit(*s)) {
me.mnt_freq = atoi(s);
while(isdigit(*s)) s++;
} else
me.mnt_freq = 0;
if(*s && !is_space_or_tab(*s))
goto err;
s = skip_spaces(s);
if(isdigit(*s)) {
me.mnt_passno = atoi(s);
while(isdigit(*s)) s++;
} else
me.mnt_passno = 0;
if(*s && !is_space_or_tab(*s))
goto err;
/* allow more stuff, e.g. comments, on this line */
return &me;
err:
mfp->mntent_softerrs++;
fprintf(stderr, _("[mntent]: line %d in %s is bad%s\n"),
mfp->mntent_lineno, mfp->mntent_file,
(mfp->mntent_errs || mfp->mntent_softerrs >= ERR_MAX) ?
_("; rest of file ignored") : "");
goto again;
}