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 }