From 987b75d8db9d536a534c18d668583b05536b051a Mon Sep 17 00:00:00 2001 From: yilabs Date: Mon, 14 Nov 2022 00:15:06 -0500 Subject: [PATCH] add Recyclable! --- source/jdiutil/memory.d | 70 ++++++++++++++++++++++++++++++++++++++++- 1 file changed, 69 insertions(+), 1 deletion(-) diff --git a/source/jdiutil/memory.d b/source/jdiutil/memory.d index 8773faa..6ecef91 100644 --- a/source/jdiutil/memory.d +++ b/source/jdiutil/memory.d @@ -1,12 +1,14 @@ module jdiutil.memory; import core.memory; + +import std.exception; import std.experimental.logger; import std.stdio; import std.traits; version (unittest) { -import fluent.asserts; +public import fluent.asserts; } public import jdiutil.jdiutil; @@ -137,6 +139,72 @@ unittest { } +/* ========================================================================== *\ + object pool +\* ========================================================================== */ +const int EACH_WAIT_MILLISECONDS = 100; + +// allocate an array of T at init, and recycle the object to reduce memory usage +// TODO: add linked-list for size unbounded pool +template Recyclable(T) { + static if (is(T == struct)) { + alias RefT = T*; + } else { + alias RefT = T; + } + + // https://issues.dlang.org/show_bug.cgi?id=20838#c12 + // manually take care of required 16-bytes alignment: for cas(CMPXCHG16B) + public align(16) shared bool taken; // core.atomic.cas to select free one to use + + static __gshared T[] pool; + + static void initObjectPool(long size) { + pool = new T[size]; // why in actual run, we need more than numWorkerThreads? + + static if (is(T == class)) { // TODO: heapAllocN + foreach (i; 0 .. size) { + pool[i] = new T(); + } + } + } + + static RefT make(int wait=int.max) { // how many times should wait, by default will wait + enforce(pool.length > 0); + for (;;) { + foreach (ref nd; pool) { // loop on the pool array + if (core.atomic.cas(&(nd.taken), false, true)) { // atomic op! + enforce(nd.taken); + static if (is(T == struct)) { + return &nd; + } else { + return nd; + } + } + } + + if (wait-- > 0) { + Thread.sleep(dur!("msecs")(EACH_WAIT_MILLISECONDS)); + } else { + return null; + } + } + return null; + } + +} + +// release the object, and put back into the recycle bin +void recycle(T)(T needleDev) { + if (needleDev is null) { + return; + } + bool released = (core.atomic.cas(&(needleDev.taken), true, false)); // atomic op! + enforce(released); +} + + + /* ========================================================================== *\ \* ========================================================================== */ // from: https://forum.dlang.org/post/acafsosotrjdswwuklob@forum.dlang.org