File Coverage

xs/i16s.xsi
Criterion Covered Total %
statement 79 290 27.2
branch 120 674 17.8
condition n/a
subroutine n/a
pod n/a
total 199 964 20.6


line stmt bran cond sub pod time code
1             MODULE = Data::HashMap PACKAGE = Data::HashMap::I16S
2             PROTOTYPES: DISABLE
3              
4             SV*
5             new(char* class, ...)
6             CODE:
7 15 100         EXTRACT_NEW_ARGS(_max_size, _ttl, _lru_skip);
    50          
    50          
8 15           HashMapI16S* map = hashmap_i16s_create(_max_size, _ttl, _lru_skip);
9 15 50         if (!map) croak("Failed to create HashMap::I16S");
10 15           RETVAL = sv_setref_pv(newSV(0), class, (void*)map);
11             OUTPUT:
12             RETVAL
13              
14             void
15             DESTROY(SV* self_sv)
16             CODE:
17 15 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
18 15           hashmap_i16s_destroy(self);
19 15           sv_setiv(SvRV(self_sv), 0);
20              
21             bool
22             put(SV* self_sv, int16_t key, SV* value)
23             CODE:
24 60316 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
25 60316 50         EXTRACT_STR_VAL(value);
26 60316           RETVAL = hashmap_i16s_put(self, key, _vstr, (uint32_t)_vlen, _vutf8, 0);
27             OUTPUT:
28             RETVAL
29              
30             SV*
31             get(SV* self_sv, int16_t key)
32             CODE:
33 30009 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
34             const char* val;
35             uint32_t val_len;
36             bool val_utf8;
37 30009 100         if (!hashmap_i16s_get(self, key, &val, &val_len, &val_utf8))
38 3           XSRETURN_UNDEF;
39 30006           RETVAL = newSVpvn(val, val_len);
40 30006 100         if (val_utf8) SvUTF8_on(RETVAL);
41             OUTPUT:
42             RETVAL
43              
44             SV*
45             get_direct(SV* self_sv, int16_t key)
46             CODE:
47 2 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
48             const char* val;
49             uint32_t val_len;
50             bool val_utf8;
51 2 100         if (!hashmap_i16s_get(self, key, &val, &val_len, &val_utf8))
52 1           XSRETURN_UNDEF;
53 1           RETVAL = hm_zerocopy_sv(aTHX_ val, val_len, val_utf8);
54             OUTPUT:
55             RETVAL
56              
57             bool
58             remove(SV* self_sv, int16_t key)
59             CODE:
60 60006 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
61 60006           RETVAL = hashmap_i16s_remove(self, key);
62             OUTPUT:
63             RETVAL
64              
65             SV*
66             take(SV* self_sv, int16_t key)
67             CODE:
68 1 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
69             const char* val; uint32_t val_len; bool val_utf8;
70 1 50         if (!hashmap_i16s_take(self, key, &val, &val_len, &val_utf8)) XSRETURN_UNDEF;
71 1           RETVAL = newSVpvn(val, val_len);
72 1 50         if (val_utf8) SvUTF8_on(RETVAL);
73 1           free((void*)val);
74             OUTPUT:
75             RETVAL
76              
77             bool
78             exists(SV* self_sv, int16_t key)
79             CODE:
80 4 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
81 4           RETVAL = hashmap_i16s_exists(self, key);
82             OUTPUT:
83             RETVAL
84              
85             size_t
86             size(SV* self_sv)
87             CODE:
88 8 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
89 8 50         RETVAL = self->size;
90             OUTPUT:
91             RETVAL
92              
93             size_t
94             max_size(SV* self_sv)
95             CODE:
96 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
97 0 0         RETVAL = self->max_size;
98             OUTPUT:
99             RETVAL
100              
101             UV
102             ttl(SV* self_sv)
103             CODE:
104 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
105 0 0         RETVAL = (UV)self->default_ttl;
106             OUTPUT:
107             RETVAL
108              
109             UV
110             lru_skip(SV* self_sv)
111             CODE:
112 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
113 0 0         RETVAL = (UV)self->lru_skip;
114             OUTPUT:
115             RETVAL
116              
117             void
118             keys(SV* self_sv)
119             PPCODE:
120 2 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
121 2 50         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
122 2 50         EXTEND(SP, self->size);
123             size_t i;
124 34 100         for (i = 0; i < self->capacity; i++) {
125 32 100         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now))
    50          
    50          
    0          
    0          
126 2 50         mXPUSHi(self->nodes[i].key);
127             }
128              
129             void
130             values(SV* self_sv)
131             PPCODE:
132 1 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
133 1 50         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
134 1 50         EXTEND(SP, self->size);
135             size_t i;
136 17 100         for (i = 0; i < self->capacity; i++) {
137 16 100         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    50          
    50          
    0          
    0          
138 2 50         if (self->nodes[i].value) {
139 2           SV* sv = newSVpvn(self->nodes[i].value,
140             HM_UNPACK_LEN(self->nodes[i].val_len));
141 2 50         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(sv);
142 2 50         mXPUSHs(sv);
143             } else {
144 0 0         XPUSHs(&PL_sv_undef);
145             }
146             }
147             }
148              
149             void
150             items(SV* self_sv)
151             PPCODE:
152 1 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
153 1 50         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
154 1 50         EXTEND(SP, self->size * 2);
155             size_t i;
156 17 100         for (i = 0; i < self->capacity; i++) {
157 16 100         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    50          
    50          
    0          
    0          
158 2 50         mXPUSHi(self->nodes[i].key);
159 2 50         if (self->nodes[i].value) {
160 2           SV* sv = newSVpvn(self->nodes[i].value,
161             HM_UNPACK_LEN(self->nodes[i].val_len));
162 2 50         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(sv);
163 2 50         mXPUSHs(sv);
164             } else {
165 0 0         XPUSHs(&PL_sv_undef);
166             }
167             }
168             }
169              
170             void
171             each(SV* self_sv)
172             PPCODE:
173 3 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
174 3 50         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
175 17 100         while (self->iter_pos < self->capacity) {
176 16           size_t i = self->iter_pos++;
177 16 100         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    50          
    50          
    0          
    0          
178 2 50         EXTEND(SP, 2);
179 2 50         mXPUSHi(self->nodes[i].key);
180 2 50         if (GIMME_V == G_SCALAR) XSRETURN(1);
181 2 50         if (self->nodes[i].value) {
182 2           SV* vsv = newSVpvn(self->nodes[i].value,
183             HM_UNPACK_LEN(self->nodes[i].val_len));
184 2 50         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(vsv);
185 2 50         mXPUSHs(vsv);
186             } else {
187 0 0         XPUSHs(&PL_sv_undef);
188             }
189 2           XSRETURN(2);
190             }
191             }
192 1           self->iter_pos = 0;
193 1           XSRETURN_EMPTY;
194              
195             void
196             iter_reset(SV* self_sv)
197             CODE:
198 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
199 0           self->iter_pos = 0;
200              
201             void
202             drain(SV* self_sv, UV count)
203             PPCODE:
204 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
205 0 0         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
206 0           UV n = 0;
207 0 0         EXTEND(SP, (count < self->size ? count : self->size) * 2);
208 0 0         while (self->iter_pos < self->capacity && n < count) {
    0          
209 0           size_t i = self->iter_pos++;
210 0 0         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    0          
    0          
    0          
    0          
211 0 0         mXPUSHi(self->nodes[i].key);
212 0 0         if (self->nodes[i].value) {
213 0           SV* sv = newSVpvn(self->nodes[i].value, HM_UNPACK_LEN(self->nodes[i].val_len));
214 0 0         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(sv);
215 0 0         mXPUSHs(sv);
216 0 0         } else { XPUSHs(&PL_sv_undef); }
217 0 0         if (HM_UNLIKELY(self->lru_prev)) hashmap_i16s_lru_unlink(self, (uint32_t)i);
218 0           hashmap_i16s_tombstone_at(self, i);
219 0           n++;
220             }
221             }
222 0 0         if (self->iter_pos >= self->capacity) self->iter_pos = 0;
223 0 0         HM_MAYBE_COMPACT_XS(self, hashmap_i16s_compact);
    0          
    0          
224              
225             void
226             pop(SV* self_sv)
227             PPCODE:
228 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
229 0 0         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
230 0 0         if (self->lru_prev) {
231 0 0         while (self->lru_tail != HM_LRU_NONE) {
232 0           uint32_t idx = self->lru_tail;
233 0 0         if (HM_UNLIKELY(HM_TTL_SKIP_EXPIRED(self, idx, now))) {
    0          
    0          
    0          
234 0           hashmap_i16s_expire_at(self, idx, true); continue;
235             }
236 0 0         EXTEND(SP, 2);
237 0 0         mXPUSHi(self->nodes[idx].key);
238 0 0         if (self->nodes[idx].value) {
239 0           SV* sv = newSVpvn(self->nodes[idx].value, HM_UNPACK_LEN(self->nodes[idx].val_len));
240 0 0         if (HM_UNPACK_UTF8(self->nodes[idx].val_len)) SvUTF8_on(sv);
241 0 0         mXPUSHs(sv);
242 0 0         } else { XPUSHs(&PL_sv_undef); }
243 0           hashmap_i16s_lru_unlink(self, idx);
244 0           hashmap_i16s_tombstone_at(self, idx);
245 0 0         HM_MAYBE_COMPACT_XS(self, hashmap_i16s_compact);
    0          
    0          
246 0           XSRETURN(2);
247             }
248 0           XSRETURN_EMPTY;
249             } else {
250 0 0         while (self->iter_pos < self->capacity) {
251 0           size_t i = self->iter_pos++;
252 0 0         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    0          
    0          
    0          
    0          
253 0 0         EXTEND(SP, 2);
254 0 0         mXPUSHi(self->nodes[i].key);
255 0 0         if (self->nodes[i].value) {
256 0           SV* sv = newSVpvn(self->nodes[i].value, HM_UNPACK_LEN(self->nodes[i].val_len));
257 0 0         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(sv);
258 0 0         mXPUSHs(sv);
259 0 0         } else { XPUSHs(&PL_sv_undef); }
260 0           hashmap_i16s_tombstone_at(self, i);
261 0 0         HM_MAYBE_COMPACT_XS(self, hashmap_i16s_compact);
    0          
    0          
262 0           XSRETURN(2);
263             }
264             }
265 0           self->iter_pos = 0;
266 0           XSRETURN_EMPTY;
267             }
268              
269             void
270             shift(SV* self_sv)
271             PPCODE:
272 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
273 0 0         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
274 0 0         if (self->lru_prev) {
275 0 0         while (self->lru_head != HM_LRU_NONE) {
276 0           uint32_t idx = self->lru_head;
277 0 0         if (HM_UNLIKELY(HM_TTL_SKIP_EXPIRED(self, idx, now))) {
    0          
    0          
    0          
278 0           hashmap_i16s_expire_at(self, idx, true); continue;
279             }
280 0 0         EXTEND(SP, 2);
281 0 0         mXPUSHi(self->nodes[idx].key);
282 0 0         if (self->nodes[idx].value) {
283 0           SV* sv = newSVpvn(self->nodes[idx].value, HM_UNPACK_LEN(self->nodes[idx].val_len));
284 0 0         if (HM_UNPACK_UTF8(self->nodes[idx].val_len)) SvUTF8_on(sv);
285 0 0         mXPUSHs(sv);
286 0 0         } else { XPUSHs(&PL_sv_undef); }
287 0           hashmap_i16s_lru_unlink(self, idx);
288 0           hashmap_i16s_tombstone_at(self, idx);
289 0 0         HM_MAYBE_COMPACT_XS(self, hashmap_i16s_compact);
    0          
    0          
290 0           XSRETURN(2);
291             }
292 0           XSRETURN_EMPTY;
293             } else {
294 0 0         if (self->iter_pos == 0) self->iter_pos = self->capacity;
295 0 0         while (self->iter_pos > 0) {
296 0           size_t i = --self->iter_pos;
297 0 0         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    0          
    0          
    0          
    0          
298 0 0         EXTEND(SP, 2);
299 0 0         mXPUSHi(self->nodes[i].key);
300 0 0         if (self->nodes[i].value) {
301 0           SV* sv = newSVpvn(self->nodes[i].value, HM_UNPACK_LEN(self->nodes[i].val_len));
302 0 0         if (HM_UNPACK_UTF8(self->nodes[i].val_len)) SvUTF8_on(sv);
303 0 0         mXPUSHs(sv);
304 0 0         } else { XPUSHs(&PL_sv_undef); }
305 0           hashmap_i16s_tombstone_at(self, i);
306 0 0         HM_MAYBE_COMPACT_XS(self, hashmap_i16s_compact);
    0          
    0          
307 0           XSRETURN(2);
308             }
309             }
310 0           XSRETURN_EMPTY;
311             }
312              
313              
314             void
315             clear(SV* self_sv)
316             CODE:
317 3 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
318 3           hashmap_i16s_clear(self);
319              
320             void
321             reserve(SV* self_sv, UV count)
322             CODE:
323 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
324 0 0         if (!hashmap_i16s_reserve(self, (size_t)count))
325 0           croak("Failed to reserve capacity");
326              
327             void
328             purge(SV* self_sv)
329             CODE:
330 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
331 0           hashmap_i16s_purge(self);
332              
333             SV*
334             freeze(SV* self_sv)
335             CODE:
336 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
337 0 0         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
338 0           SV* buf = newSV(22 + self->size * 32);
339 0           SvPOK_on(buf);
340 0           SvCUR_set(buf, 0);
341             /* Header: magic(4) + version(1) + variant_id(1) + count(4) + max_size(4) + ttl(4) + lru_skip(4) */
342 0           sv_catpvn(buf, "DHMP", 4);
343 0           { uint8_t ver = 1; sv_catpvn(buf, (const char*)&ver, 1); }
344 0           { uint8_t vid = 8; sv_catpvn(buf, (const char*)&vid, 1); }
345 0           STRLEN cnt_offset = SvCUR(buf);
346 0           { uint32_t cnt = 0; sv_catpvn(buf, (const char*)&cnt, 4); }
347 0           UV n_written = 0;
348 0           { uint32_t ms = (uint32_t)self->max_size; sv_catpvn(buf, (const char*)&ms, 4); }
349 0           { uint32_t dt = self->default_ttl; sv_catpvn(buf, (const char*)&dt, 4); }
350 0           { uint32_t ls = self->lru_skip; sv_catpvn(buf, (const char*)&ls, 4); }
351             { size_t i;
352 0 0         for (i = 0; i < self->capacity; i++) {
353 0 0         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    0          
    0          
    0          
    0          
354 0           sv_catpvn(buf, (const char*)&self->nodes[i].key, 2);
355 0           uint32_t vl = HM_UNPACK_LEN(self->nodes[i].val_len);
356 0           uint8_t vu = HM_UNPACK_UTF8(self->nodes[i].val_len) ? 1 : 0;
357 0           sv_catpvn(buf, (const char*)&vl, 4);
358 0           sv_catpvn(buf, (const char*)&vu, 1);
359 0 0         if (self->nodes[i].value) sv_catpvn(buf, self->nodes[i].value, vl);
360             /* TTL: remaining seconds (0 = none) */
361 0           { uint32_t rem = 0;
362 0 0         if (self->expires_at && self->expires_at[i] && self->expires_at[i] >= now) {
    0          
    0          
363 0           rem = self->expires_at[i] - now;
364 0 0         if (!rem) rem = 1;
365             }
366 0           sv_catpvn(buf, (const char*)&rem, 4);
367             }
368 0           n_written++;
369             }
370             }
371             }
372 0           { uint32_t actual = (uint32_t)n_written;
373 0           memcpy(SvPVX(buf) + cnt_offset, &actual, 4); }
374 0           RETVAL = buf;
375             OUTPUT:
376             RETVAL
377              
378             SV*
379             thaw(char* class, SV* data)
380             CODE:
381             STRLEN dlen;
382 0           const uint8_t* p = (const uint8_t*)SvPV(data, dlen);
383 0           const uint8_t* end = p + dlen;
384 0 0         if (dlen < 22 || memcmp(p, "DHMP", 4) != 0) croak("Invalid freeze data");
    0          
385 0           p += 4;
386 0 0         uint8_t ver = *p++; if (ver != 1) croak("Unsupported freeze version %d", ver);
387 0 0         uint8_t vid = *p++; if (vid != 8) croak("Variant mismatch: expected 8, got %d", vid);
388 0           uint32_t cnt; memcpy(&cnt, p, 4); p += 4;
389 0           uint32_t ms; memcpy(&ms, p, 4); p += 4;
390 0           uint32_t dt; memcpy(&dt, p, 4); p += 4;
391 0           uint32_t ls; memcpy(&ls, p, 4); p += 4;
392 0           HashMapI16S* map = hashmap_i16s_create((size_t)ms, dt, ls);
393 0 0         if (!map) croak("Failed to create map for thaw");
394 0 0         if (cnt > 0) hashmap_i16s_reserve(map, (size_t)cnt);
395             { uint32_t j;
396 0 0         for (j = 0; j < cnt; j++) {
397 0 0         if (p + 11 > end) { hashmap_i16s_destroy(map); croak("Truncated freeze data"); }
398 0           int16_t key; memcpy(&key, p, 2); p += 2;
399 0           uint32_t vl; memcpy(&vl, p, 4); p += 4;
400 0           uint8_t vu = *p++; bool vutf8 = vu ? true : false;
401 0 0         if (p + vl + 4 > end) { hashmap_i16s_destroy(map); croak("Truncated freeze data"); }
402 0           const char* vstr = (const char*)p; p += vl;
403 0           uint32_t ttl; memcpy(&ttl, p, 4); p += 4;
404 0 0         if (!hashmap_i16s_put(map, key, vstr, vl, vutf8, ttl)) { hashmap_i16s_destroy(map); croak("OOM during thaw"); }
405 0 0         if (ttl == 0 && map->expires_at) hashmap_i16s_persist(map, key);
    0          
406             }
407             }
408 0 0         if (p > end) croak("Truncated freeze data");
409 0           RETVAL = sv_setref_pv(newSV(0), class, (void*)map);
410             OUTPUT:
411             RETVAL
412              
413              
414             UV
415             capacity(SV* self_sv)
416             CODE:
417 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
418 0 0         RETVAL = (UV)self->capacity;
419             OUTPUT:
420             RETVAL
421              
422             bool
423             persist(SV* self_sv, int16_t key)
424             CODE:
425 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
426 0           RETVAL = hashmap_i16s_persist(self, key);
427             OUTPUT:
428             RETVAL
429              
430             SV*
431             swap(SV* self_sv, int16_t key, SV* new_sv)
432             CODE:
433 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
434 0           STRLEN _nvl; const char* _nvs = SvPV(new_sv, _nvl); bool _nvu = SvUTF8(new_sv) ? true : false;
435             const char* old; uint32_t old_len; bool old_utf8;
436 0 0         if (!hashmap_i16s_swap(self, key, _nvs, (uint32_t)_nvl, _nvu, &old, &old_len, &old_utf8)) XSRETURN_UNDEF;
437 0           RETVAL = newSVpvn(old, old_len);
438 0 0         if (old_utf8) SvUTF8_on(RETVAL);
439 0           free((void*)old);
440             OUTPUT:
441             RETVAL
442              
443              
444             SV*
445             clone(SV* self_sv)
446             CODE:
447 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
448 0           HashMapI16S* clone = hashmap_i16s_clone(self);
449 0 0         if (!clone) croak("Failed to clone");
450 0 0         RETVAL = sv_setref_pv(newSV(0), HvNAME(SvSTASH(SvRV(self_sv))), (void*)clone);
    0          
    0          
    0          
    0          
    0          
451             OUTPUT:
452             RETVAL
453              
454             void
455             from_hash(SV* self_sv, SV* href)
456             CODE:
457 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
458 0 0         if (!SvROK(href) || SvTYPE(SvRV(href)) != SVt_PVHV)
    0          
459 0           croak("from_hash requires a hashref");
460 0           HV* hv = (HV*)SvRV(href);
461 0 0         hashmap_i16s_reserve(self, (size_t)HvUSEDKEYS(hv));
462 0           hv_iterinit(hv);
463             HE* he;
464 0 0         while ((he = hv_iternext(hv))) {
465 0           SV* ksv = hv_iterkeysv(he);
466 0           int16_t key = (int16_t)SvIV(ksv);
467 0           SV* val = HeVAL(he);
468 0           STRLEN vlen; const char* vstr = SvPV(val, vlen); bool vutf8 = SvUTF8(val) ? true : false;
469 0           hashmap_i16s_put(self, key, vstr, (uint32_t)vlen, vutf8, 0);
470             }
471              
472             void
473             merge(SV* self_sv, SV* other_sv)
474             CODE:
475 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
476 0 0         if (!SvROK(other_sv) || !SvOBJECT(SvRV(other_sv)) || SvSTASH(SvRV(other_sv)) != stash_i16s)
    0          
    0          
477 0           croak("Expected a Data::HashMap::I16S object");
478 0           HashMapI16S* other = INT2PTR(HashMapI16S*, SvIV(SvRV(other_sv)));
479 0           hashmap_i16s_reserve(self, self->size + other->size);
480 0 0         uint32_t now = other->expires_at ? (uint32_t)time(NULL) : 0;
481             size_t i;
482 0 0         for (i = 0; i < other->capacity; i++) {
483 0 0         if (I16S_NODE_LIVE(other->nodes[i]) && !HM_TTL_SKIP_EXPIRED(other, i, now))
    0          
    0          
    0          
    0          
484 0           hashmap_i16s_put(self, other->nodes[i].key, other->nodes[i].value, HM_UNPACK_LEN(other->nodes[i].val_len), HM_UNPACK_UTF8(other->nodes[i].val_len), 0);
485             }
486              
487              
488             SV*
489             to_hash(SV* self_sv)
490             CODE:
491 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
492 0           HV* hv = newHV();
493 0 0         uint32_t now = self->expires_at ? (uint32_t)time(NULL) : 0;
494             size_t i;
495 0 0         for (i = 0; i < self->capacity; i++) {
496 0 0         if (I16S_NODE_LIVE(self->nodes[i]) && !HM_TTL_SKIP_EXPIRED(self, i, now)) {
    0          
    0          
    0          
    0          
497 0           uint32_t vlen = HM_UNPACK_LEN(self->nodes[i].val_len);
498 0           bool vutf8 = HM_UNPACK_UTF8(self->nodes[i].val_len);
499 0           SV* val = self->nodes[i].value
500 0           ? newSVpvn(self->nodes[i].value, vlen)
501 0 0         : newSV(0);
502 0 0         if (self->nodes[i].value && vutf8) SvUTF8_on(val);
    0          
503             char kbuf[24];
504 0           int klen = my_snprintf(kbuf, sizeof(kbuf), "%" IVdf, (IV)self->nodes[i].key);
505 0           (void)hv_store(hv, kbuf, klen, val, 0);
506             }
507             }
508 0           RETVAL = newRV_noinc((SV*)hv);
509             OUTPUT:
510             RETVAL
511              
512             bool
513             put_ttl(SV* self_sv, int16_t key, SV* value, UV ttl)
514             CODE:
515 0 0         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    0          
    0          
    0          
516 0 0         EXTRACT_STR_VAL(value);
517 0           RETVAL = hashmap_i16s_put(self, key, _vstr, (uint32_t)_vlen, _vutf8, (uint32_t)ttl);
518             OUTPUT:
519             RETVAL
520              
521             SV*
522             get_or_set(SV* self_sv, int16_t key, SV* default_sv)
523             CODE:
524 1 50         EXTRACT_MAP(HashMapI16S, stash_i16s, "Data::HashMap::I16S", self_sv);
    50          
    50          
    50          
525 1 50         EXTRACT_STR_VAL(default_sv);
526             bool was_found;
527 1           size_t idx = hashmap_i16s_get_or_set(self, key, _vstr, (uint32_t)_vlen, _vutf8, 0, &was_found);
528 1 50         if (idx >= self->capacity) XSRETURN_UNDEF;
529 1 50         if (self->nodes[idx].value) {
530 1           RETVAL = newSVpvn(self->nodes[idx].value, HM_UNPACK_LEN(self->nodes[idx].val_len));
531 1 50         if (HM_UNPACK_UTF8(self->nodes[idx].val_len)) SvUTF8_on(RETVAL);
532             } else {
533 0           RETVAL = newSV(0);
534             }
535             OUTPUT:
536             RETVAL
537              
538              
539