1 /** Various extensions to core.bitop and std.bitmanip.
2  *
3  * Copyright: Per Nordlöw 2018-.
4  * License: $(WEB boost.org/LICENSE_1_0.txt, Boost License 1.0).
5  * Authors: $(WEB Per Nordlöw)
6  *
7  * TODO: Add range checking of bit indexes.
8  *
9  * TODO: Make use of TZCNT and LZCNT either as inline assembly or as builtins: https://github.com/dlang/dmd/pull/6364
10  */
11 module nxt.bitop_ex;
12 
13 import std.meta : allSatisfy;
14 import std.traits : isIntegral;
15 
16 @safe pure nothrow @nogc:
17 
18 /** Get an unsigned type of size as `T` if possible. */
19 template UnsignedOfSameSizeAs(T)
20 {
21     enum nBits = 8*T.sizeof;
22     static      if (nBits ==  8) alias UnsignedOfSameSizeAs = ubyte;
23     else static if (nBits == 16) alias UnsignedOfSameSizeAs = ushort;
24     else static if (nBits == 32) alias UnsignedOfSameSizeAs = uint;
25     else static if (nBits == 64) alias UnsignedOfSameSizeAs = ulong;
26     else static if (nBits == 128) alias UnsignedOfSameSizeAs = ucent;
27     else
28     {
29         import std.conv: to;
30         static assert(0, "No Unsigned type of size " ~ to!string(nBits) ~ " found");
31     }
32 }
33 
34 /** Returns: `T` with only `bix`:th bit set. */
35 T makeBit(T, I...)(I bixs) @safe
36 if (isIntegral!T &&
37     allSatisfy!(isIntegral, I) &&
38     I.length >= 1)
39 in
40 {
41     foreach (n, const bix; bixs)
42     {
43         assert(0 <= bix && bix < 8*T.sizeof,
44                "Bit index " ~ n.stringof ~ " is out of range");
45     }
46 }
47 do
48 {
49     T x = 0;
50     foreach (const bix; bixs)
51     {
52         x |= cast(T)((cast(T)1) << bix);
53     }
54     return x;
55 }
56 alias btm = makeBit;
57 
58 /** Returns: `true` iff all `bix`:th bits of `a` are set. */
59 bool testBit(T, I...)(in T a, I bixs) @safe
60 if (isIntegral!T &&
61     allSatisfy!(isIntegral, I) &&
62     I.length >= 1)
63 {
64     pragma(inline, true);
65     return a & makeBit!T(bixs) ? true : false;
66 }
67 
68 /** Returns: `true` iff all `bix`:th bits of `a` are set. */
69 bool testBit(T, I...)(in T a, I bixs) @trusted
70 if ((!(isIntegral!T)) &&
71     allSatisfy!(isIntegral, I))
72 {
73     pragma(inline, true);
74     return (*(cast(UnsignedOfSameSizeAs!T*)&a)).testBit(bixs); // reuse integer variant
75 }
76 
77 /** Returns: `true` iff all `bix`:th bits of `*a` are set. */
78 bool testBit(T, I...)(in T* a, I bixs) @safe
79 if ((!(isIntegral!T)) &&
80     !is(T == size_t) &&     // avoid stealing `core.bitop.bt`
81     allSatisfy!(isIntegral, I) &&
82     I.length >= 1)
83 {
84     pragma(inline, true);
85     return testBit(*a, bixs);
86 }
87 alias bt = testBit;
88 
89 ///
90 @safe pure nothrow @nogc unittest
91 {
92     static void test(T)()
93     {
94         const mn = T.min, mx = T.max;
95         enum nBits = 8*T.sizeof;
96         foreach (const ix; 0 .. nBits-1)
97         {
98             assert(!mn.bt(ix));
99         }
100         assert(mn.bt(nBits - 1));
101         foreach (const ix; 0 .. T.sizeof)
102         {
103             assert(mx.bt(ix));
104         }
105     }
106     test!byte;
107     test!short;
108     test!int;
109     test!long;
110 }
111 
112 /** Test and sets the `bix`:th bit of `a` to one.
113  *
114  * Returns: A non-zero value if the bit was set, and a zero if it was clear.
115 */
116 void setBit(T, I...)(ref T a, I bixs) @safe
117 if (isIntegral!T &&
118     allSatisfy!(isIntegral, I) &&
119     I.length >= 1)
120 {
121     pragma(inline, true);
122     a |= makeBit!T(bixs);
123 }
124 
125 /** Sets the `bix`:th bit of `*a` to one.
126  */
127 void setBit(T, I...)(T* a, I bixs) @safe
128 if (isIntegral!T &&
129     !is(T == size_t) && // avoid stealing core.bitop.bt
130     allSatisfy!(isIntegral, I) &&
131     I.length >= 1)
132 {
133     pragma(inline, true);
134     *a |= makeBit!T(bixs);
135 }
136 
137 /** Sets the `bix`:th bit of `*a` to one.
138  */
139 void setBit(T, I...)(ref T a, I bixs) @trusted
140 if ((!(isIntegral!T)) &&
141     allSatisfy!(isIntegral, I) &&
142     I.length >= 1)
143 {
144     pragma(inline, true);
145     alias U = UnsignedOfSameSizeAs!T;
146     (*(cast(U*)&a)) |= makeBit!U(bixs); // reuse integer variant
147 }
148 alias bts = setBit;
149 
150 /* alias btc = complementBit; */
151 /* alias btr = resetBit; */
152 
153 /** Set lowest bit of `a` to one. */
154 void setLowestBit(T)(ref T a) @safe
155 if (isIntegral!T)
156 {
157     pragma(inline, true);
158     setBit(a, 0);
159 }
160 alias setBottomBit = setLowestBit;
161 alias setLsbit = setLowestBit;
162 
163 /** Set highest bit of `a` to one. */
164 void setHighestBit(T)(ref T a) @safe
165 if (isIntegral!T)
166 {
167     pragma(inline, true);
168     setBit(a, 8*T.sizeof - 1);
169 }
170 alias setTopBit = setHighestBit;
171 alias setMsbit = setHighestBit;
172 
173 /** Get lowest bit of `a`. */
174 bool getLowestBit(T)(in T a) @safe
175 if (isIntegral!T)
176 {
177     pragma(inline, true);
178     return (a & 1) != 0;
179 }
180 alias getBottomBit = getLowestBit;
181 alias getLsbit = getLowestBit;
182 
183 /** Get highest bit of `a`. */
184 bool getHighestBit(T)(in T a) @safe
185 if (isIntegral!T)
186 {
187     pragma(inline, true);
188     // TODO: use core.bitop.bt when T is a size_t
189     return (a & (cast(T)1 << 8*T.sizeof - 1)) != 0;
190 }
191 alias getTopBit = getHighestBit;
192 alias getMsbit = getHighestBit;
193 
194 ///
195 @safe pure nothrow @nogc unittest
196 {
197     const ubyte x = 1;
198     assert(!x.getTopBit);
199     assert(x.getLowestBit);
200 }
201 
202 ///
203 @safe pure nothrow @nogc unittest
204 {
205     const ubyte x = 128;
206     assert(x.getTopBit);
207     assert(!x.getLowestBit);
208 }
209 
210 /** Reset bits `I` of `a` (to zero). */
211 void resetBit(T, I...)(ref T a, I bixs) @safe
212 if (isIntegral!T &&
213     allSatisfy!(isIntegral, I))
214 {
215     pragma(inline, true);
216     a &= ~makeBit!T(bixs);
217 }
218 
219 /** Reset bits `I` of `*a` (to zero). */
220 void resetBit(T, I...)(T* a, I bixs) @safe
221 if (isIntegral!T &&
222     !is(T == size_t) && // avoid stealing core.bitop.bt
223     allSatisfy!(isIntegral, I))
224 {
225     pragma(inline, true);
226     *a &= ~makeBit!T(bixs);
227 }
228 
229 /** Reset bits `I` of `a` (to zero). */
230 void resetBit(T, I...)(ref T a, I bixs)
231 if ((!(isIntegral!T)) &&
232     allSatisfy!(isIntegral, I))
233 {
234     pragma(inline, true);
235     alias U = UnsignedOfSameSizeAs!T;
236     (*(cast(U*)&a)) &= ~makeBit!U(bixs); // reuse integer variant
237 }
238 
239 /** Reset lowest bit of `a` (to zero). */
240 void resetLowestBit(T)(ref T a) @safe
241 if (isIntegral!T)
242 {
243     pragma(inline, true);
244     resetBit(a, 0);
245 }
246 alias resetBottomBit = resetLowestBit;
247 
248 /** Reset highest bit of `a` (to zero). */
249 void resetHighestBit(T)(ref T a) @safe
250 if (isIntegral!T)
251 {
252     pragma(inline, true);
253     resetBit(a, 8*T.sizeof - 1);
254 }
255 alias resetTopBit = resetHighestBit;
256 
257 alias btr = resetBit;
258 
259 ///
260 pure nothrow @nogc unittest
261 {
262     alias T = int;
263     enum nBits = 8*T.sizeof;
264     T a = 0;
265 
266     a.bts(0); assert(a == 1);
267     a.bts(1); assert(a == 3);
268     a.bts(2); assert(a == 7);
269 
270     a.btr(0); assert(a == 6);
271     a.btr(1); assert(a == 4);
272     a.btr(2); assert(a == 0);
273 
274     a.bts(0, 1, 2); assert(a == 7);
275     a.btr(0, 1, 2); assert(a == 0);
276 
277     a.bts(8*T.sizeof - 1); assert(a != 0);
278     a.btr(8*T.sizeof - 1); assert(a == 0);
279 
280     T b = 0;
281     b.bts(nBits - 1);
282     assert(b == T.min);
283 }
284 
285 ///
286 @safe pure nothrow @nogc unittest
287 {
288     static void test(T)()
289     {
290         enum nBits = 8*T.sizeof;
291         T x = 0;
292         x.bts(0);
293     }
294 
295     test!float;
296     test!double;
297 }
298 
299 ///
300 @safe pure nothrow @nogc unittest
301 {
302     assert(makeBit!int(2) == 4);
303     assert(makeBit!int(2, 3) == 12);
304     assert(makeBit!uint(0, 31) == 2^^31 + 1);
305 
306     import std.meta : AliasSeq;
307     foreach (T; AliasSeq!(ubyte, ushort, uint, ulong))
308     {
309         foreach (const n; 0 .. 8*T.sizeof)
310         {
311             const x = makeBit!T(n);
312             assert(x == 2UL^^n);
313 
314             T y = x;
315             y.resetBit(n);
316             assert(y == 0);
317 
318             y.setBit(n);
319             assert(y == x);
320         }
321     }
322 }