|
@@ -1,7 +1,12 @@
|
|
|
from libc.string cimport memcmp
|
|
|
from cpython.mem cimport PyMem_Malloc, PyMem_Realloc, PyMem_Free
|
|
|
|
|
|
-cdef class Keyset:
|
|
|
+from lakesuperior.cy_includes cimport collections as cc
|
|
|
+from lakesuperior.model.base cimport (
|
|
|
+ KeyIdx, Key, DoubleKey, TripleKey, Buffer
|
|
|
+)
|
|
|
+
|
|
|
+cdef class BaseKeyset:
|
|
|
"""
|
|
|
Pre-allocated result set.
|
|
|
|
|
@@ -17,29 +22,27 @@ cdef class Keyset:
|
|
|
``size``: Total size, in bytes, of the data set. This is the product of
|
|
|
``itemsize`` and ``ct``.
|
|
|
"""
|
|
|
- def __cinit__(self, size_t ct, unsigned char itemsize):
|
|
|
+ def __cinit__(self, size_t ct):
|
|
|
"""
|
|
|
Initialize and allocate memory for the data set.
|
|
|
|
|
|
:param size_t ct: Number of elements to be accounted for.
|
|
|
- :param unsigned char itemsize: Size of an individual item.
|
|
|
- Note that the ``itemsize`` is an unsigned char,
|
|
|
- i.e. an item can be at most 255 bytes. This is for economy reasons,
|
|
|
- since many multiplications are done between ``itemsize`` and other
|
|
|
- char variables.
|
|
|
"""
|
|
|
- self.ct = ct
|
|
|
- self.itemsize = itemsize
|
|
|
- self.size = self.itemsize * self.ct
|
|
|
+ self.conf.capacity = ct
|
|
|
+ self.itemsize = self.get_itemsize() # Set this in concrete classes
|
|
|
+ self.size = self.itemsize * self.conf.capacity
|
|
|
+
|
|
|
+ cc.array_conf_init(&self.conf)
|
|
|
+ self.conf.capacity = self.conf.capacity
|
|
|
+ cc.array_init_conf(&self.data
|
|
|
+ if not self.data:
|
|
|
+ raise MemoryError()
|
|
|
self._cur = 0
|
|
|
|
|
|
#logger.debug('Got malloc sizes: {}, {}'.format(ct, itemsize))
|
|
|
#logger.debug(
|
|
|
# 'Allocating {0} ({1}x{2}) bytes of Keyset data...'.format(
|
|
|
- # self.size, self.ct, self.itemsize))
|
|
|
- self.data = <unsigned char *>PyMem_Malloc(ct * itemsize)
|
|
|
- if not self.data:
|
|
|
- raise MemoryError()
|
|
|
+ # self.size, self.conf.capacity, self.itemsize))
|
|
|
#logger.debug('...done allocating @ {0:x}.'.format(
|
|
|
# <unsigned long>self.data))
|
|
|
|
|
@@ -53,65 +56,14 @@ cdef class Keyset:
|
|
|
"""
|
|
|
#logger.debug(
|
|
|
# 'Releasing {0} ({1}x{2}) bytes of Keyset @ {3:x}...'.format(
|
|
|
- # self.size, self.ct, self.itemsize,
|
|
|
+ # self.size, self.conf.capacity, self.itemsize,
|
|
|
# <unsigned long>self.data))
|
|
|
PyMem_Free(self.data)
|
|
|
#logger.debug('...done releasing.')
|
|
|
|
|
|
|
|
|
- cdef void resize(self, size_t ct) except *:
|
|
|
- """
|
|
|
- Resize the result set. Uses ``PyMem_Realloc``.
|
|
|
-
|
|
|
- Note that resizing to a smaller size does not copy or reallocate the
|
|
|
- data, resizing to a larger size does.
|
|
|
-
|
|
|
- Also, note that only the number of items can be changed, the item size
|
|
|
- cannot.
|
|
|
-
|
|
|
- :param size_t ct: Number of items in the result set.
|
|
|
- """
|
|
|
- cdef unsigned char *tmp
|
|
|
- self.ct = ct
|
|
|
- self.size = self.itemsize * self.ct
|
|
|
-
|
|
|
- #logger.debug(
|
|
|
- # 'Resizing Keyset to {0} ({1}x{2}) bytes @ {3:x}...'.format(
|
|
|
- # self.itemsize * ct, ct, self.itemsize,
|
|
|
- # <unsigned long>self.data))
|
|
|
- tmp = <unsigned char *>PyMem_Realloc(self.data, ct * self.itemsize)
|
|
|
- if not tmp:
|
|
|
- raise MemoryError()
|
|
|
- #logger.debug('...done resizing.')
|
|
|
-
|
|
|
- self.data = tmp
|
|
|
-
|
|
|
-
|
|
|
# Access methods.
|
|
|
|
|
|
- def to_tuple(self):
|
|
|
- """
|
|
|
- Return the data set as a Python tuple.
|
|
|
-
|
|
|
- :rtype: tuple
|
|
|
- """
|
|
|
- return tuple(
|
|
|
- self.data[i: i + self.itemsize]
|
|
|
- for i in range(0, self.size, self.itemsize))
|
|
|
-
|
|
|
-
|
|
|
- def get_item_obj(self, i):
|
|
|
- """
|
|
|
- Get an item at a given index position.
|
|
|
-
|
|
|
- :rtype: bytes
|
|
|
- """
|
|
|
- if i >= self.ct:
|
|
|
- raise ValueError(f'Index {i} out of range.')
|
|
|
-
|
|
|
- return self.get_item(i)[: self.itemsize]
|
|
|
-
|
|
|
-
|
|
|
def iter_init(self):
|
|
|
"""
|
|
|
Reset the cursor to the initial position.
|
|
@@ -149,7 +101,7 @@ cdef class Keyset:
|
|
|
:return: True if a value was found, False if the end of the buffer
|
|
|
has been reached.
|
|
|
"""
|
|
|
- if self._cur >= self.ct:
|
|
|
+ if self._cur >= self.conf.capacity:
|
|
|
val = NULL
|
|
|
return False
|
|
|
|
|
@@ -171,3 +123,17 @@ cdef class Keyset:
|
|
|
return True
|
|
|
return False
|
|
|
|
|
|
+
|
|
|
+class Keyset(BaseKeyset):
|
|
|
+ cdef size_t get_itemsize():
|
|
|
+ return KLEN
|
|
|
+
|
|
|
+
|
|
|
+class DoubleKeyset(BaseKeyset):
|
|
|
+ cdef size_t get_itemsize():
|
|
|
+ return DBL_KLEN
|
|
|
+
|
|
|
+
|
|
|
+class TripleKeyset(BaseKeyset):
|
|
|
+ cdef size_t get_itemsize():
|
|
|
+ return TRP_KLEN
|