admin
2023-03-07 8b06b1cbf112d55307ea8a6efe711db4e7506d89
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
#ifndef strings_h
#define strings_h
 
/* MSVC doesn't define ffs/ffsl. This dummy strings.h header is provided
 * for both */
#ifdef _MSC_VER
#  include <intrin.h>
#  pragma intrinsic(_BitScanForward)
static __forceinline int ffsl(long x) {
    unsigned long i;
 
    if (_BitScanForward(&i, x)) {
        return i + 1;
    }
    return 0;
}
 
static __forceinline int ffs(int x) {
    return ffsl(x);
}
 
#  ifdef  _M_X64
#    pragma intrinsic(_BitScanForward64)
#  endif
 
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
}
 
#else
#  define ffsll(x) __builtin_ffsll(x)
#  define ffsl(x) __builtin_ffsl(x)
#  define ffs(x) __builtin_ffs(x)
#endif
 
#endif /* strings_h */