OSSP CVS Repository

ossp - ossp-pkg/sio/sio_buffer.c 1.1
Not logged in
[Honeypot]  [Browse]  [Directory]  [Home]  [Login
[Reports]  [Search]  [Ticket]  [Timeline
  [Raw

ossp-pkg/sio/sio_buffer.c 1.1
#include <stddef.h>
#include <stdlib.h>
#include <string.h>

#include "al.h"
#include "sio.h"
#include "sio_module.h"

typedef struct {
    size_t outputsize;
    al_tx_t *outputtx;
    size_t inputsize;
    al_t *input, *output;
} private_t;

/*
 * create stage
 *
 * allocate private instance data
 */
static
sio_rc_t buffer_init(sio_t *sio, void **u)
{
    private_t *mydata;
    
    mydata = (private_t *)malloc(sizeof(private_t));
    if (mydata == NULL)
        return SIO_ERR_MEM;

    mydata->inputsize    = 0;
    mydata->outputsize   = 0;

    *u = mydata;

    return SIO_OK;
}

/*
 * configure stage
 *
 * pass two void pointers
 */
static
sio_rc_t buffer_configure(sio_t *sio, void *u, void *obj, void *val)
{
    private_t *mydata = (private_t *)u;
    const char *name = (const char *)obj;

    if (!strcmp(name, "inputsize")) {
        mydata->inputsize = *(size_t *)val;
    } else if (!strcmp(name, "outputsize")) {
        mydata->outputsize = *(size_t *)val;
    } else {
        return SIO_ERR_ARG;
    }

    return SIO_OK;
}

/*
 * destroy stage
 */
static
sio_rc_t buffer_cleanup(sio_t *sio, void *u)
{
    private_t *mydata = (private_t *)u;

    free(mydata);

    return SIO_OK;
}

static
sio_rc_t buffer_openr(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;
    al_rc_t arc;

    arc = al_create(&mydata->input);
    if (arc != AL_OK)
        return SIO_ERR_INT;

    return SIO_OK;
}

static
sio_rc_t buffer_closer(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;

    al_destroy(mydata->input);
    mydata->input = NULL;

    return SIO_OK;
}

static
sio_rc_t buffer_openw(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;
    al_rc_t arc;

    arc = al_txalloc(al, &mydata->outputtx);
    if (arc != AL_OK)
        return SIO_ERR_INT;

    arc = al_create(&mydata->output);
    if (arc != AL_OK) {
        al_txfree(al, mydata->outputtx);
        return SIO_ERR_INT;
    }

    return SIO_OK;
}

static
sio_rc_t buffer_closew(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;

    al_destroy(mydata->output);
    mydata->output = NULL;
    al_txfree(al, mydata->outputtx);
    mydata->outputtx = NULL;

    return SIO_OK;
}

static
sio_rc_t buffer_inout(sio_t *sio, al_t *al, al_t *buf, size_t size,
                      sio_rc_t up, sio_rc_t down)
{
    size_t avail, data, needed;
    al_rc_t arc;
    al_label_t label;

    arc = al_splice(buf, al_bytes(buf), 0, al, NULL);
    if (arc != AL_OK)
        return SIO_ERR_INT;

    avail = al_bytes(buf);
    if (avail <= 0)
        return down;

    arc = al_firstlabel(buf, 0, 1, AL_FORWARD, NULL, &label);
    if (arc != AL_OK)
        return SIO_ERR_INT;

    al_flatten(buf, 0, avail, AL_FORWARD_SPAN, label, NULL, &data);

    needed = size;
    if (data < avail)
        needed = 1;

    if (data >= needed) {
        if (data >= size)
            data = size;
        al_splice(buf, 0, data, NULL, al);
        return up;
    }

    return down;
}

static
sio_rc_t buffer_input(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;

    return buffer_inout(sio, al, mydata->input, mydata->inputsize,
                        SIO_DOWNSTREAM, SIO_UPSTREAM);
}

static
sio_rc_t buffer_output(sio_t *sio, al_t *al, void *u)
{
    private_t *mydata = (private_t *)u;

    return buffer_inout(sio, al, mydata->output, mydata->outputsize,
                        SIO_UPSTREAM, SIO_DOWNSTREAM);
}

sio_module_t sio_module_buffer = {
    "buffer",
    buffer_init,
    buffer_configure,
    buffer_cleanup,
    buffer_openr,
    buffer_closer,
    buffer_openw,
    buffer_closew,
    buffer_input,
    buffer_output
};


CVSTrac 2.0.1