libowfat/scan/scan_short.c

39 lines
1.0 KiB
C
Raw Normal View History

2001-02-02 17:54:47 +00:00
#include "scan.h"
2014-03-14 02:15:38 +00:00
static const unsigned short maxshort = ((unsigned short)-1)>>1;
2006-11-07 17:56:05 +00:00
size_t scan_short(const char* src,short* dest) {
register const char *tmp;
register short l;
register unsigned char c;
int neg;
int ok;
tmp=src; l=0; ok=neg=0;
switch (*tmp) {
case '-': neg=1;
case '+': ++tmp;
}
while ((c=(unsigned char)(*tmp-'0'))<10) {
2014-03-14 02:15:38 +00:00
unsigned short int n;
/* we want to do: l=l*10+c
* but we need to check for integer overflow.
* to check whether l*10 overflows, we could do
* if ((l*10)/10 != l)
* however, multiplication and division are expensive.
* so instead of *10 we do (l<<3) (i.e. *8) + (l<<1) (i.e. *2)
* and check for overflow on all the intermediate steps */
2014-03-14 02:15:38 +00:00
n=(unsigned short)(l<<3); if ((n>>3)!=(unsigned short)l) break;
if (n+(l<<1) < n) break;
2014-03-14 02:15:38 +00:00
n=(unsigned short)(n+(l<<1));
if (n+c < n) break;
2014-03-14 02:15:38 +00:00
n=(unsigned short)(n+c);
if (n > maxshort+neg) break;
2014-03-14 02:15:38 +00:00
l=(short)n;
++tmp;
ok=1;
}
if (!ok) return 0;
2014-03-14 02:15:38 +00:00
*dest=(short)(neg?-l:l);
return (size_t)(tmp-src);
2001-02-02 17:54:47 +00:00
}