diff --git a/.vs/deemon-v141.sln b/.vs/deemon-v141.sln
index 0f4989826..a0dbdf39d 100644
--- a/.vs/deemon-v141.sln
+++ b/.vs/deemon-v141.sln
@@ -396,6 +396,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "test", "test", "{2360741E-F
..\util\test\deemon-sequence-each-contains.dee = ..\util\test\deemon-sequence-each-contains.dee
..\util\test\deemon-sequence-each.dee = ..\util\test\deemon-sequence-each.dee
..\util\test\deemon-sequence-enumerate.dee = ..\util\test\deemon-sequence-enumerate.dee
+ ..\util\test\deemon-sequence-flatten.dee = ..\util\test\deemon-sequence-flatten.dee
..\util\test\deemon-sequence-locate.dee = ..\util\test\deemon-sequence-locate.dee
..\util\test\deemon-sequence-math.dee = ..\util\test\deemon-sequence-math.dee
..\util\test\deemon-sequence-reversed.dee = ..\util\test\deemon-sequence-reversed.dee
diff --git a/.vs/deemon-v141.vcxproj b/.vs/deemon-v141.vcxproj
index 72ca8dac6..c43cab3f4 100644
--- a/.vs/deemon-v141.vcxproj
+++ b/.vs/deemon-v141.vcxproj
@@ -245,6 +245,7 @@
+
@@ -443,6 +444,7 @@
+
diff --git a/.vs/deemon-v141.vcxproj.filters b/.vs/deemon-v141.vcxproj.filters
index 3ec72f7f8..e2e448ffa 100644
--- a/.vs/deemon-v141.vcxproj.filters
+++ b/.vs/deemon-v141.vcxproj.filters
@@ -708,6 +708,9 @@
src\objects\seq
+
+ src\objects\seq
+
src\objects\seq
@@ -1298,6 +1301,9 @@
src\objects\seq
+
+ src\objects\seq
+
src\objects\seq
diff --git a/.vs/deemon-v142.sln b/.vs/deemon-v142.sln
index 7d3b80356..4565eb487 100644
--- a/.vs/deemon-v142.sln
+++ b/.vs/deemon-v142.sln
@@ -396,6 +396,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "test", "test", "{2360741E-F
..\util\test\deemon-sequence-each-contains.dee = ..\util\test\deemon-sequence-each-contains.dee
..\util\test\deemon-sequence-each.dee = ..\util\test\deemon-sequence-each.dee
..\util\test\deemon-sequence-enumerate.dee = ..\util\test\deemon-sequence-enumerate.dee
+ ..\util\test\deemon-sequence-flatten.dee = ..\util\test\deemon-sequence-flatten.dee
..\util\test\deemon-sequence-locate.dee = ..\util\test\deemon-sequence-locate.dee
..\util\test\deemon-sequence-math.dee = ..\util\test\deemon-sequence-math.dee
..\util\test\deemon-sequence-reversed.dee = ..\util\test\deemon-sequence-reversed.dee
diff --git a/.vs/deemon-v142.vcxproj b/.vs/deemon-v142.vcxproj
index a16e30cad..50fb30e5e 100644
--- a/.vs/deemon-v142.vcxproj
+++ b/.vs/deemon-v142.vcxproj
@@ -245,6 +245,7 @@
+
@@ -443,6 +444,7 @@
+
diff --git a/.vs/deemon-v142.vcxproj.filters b/.vs/deemon-v142.vcxproj.filters
index 3ec72f7f8..e2e448ffa 100644
--- a/.vs/deemon-v142.vcxproj.filters
+++ b/.vs/deemon-v142.vcxproj.filters
@@ -708,6 +708,9 @@
src\objects\seq
+
+ src\objects\seq
+
src\objects\seq
@@ -1298,6 +1301,9 @@
src\objects\seq
+
+ src\objects\seq
+
src\objects\seq
diff --git a/src/deemon/objects/dict.c b/src/deemon/objects/dict.c
index 4e4d0b1ea..a39eaf2fb 100644
--- a/src/deemon/objects/dict.c
+++ b/src/deemon/objects/dict.c
@@ -2906,6 +2906,7 @@ INTERN_TPCONST struct type_getset tpconst dict_getsets[] = {
TYPE_GETTER_F(STR_frozen, &DeeRoDict_FromSequence, METHOD_FNOREFESCAPE,
"->?Ert:RoDict\n"
"Returns a read-only (frozen) copy of @this ?."),
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
#ifndef CONFIG_NO_DEEMON_100_COMPAT
TYPE_GETSET_F("max_load_factor",
&deprecated_d100_get_maxloadfactor,
diff --git a/src/deemon/objects/hashset.c b/src/deemon/objects/hashset.c
index 76cc252dd..3d2acfc45 100644
--- a/src/deemon/objects/hashset.c
+++ b/src/deemon/objects/hashset.c
@@ -2048,6 +2048,7 @@ INTERN_TPCONST struct type_getset tpconst hashset_getsets[] = {
TYPE_GETTER_F(STR_frozen, &DeeRoSet_FromSequence, METHOD_FNOREFESCAPE,
"->?Ert:RoSet\n"
"Returns a read-only (frozen) copy of @this HashSet"),
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
#ifndef CONFIG_NO_DEEMON_100_COMPAT
TYPE_GETSET_F("max_load_factor",
&deprecated_d100_get_maxloadfactor,
diff --git a/src/deemon/objects/list.c b/src/deemon/objects/list.c
index 4e3a2c288..c8ae2fde8 100644
--- a/src/deemon/objects/list.c
+++ b/src/deemon/objects/list.c
@@ -3608,6 +3608,7 @@ PRIVATE struct type_getset tpconst list_getsets[] = {
TYPE_GETTER_F(STR_frozen, &list_get_frozen, METHOD_FNOREFESCAPE,
"->?DTuple\n"
"Return a copy of the contents of @this List as an immutable sequence"),
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &list_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/src/deemon/objects/rodict.c b/src/deemon/objects/rodict.c
index a003ec650..63562d180 100644
--- a/src/deemon/objects/rodict.c
+++ b/src/deemon/objects/rodict.c
@@ -1082,6 +1082,7 @@ PRIVATE struct type_method tpconst rodict_methods[] = {
PRIVATE struct type_getset tpconst rodict_getsets[] = {
TYPE_GETTER(STR_frozen, &DeeObject_NewRef, "->?."),
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &rodict_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/src/deemon/objects/roset.c b/src/deemon/objects/roset.c
index 393cba32d..7ee7b5802 100644
--- a/src/deemon/objects/roset.c
+++ b/src/deemon/objects/roset.c
@@ -585,6 +585,7 @@ PRIVATE struct type_method tpconst roset_methods[] = {
PRIVATE struct type_getset tpconst roset_getsets[] = {
TYPE_GETTER(STR_frozen, &DeeObject_NewRef, "->?."),
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &roset_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/src/deemon/objects/seq.c b/src/deemon/objects/seq.c
index 541e630e9..a8960162f 100644
--- a/src/deemon/objects/seq.c
+++ b/src/deemon/objects/seq.c
@@ -2179,35 +2179,59 @@ PRIVATE struct type_getset tpconst seq_getsets[] = {
/**/ "print repr x.flatten; /* { 0, 1, 2, 5, 7, 9, 1, 10, 1 } */"
"}"),
- /* TODO: Implement this one similar to how "first" and "last" are implement (iow: using `struct Dee_type_seq_cache') */
/* TODO: Variants of this need to be added for `Set' and `Mapping' */
-//TODO: TYPE_GETTER("cached", &DeeSeq_InvokeCached,
-//TODO: "->?.\n"
-//TODO: "Returns a sequence that is a lazily-populated, read-only proxy of @this ?{.}.\n"
-//TODO: "As such, ?#cached behaves similar to ?#frozen, except that the rather than"
-//TODO: /**/ "evaluating the elements of the sequence immediately, said evaluation will"
-//TODO: /**/ "happen the first time elements are accessed. Additionally, there is no"
-//TODO: /**/ "requirement that changes to the original sequence won't propagate into the"
-//TODO: /**/ "\"cached\" sequence (even after the cache may have been populated).\n"
-//TODO: "The only difference between some $seq and ${seq.cached} is that the later"
-//TODO: /**/ "will allow for #B{O(1)} access to elements after an initial access, and"
-//TODO: /**/ "that any side-effects of accessing some element will happen exactly once.\n"
-//TODO: "In practice, this means that ?Acached?DTuple and ?Acached?DList will just"
-//TODO: /**/ "re-return the tuple/list, but for most other sequences, a proxy is returned"
-//TODO: /**/ "that will do one of the following:"
-//TODO: "#L-{"
-//TODO: /**/ "If @this's ?A__seqclass__?DType is ?. and implement ${operator []} and ${operator size},"
-//TODO: /**/ /**/ "the cache lazily computes its size the first time it is accessed, and elements"
-//TODO: /**/ /**/ "are saved into an internal buffer which is re-used during repeated calls|"
-//TODO: /**/ "If @this's ?A__seqclass__?DType is ?. and only implement ${operator []}, the same thing"
-//TODO: /**/ /**/ "happens, except that the size is determined by accessing+caching elements"
-//TODO: /**/ /**/ "until the underlying sequence's ${operator []} returns :IndexError|"
-//TODO: /**/ "If @this implements ${operator iter}, that operator is invoked the first time"
-//TODO: /**/ /**/ "the size- or an element of the cache is accessed, then proceeds to yield from"
-//TODO: /**/ /**/ "that iterator and cache its results until the requested index is reached. (In"
-//TODO: /**/ /**/ "the case of the returned cache's size being accessed, the cache is populated"
-//TODO: /**/ /**/ "fully)"
-//TODO: "}"),
+ TYPE_GETTER(STR_cached, &default_seq_cached,
+ "->?.\n"
+ "Returns a sequence that is a lazily-populated, read-only proxy of @this ?..\n"
+ "The returned proxy will ensure that any indirect side-effects¹ of accessing elements, "
+ /**/ "or iterating the sequence will only happen once upon initial access/iteration. "
+ /**/ "E.g. a yield-function is only executed one time, and only up the point that "
+ /**/ "elements have been requested for.\n"
+ "Sequence types where element access/iteration has no indirect side-effects¹, meaning "
+ /**/ "essentially every materialized ?. type (e.g. ?DList, ?DTuple, ?DDict, ...), can "
+ /**/ "implement this property as a no-op: ${property cached = { . -\\> this; }}\n"
+ "As such, ?#cached behaves similar to ?#frozen, except that the rather than "
+ /**/ "evaluating the elements of the sequence immediately, said evaluation will "
+ /**/ "happen the first time elements are accessed. Additionally, there is no "
+ /**/ "requirement that changes to the original sequence won't propagate into the "
+ /**/ "\"cached\" sequence (even after the cache may have been populated).\n"
+ "The difference between some $seq and ${seq.cached} is that the later "
+ /**/ "will allow for #B{O(1)} access to elements after an initial access, and "
+ /**/ "that any indirect side-effects¹ of accessing some element will happen "
+ /**/ "exactly once.\n"
+ "${"
+ /**/ "/* \"lines\" will lazily populate itself from the contents of \"data.txt\" */\n"
+ /**/ "local lines = Sequence.cached(File.open(\"data.txt\", \"rb\"));\n"
+ /**/ "print lines.locate(line -\\> !line.startswith(\"##\"));\n"
+ /**/ "\\\n"
+ /**/ "/* This will print the first couple of lines from the cache, up until (and\n"
+ /**/ " * including) the already-accessed line, then continue reading from \"data.txt\" */\n"
+ /**/ "for (local line: lines)\n"
+ /**/ " print line;"
+ "}\n"
+ "${"
+ /**/ "function getItems() {\n"
+ /**/ " print \"Enter\";\n"
+ /**/ " yield 10;\n"
+ /**/ " print \"Middle\";\n"
+ /**/ " yield 20;\n"
+ /**/ " print \"Leave\";\n"
+ /**/ "}\n"
+ /**/ "\\\n"
+ /**/ "local items = getItems().cached;\n"
+ /**/ "print items[0]; /* Enter 10 */\n"
+ /**/ "print items[0]; /* 10 */\n"
+ /**/ "print items[1]; /* Middle 20 */\n"
+ /**/ "print items[0]; /* 10 */\n"
+ /**/ "print items[1]; /* 20 */\n"
+ /**/ "print ##items; /* Leave 3 */\n"
+ /**/ "print repr items; /* { 10, 20, 30 } */"
+ "}\n"
+ "#L{"
+ /**/ "{¹}An indirect side-effect is a change to the program's global state, usually caused "
+ /**/ /**/ "by the invocation of user-code. Examples include: changes to global variables, "
+ /**/ /**/ "output being written to a file, or text being printed to stdout."
+ "}"),
/* TODO: itemtype->?DType
* Check if the type of @this overrides the ?#ItemType class attribute.
@@ -2233,9 +2257,9 @@ PRIVATE struct type_getset tpconst seq_getsets[] = {
"Returns a copy of @this Sequence, with all of its current elements, as well as "
/**/ "their current order frozen in place, constructing a snapshot of the Sequence's "
/**/ "current elements. - The actual type of Sequence returned is implementation- "
- /**/ "and type- specific, but a guaranty is made that nothing no non-implementation-"
- /**/ "specific functions/attributes (i.e. anything that doesn't match #C{__*__}) will "
- /**/ "be able to change the elements of the returned sequence.\n"
+ /**/ "and type- specific, but a guaranty is made that no non-implementation-specific "
+ /**/ "functions/attributes (i.e. anything that doesn't match #C{__*__}) will be able "
+ /**/ "to change the elements of the returned sequence.\n"
"By default, this attribute simply casts @this Sequence into a ?DTuple, but sequence "
/**/ "types that are known to already be immutable can override this attribute such "
/**/ "that they simple re-return themselves.\n"
diff --git a/src/deemon/objects/seq/cached-seq.c b/src/deemon/objects/seq/cached-seq.c
new file mode 100644
index 000000000..d31340d16
--- /dev/null
+++ b/src/deemon/objects/seq/cached-seq.c
@@ -0,0 +1,769 @@
+/* Copyright (c) 2018-2024 Griefer@Work *
+ * *
+ * This software is provided 'as-is', without any express or implied *
+ * warranty. In no event will the authors be held liable for any damages *
+ * arising from the use of this software. *
+ * *
+ * Permission is granted to anyone to use this software for any purpose, *
+ * including commercial applications, and to alter it and redistribute it *
+ * freely, subject to the following restrictions: *
+ * *
+ * 1. The origin of this software must not be misrepresented; you must not *
+ * claim that you wrote the original software. If you use this software *
+ * in a product, an acknowledgement (see the following) in the product *
+ * documentation is required: *
+ * Portions Copyright (c) 2018-2024 Griefer@Work *
+ * 2. Altered source versions must be plainly marked as such, and must not be *
+ * misrepresented as being the original software. *
+ * 3. This notice may not be removed or altered from any source distribution. *
+ */
+#ifndef GUARD_DEEMON_OBJECTS_CACHED_SEQ_C
+#define GUARD_DEEMON_OBJECTS_CACHED_SEQ_C 1
+
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include
+
+/**/
+#include "../../runtime/runtime_error.h"
+#include "../../runtime/strings.h"
+#include "../generic-proxy.h"
+
+/**/
+#include "cached-seq.h"
+
+DECL_BEGIN
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_ctor(CachedSeq_WithIter *__restrict self) {
+ Dee_Incref(Dee_EmptyIterator);
+ self->cswi_iter = Dee_EmptyIterator;
+ Dee_atomic_lock_init(&self->cswi_lock);
+ objectlist_init(&self->cswi_cache);
+ CachedSeq_WithIter_SetFinished(self);
+ return 0;
+}
+
+PRIVATE WUNUSED NONNULL((1, 2)) int DCALL
+cswi_copycache(CachedSeq_WithIter *__restrict self,
+ CachedSeq_WithIter *__restrict other) {
+ DREF DeeObject **cache_copy;
+ size_t cache_size;
+again:
+ CachedSeq_WithIter_LockAcquire(other);
+ cache_size = other->cswi_cache.ol_elemc;
+ cache_copy = Dee_objectlist_elemv_trymalloc(cache_size);
+ if unlikely(!cache_copy) {
+ CachedSeq_WithIter_LockRelease(other);
+ cache_copy = Dee_objectlist_elemv_malloc(cache_size);
+ if unlikely(!cache_copy)
+ goto err;
+ CachedSeq_WithIter_LockAcquire(other);
+ if unlikely(cache_size != other->cswi_cache.ol_elemc) {
+ CachedSeq_WithIter_LockRelease(other);
+ Dee_objectlist_elemv_free(cache_copy);
+ goto again;
+ }
+ }
+ cache_copy = Dee_Movrefv(cache_copy, other->cswi_cache.ol_elemv, cache_size);
+ _Dee_objectlist_setalloc(&self->cswi_cache, cache_size);
+#ifdef CachedSeq_WithIter_HAVE_cswi_finished
+ self->cswi_finished = other->cswi_finished;
+#else /* CachedSeq_WithIter_HAVE_cswi_finished */
+ if (CachedSeq_WithIter_GetFinished(other))
+ CachedSeq_WithIter_SetFinished(self);
+#endif /* !CachedSeq_WithIter_HAVE_cswi_finished */
+ CachedSeq_WithIter_LockRelease(other);
+ self->cswi_cache.ol_elemc = cache_size;
+ self->cswi_cache.ol_elemv = cache_copy;
+ return 0;
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1, 2)) int DCALL
+cswi_copy(CachedSeq_WithIter *__restrict self,
+ CachedSeq_WithIter *__restrict other) {
+ if unlikely(cswi_copycache(self, other))
+ goto err;
+ if (CachedSeq_WithIter_GetFinished(self)) {
+ Dee_Incref(Dee_EmptyIterator);
+ self->cswi_iter = Dee_EmptyIterator;
+ } else {
+ self->cswi_iter = DeeObject_Copy(other->cswi_iter);
+ if unlikely(!self->cswi_iter)
+ goto err_cache;
+ }
+ return 0;
+err_cache:
+ objectlist_fini(&self->cswi_cache);
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1, 2)) int DCALL
+cswi_deep(CachedSeq_WithIter *__restrict self,
+ CachedSeq_WithIter *__restrict other) {
+ if unlikely(cswi_copycache(self, other))
+ goto err;
+ if (CachedSeq_WithIter_GetFinished(self)) {
+ Dee_Incref(Dee_EmptyIterator);
+ self->cswi_iter = Dee_EmptyIterator;
+ } else {
+ self->cswi_iter = DeeObject_DeepCopy(other->cswi_iter);
+ if unlikely(!self->cswi_iter)
+ goto err_cache;
+ }
+ return 0;
+err_cache:
+ objectlist_fini(&self->cswi_cache);
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_deepload(CachedSeq_WithIter *__restrict self) {
+ size_t i;
+ for (i = 0;; ++i) {
+ DREF DeeObject *oldval, *newval;
+again_copy_at_i:
+ CachedSeq_WithIter_LockAcquire(self);
+ if (i >= self->cswi_cache.ol_elemc) {
+ CachedSeq_WithIter_LockRelease(self);
+ break;
+ }
+ oldval = self->cswi_cache.ol_elemv[i];
+ Dee_Incref(oldval);
+ CachedSeq_WithIter_LockRelease(self);
+ newval = DeeObject_DeepCopy(oldval);
+ Dee_Decref_unlikely(oldval);
+ if unlikely(!newval)
+ goto err;
+ CachedSeq_WithIter_LockAcquire(self);
+ if unlikely(i >= self->cswi_cache.ol_elemc) {
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_Decref(newval);
+ break;
+ }
+ if unlikely(self->cswi_cache.ol_elemv[i] != oldval) {
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_Decref(newval);
+ goto again_copy_at_i;
+ }
+ self->cswi_cache.ol_elemv[i] = newval; /* Inherit reference (x2) */
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_Decref(oldval);
+ }
+ return 0;
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_init(CachedSeq_WithIter *__restrict self,
+ size_t argc, DeeObject *const *argv) {
+ if (DeeArg_Unpack(argc, argv, "o:_CachedSeqWithIter", &self->cswi_iter))
+ goto err;
+ Dee_Incref(self->cswi_iter);
+ Dee_atomic_lock_init(&self->cswi_lock);
+ objectlist_init(&self->cswi_cache);
+ CachedSeq_WithIter_InitFinished(self);
+ return 0;
+err:
+ return -1;
+}
+
+PRIVATE NONNULL((1)) void DCALL
+cswi_fini(CachedSeq_WithIter *__restrict self) {
+ objectlist_fini(&self->cswi_cache);
+ Dee_Decref(self->cswi_iter);
+}
+
+PRIVATE NONNULL((1, 2)) void DCALL
+cswi_visit(CachedSeq_WithIter *__restrict self, dvisit_t proc, void *arg) {
+ CachedSeq_WithIter_LockAcquire(self);
+ Dee_Visitv(self->cswi_cache.ol_elemv,
+ self->cswi_cache.ol_elemc);
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_Visit(self->cswi_iter);
+}
+
+PRIVATE NONNULL((1)) void DCALL
+cswi_clear(CachedSeq_WithIter *__restrict self) {
+ size_t old_elemc;
+ DREF DeeObject **old_elemv;
+ CachedSeq_WithIter_LockAcquire(self);
+ old_elemc = self->cswi_cache.ol_elemc;
+ old_elemv = self->cswi_cache.ol_elemv;
+ self->cswi_cache.ol_elemc = 0;
+ self->cswi_cache.ol_elemv = NULL;
+ _Dee_objectlist_setalloc(&self->cswi_cache, 0);
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_Decrefv(old_elemv, old_elemc);
+ Dee_objectlist_elemv_free(old_elemv);
+}
+
+PRIVATE struct type_gc cswi_gc = {
+ /* .tp_clear = */ (void (DCALL *)(DeeObject *__restrict))&cswi_clear
+};
+
+/* @return: 1 : Failure (`index' is out-of-bounds, but the cache is now fully loaded)
+ * @return: 0 : Success (`index' is now loaded within the cache)
+ * @return: -1: Error * */
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_ensure_loaded(CachedSeq_WithIter *__restrict self, size_t index) {
+ DREF DeeObject *nextitem;
+ CachedSeq_WithIter_LockAcquire(self);
+ while (index >= self->cswi_cache.ol_elemc) {
+ size_t alloc;
+ if (CachedSeq_WithIter_GetFinished(self)) {
+ CachedSeq_WithIter_LockRelease(self);
+ return 1; /* Out-of-bounds */
+ }
+ CachedSeq_WithIter_LockRelease(self);
+
+ /* Load 1 additional element. */
+ nextitem = DeeObject_IterNext(self->cswi_iter);
+ if (!ITER_ISOK(nextitem)) {
+ if unlikely(!nextitem)
+ goto err;
+ CachedSeq_WithIter_LockAcquire(self);
+ CachedSeq_WithIter_SetFinished(self);
+ CachedSeq_WithIter_LockRelease(self);
+ return 1;
+ }
+
+ /* Store "nextitem" within the cache. */
+lock_and_append_to_cache:
+ CachedSeq_WithIter_LockAcquire(self);
+ alloc = Dee_objectlist_getalloc(&self->cswi_cache);
+ ASSERT(alloc >= self->cswi_cache.ol_elemc);
+ if unlikely(alloc <= self->cswi_cache.ol_elemc) {
+ DREF DeeObject **new_elemv;
+ size_t new_alloc = DEE_OBJECTLIST_MOREALLOC(alloc);
+ if (new_alloc < DEE_OBJECTLIST_MINALLOC)
+ new_alloc = DEE_OBJECTLIST_MINALLOC;
+ new_elemv = Dee_objectlist_elemv_tryrealloc_safe(self->cswi_cache.ol_elemv, new_alloc);
+ if unlikely(!new_elemv) {
+ size_t min_alloc = self->cswi_cache.ol_elemc;
+ new_alloc = min_alloc + 1;
+ new_elemv = Dee_objectlist_elemv_tryrealloc_safe(self->cswi_cache.ol_elemv, new_alloc);
+ if unlikely(!new_elemv) {
+ /* Cannot realloc -> try again while not holding any locks. */
+ CachedSeq_WithIter_LockRelease(self);
+ new_alloc = DEE_OBJECTLIST_MOREALLOC(alloc);
+ if (new_alloc < DEE_OBJECTLIST_MINALLOC)
+ new_alloc = DEE_OBJECTLIST_MINALLOC;
+ new_elemv = Dee_objectlist_elemv_trymalloc_safe(new_alloc);
+ if unlikely(!new_elemv) {
+ new_alloc = min_alloc + 1;
+ new_elemv = Dee_objectlist_elemv_malloc_safe(new_alloc);
+ if unlikely(!new_elemv)
+ goto err_nextitem;
+ }
+ CachedSeq_WithIter_LockAcquire(self);
+ if unlikely(self->cswi_cache.ol_elemc > min_alloc) {
+ CachedSeq_WithIter_LockRelease(self);
+ Dee_objectlist_elemv_free(new_elemv);
+ goto lock_and_append_to_cache;
+ }
+ /* Copy into the new cache-buffer. */
+ new_elemv = (DREF DeeObject **)memcpyc(new_elemv, self->cswi_cache.ol_elemv,
+ self->cswi_cache.ol_elemc,
+ sizeof(DREF DeeObject *));
+ Dee_objectlist_elemv_free(self->cswi_cache.ol_elemv);
+ self->cswi_cache.ol_elemv = new_elemv;
+ _Dee_objectlist_setalloc(&self->cswi_cache, new_alloc);
+ }
+ }
+ self->cswi_cache.ol_elemv = new_elemv;
+ _Dee_objectlist_setalloc(&self->cswi_cache, new_alloc);
+ }
+ self->cswi_cache.ol_elemv[self->cswi_cache.ol_elemc] = nextitem; /* Inherit reference */
+ ++self->cswi_cache.ol_elemc;
+ }
+ CachedSeq_WithIter_LockRelease(self);
+ return 0; /* Already in cache! */
+err_nextitem:
+ Dee_Decref(nextitem);
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+cswi_trygetitem_index(CachedSeq_WithIter *__restrict self, size_t index) {
+ DREF DeeObject *result;
+ int status;
+#ifndef CONFIG_NO_THREADS
+again:
+#endif /* !CONFIG_NO_THREADS */
+ status = cswi_ensure_loaded(self, index);
+ if unlikely(status < 0)
+ goto err;
+ if (status > 0)
+ return ITER_DONE;
+ CachedSeq_WithIter_LockAcquire(self);
+#ifndef CONFIG_NO_THREADS
+ if unlikely(index >= self->cswi_cache.ol_elemc) {
+ CachedSeq_WithIter_LockRelease(self);
+ goto again;
+ }
+#endif /* !CONFIG_NO_THREADS */
+ ASSERT(index < self->cswi_cache.ol_elemc);
+ result = self->cswi_cache.ol_elemv[index];
+ Dee_Incref(result);
+ CachedSeq_WithIter_LockRelease(self);
+ return result;
+err:
+ return NULL;
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+cswi_getitem_index(CachedSeq_WithIter *__restrict self, size_t index) {
+ DREF DeeObject *result = cswi_trygetitem_index(self, index);
+ if unlikely(result == ITER_DONE)
+ goto err_oob;
+ return result;
+ {
+ size_t size;
+err_oob:
+ CachedSeq_WithIter_LockAcquire(self);
+ size = self->cswi_cache.ol_elemc;
+ CachedSeq_WithIter_LockRelease(self);
+ err_index_out_of_bounds((DeeObject *)self, index, size);
+ }
+/*err:*/
+ return NULL;
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_bounditem_index(CachedSeq_WithIter *__restrict self, size_t index) {
+ int result = cswi_ensure_loaded(self, index);
+ if likely(result >= 0)
+ result = result ? -2 : 1;
+ return result;
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_hasitem_index(CachedSeq_WithIter *__restrict self, size_t index) {
+ int result = cswi_ensure_loaded(self, index);
+ if likely(result >= 0)
+ result = !result;
+ return result;
+}
+
+PRIVATE WUNUSED NONNULL((1)) size_t DCALL
+cswi_size(CachedSeq_WithIter *__restrict self) {
+ size_t result;
+ if unlikely(cswi_ensure_loaded(self, (size_t)-1) < 0)
+ goto err;
+ CachedSeq_WithIter_LockAcquire(self);
+ result = self->cswi_cache.ol_elemc;
+ CachedSeq_WithIter_LockRelease(self);
+ return result;
+err:
+ return (size_t)-1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) size_t DCALL
+cswi_size_fast(CachedSeq_WithIter *__restrict self) {
+ size_t result;
+ CachedSeq_WithIter_LockAcquire(self);
+ result = self->cswi_cache.ol_elemc;
+ if (!CachedSeq_WithIter_GetFinished(self))
+ result = (size_t)-1;
+ CachedSeq_WithIter_LockRelease(self);
+ return result;
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF CachedSeq_WithIter_Iterator *DCALL
+cswi_iter(CachedSeq_WithIter *__restrict self) {
+ DREF CachedSeq_WithIter_Iterator *result;
+ result = DeeObject_MALLOC(CachedSeq_WithIter_Iterator);
+ if unlikely(!result)
+ goto err;
+ Dee_Incref(self);
+ result->cswii_cache = self;
+ result->cswii_index = 0;
+ DeeObject_Init(result, &CachedSeq_WithIter_Iterator_Type);
+ return result;
+err:
+ return NULL;
+}
+
+PRIVATE WUNUSED NONNULL((1)) Dee_ssize_t DCALL
+cswi_foreach(CachedSeq_WithIter *__restrict self, Dee_foreach_t cb, void *arg) {
+ size_t i;
+ Dee_ssize_t temp, result = 0;
+ for (i = 0;; ++i) {
+ DREF DeeObject *item;
+ item = cswi_trygetitem_index(self, i);
+ if unlikely(!ITER_ISOK(item)) {
+ if unlikely(!item)
+ goto err;
+ break;
+ }
+ temp = (*cb)(arg, item);
+ Dee_Decref(item);
+ if unlikely(temp < 0)
+ goto err_temp;
+ result += temp;
+ }
+ return result;
+err_temp:
+ return temp;
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) size_t DCALL
+cswi_asvector(CachedSeq_WithIter *self, size_t dst_length, /*out*/ DREF DeeObject **dst) {
+ size_t result;
+ if unlikely(cswi_ensure_loaded(self, (size_t)-1) < 0)
+ goto err;
+ CachedSeq_WithIter_LockAcquire(self);
+ result = self->cswi_cache.ol_elemc;
+ if likely(dst_length >= result)
+ Dee_Movrefv(dst, self->cswi_cache.ol_elemv, result);
+ CachedSeq_WithIter_LockRelease(self);
+ return result;
+err:
+ return (size_t)-1;
+}
+
+
+
+PRIVATE struct type_seq cswi_seq = {
+ /* .tp_iter = */ (DREF DeeObject *(DCALL *)(DeeObject *__restrict))&cswi_iter,
+ /* .tp_sizeob = */ NULL,
+ /* .tp_contains = */ NULL,
+ /* .tp_getitem = */ NULL,
+ /* .tp_delitem = */ NULL,
+ /* .tp_setitem = */ NULL,
+ /* .tp_getrange = */ NULL,
+ /* .tp_delrange = */ NULL,
+ /* .tp_setrange = */ NULL,
+ /* .tp_nsi = */ NULL,
+ /* .tp_foreach = */ (Dee_ssize_t (DCALL *)(DeeObject *__restrict, Dee_foreach_t, void *))&cswi_foreach,
+ /* .tp_foreach_pair = */ NULL,
+ /* .tp_enumerate = */ NULL,
+ /* .tp_enumerate_index = */ NULL,
+ /* .tp_iterkeys = */ NULL,
+ /* .tp_bounditem = */ NULL,
+ /* .tp_hasitem = */ NULL,
+ /* .tp_size = */ (size_t (DCALL *)(DeeObject *__restrict))&cswi_size,
+ /* .tp_size_fast = */ (size_t (DCALL *)(DeeObject *__restrict))&cswi_size_fast,
+ /* .tp_getitem_index = */ (DREF DeeObject *(DCALL *)(DeeObject *, size_t))&cswi_getitem_index,
+ /* .tp_getitem_index_fast = */ NULL,
+ /* .tp_delitem_index = */ NULL,
+ /* .tp_setitem_index = */ NULL,
+ /* .tp_bounditem_index = */ (int (DCALL *)(DeeObject *, size_t))&cswi_bounditem_index,
+ /* .tp_hasitem_index = */ (int (DCALL *)(DeeObject *, size_t))&cswi_hasitem_index,
+ /* .tp_getrange_index = */ NULL,
+ /* .tp_delrange_index = */ NULL,
+ /* .tp_setrange_index = */ NULL,
+ /* .tp_getrange_index_n = */ NULL,
+ /* .tp_delrange_index_n = */ NULL,
+ /* .tp_setrange_index_n = */ NULL,
+ /* .tp_trygetitem = */ NULL,
+ /* .tp_trygetitem_index = */ (DREF DeeObject *(DCALL *)(DeeObject *, size_t))&cswi_trygetitem_index,
+ /* .tp_trygetitem_string_hash = */ NULL,
+ /* .tp_getitem_string_hash = */ NULL,
+ /* .tp_delitem_string_hash = */ NULL,
+ /* .tp_setitem_string_hash = */ NULL,
+ /* .tp_bounditem_string_hash = */ NULL,
+ /* .tp_hasitem_string_hash = */ NULL,
+ /* .tp_trygetitem_string_len_hash = */ NULL,
+ /* .tp_getitem_string_len_hash = */ NULL,
+ /* .tp_delitem_string_len_hash = */ NULL,
+ /* .tp_setitem_string_len_hash = */ NULL,
+ /* .tp_bounditem_string_len_hash = */ NULL,
+ /* .tp_hasitem_string_len_hash = */ NULL,
+ /* NOTE: "tp_asvector" explicitly states that "repeated" invocations mustn't produce any
+ * (additional) side-effects. As such, it *does* allow side-effects on the first
+ * invocation, meaning we're perfectly fine to implement it! */
+ /* .tp_asvector = */ (size_t (DCALL *)(DeeObject *, size_t, DREF DeeObject **))&cswi_asvector,
+};
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswi_bool(CachedSeq_WithIter *__restrict self) {
+ return cswi_hasitem_index(self, 0);
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF CachedSeq_WithIter *DCALL
+cswi_getfrozen(CachedSeq_WithIter *__restrict self) {
+ /* Once fully loaded, a cache can be considered as frozen! */
+ if unlikely(cswi_ensure_loaded(self, (size_t)-1) < 0)
+ goto err;
+ return_reference_(self);
+err:
+ return NULL;
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+cswi_getseq(CachedSeq_WithIter *__restrict self) {
+ return DeeObject_GetAttr(self->cswi_iter, (DeeObject *)&str_seq);
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+cswi_populate(CachedSeq_WithIter *__restrict self,
+ size_t argc, DeeObject *const *argv) {
+ size_t count = (size_t)-1;
+ if (DeeArg_Unpack(argc, argv, ":populate", &count))
+ goto err;
+ if likely(count) {
+ --count;
+ if unlikely(cswi_ensure_loaded(self, count) < 0)
+ goto err;
+ }
+ return_none;
+err:
+ return NULL;
+}
+
+
+PRIVATE struct type_method tpconst cswi_methods[] = {
+ TYPE_METHOD("populate", &cswi_populate,
+ "(count:?Dint=!ASIZE_MAX?Dint)\n"
+ "Ensure that the first @count elements of ?#__seq__ have been cached"),
+ TYPE_METHOD_END
+};
+
+PRIVATE struct type_getset tpconst cswi_getsets[] = {
+ TYPE_GETTER(STR_cached, &DeeObject_NewRef, "->?."),
+ TYPE_GETTER(STR_frozen, &cswi_getfrozen, "->?.\nFully populate the cache, then re-return it"),
+ TYPE_GETTER("__seq__", &cswi_getseq, "->?DSequence\nAlias for ${this.__iter__.seq}"),
+ TYPE_GETSET_END
+};
+
+PRIVATE struct type_member tpconst cswi_members[] = {
+ TYPE_MEMBER_FIELD_DOC("__iter__", STRUCT_OBJECT, offsetof(CachedSeq_WithIter, cswi_iter), "->?DIterator"),
+ TYPE_MEMBER_FIELD("__cache_size__", STRUCT_SIZE_T | STRUCT_CONST | STRUCT_ATOMIC,
+ offsetof(CachedSeq_WithIter, cswi_cache.ol_elemc)),
+ TYPE_MEMBER_END
+};
+
+PRIVATE struct type_member tpconst cswi_class_members[] = {
+ TYPE_MEMBER_CONST(STR_Iterator, &CachedSeq_WithIter_Iterator_Type),
+ TYPE_MEMBER_END
+};
+
+
+
+/* Uses an auto-growing vector for elements, that is fed by an iterator. */
+INTERN DeeTypeObject CachedSeq_WithIter_Type = {
+ OBJECT_HEAD_INIT(&DeeType_Type),
+ /* .tp_name = */ "_CachedSeqWithIter",
+ /* .tp_doc = */ DOC("()\n"
+ "(iter:?DIterator)"),
+ /* .tp_flags = */ TP_FNORMAL | TP_FFINAL | TP_FGC,
+ /* .tp_weakrefs = */ 0,
+ /* .tp_features = */ TF_NONE,
+ /* .tp_base = */ &DeeSeq_Type,
+ /* .tp_init = */ {
+ {
+ /* .tp_alloc = */ {
+ /* .tp_ctor = */ (dfunptr_t)&cswi_ctor,
+ /* .tp_copy_ctor = */ (dfunptr_t)&cswi_copy,
+ /* .tp_deep_ctor = */ (dfunptr_t)&cswi_deep,
+ /* .tp_any_ctor = */ (dfunptr_t)&cswi_init,
+ TYPE_FIXED_ALLOCATOR_GC(CachedSeq_WithIter)
+ }
+ },
+ /* .tp_dtor = */ (void (DCALL *)(DeeObject *__restrict))&cswi_fini,
+ /* .tp_assign = */ NULL,
+ /* .tp_move_assign = */ NULL,
+ /* .tp_deepload = */ (int (DCALL *)(DeeObject *__restrict))&cswi_deepload,
+ },
+ /* .tp_cast = */ {
+ /* .tp_str = */ NULL,
+ /* .tp_repr = */ NULL,
+ /* .tp_bool = */ (int (DCALL *)(DeeObject *__restrict))&cswi_bool
+ },
+ /* .tp_call = */ NULL,
+ /* .tp_visit = */ (void (DCALL *)(DeeObject *__restrict, dvisit_t, void *))&cswi_visit,
+ /* .tp_gc = */ &cswi_gc,
+ /* .tp_math = */ NULL,
+ /* .tp_cmp = */ NULL,
+ /* .tp_seq = */ &cswi_seq,
+ /* .tp_iter_next = */ NULL,
+ /* .tp_iterator = */ NULL,
+ /* .tp_attr = */ NULL,
+ /* .tp_with = */ NULL,
+ /* .tp_buffer = */ NULL,
+ /* .tp_methods = */ cswi_methods,
+ /* .tp_getsets = */ cswi_getsets,
+ /* .tp_members = */ cswi_members,
+ /* .tp_class_methods = */ NULL,
+ /* .tp_class_getsets = */ NULL,
+ /* .tp_class_members = */ cswi_class_members
+};
+
+
+STATIC_ASSERT(offsetof(CachedSeq_WithIter_Iterator, cswii_cache) == offsetof(ProxyObject, po_obj));
+#define cswiiter_fini generic_proxy_fini
+#define cswiiter_visit generic_proxy_visit
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswiiter_ctor(CachedSeq_WithIter_Iterator *__restrict self) {
+ self->cswii_cache = (DREF CachedSeq_WithIter *)DeeObject_NewDefault(&CachedSeq_WithIter_Type);
+ if unlikely(!self->cswii_cache)
+ goto err;
+ self->cswii_index = 0;
+ return 0;
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1, 2)) int DCALL
+cswiiter_copy(CachedSeq_WithIter_Iterator *__restrict self,
+ CachedSeq_WithIter_Iterator *__restrict other) {
+ self->cswii_index = atomic_read(&other->cswii_index);
+ return generic_proxy_copy_alias((ProxyObject *)self, (ProxyObject *)other);
+}
+
+PRIVATE WUNUSED NONNULL((1, 2)) int DCALL
+cswiiter_deep(CachedSeq_WithIter_Iterator *__restrict self,
+ CachedSeq_WithIter_Iterator *__restrict other) {
+ self->cswii_index = atomic_read(&other->cswii_index);
+ return generic_proxy_deepcopy((ProxyObject *)self, (ProxyObject *)other);
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswiiter_init(CachedSeq_WithIter_Iterator *__restrict self,
+ size_t argc, DeeObject *const *argv) {
+ self->cswii_index = 0;
+ if (DeeArg_Unpack(argc, argv, "o|" UNPuSIZ ":_CachedSeqWithIterIterator",
+ &self->cswii_cache, &self->cswii_index))
+ goto err;
+ if (DeeObject_AssertTypeExact(self->cswii_cache, &CachedSeq_WithIter_Type))
+ goto err;
+ Dee_Incref(self->cswii_cache);
+ return 0;
+err:
+ return -1;
+}
+
+PRIVATE WUNUSED NONNULL((1)) int DCALL
+cswiiter_bool(CachedSeq_WithIter_Iterator *__restrict self) {
+ size_t index = atomic_read(&self->cswii_index);
+ return cswi_hasitem_index(self->cswii_cache, index);
+}
+
+PRIVATE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+cswiiter_next(CachedSeq_WithIter_Iterator *__restrict self) {
+ DREF DeeObject *result;
+ for (;;) {
+ size_t index;
+ index = atomic_read(&self->cswii_index);
+ result = cswi_trygetitem_index(self->cswii_cache, index);
+ if unlikely(!ITER_ISOK(result))
+ break;
+ if likely(atomic_cmpxch_or_write(&self->cswii_index, index, index + 1))
+ break;
+ Dee_Decref(result);
+ }
+ return result;
+}
+
+PRIVATE WUNUSED NONNULL((1)) size_t DCALL
+cswiiter_advance(CachedSeq_WithIter_Iterator *__restrict self, size_t step) {
+ size_t old_index, new_index;
+ do {
+ old_index = atomic_read(&self->cswii_index);
+ if (OVERFLOW_UADD(old_index, step, &new_index))
+ new_index = (size_t)-1;
+ if likely(new_index > 0) {
+ int load_status;
+ load_status = cswi_ensure_loaded(self->cswii_cache, new_index - 1);
+ if unlikely(load_status < 0)
+ goto err;
+ if (load_status > 0) {
+ /* Clap "new_index" to the max possible value */
+ CachedSeq_WithIter_LockAcquire(self->cswii_cache);
+ new_index = self->cswii_cache->cswi_cache.ol_elemc;
+ CachedSeq_WithIter_LockRelease(self->cswii_cache);
+ }
+ }
+ } while (!atomic_cmpxch_or_write(&self->cswii_index, old_index, new_index));
+ return new_index - old_index;
+err:
+ return (size_t)-1;
+}
+
+PRIVATE struct type_iterator cswiiter_iterator = {
+ /* .tp_nextpair = */ NULL,
+ /* .tp_nextkey = */ NULL,
+ /* .tp_nextvalue = */ NULL,
+ /* .tp_advance = */ (size_t (DCALL *)(DeeObject *__restrict, size_t))&cswiiter_advance,
+};
+
+PRIVATE struct type_member tpconst cswiiter_members[] = {
+ TYPE_MEMBER_FIELD_DOC(STR_seq, STRUCT_OBJECT,
+ offsetof(CachedSeq_WithIter_Iterator, cswii_cache),
+ "->?Ert:CachedSeqWithIter"),
+ TYPE_MEMBER_FIELD("__index__", STRUCT_SIZE_T | STRUCT_ATOMIC,
+ offsetof(CachedSeq_WithIter_Iterator, cswii_index)),
+ TYPE_MEMBER_END
+};
+
+INTERN DeeTypeObject CachedSeq_WithIter_Iterator_Type = {
+ OBJECT_HEAD_INIT(&DeeType_Type),
+ /* .tp_name = */ "_CachedSeqWithIterIterator",
+ /* .tp_doc = */ DOC("()\n"
+ "(base:?Ert:CachedSeqWithIter,index=!0)"),
+ /* .tp_flags = */ TP_FNORMAL | TP_FFINAL,
+ /* .tp_weakrefs = */ 0,
+ /* .tp_features = */ TF_NONE,
+ /* .tp_base = */ &DeeIterator_Type,
+ /* .tp_init = */ {
+ {
+ /* .tp_alloc = */ {
+ /* .tp_ctor = */ (dfunptr_t)&cswiiter_ctor,
+ /* .tp_copy_ctor = */ (dfunptr_t)&cswiiter_copy,
+ /* .tp_deep_ctor = */ (dfunptr_t)&cswiiter_deep,
+ /* .tp_any_ctor = */ (dfunptr_t)&cswiiter_init,
+ TYPE_FIXED_ALLOCATOR(CachedSeq_WithIter_Iterator)
+ }
+ },
+ /* .tp_dtor = */ (void (DCALL *)(DeeObject *__restrict))&cswiiter_fini,
+ /* .tp_assign = */ NULL,
+ /* .tp_move_assign = */ NULL
+ },
+ /* .tp_cast = */ {
+ /* .tp_str = */ NULL,
+ /* .tp_repr = */ NULL,
+ /* .tp_bool = */ (int (DCALL *)(DeeObject *__restrict))&cswiiter_bool
+ },
+ /* .tp_call = */ NULL,
+ /* .tp_visit = */ (void (DCALL *)(DeeObject *__restrict, dvisit_t, void *))&cswiiter_visit,
+ /* .tp_gc = */ NULL,
+ /* .tp_math = */ NULL,
+ /* .tp_cmp = */ NULL,
+ /* .tp_seq = */ NULL,
+ /* .tp_iter_next = */ (DREF DeeObject *(DCALL *)(DeeObject *__restrict))&cswiiter_next,
+ /* .tp_iterator = */ &cswiiter_iterator,
+ /* .tp_attr = */ NULL,
+ /* .tp_with = */ NULL,
+ /* .tp_buffer = */ NULL,
+ /* .tp_methods = */ NULL,
+ /* .tp_getsets = */ NULL,
+ /* .tp_members = */ cswiiter_members,
+ /* .tp_class_methods = */ NULL,
+ /* .tp_class_getsets = */ NULL,
+ /* .tp_class_members = */ NULL
+};
+
+DECL_END
+
+#endif /* !GUARD_DEEMON_OBJECTS_CACHED_SEQ_C */
diff --git a/src/deemon/objects/seq/cached-seq.h b/src/deemon/objects/seq/cached-seq.h
new file mode 100644
index 000000000..58036db26
--- /dev/null
+++ b/src/deemon/objects/seq/cached-seq.h
@@ -0,0 +1,119 @@
+/* Copyright (c) 2018-2024 Griefer@Work *
+ * *
+ * This software is provided 'as-is', without any express or implied *
+ * warranty. In no event will the authors be held liable for any damages *
+ * arising from the use of this software. *
+ * *
+ * Permission is granted to anyone to use this software for any purpose, *
+ * including commercial applications, and to alter it and redistribute it *
+ * freely, subject to the following restrictions: *
+ * *
+ * 1. The origin of this software must not be misrepresented; you must not *
+ * claim that you wrote the original software. If you use this software *
+ * in a product, an acknowledgement (see the following) in the product *
+ * documentation is required: *
+ * Portions Copyright (c) 2018-2024 Griefer@Work *
+ * 2. Altered source versions must be plainly marked as such, and must not be *
+ * misrepresented as being the original software. *
+ * 3. This notice may not be removed or altered from any source distribution. *
+ */
+#ifndef GUARD_DEEMON_OBJECTS_CACHED_SEQ_H
+#define GUARD_DEEMON_OBJECTS_CACHED_SEQ_H 1
+
+#include
+#include
+#include
+#include
+#include
+/**/
+
+#include "../generic-proxy.h"
+
+DECL_BEGIN
+
+/************************************************************************/
+/* ITERATOR-BASED CACHE */
+/************************************************************************/
+typedef struct {
+ PROXY_OBJECT_HEAD(cswi_iter) /* [1..1][const] The iterator whose results are being cached. */
+#ifndef CONFIG_NO_THREADS
+ Dee_atomic_lock_t cswi_lock; /* The lock used to synchronize the cache below. */
+#endif /* !CONFIG_NO_THREADS */
+ struct objectlist cswi_cache; /* Cache of results returned by `cswi_iter' */
+#ifndef DEE_OBJECTLIST_HAVE_ELEMA
+#define CachedSeq_WithIter_HAVE_cswi_finished
+ bool cswi_finished; /* Set to true once `cswi_iter' has been exhausted */
+#endif /* !DEE_OBJECTLIST_HAVE_ELEMA */
+} CachedSeq_WithIter;
+
+/* Uses an auto-growing vector for elements, that is fed by an iterator. */
+INTDEF DeeTypeObject CachedSeq_WithIter_Type;
+
+#ifdef CachedSeq_WithIter_HAVE_cswi_finished
+#define CachedSeq_WithIter_InitFinished(self) (void)((self)->cswi_finished = false)
+#define CachedSeq_WithIter_GetFinished(self) ((self)->cswi_finished)
+#define CachedSeq_WithIter_SetFinished(self) (void)((self)->cswi_finished = true)
+#else /* CachedSeq_WithIter_HAVE_cswi_finished */
+#define CachedSeq_WithIter_InitFinished(self) (void)0
+#define CachedSeq_WithIter_GetFinished(self) ((self)->cswi_cache.ol_elema == (size_t)-1)
+#define CachedSeq_WithIter_SetFinished(self) (void)((self)->cswi_cache.ol_elema = (size_t)-1)
+#endif /* !CachedSeq_WithIter_HAVE_cswi_finished */
+
+#define CachedSeq_WithIter_LockAvailable(self) Dee_atomic_lock_available(&(self)->cswi_lock)
+#define CachedSeq_WithIter_LockAcquired(self) Dee_atomic_lock_acquired(&(self)->cswi_lock)
+#define CachedSeq_WithIter_LockTryAcquire(self) Dee_atomic_lock_tryacquire(&(self)->cswi_lock)
+#define CachedSeq_WithIter_LockAcquire(self) Dee_atomic_lock_acquire(&(self)->cswi_lock)
+#define CachedSeq_WithIter_LockWaitFor(self) Dee_atomic_lock_waitfor(&(self)->cswi_lock)
+#define CachedSeq_WithIter_LockRelease(self) Dee_atomic_lock_release(&(self)->cswi_lock)
+
+
+
+typedef struct {
+ PROXY_OBJECT_HEAD_EX(CachedSeq_WithIter, cswii_cache) /* [1..1][const] The cache that is being iterated. */
+ size_t cswii_index; /* [lock(ATOMIC)] index of next element to yield. */
+} CachedSeq_WithIter_Iterator;
+
+INTDEF DeeTypeObject CachedSeq_WithIter_Iterator_Type;
+
+LOCAL WUNUSED NONNULL((1)) DREF CachedSeq_WithIter *DCALL
+CachedSeq_WithIter_New(/*inherit(always)*/ DREF DeeObject *iter) {
+ DREF CachedSeq_WithIter *result;
+ result = DeeGCObject_MALLOC(CachedSeq_WithIter);
+ if unlikely(!result)
+ goto err;
+ result->cswi_iter = iter; /* Inherit reference */
+ Dee_atomic_lock_init(&result->cswi_lock);
+ objectlist_init(&result->cswi_cache);
+ CachedSeq_WithIter_InitFinished(result);
+ DeeObject_Init(result, &CachedSeq_WithIter_Type);
+ return (DREF CachedSeq_WithIter *)DeeGC_Track((DREF DeeObject *)result);
+err:
+ Dee_Decref_likely(iter);
+ return NULL;
+}
+
+
+
+
+
+
+
+/************************************************************************/
+/* INDEX-BASED CACHE */
+/************************************************************************/
+typedef struct {
+ PROXY_OBJECT_HEAD(cswgi_seq) /* [1..1][const] The sequence being cached. */
+ /* TODO */
+} CachedSeq_WithGetItem;
+
+INTDEF DeeTypeObject CachedSeq_WithGetItem_Type; /* Uses a lazily-allocated vector for small integers, and a dict for large ones, or non-integer keys */
+INTDEF DeeTypeObject CachedSeq_WithSizeObAndGetItem_Type; /* Like `CachedSeq_WithGetItem_Type', but also uses+caches `DeeSeq_OperatorSizeOb' */
+INTDEF DeeTypeObject CachedSeq_WithSizeAndGetItemIndex_Type; /* Like `CachedSeq_WithSizeObAndGetItem_Type', but uses+caches `DeeSeq_OperatorSize'. */
+
+INTDEF DeeTypeObject CachedSeq_WithGetItem_Iterator_Type;
+INTDEF DeeTypeObject CachedSeq_WithSizeObAndGetItem_Iterator_Type;
+INTDEF DeeTypeObject CachedSeq_WithSizeAndGetItemIndex_Iterator_Type;
+
+DECL_END
+
+#endif /* !GUARD_DEEMON_OBJECTS_CACHED_SEQ_H */
diff --git a/src/deemon/objects/seq/default-api-methods-attrproxy-impl.c.inl b/src/deemon/objects/seq/default-api-methods-attrproxy-impl.c.inl
index 6507f070c..15f8097e1 100644
--- a/src/deemon/objects/seq/default-api-methods-attrproxy-impl.c.inl
+++ b/src/deemon/objects/seq/default-api-methods-attrproxy-impl.c.inl
@@ -65,6 +65,7 @@ DECL_BEGIN
#define LOCAL_DeeSeq_DefaultBoundLastWithCallAttrGetLast LOCAL_DeeSeq_DefaultFooWithCallAttrBar(BoundLast, GetLast)
#define LOCAL_DeeSeq_DefaultDelLastWithCallAttrDelLast LOCAL_DeeSeq_DefaultFooWithCallAttrBar(DelLast, DelLast)
#define LOCAL_DeeSeq_DefaultSetLastWithCallAttrSetLast LOCAL_DeeSeq_DefaultFooWithCallAttrBar(SetLast, SetLast)
+#define LOCAL_DeeSeq_DefaultCachedWithCallAttrCached LOCAL_DeeSeq_DefaultFooWithCallAttrBar(Cached, Cached)
#define LOCAL_DeeSeq_DefaultAnyWithCallAttrAny LOCAL_DeeSeq_DefaultFooWithCallAttrBar(Any, Any)
#define LOCAL_DeeSeq_DefaultAnyWithKeyWithCallAttrAnyForSeq LOCAL_DeeSeq_DefaultFooWithCallAttrBar_(AnyWithKey, Any, ForSeq)
#define LOCAL_DeeSeq_DefaultAnyWithKeyWithCallAttrAnyForSetOrMap LOCAL_DeeSeq_DefaultFooWithCallAttrBar_(AnyWithKey, Any, ForSetOrMap)
@@ -1560,6 +1561,11 @@ LOCAL_DeeSeq_DefaultSetLastWithCallAttrSetLast(DeeObject *self, DeeObject *value
return LOCAL_DeeObject_SetAttr(self, tsc_seq_dellast_data, &str_last, value);
}
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+LOCAL_DeeSeq_DefaultCachedWithCallAttrCached(DeeObject *__restrict self) {
+ return LOCAL_DeeObject_GetAttr(self, tsc_seq_cached_data, &str_cached);
+}
+
INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
LOCAL_DeeMap_DefaultKeysWithCallAttrKeys(DeeObject *self) {
return LOCAL_DeeObject_GetAttr(self, tsc_map_keys_data, &str_keys);
@@ -1596,6 +1602,7 @@ LOCAL_DeeMap_DefaultIterValuesWithCallAttrIterValues(DeeObject *self) {
#undef LOCAL_DeeSeq_DefaultBoundLastWithCallAttrBoundLast
#undef LOCAL_DeeSeq_DefaultDelLastWithCallAttrDelLast
#undef LOCAL_DeeSeq_DefaultSetLastWithCallAttrSetLast
+#undef LOCAL_DeeSeq_DefaultCachedWithCallAttrCached
#undef LOCAL_DeeSeq_DefaultAnyWithCallAttrAny
#undef LOCAL_DeeSeq_DefaultAnyWithKeyWithCallAttrAnyForSeq
#undef LOCAL_DeeSeq_DefaultAnyWithKeyWithCallAttrAnyForSetOrMap
diff --git a/src/deemon/objects/seq/default-api-methods-last.c.inl b/src/deemon/objects/seq/default-api-methods-last.c.inl
index 6e96a3314..fde765554 100644
--- a/src/deemon/objects/seq/default-api-methods-last.c.inl
+++ b/src/deemon/objects/seq/default-api-methods-last.c.inl
@@ -386,4 +386,48 @@ DeeSeq_DefaultSetLastWithError(DeeObject *self, DeeObject *value) {
return err_seq_unsupportedf(self, "last = %r", value);
}
+
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+DeeSeq_DefaultCachedWithSeqIter(DeeObject *__restrict self) {
+ DREF CachedSeq_WithIter *result;
+ DREF DeeObject *iter;
+ iter = DeeSeq_OperatorIter(self);
+ if unlikely(!iter)
+ goto err;
+ result = CachedSeq_WithIter_New(iter);
+ return (DREF DeeObject *)result;
+err:
+ return NULL;
+}
+
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+DeeSeq_DefaultCachedWithSeqGetItem(DeeObject *__restrict self) {
+ /* TODO */
+ (void)self;
+ DeeError_NOTIMPLEMENTED();
+ return NULL;
+}
+
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+DeeSeq_DefaultCachedWithSeqSizeObAndSeqGetItem(DeeObject *__restrict self) {
+ /* TODO */
+ (void)self;
+ DeeError_NOTIMPLEMENTED();
+ return NULL;
+}
+
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+DeeSeq_DefaultCachedWithSeqSizeAndSeqGetItemIndex(DeeObject *__restrict self) {
+ /* TODO */
+ (void)self;
+ DeeError_NOTIMPLEMENTED();
+ return NULL;
+}
+
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *DCALL
+DeeSeq_DefaultCachedWithError(DeeObject *__restrict self) {
+ err_seq_unsupportedf(self, "cached");
+ return NULL;
+}
+
DECL_END
diff --git a/src/deemon/objects/seq/default-api-methods.c b/src/deemon/objects/seq/default-api-methods.c
index d4b856753..169dec969 100644
--- a/src/deemon/objects/seq/default-api-methods.c
+++ b/src/deemon/objects/seq/default-api-methods.c
@@ -47,6 +47,7 @@
#include
/**/
+#include "cached-seq.h"
#include "default-iterators.h"
#include "default-map-proxy.h"
#include "default-reversed.h"
@@ -6206,6 +6207,7 @@ print define_Dee_HashStr("end");
* to inject the most optimized version of getsets into top-level objects:
* - Sequence.first
* - Sequence.last
+ * - Sequence.cached
* - Mapping.keys
* - Mapping.values
* - Mapping.iterkeys
@@ -6214,6 +6216,7 @@ print define_Dee_HashStr("end");
#ifdef __OPTIMIZE_SIZE__
#define maybe_cache_optimized_seq_first_in_membercache(self) (void)0
#define maybe_cache_optimized_seq_last_in_membercache(self) (void)0
+#define maybe_cache_optimized_seq_cached_in_membercache(self) (void)0
#define maybe_cache_optimized_map_keys_in_membercache(self) (void)0
#define maybe_cache_optimized_map_values_in_membercache(self) (void)0
#define maybe_cache_optimized_map_iterkeys_in_membercache(self) (void)0
@@ -6249,6 +6252,20 @@ maybe_cache_optimized_seq_last_in_membercache(DeeTypeObject *__restrict self) {
&new_getset, &gs_default_seq_last);
}
+PRIVATE struct type_getset tpconst gs_default_seq_cached =
+TYPE_GETTER_NODOC(NULL, &default_seq_cached);
+PRIVATE NONNULL((1)) void DCALL
+maybe_cache_optimized_seq_cached_in_membercache(DeeTypeObject *__restrict self) {
+ struct type_getset new_getset;
+ new_getset.gs_name = STR_cached;
+ new_getset.gs_get = DeeType_RequireSeqCached(self);
+ new_getset.gs_del = NULL;
+ new_getset.gs_set = NULL;
+ new_getset.gs_bound = NULL;
+ DeeTypeMRO_PatchGetSet(self, &DeeSeq_Type, Dee_HashStr__cached,
+ &new_getset, &gs_default_seq_cached);
+}
+
PRIVATE struct type_getset tpconst gs_default_map_keys =
TYPE_GETTER_NODOC(NULL, &default_map_keys);
PRIVATE NONNULL((1)) void DCALL
@@ -6354,6 +6371,12 @@ default_seq_setlast(DeeObject *self, DeeObject *value) {
return DeeSeq_InvokeSetLast(self, value);
}
+INTERN WUNUSED NONNULL((1)) DREF DeeObject *
+(DCALL default_seq_cached)(DeeObject *__restrict self) {
+ maybe_cache_optimized_seq_cached_in_membercache(Dee_TYPE(self));
+ return DeeSeq_InvokeCached(self);
+}
+
PRIVATE ATTR_NOINLINE WUNUSED NONNULL((1)) DREF DeeObject *DCALL
do_seq_enumerate_with_kw(DeeObject *self, size_t argc,
DeeObject *const *argv, DeeObject *kw) {
diff --git a/src/deemon/objects/seq/default-api-require-method-impl.c.inl b/src/deemon/objects/seq/default-api-require-method-impl.c.inl
index 1a33ebd31..6f2260d02 100644
--- a/src/deemon/objects/seq/default-api-require-method-impl.c.inl
+++ b/src/deemon/objects/seq/default-api-require-method-impl.c.inl
@@ -29,6 +29,7 @@
//#define DEFINE_DeeType_RequireSeqBoundLast
//#define DEFINE_DeeType_RequireSeqDelLast
//#define DEFINE_DeeType_RequireSetLast
+#define DEFINE_DeeType_RequireCached
//#define DEFINE_DeeType_RequireSeqAny
//#define DEFINE_DeeType_RequireSeqAnyWithKey
//#define DEFINE_DeeType_RequireSeqAnyWithRange
@@ -84,7 +85,7 @@
//#define DEFINE_DeeType_RequireSeqPushFront
//#define DEFINE_DeeType_RequireSeqAppend
//#define DEFINE_DeeType_RequireSeqExtend
-#define DEFINE_DeeType_RequireSeqXchItemIndex
+//#define DEFINE_DeeType_RequireSeqXchItemIndex
//#define DEFINE_DeeType_RequireSeqClear
//#define DEFINE_DeeType_RequireSeqPop
//#define DEFINE_DeeType_RequireSeqRemove
@@ -142,6 +143,7 @@
defined(DEFINE_DeeType_RequireSeqBoundLast) + \
defined(DEFINE_DeeType_RequireSeqDelLast) + \
defined(DEFINE_DeeType_RequireSetLast) + \
+ defined(DEFINE_DeeType_RequireCached) + \
defined(DEFINE_DeeType_RequireSeqAny) + \
defined(DEFINE_DeeType_RequireSeqAnyWithKey) + \
defined(DEFINE_DeeType_RequireSeqAnyWithRange) + \
@@ -320,6 +322,13 @@ DECL_BEGIN
#define LOCAL_IS_GETSET_SET
#define LOCAL_default_seq_foo default_seq_setlast
#define LOCAL_NO_TMH
+#elif defined(DEFINE_DeeType_RequireCached)
+#define LOCAL_foo cached
+#define LOCAL_Foo Cached
+#define LOCAL_IS_GETSET_GET
+#define LOCAL_default_seq_foo default_seq_cached
+#define LOCAL_NO_TMH
+#define LOCAL_ATTR_REQUIRED_SEQCLASS Dee_SEQCLASS_SEQ
#elif defined(DEFINE_DeeType_RequireSeqAny)
#define LOCAL_foo any
#define LOCAL_Foo Any
@@ -1337,6 +1346,28 @@ LOCAL_DeeType_RequireSeqFoo_private_uncached(DeeTypeObject *orig_type, DeeTypeOb
}
}
}
+#elif defined(DEFINE_DeeType_RequireCached)
+ if (seqclass == Dee_SEQCLASS_SEQ) {
+ if (DeeType_HasPrivateOperator_in(orig_type, self, OPERATOR_GETITEM)) {
+ ASSERT(self->tp_seq);
+ ASSERT(self->tp_seq->tp_getitem);
+ ASSERT(self->tp_seq->tp_getitem_index);
+ if (self->tp_seq->tp_size && !DeeType_IsDefaultSize(self->tp_seq->tp_size))
+ return &DeeSeq_DefaultCachedWithSeqSizeAndSeqGetItemIndex;
+ if (self->tp_seq->tp_sizeob && !DeeType_IsDefaultSizeOb(self->tp_seq->tp_sizeob))
+ return &DeeSeq_DefaultCachedWithSeqSizeObAndSeqGetItem;
+ if ((self->tp_seq->tp_getitem && !DeeType_IsDefaultGetItem(self->tp_seq->tp_getitem)) ||
+ (self->tp_seq->tp_getitem_index && !DeeType_IsDefaultGetItemIndex(self->tp_seq->tp_getitem_index)))
+ return &DeeSeq_DefaultCachedWithSeqGetItem; /* custom getitem. */
+ if (DeeType_HasPrivateOperator_in(orig_type, self, OPERATOR_ITER))
+ return &DeeSeq_DefaultCachedWithSeqIter; /* custom iter/foreach/etc. */
+ if (DeeType_HasPrivateOperator_in(orig_type, self, OPERATOR_SIZE))
+ return &DeeSeq_DefaultCachedWithSeqSizeObAndSeqGetItem;
+ return &DeeSeq_DefaultCachedWithSeqGetItem;
+ }
+ }
+ if (DeeType_HasPrivateOperator_in(orig_type, self, OPERATOR_ITER))
+ return &DeeSeq_DefaultCachedWithSeqIter;
#elif defined(DEFINE_DeeType_RequireSeqAny)
/* ... */
#elif defined(DEFINE_DeeType_RequireSeqAnyWithKey)
@@ -2508,6 +2539,7 @@ DECL_END
#undef DEFINE_DeeType_RequireSeqBoundLast
#undef DEFINE_DeeType_RequireSeqDelLast
#undef DEFINE_DeeType_RequireSetLast
+#undef DEFINE_DeeType_RequireCached
#undef DEFINE_DeeType_RequireSeqAny
#undef DEFINE_DeeType_RequireSeqAnyWithKey
#undef DEFINE_DeeType_RequireSeqAnyWithRange
diff --git a/src/deemon/objects/seq/default-api.c b/src/deemon/objects/seq/default-api.c
index 29a5f470d..41b9a2180 100644
--- a/src/deemon/objects/seq/default-api.c
+++ b/src/deemon/objects/seq/default-api.c
@@ -394,6 +394,8 @@ Dee_type_seq_cache_destroy(struct Dee_type_seq_cache *__restrict self) {
Dee_tsc_uslot_fini_function(&self->tsc_seq_dellast_data);
if (self->tsc_seq_setlast == &DeeSeq_DefaultSetLastWithCallSetLastDataFunction)
Dee_tsc_uslot_fini_function(&self->tsc_seq_setlast_data);
+ if (self->tsc_seq_cached == &DeeSeq_DefaultCachedWithCallCachedDataFunction)
+ Dee_tsc_uslot_fini_function(&self->tsc_seq_cached_data);
if (self->tsc_seq_any == &DeeSeq_DefaultAnyWithCallAnyDataFunction ||
self->tsc_seq_any_with_key == &DeeSeq_DefaultAnyWithKeyWithCallAnyDataFunctionForSeq ||
self->tsc_seq_any_with_key == &DeeSeq_DefaultAnyWithKeyWithCallAnyDataFunctionForSetOrMap ||
@@ -1117,6 +1119,8 @@ DECL_END
#include "default-api-require-method-impl.c.inl"
#define DEFINE_DeeType_RequireSetLast
#include "default-api-require-method-impl.c.inl"
+#define DEFINE_DeeType_RequireCached
+#include "default-api-require-method-impl.c.inl"
#define DEFINE_DeeType_RequireSeqAny
#include "default-api-require-method-impl.c.inl"
#define DEFINE_DeeType_RequireSeqAnyWithKey
diff --git a/src/deemon/objects/seq/default-api.h b/src/deemon/objects/seq/default-api.h
index caa9bdbe2..6fdd60385 100644
--- a/src/deemon/objects/seq/default-api.h
+++ b/src/deemon/objects/seq/default-api.h
@@ -68,6 +68,9 @@ typedef WUNUSED_T NONNULL_T((1)) int (DCALL *Dee_mh_seq_boundlast_t)(DeeObject *
typedef WUNUSED_T NONNULL_T((1)) int (DCALL *Dee_mh_seq_dellast_t)(DeeObject *__restrict self);
typedef WUNUSED_T NONNULL_T((1, 2)) int (DCALL *Dee_mh_seq_setlast_t)(DeeObject *self, DeeObject *value);
+/* Returns an auto-caching variant of the sequence (s.a. "cached-seq.h") */
+typedef WUNUSED_T NONNULL_T((1)) DREF DeeObject *(DCALL *Dee_mh_seq_cached_t)(DeeObject *__restrict self);
+
typedef WUNUSED_T NONNULL_T((1)) DREF DeeObject *(DCALL *Dee_mh_map_keys_t)(DeeObject *__restrict self);
typedef WUNUSED_T NONNULL_T((1)) DREF DeeObject *(DCALL *Dee_mh_map_values_t)(DeeObject *__restrict self);
typedef WUNUSED_T NONNULL_T((1)) DREF DeeObject *(DCALL *Dee_mh_map_iterkeys_t)(DeeObject *__restrict self);
@@ -96,6 +99,10 @@ struct Dee_type_seq_cache {
Dee_mh_seq_dellast_t tsc_seq_dellast;
Dee_mh_seq_setlast_t tsc_seq_setlast;
+ /* Extra callbacks for Sequence.cached */
+ Dee_mh_seq_cached_t tsc_seq_cached;
+
+ /* Extra callbacks for Mapping.keys/values/iterkeys/itervalues */
Dee_mh_map_keys_t tsc_map_keys;
Dee_mh_map_values_t tsc_map_values;
Dee_mh_map_iterkeys_t tsc_map_iterkeys;
@@ -107,6 +114,7 @@ struct Dee_type_seq_cache {
union Dee_tsc_uslot tsc_seq_getlast_data;
union Dee_tsc_uslot tsc_seq_dellast_data;
union Dee_tsc_uslot tsc_seq_setlast_data;
+ union Dee_tsc_uslot tsc_seq_cached_data;
union Dee_tsc_uslot tsc_seq_any_data;
union Dee_tsc_uslot tsc_seq_all_data;
union Dee_tsc_uslot tsc_seq_parity_data;
@@ -188,6 +196,7 @@ INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_seq_getlast_t DCALL
INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_seq_boundlast_t DCALL DeeType_RequireSeqBoundLast(DeeTypeObject *__restrict self);
INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_seq_dellast_t DCALL DeeType_RequireSeqDelLast(DeeTypeObject *__restrict self);
INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_seq_setlast_t DCALL DeeType_RequireSeqSetLast(DeeTypeObject *__restrict self);
+INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_seq_cached_t DCALL DeeType_RequireSeqCached(DeeTypeObject *__restrict self);
INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_map_keys_t DCALL DeeType_RequireMapKeys(DeeTypeObject *__restrict self);
INTDEF ATTR_PURE ATTR_RETNONNULL WUNUSED NONNULL((1)) Dee_mh_map_values_t DCALL DeeType_RequireMapValues(DeeTypeObject *__restrict self);
@@ -687,6 +696,7 @@ INTDEF struct type_math DeeMap_OperatorMath;
#define DeeSeq_InvokeBoundLast(self) (*DeeType_RequireSeqBoundLast(Dee_TYPE(self)))(self)
#define DeeSeq_InvokeDelLast(self) (*DeeType_RequireSeqDelLast(Dee_TYPE(self)))(self)
#define DeeSeq_InvokeSetLast(self, v) (*DeeType_RequireSeqSetLast(Dee_TYPE(self)))(self, v)
+#define DeeSeq_InvokeCached(self) (*DeeType_RequireSeqCached(Dee_TYPE(self)))(self)
#define DeeSeq_InvokeAny(self) (*DeeType_RequireSeqAny(Dee_TYPE(self)))(self)
#define DeeSeq_InvokeAnyWithKey(self, key) (*DeeType_RequireSeqAnyWithKey(Dee_TYPE(self)))(self, key)
#define DeeSeq_InvokeAnyWithRange(self, start, end) (*DeeType_RequireSeqAnyWithRange(Dee_TYPE(self)))(self, start, end)
@@ -1299,6 +1309,15 @@ INTDEF WUNUSED NONNULL((1, 2)) int DCALL DeeSeq_DefaultSetLastWithSizeAndSetItem
INTDEF WUNUSED NONNULL((1, 2)) int DCALL DeeSeq_DefaultSetLastWithSizeAndSetItem(DeeObject *self, DeeObject *value);
INTDEF WUNUSED NONNULL((1, 2)) int DCALL DeeSeq_DefaultSetLastWithError(DeeObject *self, DeeObject *value);
+/* Implementations for `Sequence.cached' */
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithCallAttrCached(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithCallCachedDataFunction(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithSeqIter(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithSeqGetItem(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithSeqSizeObAndSeqGetItem(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithSeqSizeAndSeqGetItemIndex(DeeObject *__restrict self);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL DeeSeq_DefaultCachedWithError(DeeObject *__restrict self);
+
/* Functions that need additional variants for sequence sub-types that don't have indices (sets, maps) */
INTDEF WUNUSED NONNULL((1)) int DCALL DeeSeq_DefaultAnyWithCallAttrAny(DeeObject *self);
@@ -2081,10 +2100,12 @@ INTDEF WUNUSED NONNULL((1)) DREF DeeObject *DCALL default_seq_getlast(DeeObject
INTDEF WUNUSED NONNULL((1)) int DCALL default_seq_boundlast(DeeObject *__restrict self);
INTDEF WUNUSED NONNULL((1)) int DCALL default_seq_dellast(DeeObject *__restrict self);
INTDEF WUNUSED NONNULL((1, 2)) int DCALL default_seq_setlast(DeeObject *self, DeeObject *value);
+INTDEF WUNUSED NONNULL((1)) DREF DeeObject *(DCALL default_seq_cached)(DeeObject *__restrict self);
INTDEF WUNUSED NONNULL((1)) DREF DeeObject *(DCALL default_map_keys)(DeeObject *self);
INTDEF WUNUSED NONNULL((1)) DREF DeeObject *(DCALL default_map_values)(DeeObject *self);
INTDEF WUNUSED NONNULL((1)) DREF DeeObject *(DCALL default_map_iterkeys)(DeeObject *self);
INTDEF WUNUSED NONNULL((1)) DREF DeeObject *(DCALL default_map_itervalues)(DeeObject *self);
+#define default_seq_cached(self) DeeSeq_InvokeCached(self)
#define default_map_keys(self) DeeMap_InvokeKeys(self)
#define default_map_values(self) DeeMap_InvokeValues(self)
#define default_map_iterkeys(self) DeeMap_InvokeIterKeys(self)
diff --git a/src/deemon/objects/seq/flat.c b/src/deemon/objects/seq/flat.c
index e7462d8f8..63cce47e7 100644
--- a/src/deemon/objects/seq/flat.c
+++ b/src/deemon/objects/seq/flat.c
@@ -840,7 +840,7 @@ INTERN DeeTypeObject SeqFlat_Type = {
OBJECT_HEAD_INIT(&DeeType_Type),
/* .tp_name = */ "_SeqFlat",
/* .tp_doc = */ DOC("()\n"
- "(seq:??S?S?O)"),
+ "(seq:?S?S?O)"),
/* .tp_flags = */ TP_FNORMAL | TP_FFINAL,
/* .tp_weakrefs = */ 0,
/* .tp_features = */ TF_NONE,
diff --git a/src/deemon/objects/tuple.c b/src/deemon/objects/tuple.c
index fcc61573d..ebab6522e 100644
--- a/src/deemon/objects/tuple.c
+++ b/src/deemon/objects/tuple.c
@@ -1855,6 +1855,7 @@ PRIVATE struct type_getset tpconst tuple_getsets[] = {
TYPE_GETTER_F_NODOC(STR_last, &tuple_last, METHOD_FNOREFESCAPE | METHOD_FCONSTCALL),
#define nullable_tuple_getsets (tuple_getsets + 2)
TYPE_GETTER_F(STR_frozen, &DeeObject_NewRef, METHOD_FCONSTCALL, "->?."),
+ TYPE_GETTER_F(STR_cached, &DeeObject_NewRef, METHOD_FCONSTCALL, "->?."),
TYPE_GETTER_F("__sizeof__", &tuple_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/src/deemon/runtime/strings.h b/src/deemon/runtime/strings.h
index e66d7f21b..ca4f82433 100644
--- a/src/deemon/runtime/strings.h
+++ b/src/deemon/runtime/strings.h
@@ -188,6 +188,7 @@ local STRINGS = {
"__module__",
"first",
"last",
+ "cached",
"size",
"filename",
@@ -685,6 +686,9 @@ DEF_STRING(str_first, "first", 0xa9f0e818, 0x9d12a485470a29a7)
#define Dee_HashStr__last _Dee_HashSelectC(0x185a4f9a, 0x760894ca6d41e4dc)
DEF_STRING(str_last, "last", 0x185a4f9a, 0x760894ca6d41e4dc)
#define STR_last DeeString_STR(&str_last)
+#define Dee_HashStr__cached _Dee_HashSelectC(0x915e175e, 0xddfd408a14eae4b4)
+DEF_STRING(str_cached, "cached", 0x915e175e, 0xddfd408a14eae4b4)
+#define STR_cached DeeString_STR(&str_cached)
#define Dee_HashStr__size _Dee_HashSelectC(0xed8917fa, 0x3fe8023bdf261c0f)
DEF_STRING(str_size, "size", 0xed8917fa, 0x3fe8023bdf261c0f)
#define STR_size DeeString_STR(&str_size)
diff --git a/src/dex/collections/bitset.c b/src/dex/collections/bitset.c
index 1d7e25ab0..789262ee0 100644
--- a/src/dex/collections/bitset.c
+++ b/src/dex/collections/bitset.c
@@ -1789,6 +1789,7 @@ PRIVATE struct type_method tpconst bs_methods[] = {
PRIVATE struct type_getset tpconst bs_getsets[] = {
TYPE_GETTER_F("frozen", &bs_frozen, METHOD_FNOREFESCAPE, "->?#Frozen"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &bs_sizeof, METHOD_FCONSTCALL | METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
@@ -2358,6 +2359,7 @@ PRIVATE struct type_method tpconst robs_methods[] = {
PRIVATE struct type_getset tpconst robs_getsets[] = {
TYPE_GETTER_F("frozen", &DeeObject_NewRef, METHOD_FCONSTCALL, "->?Dint"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &robs_sizeof, METHOD_FCONSTCALL | METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
@@ -4070,6 +4072,7 @@ PRIVATE struct type_getset tpconst bsv_getsets[] = {
/**/ "read-only, and the underlying object is also read-only, or a @this view "
/**/ "wrapped as a ?AFrozen?GBitset when the underlying bits may be modified "
/**/ "by something"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("nbits", &bsv_nbits, METHOD_FNOREFESCAPE | METHOD_FCONSTCALL,
"->?Dint\n"
"The # of bits stored in this bitset. Attempting to alter the state of "
diff --git a/src/dex/collections/deque.c b/src/dex/collections/deque.c
index 3a22de36c..fff967123 100644
--- a/src/dex/collections/deque.c
+++ b/src/dex/collections/deque.c
@@ -1355,6 +1355,7 @@ PRIVATE struct type_method_hint tpconst deq_method_hints[] = {
PRIVATE struct type_getset tpconst deq_getsets[] = {
TYPE_GETTER_F("__sizeof__", &deq_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETSET_END
};
diff --git a/src/dex/collections/fixedlist.c b/src/dex/collections/fixedlist.c
index c38c9df9f..a9a7e1c8a 100644
--- a/src/dex/collections/fixedlist.c
+++ b/src/dex/collections/fixedlist.c
@@ -1357,6 +1357,7 @@ fl_sizeof(FixedList *self) {
PRIVATE struct type_getset tpconst fl_getsets[] = {
TYPE_GETTER_F("__sizeof__", &fl_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETSET_END
};
diff --git a/src/dex/collections/rbtree.c b/src/dex/collections/rbtree.c
index 9d247c05c..56703fee3 100644
--- a/src/dex/collections/rbtree.c
+++ b/src/dex/collections/rbtree.c
@@ -3620,22 +3620,23 @@ PRIVATE struct type_method tpconst rbtree_methods[] = {
PRIVATE struct type_getset tpconst rbtree_getsets[] = {
TYPE_GETTER_F("first", &rbtree_get_first, METHOD_FNOREFESCAPE,
- "->?T3?O?O?O\n"
- DOC_ERROR_ValueError_EMPTY_SEQUENCE
- "Return the triple #C{minkey,maxkey,value} for the lowest range in @this ?."),
+ "->?T3?O?O?O\n"
+ DOC_ERROR_ValueError_EMPTY_SEQUENCE
+ "Return the triple #C{minkey,maxkey,value} for the lowest range in @this ?."),
TYPE_GETTER_F("last", &rbtree_get_last, METHOD_FNOREFESCAPE,
- "->?T3?O?O?O\n"
- DOC_ERROR_ValueError_EMPTY_SEQUENCE
- "Return the triple #C{minkey,maxkey,value} for the greatest range in @this ?."),
+ "->?T3?O?O?O\n"
+ DOC_ERROR_ValueError_EMPTY_SEQUENCE
+ "Return the triple #C{minkey,maxkey,value} for the greatest range in @this ?."),
TYPE_GETTER_F("__root__", &rbtree_get_itroot, METHOD_FNOREFESCAPE,
- "->?#Iterator\n"
- DOC_ERROR_ValueError_EMPTY_SEQUENCE
- "Return an ?#Iterator for the root node of the tree"),
+ "->?#Iterator\n"
+ DOC_ERROR_ValueError_EMPTY_SEQUENCE
+ "Return an ?#Iterator for the root node of the tree"),
TYPE_GETTER_F("__depth__", &rbtree_get_depth, METHOD_FNOREFESCAPE,
- "->?Dint\n"
- "Depth of the left-most tree node (since the tree is balanced, "
- /**/ "this is either the tree's max-depth, or one less than that)"),
+ "->?Dint\n"
+ "Depth of the left-most tree node (since the tree is balanced, "
+ /**/ "this is either the tree's max-depth, or one less than that)"),
TYPE_GETTER_F("__sizeof__", &rbtree_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETSET_END
};
diff --git a/src/dex/collections/udict.c b/src/dex/collections/udict.c
index cd20e8df6..a91ad1565 100644
--- a/src/dex/collections/udict.c
+++ b/src/dex/collections/udict.c
@@ -1282,6 +1282,7 @@ PRIVATE struct type_getset tpconst udict_getsets[] = {
TYPE_GETTER_F("frozen", &URoDict_FromUDict, METHOD_FNOREFESCAPE,
"->?AFrozen?.\n"
"Returns a read-only (frozen) copy of @this dict"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &udict_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
@@ -2084,6 +2085,7 @@ urodict_sizeof(URoDict *self) {
PRIVATE struct type_getset tpconst urodict_getsets[] = {
TYPE_GETTER("frozen", &DeeObject_NewRef, "->?."),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &urodict_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/src/dex/collections/uset.c b/src/dex/collections/uset.c
index d30d9633e..792c2f084 100644
--- a/src/dex/collections/uset.c
+++ b/src/dex/collections/uset.c
@@ -706,6 +706,7 @@ PRIVATE struct type_getset tpconst uset_getsets[] = {
TYPE_GETTER_F("frozen", &URoSet_FromUSet, METHOD_FNOREFESCAPE,
"->?AFrozen?.\n"
"Returns a read-only (frozen) copy of @this set"),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &uset_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
@@ -1983,9 +1984,8 @@ uroset_sizeof(URoSet *self) {
}
PRIVATE struct type_getset tpconst uroset_getsets[] = {
- TYPE_GETTER("frozen", &DeeObject_NewRef,
- "->?AFrozen?.\n"
- "Simply re-return @this object"),
+ TYPE_GETTER("frozen", &DeeObject_NewRef, "->?."),
+ TYPE_GETTER("cached", &DeeObject_NewRef, "->?."),
TYPE_GETTER_F("__sizeof__", &uroset_sizeof, METHOD_FNOREFESCAPE, "->?Dint"),
TYPE_GETSET_END
};
diff --git a/util/test/deemon-sequence-cached.dee b/util/test/deemon-sequence-cached.dee
new file mode 100644
index 000000000..5eb47ff27
--- /dev/null
+++ b/util/test/deemon-sequence-cached.dee
@@ -0,0 +1,68 @@
+#!/usr/bin/deemon
+/* Copyright (c) 2018-2024 Griefer@Work *
+ * *
+ * This software is provided 'as-is', without any express or implied *
+ * warranty. In no event will the authors be held liable for any damages *
+ * arising from the use of this software. *
+ * *
+ * Permission is granted to anyone to use this software for any purpose, *
+ * including commercial applications, and to alter it and redistribute it *
+ * freely, subject to the following restrictions: *
+ * *
+ * 1. The origin of this software must not be misrepresented; you must not *
+ * claim that you wrote the original software. If you use this software *
+ * in a product, an acknowledgement (see the following) in the product *
+ * documentation is required: *
+ * Portions Copyright (c) 2018-2024 Griefer@Work *
+ * 2. Altered source versions must be plainly marked as such, and must not be *
+ * misrepresented as being the original software. *
+ * 3. This notice may not be removed or altered from any source distribution. *
+ */
+
+import * from deemon;
+import * from errors;
+
+global hits = [];
+
+function producer() {
+ hits.append(1);
+ yield 10;
+ hits.append(2);
+ yield 20;
+ hits.append(3);
+ yield 30;
+ hits.append(4);
+ yield 40;
+ hits.append(5);
+}
+
+local items = producer().cached;
+assert hits == {};
+
+assert items[0] == 10;
+assert items[0] == 10;
+assert hits == {1};
+
+assert items[2] == 30;
+assert items[2] == 30;
+assert hits == {1,2,3};
+
+assert items[1] == 20;
+assert items[1] == 20;
+assert hits == {1,2,3};
+
+assert items[3] == 40;
+assert items[3] == 40;
+assert hits == {1,2,3,4};
+
+assert #items == 4;
+assert #items == 4;
+assert hits == {1,2,3,4,5};
+
+assert (try items[4] catch (e...) e) is IndexError;
+assert (try items[4] catch (e...) e) is IndexError;
+assert hits == {1,2,3,4,5};
+
+assert items == {10,20,30,40};
+assert items == {10,20,30,40};
+assert hits == {1,2,3,4,5};