1 /** Lightweight versions of polymorphism packed inside one single
2     word/pointer.
3 
4     Most significant bits are used to store type information.
5 
6     These higher bits are normally unused on 64-bit systems (tested on
7     Linux). 16 higher bits are, on Linux, either 1 (kernel-space) or 0 (user-space).
8 
9     See_Also: http://forum.dlang.org/post/sybuoliqhhefcovxjfjv@forum.dlang.org
10 
11     TODO: Ask forum.dlang.org: Is it safe to assume that `typeBits` most significant bits of a
12     pointer are zero? If not put them in least significant part.
13 
14     TODO: What todo with the fact that the GC will fail to scan WordVariant?
15     Can the GC be tweaked to mask out the type bits before scanning?
16 
17     TODO: Enable support for `is null` instead of `isNull`?
18 
19     TODO: Use `enforce()` instead of `assert()` in WordVariant:initialize()
20 
21     TODO: Move to Phobos std.variant
22 */
23 module nxt.variant_ex;
24 
25 /** A variant of `Types` packed into a word (`size_t`).
26  *
27  * Suitable for use in tree-data containers, such as radix trees (tries), where
28  * hybrid value (sparsely packed sub-tree) and pointer (to dense sub-tree)
29  * packing of sub-nodes is needed.
30  */
31 struct WordVariant(Types...)
32 {
33     import nxt.traits_ex : allSame, sizesOf;
34     static assert(allSame!(sizesOf!(Types)), "Types must have equal size");
35     static assert(this.sizeof == (void*).sizeof, "Size must be same as word (pointer)");
36 
37     alias S = size_t; // TODO: templatize?
38 
39     import nxt.typecons_ex : makeEnumFromSymbolNames;
40     alias Ix = makeEnumFromSymbolNames!(`ix_`, ``, true, false, Types);
41     static assert(Ix.undefined == 0);
42 
43     import nxt.bit_traits : bitsNeeded;
44     enum typeBits = bitsNeeded!(1 + Types.length); // number of bits needed to represent variant type, plus one for undefined state
45     enum typeShift = 8*S.sizeof - typeBits;
46     enum typeMask = cast(S)(2^^typeBits - 1) << typeShift;
47 
48     import std.meta : staticIndexOf;
49     enum indexOf(T) = staticIndexOf!(T, Types); // TODO: cast to ubyte if Types.length is <= 256
50 
51     /// Is `true` iff a `T` can be stored.
52     enum canStore(T) = indexOf!T >= 0;
53 
54     pure:
55 
56     @property string toString() const @trusted // TODO: pure nothrow
57     {
58         import std.traits : isPointer;
59         import std.conv : to;
60         final switch (typeIndex) // typeIndex starts at 0 (undefined)
61         {
62         case 0: return "null";
63             foreach (const i, T; Types)
64             {
65             case i + 1:
66                 // TODO: improve this to look more like D-code:
67                 static if (isPointer!T)
68                     return T.stringof ~ `@` ~ as!T.to!string; // TODO: ~ `:` ~ (*as!T).to!string;
69                 else
70                     return T.stringof ~ `@` ~ as!T.to!string;
71             }
72         }
73     }
74 
75     nothrow:
76 
77     extern (D) auto toHash() const
78     {
79         import core.internal.hash : hashOf;
80         return _raw.hashOf;
81     }
82 
83     @nogc:
84 
85     /// Construction from `value`.
86     this(T)(T value) if (canStore!T) { initialize(value); }
87     /// ditto
88     this(typeof(null) value) { _raw = S.init; }
89     /// Construction from sub-variant `value`.
90     this(SubTypes...)(WordVariant!(SubTypes) value) { initializeFromSub(value); }
91 
92     /// Assignment from `that`.
93     auto ref opAssign(typeof(this) value) { _raw = value._raw; return this; }
94     /// ditto
95     auto ref opAssign(T)(T that) if (canStore!T) { initialize(that); return this; }
96     /// ditto
97     auto ref opAssign(typeof(null) that) { _raw = S.init; return this; }
98     /// Assignment from sub-variant `value`.
99     auto ref opAssign(SubTypes...)(WordVariant!(SubTypes) value) { initializeFromSub(value); return this; }
100 
101 pragma(inline):
102 
103     /** Reference to peek of type `T`. */
104     static private struct Ref(T)
105     {
106         S _raw;
107         const bool defined;
108         pragma(inline):
109         bool opCast(T : bool)() const { return defined; }
110         inout(T) opUnary(string op : `*`)() @trusted inout
111         {
112             auto data = (_raw & ~typeMask); // mask away type bits
113             return *(cast(typeof(return)*)&data);
114         }
115     }
116 
117     @property inout(Ref!T) peek(T)() inout @trusted if (canStore!T)
118     {
119         if (!ofType!T)
120             return typeof(return).init;
121         return typeof(return)(_raw, true);
122     }
123 
124     bool opEquals(WordVariant that) const
125     {
126         return _raw == that._raw;
127     }
128 
129     bool opEquals(T)(T that) const
130     {
131         auto x = peek!T; // if `this` contains pointer of to instance of type `T`
132         return x && *x == that; // and is equal to it
133     }
134 
135     bool isNull() const
136     {
137         return _raw == S.init;
138     }
139 
140     bool opCast(T : bool)() const
141     {
142         return !isNull;
143     }
144 
145     private void initialize(T)(T that) @trusted
146     {
147         const thatRaw = (*(cast(S*)(&that)));
148         assert(!(thatRaw & typeMask), `Top-most bits of parameter is already occupied`); // TODO: use enforce instead?
149         _raw = (thatRaw | // data in lower part
150                 (cast(S)(indexOf!T + 1) << typeShift)); // use higher bits for type information
151     }
152 
153     private void initializeFromSub(SubTypes...)(WordVariant!(SubTypes) that) @trusted
154     {
155         import std.meta : staticIndexOf;
156         final switch (that.typeIndex)
157         {
158         case 0:
159             this._raw = 0;      // propagate `undefined`/`null`
160             foreach (const i, SubType; SubTypes)
161             {
162                 static assert(staticIndexOf!(SubType, Types) != -1,
163                               "Subtype " ~ SubType.stringof ~ " must be part of the supertypes " ~ Types.stringof);
164             case i + 1:
165                 this._raw = (that.rawValue | // raw value
166                              (cast(S)(indexOf!SubType + 1) << typeShift)); // super type index
167                 break;
168             }
169         }
170     }
171 
172     private bool ofType(T)() const if (canStore!T)
173     {
174         return (!isNull &&
175                 typeIndex == indexOf!T + 1);
176     }
177 
178     inout(T) as(T)() inout @trusted if (canStore!T)
179     {
180         // only in debug mode because it's meant to be called in conjunction with `typeIndex`
181         assert(ofType!T);
182         inout x = rawValue;
183         return *(cast(typeof(return)*)(cast(void*)&x)); // reinterpret
184     }
185 
186     /** Get zero-offset index as `Ix` of current variant type. */
187     Ix typeIx() const
188     {
189         return cast(typeof(return))typeIndex;
190     }
191 
192     /** Get zero-offset index of current variant type. */
193     private auto typeIndex() const
194     {
195         return ((_raw & typeMask) >> typeShift);
196     }
197 
198     private S rawValue() const
199     {
200         return _raw & ~typeMask;
201     }
202 
203     private S _raw;             // raw untyped word
204 
205 }
206 
207 ///
208 pure nothrow unittest
209 {
210     import std.meta : AliasSeq;
211 
212     alias SubType = WordVariant!(byte*, short*);
213     alias SuperType = WordVariant!(bool*, byte*, short*, long*);
214 
215     byte* byteValue = cast(byte*)(0x11);
216     short* shortValue = cast(short*)(0x22);
217 
218     SubType sub = byteValue;
219     assert(sub.typeIndex == 1);
220     assert(sub.peek!(byte*));
221     assert(*(sub.peek!(byte*)) == byteValue);
222 
223     SuperType sup = sub;
224     assert(sup.typeIndex == 2);
225     assert(sup.peek!(byte*));
226     assert(*(sup.peek!(byte*)) == byteValue);
227 
228     sub = shortValue;
229     assert(sub.typeIndex == 2);
230     assert(sub.peek!(short*));
231     assert(*(sub.peek!(short*)) == shortValue);
232 
233     sup = sub;
234     assert(sup.typeIndex == 3);
235     assert(sup.peek!(short*));
236     assert(*(sup.peek!(short*)) == shortValue);
237 }
238 
239 ///
240 pure nothrow unittest
241 {
242     import std.meta : AliasSeq;
243 
244     alias Types = AliasSeq!(byte*, short*, int*, long*,
245                             ubyte*, ushort*, uint*, ulong*,
246                             float*, double*, real*,
247                             char*, wchar*, dchar*);
248 
249     alias V = WordVariant!Types;
250     V v;
251 
252     try
253     {
254         assert(v.toString == "null");
255     }
256     catch (Exception e) {}
257 
258     assert(v.isNull);
259     v = null;
260     assert(v.isNull);
261     assert(!v);
262 
263     foreach (Tp; Types)
264     {
265         alias T = typeof(*Tp.init);
266 
267         static assert(!__traits(compiles, { T[] a; v = &a; }));
268         static assert(!__traits(compiles, { v.peek!(T[]*); }));
269 
270         // assignment from stack pointer
271         T a = 73;
272         T a_ = 73;
273 
274         v = &a;
275         assert(v);
276         assert(!v.isNull);
277         assert(v.typeIndex != 0);
278         assert(v.ofType!Tp);
279 
280         assert(v == &a);
281         assert(v != &a_);
282         assert(v);
283 
284         foreach (Up; Types)
285         {
286             alias U = typeof(*Up.init);
287             static if (is(T == U))
288             {
289                 assert(v.peek!Up);
290                 assert(*(v.peek!Up) == &a);
291                 assert(v.as!Up == &a);
292             }
293             else
294                 assert(!v.peek!Up);
295         }
296 
297         // assignment from heap pointer
298         T* b = new T;
299         T* b_ = new T;
300         *b = 73;
301         *b_ = 73;
302         v = b;
303         assert(v == b);
304         assert(v != b_);
305         assert(v);
306         foreach (Up; Types)
307         {
308             alias U = typeof(*Up.init);
309             static if (is(T == U))
310             {
311                 assert(v.peek!Up);
312                 assert(*(v.peek!Up) == b);
313             }
314             else
315                 assert(!v.peek!Up);
316         }
317 
318     }
319 }
320 
321 /** A typed pointer to a variant of `Types`, packed into a word (`size_t`).
322  *
323  * See_Also: `std.bitmanip.taggedPointer`.
324  */
325 struct VariantPointerTo(Types...)
326 {
327     static assert(this.sizeof == (void*).sizeof, "Size must be same as word (pointer)");
328 
329     alias S = size_t; // TODO: templatize?
330 
331     import nxt.bit_traits : bitsNeeded;
332     enum typeBits = bitsNeeded!(Types.length); // number of bits needed to represent variant type
333     enum typeShift = 8*S.sizeof - typeBits;
334     enum typeMask = cast(S)(2^^typeBits - 1) << typeShift;
335 
336     import std.meta : staticIndexOf;
337     enum indexOf(T) = staticIndexOf!(T, Types); // TODO: cast to ubyte if Types.length is <= 256
338 
339     /// Is `true` iff a pointer to a `T` can be stored.
340     enum canStorePointerTo(T) = indexOf!T >= 0;
341 
342     pure:
343 
344     @property string toString() const @trusted // TODO: pure
345     {
346         import std.conv : to;
347         final switch (typeIndex)
348         {
349             foreach (const i, T; Types)
350             {
351             case i: return T.stringof ~ `@` ~ ptr.to!string;
352             }
353         }
354     }
355 
356     nothrow:
357 
358     extern (D) auto toHash() const
359     {
360         import core.internal.hash : hashOf;
361         return _raw.hashOf;
362     }
363 
364     @nogc:
365 
366     /// Construction from `value`.
367     this(T)(T* value)
368     if (canStorePointerTo!T)
369     {
370         init(value);
371     }
372     /// ditto
373     this(typeof(null) value) { /* null is the default */ }
374 
375     /// Assignment from `that`.
376     auto ref opAssign(T)(T* that)
377     if (canStorePointerTo!T)
378     {
379         init(that);
380         return this;
381     }
382     /// ditto
383     auto ref opAssign(typeof(null) that)
384     {
385         _raw = S.init;
386         return this;
387     }
388 
389     @property inout(void)* ptr() inout @trusted
390     {
391         return cast(void*)(_raw & ~typeMask);
392     }
393 
394     @property inout(T)* peek(T)() inout @trusted if (canStorePointerTo!T)
395     {
396         static if (is(T == void))
397             static assert(canStorePointerTo!T, `Cannot store a ` ~ T.stringof ~ ` in a ` ~ name);
398         if (!pointsTo!T)
399             return null;
400         return cast(inout T*)ptr;
401     }
402 
403     bool opEquals(T)(T* that) const @trusted
404     {
405         auto x = peek!T; // if `this` contains pointer of to instance of type `T`
406         return (x &&
407                 x == that); // and is equal to it
408     }
409 
410     bool isNull() const
411     {
412         return ptr is null;
413     }
414 
415     bool opCast(T : bool)() const
416     {
417         return ptr !is null;
418     }
419 
420     private void init(T)(T* that)
421     {
422         const thatRaw = cast(S)that;
423         assert(!(thatRaw & typeMask), `Top-most bits of pointer are already occupied`); // TODO: use enforce instead?
424         _raw = (thatRaw | // pointer in lower part
425                 (cast(S)(indexOf!T) << typeShift)); // use higher bits for type information
426     }
427 
428     /** Get zero-offset index of current variant type. */
429     private auto typeIndex() const
430     {
431         return (_raw & typeMask) >> typeShift;
432     }
433 
434     private bool pointsTo(T)() const if (canStorePointerTo!T)
435     {
436         return typeIndex == indexOf!T;
437     }
438 
439     private S _raw;             // raw untyped word
440 
441 }
442 
443 ///
444 pure nothrow unittest
445 {
446     import std.meta : AliasSeq;
447 
448     alias Types = AliasSeq!(byte, short, int, long,
449                             ubyte, ushort, uint, ulong,
450                             float, double, real,
451                             char, wchar, dchar);
452 
453     alias V = VariantPointerTo!Types;
454 
455     V v;
456     assert(v.isNull);
457     v = null;
458     assert(v.isNull);
459     assert(!v);
460 
461     foreach (T; Types)
462     {
463         static assert(!__traits(compiles, { T[] a; v = &a; }));
464         static assert(!__traits(compiles, { v.peek!(T[]*); }));
465 
466         // assignment from stack pointer
467         T a = 73;
468         T a_ = 73;
469         v = &a;
470         assert(v == &a);
471         assert(v != &a_);
472         assert(v);
473         foreach (U; Types)
474         {
475             static if (is(T == U))
476             {
477                 assert(v.peek!U);
478                 assert(*(v.peek!U) == a);
479             }
480             else
481                 assert(!v.peek!U);
482         }
483 
484         // assignment from heap pointer
485         T* b = new T;
486         T* b_ = new T;
487         *b = 73;
488         *b_ = 73;
489         v = b;
490         assert(v == b);
491         assert(v != b_);
492         assert(v);
493         foreach (U; Types)
494         {
495             static if (is(T == U))
496             {
497                 assert(v.peek!U);
498                 assert(*(v.peek!U) == *b);
499             }
500             else
501                 assert(!v.peek!U);
502         }
503 
504     }
505 }