Constructor function merely taking a function/delegate pointer, to call when invoking this fiber (via .call()).
Constructor function taking a T payload to assign to its own internal this.payload, as well as a function/delegate pointer to call when invoking this fiber (via .call()).
Hijacks the invocation of the Fiber and injects the string name of the calling function into the CarryingFiber.caller member before calling the Fiber's own .call().
Resets the payload to its .init value.
Safely returns the state of the fiber, taking into consideration it may have been reset.
How many times this CarryingFiber has been called (via .call()).
String name of the function that last called this CarryingFiber (via .call()).
String name of the function that created this CarryingFiber.
Whether or not this CarryingFiber has been called to completion and should be considered expired.
Embedded payload value in this fiber; what distinguishes it from plain Fibers.
Type to embed into the class as the type of CarryingFiber.payload.
void dg() { auto fiber = cast(CarryingFiber!bool)Fiber.getThis(); assert(fiber !is null); // Correct cast assert(fiber.payload); Fiber.yield(); assert(!fiber.payload); } auto fiber = new CarryingFiber!bool(&dg, true, BufferSize.fiberStack); assert(fiber.called == 0); fiber.call(); assert(fiber.called == 1); fiber.payload = false; fiber.call();
1 import std.conv : to; 2 3 static struct Payload 4 { 5 string s = "Hello"; 6 size_t i = 42; 7 8 static auto getCompileTimeRandomPayload() 9 { 10 enum randomString = __TIMESTAMP__; 11 return Payload(randomString, hashOf(randomString)); 12 } 13 } 14 15 static auto creatorTest(void delegate() dg) 16 { 17 import kameloso.constants : BufferSize; 18 19 auto fiber = new CarryingFiber!Payload 20 (dg, 21 Payload.getCompileTimeRandomPayload(), 22 BufferSize.fiberStack); 23 assert((fiber.called == 0), fiber.called.to!string); 24 return fiber; 25 } 26 27 static void callerTest1(CarryingFiber!Payload fiber) 28 { 29 immutable payload = Payload.getCompileTimeRandomPayload(); 30 31 assert((fiber.payload.s == payload.s), fiber.payload.s); 32 assert((fiber.payload.i == payload.i), fiber.payload.i.to!string); 33 fiber.call(); 34 assert((fiber.payload.s == Payload.init.s), fiber.payload.s); 35 assert((fiber.payload.i == Payload.init.i), fiber.payload.i.to!string); 36 } 37 38 static void callerTest2(CarryingFiber!Payload fiber) 39 { 40 fiber.call(); 41 } 42 43 void dg() 44 { 45 auto thisFiber = cast(CarryingFiber!Payload)Fiber.getThis(); 46 assert(thisFiber, "Incorrectly cast fiber: " ~ typeof(thisFiber).stringof); 47 48 // __FUNCTION__ will be something like "kameloso.thread.__unittest_L577_C1.dg" 49 enum expectedFunction = __FUNCTION__[0..$-2] ~ "dg"; 50 51 static if (__FUNCTION__ == expectedFunction) 52 { 53 enum expectedCreator = __FUNCTION__[0..$-2] ~ "creatorTest"; 54 enum expectedCaller1 = __FUNCTION__[0..$-2] ~ "callerTest1"; 55 enum expectedCaller2 = __FUNCTION__[0..$-2] ~ "callerTest2"; 56 57 // First state 58 assert((thisFiber.creator == expectedCreator), thisFiber.creator); 59 assert((thisFiber.caller == expectedCaller1), thisFiber.caller); 60 assert((thisFiber.called == 1), thisFiber.called.to!string); 61 thisFiber.resetPayload(); 62 Fiber.yield(); 63 64 // Second state 65 assert((thisFiber.caller == expectedCaller2), thisFiber.caller); 66 assert((thisFiber.called == 2), thisFiber.called.to!string); 67 } 68 else 69 { 70 enum message = "Bad logic slicing function names in `CarryingFiber` unit test; " ~ 71 "please report this as a bug. (Was there a change in the compiler?)"; 72 pragma(msg, message); 73 74 // Yield once so the tests still pass 75 Fiber.yield(); 76 } 77 } 78 79 auto fiber = creatorTest(&dg); 80 callerTest1(fiber); 81 callerTest2(fiber);
A Fiber carrying a payload of type T, along with some metadata.
Used interchangeably with Fiber, but allows for casting to true CarryingFiber!T-ness to access the payload member.