You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

50 lines
934 B
C

#if defined(__GNUC__) && (__GNUC__ >= 5)
#include "uint64.h"
int umult64(uint64 a,uint64 b,uint64* c) { return !__builtin_mul_overflow(a,b,c); }
#else
11 years ago
#include "haveuint128.h"
#include "safemult.h"
11 years ago
#if defined(HAVE_UINT128)
int umult64(uint64 a,uint64 b,uint64* c) {
__uint128_t x=((__uint128_t)a)*b;
if ((*c=(uint64)x) != x) return 0;
return 1;
}
#else
/* return 1 for overflow, 0 for ok */
int umult64(uint64 a,uint64 b,uint64* c) {
uint32 ahi=a>>32;
uint32 alo=(a&0xffffffff);
uint32 bhi=b>>32;
uint32 blo=(b&0xffffffff);
// a=ahi*x+alo, b=bhi*x+blo
// a*b = (ahi*x+alo) * (bhi*x+blo)
// = ahi*x*bhi*x + ahi*x*blo + alo*bhi*x + alo*blo
// -> overflow if ahi*bhi != zero */
if (ahi && bhi) return 0;
a=(uint64)(ahi)*blo+(uint64)(alo)*bhi;
if (a>0xffffffff) return 0;
{
uint64 x=(uint64)(alo)*blo;
if (x+(a<<32) < x) return 0;
*c=x+(a<<32);
}
return 1;
}
#endif
#endif