|
Message-ID: <20220109031250.GN7074@brightrain.aerifal.cx> Date: Sat, 8 Jan 2022 22:12:51 -0500 From: Rich Felker <dalias@...c.org> To: Kaihang Zhang <kaihang.zhang@...rtx.com> Cc: musl@...ts.openwall.com, 2010267516@...com Subject: Re: [PATCH v2] fix: Truncate the too-long mntent in function getmntent_r On Fri, Oct 15, 2021 at 08:20:00AM -0400, Kaihang Zhang wrote: > In function getmntent_r in source misc/mntent.c, entry that is too long > will be truncated rather than discarded. The caller can tell by errno > whether the supplied buffer is too small, and retry from the beginning > of the file. > --- > src/misc/mntent.c | 53 +++++++++++++++++++++++++++++------------------ > 1 file changed, 33 insertions(+), 20 deletions(-) > > diff --git a/src/misc/mntent.c b/src/misc/mntent.c > index eabb8200..085ce45d 100644 > --- a/src/misc/mntent.c > +++ b/src/misc/mntent.c > @@ -21,12 +21,12 @@ int endmntent(FILE *f) > > struct mntent *getmntent_r(FILE *f, struct mntent *mnt, char *linebuf, int buflen) > { > - int cnt, n[8], use_internal = (linebuf == SENTINEL); > - > - mnt->mnt_freq = 0; > - mnt->mnt_passno = 0; > + int use_internal = (linebuf == SENTINEL); > + char *sub; > > do { > + char *end_ptr; > + > if (use_internal) { > getline(&internal_buf, &internal_bufsize, f); > linebuf = internal_buf; > @@ -34,25 +34,38 @@ struct mntent *getmntent_r(FILE *f, struct mntent *mnt, char *linebuf, int bufle > fgets(linebuf, buflen, f); > } > if (feof(f) || ferror(f)) return 0; > - if (!strchr(linebuf, '\n')) { > + > + end_ptr = strchr(linebuf, '\n'); > + if (end_ptr != NULL) { > + while ((end_ptr[-1] == ' ' || end_ptr[-1] == '\t') && end_ptr != linebuf) end_ptr--; > + *end_ptr = '\0'; Unless I'm misreading, this seems to invoke UB by reading the [-1] index before checking if that's valid. It could be fixed by just changing the order of the comparison expressions, but I'm not clear why this needs to be done anyway. > + } else { > fscanf(f, "%*[^\n]%*[\n]"); > errno = ERANGE; > - return 0; > } > - cnt = sscanf(linebuf, " %n%*s%n %n%*s%n %n%*s%n %n%*s%n %d %d", > - n, n+1, n+2, n+3, n+4, n+5, n+6, n+7, > - &mnt->mnt_freq, &mnt->mnt_passno); > - } while (cnt < 2 || linebuf[n[0]] == '#'); > - > - linebuf[n[1]] = 0; > - linebuf[n[3]] = 0; > - linebuf[n[5]] = 0; > - linebuf[n[7]] = 0; > - > - mnt->mnt_fsname = linebuf+n[0]; > - mnt->mnt_dir = linebuf+n[2]; > - mnt->mnt_type = linebuf+n[4]; > - mnt->mnt_opts = linebuf+n[6]; > + > + linebuf += strspn(linebuf, " \t"); > + } while (linebuf[0] == '\0' || linebuf[0] == '#'); > + > + mnt->mnt_fsname = strsep(&linebuf, " \t"); > + > + if (linebuf) linebuf += strspn(linebuf, " \t"); > + sub = strsep(&linebuf, " \t"); > + mnt->mnt_dir = sub ? sub : (char *) ""; "" already has type char[1] and decays to char *; no cast is needed here. > + > + if (linebuf) linebuf += strspn(linebuf, " \t"); > + sub = strsep (&linebuf, " \t"); > + mnt->mnt_type = sub ? sub : (char *) ""; > + > + if (linebuf) linebuf += strspn(linebuf, " \t"); > + sub = strsep(&linebuf, " \t"); > + mnt->mnt_opts = sub ? sub : (char *) ""; > + > + switch (linebuf ? sscanf(linebuf, " %d %d", &mnt->mnt_freq, &mnt->mnt_passno) : 0) { > + case 0: mnt->mnt_freq = 0; > + case 1: mnt->mnt_passno = 0; > + case 2: break; > + } > > return mnt; > } > -- > 2.25.4 This is gratuitously rewriting a lot of parsing logic in a form that doesn't seem like it's better, and even if it were, the change is orthogonal to fixing the behavior. I'm sorry for taking so long to get back to you and say this clearly. I do want to move forward on this because I know folks have been waiting on an upstream fix for a long time. But I need a patch accompanied by a clear explanation of the behavioral changes it's making, and just those changes. Or if we're in agreement on what the behavioral changes should be, I can just write the patch. The patch by Alyssa Ross is more minimal and better documented, but I'm not sure it covers everything you're concerned about. Could you let me know whether it does? I'll follow up on that thread about whether there are open issues with it. Rich
Powered by blists - more mailing lists
Confused about mailing lists and their use? Read about mailing lists on Wikipedia and check out these guidelines on proper formatting of your messages.