git.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] strtoul_ui: reject negative values
@ 2015-09-17 14:37 Matthieu Moy
  2015-09-17 15:17 ` Marc Branchaud
  2015-09-17 16:18 ` Junio C Hamano
  0 siblings, 2 replies; 6+ messages in thread
From: Matthieu Moy @ 2015-09-17 14:37 UTC (permalink / raw)
  To: gitster; +Cc: git, max, Matthieu Moy

strtoul_ui uses strtoul to get a long unsigned, then checks that casting
to unsigned does not lose information and return the casted value.

On 64 bits architecture, checking that the cast does not change the value
catches most errors, but when sizeof(int) == sizeof(long) (e.g. i386),
the check does nothing. Unfortunately, strtoul silently accepts negative
values, and as a result strtoul_ui("-1", ...) raised no error.

This patch catches negative values before it's too late, i.e. before
calling strtoul. We still silently accept very large integers that wrap
to a valid "unsigned int".

Reported-by: Max Kirillov <max@max630.net>
Signed-off-by: Matthieu Moy <Matthieu.Moy@imag.fr>
---
So, here's a proper patch (I mean, a band-aid patch, but properly
send ;-) ).

It should be merged before Kartik's series (or inserted at the start
of the series) so that we get the fix before the test breakage.

 git-compat-util.h | 3 +++
 1 file changed, 3 insertions(+)

diff --git a/git-compat-util.h b/git-compat-util.h
index f649e81..1df82fa 100644
--- a/git-compat-util.h
+++ b/git-compat-util.h
@@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
 	char *p;
 
 	errno = 0;
+	/* negative values would be accepted by strtoul */
+	if (strchr(s, '-'))
+		return -1;
 	ul = strtoul(s, &p, base);
 	if (errno || *p || p == s || (unsigned int) ul != ul)
 		return -1;
-- 
2.5.0.402.g8854c44

^ permalink raw reply related	[flat|nested] 6+ messages in thread

* Re: [PATCH] strtoul_ui: reject negative values
  2015-09-17 14:37 [PATCH] strtoul_ui: reject negative values Matthieu Moy
@ 2015-09-17 15:17 ` Marc Branchaud
  2015-09-17 15:34   ` Matthieu Moy
  2015-09-17 16:18 ` Junio C Hamano
  1 sibling, 1 reply; 6+ messages in thread
From: Marc Branchaud @ 2015-09-17 15:17 UTC (permalink / raw)
  To: Matthieu Moy, gitster; +Cc: git, max

On 15-09-17 10:37 AM, Matthieu Moy wrote:
> strtoul_ui uses strtoul to get a long unsigned, then checks that casting
> to unsigned does not lose information and return the casted value.
> 
> On 64 bits architecture, checking that the cast does not change the value
> catches most errors, but when sizeof(int) == sizeof(long) (e.g. i386),
> the check does nothing. Unfortunately, strtoul silently accepts negative
> values, and as a result strtoul_ui("-1", ...) raised no error.
> 
> This patch catches negative values before it's too late, i.e. before
> calling strtoul. We still silently accept very large integers that wrap
> to a valid "unsigned int".
> 
> Reported-by: Max Kirillov <max@max630.net>
> Signed-off-by: Matthieu Moy <Matthieu.Moy@imag.fr>
> ---
> So, here's a proper patch (I mean, a band-aid patch, but properly
> send ;-) ).
> 
> It should be merged before Kartik's series (or inserted at the start
> of the series) so that we get the fix before the test breakage.
> 
>  git-compat-util.h | 3 +++
>  1 file changed, 3 insertions(+)
> 
> diff --git a/git-compat-util.h b/git-compat-util.h
> index f649e81..1df82fa 100644
> --- a/git-compat-util.h
> +++ b/git-compat-util.h
> @@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
>  	char *p;
>  
>  	errno = 0;
> +	/* negative values would be accepted by strtoul */
> +	if (strchr(s, '-'))
> +		return -1;

I think this is broken, in that it doesn't match strtoul's normal behaviour,
for strings like "1234-5678", no?

The test also doesn't work if the string has leading whitespace ("  -5").

>  	ul = strtoul(s, &p, base);
>  	if (errno || *p || p == s || (unsigned int) ul != ul)
>  		return -1;

Hmm, but we check *p here, so IIUC it's an error if the string has any
trailing non-digits.  Weird.

		M.

^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] strtoul_ui: reject negative values
  2015-09-17 15:17 ` Marc Branchaud
@ 2015-09-17 15:34   ` Matthieu Moy
  2015-09-17 16:12     ` Marc Branchaud
  0 siblings, 1 reply; 6+ messages in thread
From: Matthieu Moy @ 2015-09-17 15:34 UTC (permalink / raw)
  To: Marc Branchaud; +Cc: gitster, git, max

Marc Branchaud <marcnarc@xiplink.com> writes:

>> --- a/git-compat-util.h
>> +++ b/git-compat-util.h
>> @@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
>>  	char *p;
>>  
>>  	errno = 0;
>> +	/* negative values would be accepted by strtoul */
>> +	if (strchr(s, '-'))
>> +		return -1;
>
> I think this is broken, in that it doesn't match strtoul's normal behaviour,
> for strings like "1234-5678", no?

The goal here is just to read a positive integer value. Rejecting
"1234-5678" is indeed a good thing. We already rejected it before my
patch by checking for p (AKA endptr for strtoul), as you noted below.

> The test also doesn't work if the string has leading whitespace ("
> -5").

Why? It rejects any string that contain the character '-', regardless of
trailing spaces.

>>  	ul = strtoul(s, &p, base);
>>  	if (errno || *p || p == s || (unsigned int) ul != ul)
>>  		return -1;
>
> Hmm, but we check *p here, so IIUC it's an error if the string has any
> trailing non-digits.  Weird.

strtoul_ui is more defensive than strtoul, by design.

-- 
Matthieu Moy
http://www-verimag.imag.fr/~moy/

^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] strtoul_ui: reject negative values
  2015-09-17 15:34   ` Matthieu Moy
@ 2015-09-17 16:12     ` Marc Branchaud
  0 siblings, 0 replies; 6+ messages in thread
From: Marc Branchaud @ 2015-09-17 16:12 UTC (permalink / raw)
  To: Matthieu Moy; +Cc: gitster, git, max

On 15-09-17 11:34 AM, Matthieu Moy wrote:
> Marc Branchaud <marcnarc@xiplink.com> writes:
> 
>>> --- a/git-compat-util.h
>>> +++ b/git-compat-util.h
>>> @@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
>>>  	char *p;
>>>  
>>>  	errno = 0;
>>> +	/* negative values would be accepted by strtoul */
>>> +	if (strchr(s, '-'))
>>> +		return -1;
>>
>> I think this is broken, in that it doesn't match strtoul's normal behaviour,
>> for strings like "1234-5678", no?
> 
> The goal here is just to read a positive integer value. Rejecting
> "1234-5678" is indeed a good thing. We already rejected it before my
> patch by checking for p (AKA endptr for strtoul), as you noted below.
> 
>> The test also doesn't work if the string has leading whitespace ("
>> -5").
> 
> Why? It rejects any string that contain the character '-', regardless of
> trailing spaces.

Right, sorry.

>>>  	ul = strtoul(s, &p, base);
>>>  	if (errno || *p || p == s || (unsigned int) ul != ul)
>>>  		return -1;
>>
>> Hmm, but we check *p here, so IIUC it's an error if the string has any
>> trailing non-digits.  Weird.
> 
> strtoul_ui is more defensive than strtoul, by design.

Fair enough, just not what I expected from a function with that name.

		M.

^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] strtoul_ui: reject negative values
  2015-09-17 14:37 [PATCH] strtoul_ui: reject negative values Matthieu Moy
  2015-09-17 15:17 ` Marc Branchaud
@ 2015-09-17 16:18 ` Junio C Hamano
  2015-09-17 16:28   ` [PATCH v2] " Matthieu Moy
  1 sibling, 1 reply; 6+ messages in thread
From: Junio C Hamano @ 2015-09-17 16:18 UTC (permalink / raw)
  To: Matthieu Moy; +Cc: git, max

Matthieu Moy <Matthieu.Moy@imag.fr> writes:

> This patch catches negative values before it's too late, i.e. before
> calling strtoul. We still silently accept very large integers that wrap
> to a valid "unsigned int".

Is the last statement correct?  A very large long uint that wrap to
uint would not fit in long uint and you would get ERANGE, no?

> So, here's a proper patch (I mean, a band-aid patch, but properly
> send ;-) ).

Yup.

> It should be merged before Kartik's series (or inserted at the start
> of the series) so that we get the fix before the test breakage.

Which one of his series?

>
>  git-compat-util.h | 3 +++
>  1 file changed, 3 insertions(+)
>
> diff --git a/git-compat-util.h b/git-compat-util.h
> index f649e81..1df82fa 100644
> --- a/git-compat-util.h
> +++ b/git-compat-util.h
> @@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
>  	char *p;
>  
>  	errno = 0;
> +	/* negative values would be accepted by strtoul */
> +	if (strchr(s, '-'))
> +		return -1;
>  	ul = strtoul(s, &p, base);
>  	if (errno || *p || p == s || (unsigned int) ul != ul)
>  		return -1;

^ permalink raw reply	[flat|nested] 6+ messages in thread

* [PATCH v2] strtoul_ui: reject negative values
  2015-09-17 16:18 ` Junio C Hamano
@ 2015-09-17 16:28   ` Matthieu Moy
  0 siblings, 0 replies; 6+ messages in thread
From: Matthieu Moy @ 2015-09-17 16:28 UTC (permalink / raw)
  To: gitster; +Cc: git, max, Matthieu Moy

strtoul_ui uses strtoul to get a long unsigned, then checks that casting
to unsigned does not lose information and return the casted value.

On 64 bits architecture, checking that the cast does not change the value
catches most errors, but when sizeof(int) == sizeof(long) (e.g. i386),
the check does nothing. Unfortunately, strtoul silently accepts negative
values, and as a result strtoul_ui("-1", ...) raised no error.

This patch catches negative values before it's too late, i.e. before
calling strtoul.

Reported-by: Max Kirillov <max@max630.net>
Signed-off-by: Matthieu Moy <Matthieu.Moy@imag.fr>
---
Junio C Hamano <gitster@pobox.com> writes:

> Matthieu Moy <Matthieu.Moy@imag.fr> writes:
>
>> This patch catches negative values before it's too late, i.e. before
>> calling strtoul. We still silently accept very large integers that wrap
>> to a valid "unsigned int".
>
> Is the last statement correct?  A very large long uint that wrap to
> uint would not fit in long uint and you would get ERANGE, no?

Indeed. strtoul happily accepts negative values, but not overly large
ones.

I removed the sentence from the message. Actually, I think we are now
accepting exactly the right interval of values.

>> It should be merged before Kartik's series (or inserted at the start
>> of the series) so that we get the fix before the test breakage.
>
> Which one of his series?

kn/for-each-tag, which uses strtoul_ui for align:<num> and
content:lines=<num>.

 git-compat-util.h | 3 +++
 1 file changed, 3 insertions(+)

diff --git a/git-compat-util.h b/git-compat-util.h
index f649e81..1df82fa 100644
--- a/git-compat-util.h
+++ b/git-compat-util.h
@@ -814,6 +814,9 @@ static inline int strtoul_ui(char const *s, int base, unsigned int *result)
 	char *p;
 
 	errno = 0;
+	/* negative values would be accepted by strtoul */
+	if (strchr(s, '-'))
+		return -1;
 	ul = strtoul(s, &p, base);
 	if (errno || *p || p == s || (unsigned int) ul != ul)
 		return -1;
-- 
2.5.0.402.g8854c44

^ permalink raw reply related	[flat|nested] 6+ messages in thread

end of thread, other threads:[~2015-09-17 16:28 UTC | newest]

Thread overview: 6+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2015-09-17 14:37 [PATCH] strtoul_ui: reject negative values Matthieu Moy
2015-09-17 15:17 ` Marc Branchaud
2015-09-17 15:34   ` Matthieu Moy
2015-09-17 16:12     ` Marc Branchaud
2015-09-17 16:18 ` Junio C Hamano
2015-09-17 16:28   ` [PATCH v2] " Matthieu Moy

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).