146 {
147 if(!value)
148 return 0;
149#if defined(_MSC_VER) && !defined(__cplusplus_cli)
150 {
151 unsigned long bitpos;
152#if defined(_M_IA64) || defined(_M_X64) || defined(WIN64) || defined(_WIN64)
153 assert(8 == sizeof(size_t));
154 _BitScanReverse64(&bitpos, value);
155#else
156 assert(4 == sizeof(size_t));
157 _BitScanReverse(&bitpos, (unsigned) value);
158#endif
159 return (unsigned) bitpos;
160 }
161#elif defined(__GNUC__)
162 return sizeof(value) * 8 - 1 - (unsigned) __builtin_clzl(value);
163#else
164
165
166#if !defined(__cplusplus_cli)
167 union
168 {
169 unsigned asInt[2];
170 double asDouble;
171 };
172 int n;
173
174 asDouble = (double) value + 0.5;
175 n = (asInt[0 ] >> 20) - 1023;
176#ifdef _MSC_VER
177#pragma message(__FILE__ ": WARNING: Make sure you change the line above me if your CPU is big endian!")
178#else
179#warning Make sure you change the line above me if your CPU is big endian!
180#endif
181 return (unsigned) n;
182#else
183#if CHAR_BIT != 8
184#error CHAR_BIT is not eight, and therefore this generic bitscan routine will need adjusting!
185#endif
186
187 size_t x = value;
188 x = x | (x >> 1);
189 x = x | (x >> 2);
190 x = x | (x >> 4);
191 x = x | (x >> 8);
192 if(16 < sizeof(x) * CHAR_BIT)
193 x = x | (x >> 16);
194 if(32 < sizeof(x) * CHAR_BIT)
195 x = x | (x >> 32);
196 x = ~x;
197 x = x - ((x >> 1) & (SIZE_MAX / 3));
198 x = (x & (SIZE_MAX / 15 * 3)) + ((x >> 2) & (SIZE_MAX / 15 * 3));
199 x = ((x + (x >> 4)) & (SIZE_MAX / UCHAR_MAX * 15)) * (SIZE_MAX / UCHAR_MAX);
200 x = (CHAR_BIT * sizeof(x) - 1) - (x >> (CHAR_BIT * (sizeof(x) - 1)));
201 return (unsigned) x;
202#endif
203#endif
204 }