X-Git-Url: https://git.distorted.org.uk/~mdw/mLib-python/blobdiff_plain/20bce5e92b01cd928f26b61be78215117039c561..579d01693c86259110fe7a2c2a6f005f1bdbad5b:/pkbuf.pyx diff --git a/pkbuf.pyx b/pkbuf.pyx new file mode 100644 index 0000000..9626a98 --- /dev/null +++ b/pkbuf.pyx @@ -0,0 +1,127 @@ +# -*-pyrex-*- +# +# $Id$ +# +# Packet buffering +# +# (c) 2005 Straylight/Edgeware +# + +#----- Licensing notice ----------------------------------------------------- +# +# This file is part of the Python interface to mLib. +# +# mLib/Python is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# (at your option) any later version. +# +# mLib/Python is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with mLib/Python; if not, write to the Free Software Foundation, +# Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + +cdef class PacketBuffer: + cdef pkbuf pk + cdef _packet + cdef _eof + def __new__(me, packetproc = None, eofproc = None, *hunoz, **hukairz): + pkbuf_init(&me.pk, _pkfunc, me) + me._packet = _checkcallable(packetproc, 'packet proc') + me._eof = _checkcallable(eofproc, 'eof proc') + def __dealloc__(me): + pkbuf_destroy(&me.pk) + property activep: + def __get__(me): + return _tobool(me.pk.f & PKBUF_ENABLE) + property want: + def __get__(me): + return me.pk.want + def __set__(me, want): + if want <= 0: + raise TypeError, 'want must be positive' + pkbuf_want(&me.pk, pk) + property packetproc: + def __get__(me): + return me._packet + def __set__(me, proc): + me._line = _checkcallable(proc, 'packet proc') + def __del__(me): + me._line = None + property eofproc: + def __get__(me): + return me._eof + def __set__(me, proc): + me._eof = _checkcallable(proc, 'eof proc') + def __del__(me): + me._eof = None + def enable(me): + if me.pk.f & PKBUF_ENABLE: + raise ValueError, 'already enabled' + me.pk.f = me.pk.f | PKBUF_ENABLE + me.enabled() + return me + def disable(me): + if not (me.pk.f & PKBUF_ENABLE): + raise ValueError, 'already disabled' + me.pk.f = me.pk.f & ~PKBUF_ENABLE + me.disabled() + return me + def close(me): + if not (me.pk.f & PKBUF_ENABLE): + raise ValueError, 'buffer disabled' + pkbuf_close(&me.pk) + return me + property free: + def __get__(me): + cdef unsigned char *p + return pkbuf_free(&me.pk, &p) + def flush(me, str): + cdef int len + cdef unsigned char *p + cdef unsigned char *q + cdef size_t n + PyObject_AsReadBuffer(str, &p, &len) + while len > 0: + n = pkbuf_free(&me.pk, &q) + if n > len: + n = len + memcpy(q, p, n) + p = p + n + len = len - n + if not (me.pk.f & PKBUF_ENABLE): + break + pkbuf_flush(&me.pk, q, n) + return PyString_FromStringAndSize(p, len) + def enabled(me): + pass + def disabled(me): + pass + def packet(me, pk): + return _maybecall(me._packet, (pk,)) + def eof(me): + return _maybecall(me._eof, ()) + +cdef void _pkfunc(unsigned char *p, size_t n, pkbuf *pk, + size_t *keep, void *arg): + cdef PacketBuffer pb + cdef void *rp + cdef int rn + pb = arg + if p is NULL: + pb.eof() + else: + r = pb.packet(PyString_FromStringAndSize(p, n)) + if r is not None: + PyObject_AsReadBuffer(r, &rp, &rn) + if rn > n: + raise ValueError, 'remaining buffer too large' + if rn: + memcpy(p + n - rn, rp, rn) + keep[0] = rn + +#----- That's all, folks ----------------------------------------------------