#include "scan.h" #include "haveuint128.h" size_t scan_ulongn(const char* src,size_t n,unsigned long int* dest) { register const char *tmp=src; register unsigned long int l=0; register unsigned char c; /* Since the conditions can be computed at compile time, the compiler * should only emit code for one of the implementations, depending on * which architecture the code is compiled for. */ #if defined(__GNUC__) && (__GNUC__ >= 5) /* implementation for recent gcc or clang */ int ok=0; for (; n-->0 && (c=(unsigned char)(*tmp-'0'))<10; ++tmp) { unsigned long v; if (__builtin_mul_overflow(l,10,&v) || __builtin_add_overflow(v,c,&v)) break; l=v; ok=1; } if (!ok) return 0; *dest=l; return (size_t)(tmp-src); #else #ifdef HAVE_UINT128 if (sizeof(unsigned long)==sizeof(unsigned long long) && sizeof(unsigned long)<sizeof(__uint128_t)) { /* implementation for 64-bit platforms with gcc */ for (; n-->0 && (c=(unsigned char)(*tmp-'0'))<10; ++tmp) { __uint128_t L=(__uint128_t)l*10+c; if ((L >> ((sizeof(L)-sizeof(l))*8))) break; l=(unsigned long)L; } *dest=l; return (size_t)(tmp-src); } else #endif if (sizeof(unsigned long)<sizeof(unsigned long long)) { /* implementation for 32-bit platforms */ for (; n-->0 && (c=(unsigned char)(*tmp-'0'))<10; ++tmp) { unsigned long long L=(unsigned long long)l*10+c; if ((unsigned long)L != L) break; l=(unsigned long)L; } *dest=l; return (size_t)(tmp-src); } else { /* implementation for 64-bit platforms without gcc */ while (n-->0 && (c=(unsigned char)(*tmp-'0'))<10) { unsigned long 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 */ n=l<<3; if ((n>>3)!=l) break; if (n+(l<<1)+c < n) break; l=n+(l<<1)+c; ++tmp; } if (tmp-src) *dest=l; return (size_t)(tmp-src); } #endif } #ifdef UNITTEST #include <assert.h> int main() { unsigned long l; assert(scan_ulongn("4294967295",10,&l) == 10 && l==4294967295ul); if (sizeof(unsigned long)==4) { assert(scan_ulongn("4294967296",10,&l) == 9 && l==429496729); } else { assert(scan_ulongn("18446744073709551615",20,&l) == 20 && l==18446744073709551615ull); assert(scan_ulongn("18446744073709551616",20,&l) == 19 && l==1844674407370955161ull); } assert(scan_ulongn("1234",3,&l)==3 && l==123); return 0; } #endif