2015-09-29 14:42:40 +00:00
|
|
|
#if defined(__GNUC__) && (__GNUC__ >= 5)
|
|
|
|
|
|
|
|
#include "uint32.h"
|
|
|
|
|
|
|
|
int umult32(uint32 a,uint32 b,uint32* c) { return !__builtin_mul_overflow(a,b,c); }
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2003-08-22 15:03:10 +00:00
|
|
|
#include "safemult.h"
|
|
|
|
|
|
|
|
int umult32(uint32 a,uint32 b,uint32* c) {
|
|
|
|
unsigned long long x=(unsigned long long)a*b;
|
2003-09-05 21:25:51 +00:00
|
|
|
if (x>0xffffffff) return 0;
|
2003-08-22 15:03:10 +00:00
|
|
|
*c=x&0xffffffff;
|
2003-09-05 21:25:51 +00:00
|
|
|
return 1;
|
2003-08-22 15:03:10 +00:00
|
|
|
}
|
2015-09-29 14:42:40 +00:00
|
|
|
|
|
|
|
#endif
|
2017-05-13 22:52:21 +00:00
|
|
|
|
|
|
|
#ifdef UNITTEST
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
int main() {
|
|
|
|
uint32 a;
|
|
|
|
assert(umult32(4,0x80000000,&a)==0);
|
|
|
|
assert(umult32(16,0x45000000,&a)==0); // make sure we don't fall for "if a*b<a && a*b<b"
|
|
|
|
assert(umult32(5,10,&a)==1 && a==50);
|
|
|
|
assert(umult32(6,0x10000000,&a)==1 && a==0x60000000);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|