Source code for spiceypy.utils.support_types

"""
The MIT License (MIT)

Copyright (c) [2015-2019] [Andrew Annex]

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

The MIT License (MIT)

Copyright (c) 2013 Philipp Rasch

Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
the Software, and to permit persons to whom the Software is furnished to do so,
subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
"""
import six

from ctypes import c_char_p, c_int, c_double,\
    c_char, c_void_p, sizeof, \
    Array, create_string_buffer, cast, Structure, \
    string_at

import numpy
from numpy import ctypeslib as numpc
# Collection of supporting functions for wrapper functions
__author__ = 'AndrewAnnex'

errorformat = """
================================================================================

Toolkit version: {tkvsn}

{short} --
{explain}
{long}

{traceback}

================================================================================\
"""


[docs]class SpiceyError(Exception): """ SpiceyError wraps CSPICE errors. :type value: str """ def __init__(self, value, found=None): self.value = value self.found = found def __str__(self): return self.value
[docs]def toDoubleVector(x): return DoubleArray.from_param(param=x)
[docs]def toDoubleMatrix(x): return DoubleMatrix.from_param(param=x)
[docs]def toIntVector(x): return IntArray.from_param(param=x)
[docs]def toIntMatrix(x): return IntMatrix.from_param(param=x)
[docs]def toPythonString(inString): if six.PY2: if isinstance(inString, c_char_p): return toPythonString(inString.value) return string_at(inString).rstrip() elif six.PY3: if isinstance(inString, c_char_p): return toPythonString(inString.value) return bytes.decode(string_at(inString), errors="ignore").rstrip()
[docs]def emptyCharArray(xLen=None, yLen=None): if not yLen: yLen = 1 if not xLen: xLen = 1 if isinstance(xLen, c_int): xLen = xLen.value if isinstance(yLen, c_int): yLen = yLen.value return ((c_char * xLen) * yLen)()
[docs]def emptyDoubleMatrix(x=3, y=3): if isinstance(x, c_int): x = x.value if isinstance(y, c_int): y = y.value return ((c_double * x) * y)()
[docs]def emptyDoubleVector(n): if isinstance(n, c_int): n = n.value assert(isinstance(n, int)) return (c_double * n)()
[docs]def emptyIntMatrix(x=3, y=3): if isinstance(x, c_int): x = x.value if isinstance(y, c_int): y = y.value return ((c_int * x) * y)()
[docs]def emptyIntVector(n): if isinstance(n, c_int): n = n.value assert (isinstance(n, int)) return (c_int * n)()
[docs]def cVectorToPython(x): """ Convert the c vector data into the correct python data type (numpy arrays or strings) :param x: :return: """ if isinstance(x[0], bool): return numpy.frombuffer(x, dtype=numpy.bool).copy() elif isinstance(x[0], int): return numpy.frombuffer(x, dtype=numpy.int32).copy() elif isinstance(x[0], float): return numpy.frombuffer(x, dtype=numpy.float64).copy() elif isinstance(x[0].value, bytes): return [toPythonString(y) for y in x]
[docs]def cIntVectorToBoolPython(x): return numpc.as_array(x).copy().astype(bool)
[docs]def cMatrixToNumpy(x): """ Convert a ctypes 2d array (or matrix) into a numpy array for python use :param x: thing to convert :return: numpy.ndarray """ return numpc.as_array(x).copy()
[docs]def stringToCharP(inobject, inlen=None): """ :param inobject: input string, int for getting null string of length of int :param inlen: optional parameter, length of a given string can be specified :return: """ if inlen and isinstance(inobject, str): return create_string_buffer(inobject.encode(encoding='UTF-8'), inlen) if isinstance(inobject, bytes): return inobject if isinstance(inobject, c_int): return stringToCharP(" " * inobject.value) if isinstance(inobject, int): return stringToCharP(" " * inobject) return c_char_p(inobject.encode(encoding='UTF-8'))
[docs]def listToCharArray(inList, xLen=None, yLen=None): assert (isinstance(inList, list)) if not yLen: yLen = len(inList) if not xLen: xLen = max(len(s) for s in inList) + 1 if isinstance(xLen, c_int): xLen = xLen.value if isinstance(yLen, c_int): yLen = yLen.value return ((c_char * xLen) * yLen)(*[stringToCharP(l, inlen=xLen) for l in inList])
[docs]def listToCharArrayPtr(inList, xLen=None, yLen=None): assert (isinstance(inList, list)) if not yLen: yLen = len(inList) if not xLen: xLen = max(len(s) for s in inList) + 1 if isinstance(xLen, c_int): xLen = xLen.value if isinstance(yLen, c_int): yLen = yLen.value return cast(((c_char * xLen) * yLen)(*[stringToCharP(l, inlen=xLen) for l in inList]), c_char_p)
[docs]class DoubleArrayType: # Class type that will handle all double vectors, # inspiration from python cookbook 3rd edition
[docs] def from_param(self, param): typename = type(param).__name__ if hasattr(self, 'from_' + typename): return getattr(self, 'from_' + typename)(param) elif isinstance(param, Array): return param else: raise TypeError("Can't convert %s" % typename)
# Cast from lists/tuples
[docs] def from_list(self, param): val = ((c_double) * len(param))(*param) return val
# Cast from Tuple
[docs] def from_tuple(self, param): val = ((c_double) * len(param))(*param) return val
# Cast from a numpy array,
[docs] def from_ndarray(self, param): # return param.data_as(POINTER(c_double)) # the above older method does not work with # functions which take vectors of known size return numpc.as_ctypes(param.astype(numpy.float64, casting='same_kind', copy=False))
# Cast from array.array objects
[docs] def from_array(self, param): if param.typecode != 'd': raise TypeError('must be an array of doubles') return self.from_list(param)
[docs]class DoubleMatrixType: # Class type that will handle all double matricies, # inspiration from python cookbook 3rd edition
[docs] def from_param(self, param): typename = type(param).__name__ if hasattr(self, 'from_' + typename): return getattr(self, 'from_' + typename)(param) elif isinstance(param, Array): return param else: raise TypeError("Can't convert %s" % typename)
# Cast from lists/tuples
[docs] def from_list(self, param): val = ((c_double * len(param[0])) * len(param))(*[DoubleArray.from_param(x) for x in param]) return val
# Cast from Tuple
[docs] def from_tuple(self, param): val = ((c_double * len(param[0])) * len(param))(*[DoubleArray.from_param(x) for x in param]) return val
# Cast from a numpy array
[docs] def from_ndarray(self, param): return numpc.as_ctypes(param.astype(numpy.float64, casting='same_kind', copy=False))
# Cast from a numpy matrix
[docs] def from_matrix(self, param): return numpc.as_ctypes(param.astype(numpy.float64, casting='same_kind', copy=False))
[docs]class IntArrayType: # Class type that will handle all int vectors, # inspiration from python cookbook 3rd edition
[docs] def from_param(self, param): typename = type(param).__name__ if hasattr(self, 'from_' + typename): return getattr(self, 'from_' + typename)(param) elif isinstance(param, Array): return param else: raise TypeError("Can't convert %s" % typename)
# Cast from lists/tuples
[docs] def from_list(self, param): val = ((c_int) * len(param))(*param) return val
# Cast from Tuple
[docs] def from_tuple(self, param): val = ((c_int) * len(param))(*param) return val
# Cast from a numpy array
[docs] def from_ndarray(self, param): # cspice always uses a int size half as big as the float, ie int32 if a float64 system default return numpc.as_ctypes(param.astype(numpy.int32, casting='same_kind', copy=False))
# Cast from array.array objects
[docs] def from_array(self, param): if param.typecode != 'i': raise TypeError('must be an array of ints') return self.from_list(param)
[docs]class IntMatrixType: # Class type that will handle all int matricies, # inspiration from python cookbook 3rd edition
[docs] def from_param(self, param): typename = type(param).__name__ if hasattr(self, 'from_' + typename): return getattr(self, 'from_' + typename)(param) elif isinstance(param, Array): return param else: raise TypeError("Can't convert %s" % typename)
# Cast from lists/tuples
[docs] def from_list(self, param): val = ((c_int * len(param[0])) * len(param))(*[IntArray.from_param(x) for x in param]) return val
# Cast from Tuple
[docs] def from_tuple(self, param): val = ((c_int * len(param[0])) * len(param))(*[IntArray.from_param(x) for x in param]) return val
# Cast from a numpy array
[docs] def from_ndarray(self, param): # cspice always uses a int size half as big as the float, ie int32 if a float64 system default return numpc.as_ctypes(param.astype(numpy.int32, casting='same_kind', copy=False))
# Cast from a numpy matrix
[docs] def from_matrix(self, param): # cspice always uses a int size half as big as the float, ie int32 if a float64 system default return numpc.as_ctypes(param.astype(numpy.int32, casting='same_kind', copy=False))
DoubleArray = DoubleArrayType() IntArray = IntArrayType() IntMatrix = IntMatrixType() DoubleMatrix = DoubleMatrixType()
[docs]class Plane(Structure): _fields_ = [ ('_normal', c_double * 3), ('_constant', c_double) ] @property def normal(self): return cVectorToPython(self._normal) @property def constant(self): return self._constant def __str__(self): return '<SpicePlane: normal=%s; constant=%s>' % (', '.join([str(x) for x in self._normal]), self._constant)
[docs]class Ellipse(Structure): _fields_ = [ ('_center', c_double * 3), ('_semi_major', c_double * 3), ('_semi_minor', c_double * 3) ] @property def center(self): return cVectorToPython(self._center) @property def semi_major(self): return cVectorToPython(self._semi_major) @property def semi_minor(self): return cVectorToPython(self._semi_minor) def __str__(self): return '<SpiceEllipse: center = %s, semi_major = %s, semi_minor = %s>' % \ (self.center, self.semi_major, self.semi_minor)
[docs]class DataType(object): SPICE_CHR = 0 SPICE_DP = 1 SPICE_INT = 2 SPICE_TIME = 3 SPICE_BOOL = 4 CHR = 0 DP = 1 INT = 2 TIME = 3 BOOL = 4 def __init__(self): pass
[docs]class SpiceDSKDescr(Structure): _fields_ = [ ('_surfce', c_int), ('_center', c_int), ('_dclass', c_int), ('_dtype', c_int), ('_frmcde', c_int), ('_corsys', c_int), ('_corpar', c_double * 10), ('_co1min', c_double), ('_co1max', c_double), ('_co2min', c_double), ('_co2max', c_double), ('_co3min', c_double), ('_co3max', c_double), ('_start', c_double), ('_stop', c_double), ] @property def surfce(self): return self._surfce @property def center(self): return self._center @property def dclass(self): return self._dclass @property def dtype(self): return self._dtype @property def frmcde(self): return self._frmcde @property def corsys(self): return self._corsys @property def corpar(self): return cVectorToPython(self._corpar) @property def co1min(self): return self._co1min @property def co1max(self): return self._co1max @property def co2min(self): return self._co2min @property def co2max(self): return self._co2max @property def co3min(self): return self._co3min @property def co3max(self): return self._co3max @property def start(self): return self._start @property def stop(self): return self._stop
[docs]class SpiceDLADescr(Structure): _fields_ = [ ('_bwdptr', c_int), ('_fwdptr', c_int), ('_ibase', c_int), ('_isize', c_int), ('_dbase', c_int), ('_dsize', c_int), ('_cbase', c_int), ('_csize', c_int) ] @property def bwdptr(self): return self._bwdptr @property def fwdptr(self): return self._fwdptr @property def ibase(self): return self._ibase @property def isize(self): return self._isize @property def dbase(self): return self._dbase @property def dsize(self): return self._dsize @property def cbase(self): return self._cbase @property def csize(self): return self._csize
[docs]class SpiceEKDataType(c_int): _SPICE_CHR = c_int(0) _SPICE_DP = c_int(1) _SPICE_INT = c_int(2) _SPICE_TIME = c_int(3) _SPICE_BOOL = c_int(4) _fields_ = [ ('SPICE_CHR', _SPICE_CHR), ('SPICE_DP', _SPICE_DP), ('SPICE_INT', _SPICE_INT), ('SPICE_TIME', _SPICE_TIME), ('SPICE_BOOL', _SPICE_BOOL), ] SPICE_CHR = _SPICE_CHR.value SPICE_DP = _SPICE_DP.value SPICE_INT = _SPICE_INT.value SPICE_TIME = _SPICE_TIME.value SPICE_BOOL = _SPICE_BOOL.value
[docs]def emptySpiceEKDataTypeVector(n): if isinstance(n, c_int): n = n.value assert(isinstance(n, int)) return (SpiceEKDataType * n)()
[docs]class SpiceEKExprClass(c_int): _SPICE_EK_EXP_COL = c_int(0) _SPICE_EK_EXP_FUNC = c_int(1) _SPICE_EK_EXP_EXPR = c_int(2) _fields_ = [ ('SPICE_EK_EXP_COL', _SPICE_EK_EXP_COL), ('SPICE_EK_EXP_FUNC', _SPICE_EK_EXP_FUNC), ('SPICE_EK_EXP_EXPR', _SPICE_EK_EXP_EXPR) ] SPICE_EK_EXP_COL = _SPICE_EK_EXP_COL.value SPICE_EK_EXP_FUNC = _SPICE_EK_EXP_FUNC.value SPICE_EK_EXP_EXPR = _SPICE_EK_EXP_EXPR.value
[docs]class SpiceSPK18Subtype(c_int): _S18TP0 = c_int(0) _S18TP1 = c_int(1) S18TP0 = _S18TP0.value S18TP1 = _S18TP1.value _fields_ = [ ('S18TP0', _S18TP0), ('S18TP1', _S18TP1) ]
[docs]def emptySpiceEKExprClassVector(n): if isinstance(n, c_int): n = n.value assert(isinstance(n, int)) return (SpiceEKExprClass * n)()
[docs]class SpiceEKAttDsc(Structure): _fields_ = [ ('_cclass', c_int), ('_dtype', SpiceEKDataType), ('_strlen', c_int), ('_size', c_int), ('_indexd', c_int), ('_nullok', c_int) ] @property def cclass(self): return self._cclass @property def dtype(self): return self._dtype.value @property def strlen(self): return self._strlen @property def size(self): return self._size @property def indexd(self): return bool(self._indexd) @property def nullok(self): return bool(self._nullok) def __str__(self): return '<SpiceEKAttDsc cclass = %s, dtype = %s, strlen = %s, size = %s, indexd = %s, nullok = %s >' % \ (self.cclass, self.dtype, self.strlen, self.size, self.indexd, self.nullok)
[docs]class SpiceEKSegSum(Structure): _fields_ = [ ('_tabnam', c_char * 65), ('_nrows', c_int), ('_ncols', c_int), ('_cnames', (c_char * 100) * 33), ('_cdescrs', SpiceEKAttDsc * 100) ] @property def tabnam(self): return toPythonString(self._tabnam) @property def nrows(self): return self._nrows @property def ncols(self): return self._ncols @property def cnames(self): return cVectorToPython(self._cnames)[0:self.ncols] @property def cdescrs(self): return self._cdescrs[0:self.ncols] def __str__(self): return '<SpiceEKSegSum tabnam = %s, nrows = %s, ncols = %s, cnames = %s, cdescrs = %s >' % (self.tabnam, self.nrows, self.ncols, self.cnames, self.cdescrs)
# SpiceCell implementation below is inpart from github.com/DaRasch/spiceminer/ # and modified as needed for this author, maybe we should work together? ### helper classes/functions ### BITSIZE = {'char': sizeof(c_char), 'int': sizeof(c_int), 'double': sizeof(c_double), 'bool': sizeof(c_int), 'time': sizeof(c_int)} def _char_getter(data_p, index, length): return toPythonString((c_char * length).from_address(data_p + index * length * BITSIZE['char'])) def _double_getter(data_p, index, length): return c_double.from_address(data_p + index * BITSIZE['double']).value def _int_getter(data_p, index, length): return c_int.from_address(data_p + index * BITSIZE['int']).value
[docs]def SPICEDOUBLE_CELL(size): """ Returns a Double Spice Cell with a given size :param size: number of elements :type size: int :return: empty Spice Cell :rtype: spiceypy.utils.support_types.SpiceCell """ return SpiceCell.double(size)
[docs]def SPICEINT_CELL(size): """ Returns a Int Spice Cell with a given size :param size: number of elements :type size: int :return: empty Spice Cell :rtype: spiceypy.utils.support_types.SpiceCell """ return SpiceCell.integer(size)
[docs]def SPICECHAR_CELL(size, length): """ Returns a Char Spice Cell with a given size :param size: number of elements :type size: int :param length: width of elements :type length: int :return: empty Spice Cell :rtype: spiceypy.utils.support_types.SpiceCell """ return SpiceCell.character(size, length)
[docs]def SPICEBOOL_CELL(size): """ Returns a Bool Spice Cell with a given size :param size: number of elements :type size: int :return: empty Spice Cell :rtype: spiceypy.utils.support_types.SpiceCell """ return SpiceCell.bool(size)
[docs]def SPICETIME_CELL(size): """ Returns a Time Spice Cell with a given size :param size: number of elements :type size: int :return: empty Spice Cell :rtype: spiceypy.utils.support_types.SpiceCell """ return SpiceCell.time(size)
[docs]class SpiceCell(Structure): #Most written by DaRasch, see included MIT license at file header DATATYPES_ENUM = {'char': 0, 'double': 1, 'int': 2, 'time': 3, 'bool': 4} DATATYPES_GET = [_char_getter, _double_getter] + [_int_getter] * 3 baseSize = 6 minCharLen = 6 CTRLBLOCK = 6 _fields_ = [ ('dtype', c_int), ('length', c_int), ('size', c_int), ('card', c_int), ('isSet', c_int), ('adjust', c_int), ('init', c_int), ('base', c_void_p), ('data', c_void_p) ] def __init__(self, dtype=None, length=None, size=None, card=None, isSet=None, base=None, data=None): super(SpiceCell, self).__init__() self.dtype = dtype self.length = length self.size = size self.card = card self.isSet = isSet self.adjust = 0 # Always False, because not implemented self.init = 0 # Always False, because this is the constructor self.base = base # void pointer self.data = data def __str__(self): return '<SpiceCell dtype = %s, length = %s, size = %s, card = %s,' \ ' isSet = %s, adjust = %s, init = %s, base = %s, data = %s>' % \ (self.dtype, self.length, self.size, self.card, self.isSet, self.adjust, self.init, self.base, self.data)
[docs] def is_int(self): return self.dtype == 2
[docs] def is_double(self): return self.dtype == 1
[docs] def is_char(self): return self.dtype == 0
[docs] def is_time(self): return self.dtype == 3
[docs] def is_bool(self): return self.dtype == 4
[docs] def is_set(self): return self.isSet == 1
[docs] @classmethod def character(cls, size, length): base = (c_char * ((cls.CTRLBLOCK + size) * length))() data = (c_char * (size * length)).from_buffer(base, cls.CTRLBLOCK * BITSIZE['char'] * length) instance = cls(cls.DATATYPES_ENUM['char'], length, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p)) return instance
[docs] @classmethod def integer(cls, size): base = (c_int * (cls.CTRLBLOCK + size))() data = (c_int * size).from_buffer(base, cls.CTRLBLOCK * BITSIZE['int']) instance = cls(cls.DATATYPES_ENUM['int'], 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p)) return instance
[docs] @classmethod def double(cls, size): base = (c_double * (cls.CTRLBLOCK + size))() data = (c_double * size).from_buffer(base, cls.CTRLBLOCK * BITSIZE['double']) instance = cls(cls.DATATYPES_ENUM['double'], 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p)) return instance
[docs] @classmethod def bool(cls, size): base = (c_int * (cls.CTRLBLOCK + size))() data = (c_int * size).from_buffer(base, cls.CTRLBLOCK * BITSIZE['bool']) instance = cls(cls.DATATYPES_ENUM['bool'], 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p)) return instance
[docs] @classmethod def time(cls, size): base = (c_int * (cls.CTRLBLOCK + size))() data = (c_int * size).from_buffer(base, cls.CTRLBLOCK * BITSIZE['time']) instance = cls(cls.DATATYPES_ENUM['time'], 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p)) return instance
def __len__(self): return self.card def __iter__(self): getter = SpiceCell.DATATYPES_GET[self.dtype] length, card, data = self.length, self.card, self.data for i in six.moves.range(card): yield (getter(data, i, length)) def __contains__(self, key): return key in self.__iter__() def __getitem__(self, key): getter = SpiceCell.DATATYPES_GET[self.dtype] if isinstance(key, slice): #TODO Typechecking if self.card == 0: return [] else: start, stop, step = key.indices(self.card) return [getter(self.data, i, self.length) for i in six.moves.range(start, stop, step)] elif key in six.moves.range(-self.card, self.card): index = key if key >= 0 else self.card - abs(key) return getter(self.data, index, self.length) elif not isinstance(key, int): raise TypeError('SpiceCell indices must be integers, not {}'.format(type(key))) else: raise IndexError('SpiceCell index out of range')
[docs] def reset(self): self.card = 0 self.init = 0
def __eq__(self, other): """ element wise equality, other can be a list or cell I think sets should not equal a non set even if elements are equal... might be a bad idea :param other: :return: """ if len(self) != len(other): return False if not hasattr(other, '__iter__'): return False if isinstance(other, SpiceCell): if other.dtype != self.dtype: return False if other.isSet != self.isSet: return False for x, y in zip(self, other): if x != y: return False return True
# Spice Cell classes
[docs]class Cell_Time(SpiceCell): def __init__(self, size): """ Init a Time Spice Cell with a given size and length :param size: number of elements :type size: int """ base = (c_int * (6 + size))() data = (c_int * size).from_buffer(base, 6 * BITSIZE['time']) super(Cell_Time, self).__init__(3, 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p))
[docs]class Cell_Bool(SpiceCell): def __init__(self, size): """ Init a Bool Spice Cell with a given size and length :param size: number of elements :type size: int """ base = (c_int * (6 + size))() data = (c_int * size).from_buffer(base, 6 * BITSIZE['bool']) super(Cell_Bool, self).__init__(4, 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p))
[docs]class Cell_Int(SpiceCell): def __init__(self, size): """ Init a Int Spice Cell with a given size and length :param size: number of elements :type size: int """ base = (c_int * (6 + size))() data = (c_int * size).from_buffer(base, 6 * BITSIZE['int']) super(Cell_Int, self).__init__(2, 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p))
[docs]class Cell_Double(SpiceCell): def __init__(self, size): """ Init a Double Spice Cell with a given size and length :param size: number of elements :type size: int """ base = (c_double * (6 + size))() data = (c_double * size).from_buffer(base, 6 * BITSIZE['double']) super(Cell_Double, self).__init__(1, 0, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p))
[docs]class Cell_Char(SpiceCell): def __init__(self, size, length): """ Init a Char Spice Cell with a given size and length :param size: number of elements :type size: int :param length: width of elements :type length: int """ base = (c_char * ((6 + size) * length))() data = (c_char * (size * length)).from_buffer(base, 6 * BITSIZE['char'] * length) super(Cell_Char, self).__init__(0, length, size, 0, 1, cast(base, c_void_p), cast(data, c_void_p))