2012-05-15 09:27:12 -04:00
|
|
|
#ifndef strings_h
|
|
|
|
#define strings_h
|
|
|
|
|
|
|
|
/* MSVC doesn't define ffs/ffsl. This dummy strings.h header is provided
|
|
|
|
* for both */
|
2015-10-06 10:18:30 -04:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
# include <intrin.h>
|
|
|
|
# pragma intrinsic(_BitScanForward)
|
2018-05-24 11:17:37 -04:00
|
|
|
static __forceinline int ffsl(long x) {
|
2012-05-15 09:27:12 -04:00
|
|
|
unsigned long i;
|
|
|
|
|
2018-05-24 11:17:37 -04:00
|
|
|
if (_BitScanForward(&i, x)) {
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
return 0;
|
2012-05-15 09:27:12 -04:00
|
|
|
}
|
|
|
|
|
2018-05-24 11:17:37 -04:00
|
|
|
static __forceinline int ffs(int x) {
|
|
|
|
return ffsl(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
# ifdef _M_X64
|
|
|
|
# pragma intrinsic(_BitScanForward64)
|
|
|
|
# endif
|
2012-05-15 09:27:12 -04:00
|
|
|
|
2018-05-24 11:17:37 -04:00
|
|
|
static __forceinline int ffsll(unsigned __int64 x) {
|
|
|
|
unsigned long i;
|
|
|
|
#ifdef _M_X64
|
|
|
|
if (_BitScanForward64(&i, x)) {
|
|
|
|
return i + 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
// Fallback for 32-bit build where 64-bit version not available
|
|
|
|
// assuming little endian
|
|
|
|
union {
|
|
|
|
unsigned __int64 ll;
|
|
|
|
unsigned long l[2];
|
|
|
|
} s;
|
|
|
|
|
|
|
|
s.ll = x;
|
|
|
|
|
|
|
|
if (_BitScanForward(&i, s.l[0])) {
|
|
|
|
return i + 1;
|
|
|
|
} else if(_BitScanForward(&i, s.l[1])) {
|
|
|
|
return i + 33;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#endif
|
2012-05-15 09:27:12 -04:00
|
|
|
}
|
|
|
|
|
2015-10-06 10:18:30 -04:00
|
|
|
#else
|
2018-05-24 11:17:37 -04:00
|
|
|
# define ffsll(x) __builtin_ffsll(x)
|
2015-10-06 10:18:30 -04:00
|
|
|
# define ffsl(x) __builtin_ffsl(x)
|
|
|
|
# define ffs(x) __builtin_ffs(x)
|
2012-05-15 09:27:12 -04:00
|
|
|
#endif
|
2015-10-06 10:18:30 -04:00
|
|
|
|
|
|
|
#endif /* strings_h */
|