Skip site navigation (1)Skip section navigation (2)
Date:      Fri, 2 Dec 2016 10:30:31 +1100 (EST)
From:      Bruce Evans <brde@optusnet.com.au>
To:        Eric van Gyzen <vangyzen@freebsd.org>
Cc:        src-committers@freebsd.org, svn-src-all@freebsd.org,  svn-src-head@freebsd.org
Subject:   Re: svn commit: r309364 - head/usr.bin/locale
Message-ID:  <20161202094516.Q1025@besplex.bde.org>
In-Reply-To: <201612011654.uB1Gs2s4023355@repo.freebsd.org>
References:  <201612011654.uB1Gs2s4023355@repo.freebsd.org>

next in thread | previous in thread | raw e-mail | index | archive | help
On Thu, 1 Dec 2016, Eric van Gyzen wrote:

> Log:
>  locale: fix buffer management
>
>  Also, handle signed and unsigned chars, and more gracefully handle
>  invalid input.
>
>  Submitted by:	bde in response to r309331
>  MFC after:	1 week
>  Sponsored by:	Dell EMC

Thanks.

> Modified: head/usr.bin/locale/locale.c
> ==============================================================================
> --- head/usr.bin/locale/locale.c	Thu Dec  1 15:46:26 2016	(r309363)
> +++ head/usr.bin/locale/locale.c	Thu Dec  1 16:54:02 2016	(r309364)
> @@ -495,29 +495,29 @@ format_grouping(const char *binary)
> {
> 	static char rval[64];
> 	const char *cp;
> -	size_t len;
> +	size_t roff;
> +	int len;
> ...
> +		if (*cp < 0)
> +			break;		/* garbage input */
> +		len = snprintf(&rval[roff], sizeof(rval) - roff, "%u;", *cp);
> +		if (len < 0 || (unsigned)len >= sizeof(rval) - roff)
> +			break;		/* insufficient space for output */

I don't like casting len to fix compiler warnings, and intentionally
left out this cast.  len < 0 ensures that there is no sign extension bug
or dependency on sign extension bugs.  snprintf() returns < 0 if an
encoding error occurs, and the check is mainly to detect this unlikely
error (the comment is too short to describe this), but the check also
allows compilers to easily see that there is no sign extension bug and
only low-quality ones warn.

I used the bogus type size_t instead of int for roff since I didn't
want to fight possible compiler warnings about sign extension in the
expression sizeof(rval) - roff.  Here it is not so easy to see that
roff <= sizeof(rval).

You didn't do anything to avoid bad -Wtautological-compare compiler
warnings for (*cp < 0).  I think such warnings occur at high WARNS
on arches with unsigned char.  These warnings are bad since the
comparison is only tautological on some arches so it is not really
tautological.  Avoiding it requires code like '#if CHAR_MIN < 0', but
that is equally tautological and only escapes the warning because cpp
expressions are weaker than C expressions.

clang -Wall -Wtautological-compare doesn't actually warn for either
(*cp < 0) (where cp is unsigned char *) or (1 < 0).  gcc-4.2.1 -Wall
warns for the former but not for the latter.  This seems sort of
backwards, but ISTR discussions in gcc lists ~20 years ago about it
being intentional.  Warning for (1 < 0) mainly breaks conditional
compilation where the condtions are intentially written using C
const expressions instead of cpp expressions since cpp is harder to
work with.  (*cp < 0) is more likely to be a bug.

Bruce



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20161202094516.Q1025>