From: Zack Weinberg <zack@owlfolio.org>
To: Alejandro Colomar <alx.manpages@gmail.com>
Cc: libc-alpha@sourceware.org,
'linux-man' <linux-man@vger.kernel.org>,
Ian Abbott <abbotti@mev.co.uk>
Subject: Re: [PATCH] scanf.3: Do not mention the ERANGE error
Date: Thu, 29 Dec 2022 01:39:58 -0500 [thread overview]
Message-ID: <ypikv8luogqp.wl-zack@owlfolio.org> (raw)
In-Reply-To: <4fe9ed93-8fb9-64d0-26f1-a9560387d108@gmail.com>
On Wed, 14 Dec 2022 05:47:12 -0500, Alejandro Colomar wrote:
> On 12/14/22 03:13, Zack Weinberg wrote:
> > The scanf numeric conversions could be fixed with a one-sentence edit to
> > the C standard: change the last sentence of http://port70.net/~nsz/c/c11/n1570.html#7.21.6.2p10
> > from “If this object does not have an appropriate type, or if the result
> > of the conversion cannot be represented in the object, the behavior is
> > undefined” to “If this object does not have an appropriate type, the
> > behavior is undefined. If the result of the conversion cannot be
> > represented in the object, the execution of the directive fails; this
> > condition is a matching failure.” And, even if the C committee doesn’t
> > want to make that change, open-source C libraries can and should do it
> > unilaterally, as a documented implementation extension. I think that’s
> > a better plan than declaring most uses of *scanf “deprecated.”
>
> Yeah, if you have plans to fix it, I'm fine removing the deprecation now. :)
To be clear, I personally don’t have plans to do any of the actual
programming or standard-changing work involved here. :-)
> > Yes, I think it would be fair to say that it is almost always a mistake
> > to use the scanf variants that read directly from a FILE. The issue
> > here is, at its root, that people new to C _expect_ a scanf call to read
> > an entire line of input, but it doesn’t. This is especially problematic
> > for interactive input ― they try to use plain scanf to read numeric
> > input, don’t realize that `scanf("%d", &arg)` doesn’t consume the \n in
> > the terminal’s line buffer _after_ the number, and get very confused
> > when a subsequent getchar() reads that \n instead of the ‘y’ or ‘n’ they
> > were expecting as a response to the _next_ prompt. But it’s _also_ a
> > problem for error recovery, because scanf will stop in the middle of the
> > line when a matching failure occurs, and if you naively assumed it would
> > throw away the rest of the line, you get an error cascade.
> >
> > The recommended practice to avoid this trap, is that you should use one
> > of the functions that _does_ read an entire line of input, i.e. fgets or
> > getline, and then parse the line as a string. It would make sense for
> > the [f]scanf manpage to say that.
>
> Please clarify; do you consider [v][f]scanf something that "we think
> we should never have added this in the first place, there’s no
> plausible way to fix it, but we have to keep it around for backward
> compatibility"?
_I_ wouldn’t have added them in the first place, but I care more than
the average about robustness in the face of unexpected input, even for
“throwaway” programs. I doubt the C committee would be prepared to
say the same thing. They _can_ be used legitimately, and they can
even be used in ways that meet my robustness standards if you go
to enough trouble. It’s just that (IMNSHO) there are better ways to
reach those standards.
In terms of text for the [v][f]scanf manpage, maybe something like
[NOTES? BUGS?]
When scanf() or fscanf() report a _matching failure_, all of the
text that was matched successfully has still been read from
_stdin_ or the _stream_ (respectively), and so have an
unpredictable number of characters associated with the conversion
that failed to match. The latter characters are lost. This may
make it difficult to recover from invalid input.
One way to make recovery easier is to separate reading from
parsing: use fgets() or getline() to read an entire line of text
into a string, then use sscanf() to parse the string. If a
_matching failure_ occurs, you can try sscanf() again with a
different _format_; the equivalent is not possible using fscanf().
_Successful_ calls to scanf() and fscanf() frequently consume
either more, or fewer, characters from the input than was
expected. For example, assuming the next six characters readable
from `stdin` are `"123\n a"‘, `scanf("%d", &val)` will consume the
digits but _not_ the newline, and `scanf("%d\n", &val)‘ will
consume the digits, the newline, _and_ the space. Either of these
is likely to cause trouble when mixing calls to scanf() with calls
to fgets() or fgetc(). As above, it helps to read entire lines of
text with fgets() or getline() and then parse them with sscanf().
> In short:
>
> (1) Numeric conversion specifiers are broken but can be fixed…
Yes.
> …and you plan to fix them.
No :)
> (1.1) I'll revert the deprecation warning now; since they are
> only broken because the _current_ standard and implementations are
> broken, but not by inherent design problems.
OK.
> (1.2) When [someone fixes] the implementation to not be UB
> anymore, it will also make sense to revert the patch that removed
> the ERANGE error, since you'll need to report it.
Yes.
> (2) For the string conversion specifiers, there are ways to use them
> safely.
Yes.
> and you plan to add a way to specify a size at runtime to the
> function,
No again :)
> so it will be even better in the future. No action required.
Concur.
> (3) [v][f]scanf seem to be really broken by design. Please confirm.
See above.
If you remind me where to find the git repo for the manpages, I _may_
have time to write a patch for all this sometime next week.
zw
next prev parent reply other threads:[~2022-12-29 6:42 UTC|newest]
Thread overview: 32+ messages / expand[flat|nested] mbox.gz Atom feed top
2022-12-08 12:34 [PATCH] scanf.3: Do not mention the ERANGE error Ian Abbott
2022-12-09 18:59 ` Alejandro Colomar
2022-12-09 19:28 ` Ian Abbott
2022-12-09 19:33 ` Alejandro Colomar
2022-12-09 21:41 ` Zack Weinberg
2022-12-11 15:58 ` Alejandro Colomar
2022-12-11 16:03 ` Alejandro Colomar
2022-12-12 2:11 ` Zack Weinberg
2022-12-12 10:21 ` Alejandro Colomar
2022-12-14 2:13 ` Zack Weinberg
2022-12-14 10:47 ` Alejandro Colomar
2022-12-14 11:03 ` Ian Abbott
2022-12-29 6:42 ` Zack Weinberg
2022-12-29 6:39 ` Zack Weinberg [this message]
2022-12-29 10:47 ` Alejandro Colomar
2022-12-29 16:35 ` Zack Weinberg
2022-12-29 16:39 ` Alejandro Colomar
2022-12-12 15:22 ` Ian Abbott
2022-12-14 2:18 ` Zack Weinberg
2022-12-14 10:22 ` Ian Abbott
2022-12-14 10:39 ` Alejandro Colomar
2022-12-14 10:52 ` Ian Abbott
2022-12-14 11:23 ` Alejandro Colomar
2022-12-14 14:10 ` Ian Abbott
2022-12-14 16:38 ` Joseph Myers
2022-12-12 10:07 ` Ian Abbott
2022-12-12 11:33 ` Alejandro Colomar
2023-01-20 4:09 ` Eric Biggers
2023-01-20 13:12 ` Alejandro Colomar
2023-01-20 17:55 ` G. Branden Robinson
2023-01-20 22:02 ` Alejandro Colomar
2023-01-20 19:41 ` Eric Biggers
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=ypikv8luogqp.wl-zack@owlfolio.org \
--to=zack@owlfolio.org \
--cc=abbotti@mev.co.uk \
--cc=alx.manpages@gmail.com \
--cc=libc-alpha@sourceware.org \
--cc=linux-man@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox