| """Buffers are character arrays that may contain null bytes. |
| |
| There are a number of variants depending on: |
| - how the buffer is allocated (for output buffers), and |
| - whether and how the size is passed into and/or out of the called function. |
| """ |
| |
| |
| from bgenType import Type, InputOnlyMixIn, OutputOnlyMixIn, InputOnlyType, OutputOnlyType |
| from bgenOutput import * |
| |
| |
| # Map common types to their format characters |
| type2format = { |
| 'long': 'l', |
| 'int': 'i', |
| 'short': 'h', |
| 'char': 'b', |
| 'unsigned long': 'l', |
| 'unsigned int': 'i', |
| 'unsigned short': 'h', |
| 'unsigned char': 'b', |
| } |
| |
| |
| # ----- PART 1: Fixed character buffers ----- |
| |
| |
| class FixedInputOutputBufferType(InputOnlyType): |
| |
| """Fixed buffer -- passed as (inbuffer, outbuffer).""" |
| |
| def __init__(self, size, datatype = 'char', sizetype = 'int', sizeformat = None): |
| self.typeName = "Buffer" |
| self.size = str(size) |
| self.datatype = datatype |
| self.sizetype = sizetype |
| self.sizeformat = sizeformat or type2format[sizetype] |
| |
| def declare(self, name): |
| self.declareBuffer(name) |
| self.declareSize(name) |
| |
| def declareBuffer(self, name): |
| self.declareInputBuffer(name) |
| self.declareOutputBuffer(name) |
| |
| def declareInputBuffer(self, name): |
| Output("%s *%s__in__;", self.datatype, name) |
| |
| def declareOutputBuffer(self, name): |
| Output("%s %s__out__[%s];", self.datatype, name, self.size) |
| |
| def declareSize(self, name): |
| Output("%s %s__len__;", self.sizetype, name) |
| Output("int %s__in_len__;", name) |
| |
| def getargsFormat(self): |
| return "s#" |
| |
| def getargsArgs(self, name): |
| return "&%s__in__, &%s__in_len__" % (name, name) |
| |
| def getargsCheck(self, name): |
| Output("if (%s__in_len__ != %s)", name, self.size) |
| OutLbrace() |
| Output('PyErr_SetString(PyExc_TypeError, "buffer length should be %s");', |
| self.size) |
| Output("goto %s__error__;", name) |
| OutRbrace() |
| self.transferSize(name) |
| |
| def transferSize(self, name): |
| Output("%s__len__ = %s__in_len__;", name, name) |
| |
| def passOutput(self, name): |
| return "%s__in__, %s__out__" % (name, name) |
| |
| def mkvalueFormat(self): |
| return "s#" |
| |
| def mkvalueArgs(self, name): |
| return "%s__out__, (int)%s" % (name, self.size) |
| |
| def cleanup(self, name): |
| DedentLevel() |
| Output(" %s__error__: ;", name) |
| IndentLevel() |
| |
| |
| class FixedCombinedInputOutputBufferType(FixedInputOutputBufferType): |
| |
| """Like fixed buffer -- but same parameter is input and output.""" |
| |
| def passOutput(self, name): |
| return "(%s *)memcpy(%s__out__, %s__in__, %s)" % \ |
| (self.datatype, name, name, self.size) |
| |
| |
| class InputOnlyBufferMixIn(InputOnlyMixIn): |
| |
| def declareOutputBuffer(self, name): |
| pass |
| |
| |
| class OutputOnlyBufferMixIn(OutputOnlyMixIn): |
| |
| def declareInputBuffer(self, name): |
| pass |
| |
| |
| class FixedInputBufferType(InputOnlyBufferMixIn, FixedInputOutputBufferType): |
| |
| """Fixed size input buffer -- passed without size information. |
| |
| Instantiate with the size as parameter. |
| """ |
| |
| def passInput(self, name): |
| return "%s__in__" % name |
| |
| |
| class FixedOutputBufferType(OutputOnlyBufferMixIn, FixedInputOutputBufferType): |
| |
| """Fixed size output buffer -- passed without size information. |
| |
| Instantiate with the size as parameter. |
| """ |
| |
| def passOutput(self, name): |
| return "%s__out__" % name |
| |
| |
| class VarInputBufferType(FixedInputBufferType): |
| |
| """Variable size input buffer -- passed as (buffer, size). |
| |
| Instantiate without size parameter. |
| """ |
| |
| def __init__(self, datatype = 'char', sizetype = 'int', sizeformat = None): |
| FixedInputBufferType.__init__(self, "0", datatype, sizetype, sizeformat) |
| |
| def getargsCheck(self, name): |
| Output("%s__len__ = %s__in_len__;", name, name) |
| |
| def passInput(self, name): |
| return "%s__in__, %s__len__" % (name, name) |
| |
| |
| # ----- PART 2: Structure buffers ----- |
| |
| |
| class StructInputOutputBufferType(FixedInputOutputBufferType): |
| |
| """Structure buffer -- passed as a structure pointer. |
| |
| Instantiate with the struct type as parameter. |
| """ |
| |
| def __init__(self, type): |
| FixedInputOutputBufferType.__init__(self, "sizeof(%s)" % type) |
| self.typeName = self.type = type |
| |
| def declareInputBuffer(self, name): |
| Output("%s *%s__in__;", self.type, name) |
| |
| def declareSize(self, name): |
| Output("int %s__in_len__;", name) |
| |
| def declareOutputBuffer(self, name): |
| Output("%s %s__out__;", self.type, name) |
| |
| def getargsArgs(self, name): |
| return "(char **)&%s__in__, &%s__in_len__" % (name, name) |
| |
| def transferSize(self, name): |
| pass |
| |
| def passInput(self, name): |
| return "%s__in__" % name |
| |
| def passOutput(self, name): |
| return "%s__in__, &%s__out__" % (name, name) |
| |
| def mkvalueArgs(self, name): |
| return "(char *)&%s__out__, (int)%s" % (name, self.size) |
| |
| |
| class StructCombinedInputOutputBufferType(StructInputOutputBufferType): |
| |
| """Like structure buffer -- but same parameter is input and output.""" |
| |
| def passOutput(self, name): |
| return "(%s *)memcpy((char *)%s__out__, (char *)%s__in__, %s)" % \ |
| (self.type, name, name, self.size) |
| |
| |
| class StructInputBufferType(InputOnlyBufferMixIn, StructInputOutputBufferType): |
| |
| """Fixed size input buffer -- passed as a pointer to a structure. |
| |
| Instantiate with the struct type as parameter. |
| """ |
| |
| |
| class StructByValueBufferType(StructInputBufferType): |
| |
| """Fixed size input buffer -- passed as a structure BY VALUE. |
| |
| Instantiate with the struct type as parameter. |
| """ |
| |
| def passInput(self, name): |
| return "*%s__in__" % name |
| |
| |
| class StructOutputBufferType(OutputOnlyBufferMixIn, StructInputOutputBufferType): |
| |
| """Fixed size output buffer -- passed as a pointer to a structure. |
| |
| Instantiate with the struct type as parameter. |
| """ |
| |
| def declareSize(self, name): |
| pass |
| |
| def passOutput(self, name): |
| return "&%s__out__" % name |
| |
| |
| class ArrayOutputBufferType(OutputOnlyBufferMixIn, StructInputOutputBufferType): |
| |
| """Fixed size output buffer -- declared as a typedef, passed as an array. |
| |
| Instantiate with the struct type as parameter. |
| """ |
| |
| def declareSize(self, name): |
| pass |
| |
| def passOutput(self, name): |
| return "%s__out__" % name |