Data-HashMap
view release on metacpan or search on metacpan
RETVAL
SV*
decr(SV* self_sv, SV* key_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
EXTRACT_STR_KEY(key_sv);
int64_t val;
if (!hashmap_si_decrement(self, _kstr, (uint32_t)_klen, _khash, _kutf8, &val))
croak("HashMap::SI: decrement failed");
RETVAL = newSViv(val);
OUTPUT:
RETVAL
SV*
incr_by(SV* self_sv, SV* key_sv, int64_t delta)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
EXTRACT_STR_KEY(key_sv);
int64_t val;
if (!hashmap_si_increment_by(self, _kstr, (uint32_t)_klen, _khash, _kutf8, delta, &val))
croak("HashMap::SI: incr_by failed");
RETVAL = newSViv(val);
OUTPUT:
RETVAL
size_t
size(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
RETVAL = self->size;
OUTPUT:
RETVAL
size_t
max_size(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
RETVAL = self->max_size;
OUTPUT:
RETVAL
UV
ttl(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
RETVAL = (UV)self->default_ttl;
OUTPUT:
RETVAL
UV
lru_skip(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
RETVAL = (UV)self->lru_skip;
OUTPUT:
RETVAL
void
keys(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
EXTEND(SP, self->size);
size_t i;
for (i = 0; i < self->capacity; i++) {
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
uint32_t klen = HM_UNPACK_LEN(self->nodes[i].key_len);
SV* sv = newSVpvn(self->nodes[i].key, klen);
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(sv);
mXPUSHs(sv);
}
}
void
values(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
EXTEND(SP, self->size);
size_t i;
for (i = 0; i < self->capacity; i++) {
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now))
mXPUSHi(self->nodes[i].value);
}
void
items(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
EXTEND(SP, self->size * 2);
size_t i;
for (i = 0; i < self->capacity; i++) {
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
uint32_t klen = HM_UNPACK_LEN(self->nodes[i].key_len);
SV* sv = newSVpvn(self->nodes[i].key, klen);
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(sv);
mXPUSHs(sv);
mXPUSHi(self->nodes[i].value);
}
}
void
each(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
while (self->iter_pos < self->capacity) {
size_t i = self->iter_pos++;
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
EXTEND(SP, 2);
{
uint32_t klen = HM_UNPACK_LEN(self->nodes[i].key_len);
SV* ksv = newSVpvn(self->nodes[i].key, klen);
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
if (GIMME_V == G_SCALAR) XSRETURN(1);
mXPUSHi(self->nodes[i].value);
XSRETURN(2);
}
}
self->iter_pos = 0;
XSRETURN_EMPTY;
void
iter_reset(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
self->iter_pos = 0;
void
drain(SV* self_sv, UV count)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
UV n = 0;
EXTEND(SP, (count < self->size ? count : self->size) * 2);
while (self->iter_pos < self->capacity && n < count) {
size_t i = self->iter_pos++;
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
{
SV* ksv = newSVpvn(self->nodes[i].key, HM_UNPACK_LEN(self->nodes[i].key_len));
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
mXPUSHi(self->nodes[i].value);
if (HM_UNLIKELY(self->lru_prev)) hashmap_si_lru_unlink(self, (uint32_t)i);
hashmap_si_tombstone_at(self, i);
n++;
}
}
if (self->iter_pos >= self->capacity) self->iter_pos = 0;
HM_MAYBE_COMPACT_XS(self, hashmap_si_compact);
void
pop(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
if (self->lru_prev) {
while (self->lru_tail != HM_LRU_NONE) {
uint32_t idx = self->lru_tail;
if (HM_UNLIKELY(HM_TTL_SKIP_EXPIRED(self, idx, now))) {
hashmap_si_expire_at(self, idx, true); continue;
}
EXTEND(SP, 2);
{
SV* ksv = newSVpvn(self->nodes[idx].key, HM_UNPACK_LEN(self->nodes[idx].key_len));
if (HM_UNPACK_UTF8(self->nodes[idx].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
mXPUSHi(self->nodes[idx].value);
hashmap_si_lru_unlink(self, idx);
hashmap_si_tombstone_at(self, idx);
HM_MAYBE_COMPACT_XS(self, hashmap_si_compact);
XSRETURN(2);
}
XSRETURN_EMPTY;
} else {
while (self->iter_pos < self->capacity) {
size_t i = self->iter_pos++;
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
EXTEND(SP, 2);
{
SV* ksv = newSVpvn(self->nodes[i].key, HM_UNPACK_LEN(self->nodes[i].key_len));
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
mXPUSHi(self->nodes[i].value);
hashmap_si_tombstone_at(self, i);
HM_MAYBE_COMPACT_XS(self, hashmap_si_compact);
XSRETURN(2);
}
}
self->iter_pos = 0;
XSRETURN_EMPTY;
}
void
shift(SV* self_sv)
PPCODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
if (self->lru_prev) {
while (self->lru_head != HM_LRU_NONE) {
uint32_t idx = self->lru_head;
if (HM_UNLIKELY(HM_TTL_SKIP_EXPIRED(self, idx, now))) {
hashmap_si_expire_at(self, idx, true); continue;
}
EXTEND(SP, 2);
{
SV* ksv = newSVpvn(self->nodes[idx].key, HM_UNPACK_LEN(self->nodes[idx].key_len));
if (HM_UNPACK_UTF8(self->nodes[idx].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
mXPUSHi(self->nodes[idx].value);
hashmap_si_lru_unlink(self, idx);
hashmap_si_tombstone_at(self, idx);
HM_MAYBE_COMPACT_XS(self, hashmap_si_compact);
XSRETURN(2);
}
XSRETURN_EMPTY;
} else {
if (self->iter_pos == 0) self->iter_pos = self->capacity;
while (self->iter_pos > 0) {
size_t i = --self->iter_pos;
if (SI_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
EXTEND(SP, 2);
{
SV* ksv = newSVpvn(self->nodes[i].key, HM_UNPACK_LEN(self->nodes[i].key_len));
if (HM_UNPACK_UTF8(self->nodes[i].key_len)) SvUTF8_on(ksv);
mXPUSHs(ksv);
}
mXPUSHi(self->nodes[i].value);
hashmap_si_tombstone_at(self, i);
HM_MAYBE_COMPACT_XS(self, hashmap_si_compact);
XSRETURN(2);
}
}
XSRETURN_EMPTY;
}
void
clear(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
hashmap_si_clear(self);
void
reserve(SV* self_sv, UV count)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
if (!hashmap_si_reserve(self, (size_t)count))
croak("Failed to reserve capacity");
void
purge(SV* self_sv)
CODE:
EXTRACT_MAP(HashMapSI, stash_si, "Data::HashMap::SI", self_sv);
hashmap_si_purge(self);
( run in 1.459 second using v1.01-cache-2.11-cpan-71847e10f99 )