| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 | /* vim: set tabstop=8 shiftwidth=4 softtabstop=4 expandtab smarttab colorcolumn=80: *//* * Copyright 2017 Red Hat, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */#include <jose/io.h>#include "misc.h"#include <string.h>typedef struct {    jose_io_t io;    void **buf;    size_t *len;} io_malloc_t;typedef struct {    jose_io_t io;    uint8_t *buf;    size_t max;    size_t *len;} io_buffer_t;typedef struct {    jose_io_t io;    FILE *file;} io_file_t;typedef struct {    jose_io_t io;    bool all;    size_t nnexts;    jose_io_t *nexts[];} io_plex_t;voidjose_io_auto(jose_io_t **io){    if (!io || !*io)        return;    jose_io_decref(*io);    *io = NULL;}jose_io_t *jose_io_incref(jose_io_t *io){    if (!io)        return NULL;    io->refs++;    return io;}voidjose_io_decref(jose_io_t *io){    if (!io)        return;    if (io->refs-- == 1)        io->free(io);}static boolmalloc_feed(jose_io_t *io, const void *in, size_t len){    io_malloc_t *i = containerof(io, io_malloc_t, io);    uint8_t *tmp = NULL;    if (len == 0)        return true;    tmp = realloc(*i->buf, *i->len + len);    if (!tmp)        return false;    memcpy(&tmp[*i->len], in, len);    *i->buf = tmp;    *i->len += len;    return true;}static boolmalloc_done(jose_io_t *io){    return true;}static voidmalloc_free(jose_io_t *io){    io_malloc_t *i = containerof(io, io_malloc_t, io);    if (i->buf && *i->buf && i->len) {        zero(*i->buf, *i->len);        free(*i->buf);        *i->len = 0;    }    zero(i, sizeof(*i));    free(i);}jose_io_t *jose_io_malloc(jose_cfg_t *cfg, void **buf, size_t *len){    jose_io_auto_t *io = NULL;    io_malloc_t *i = NULL;    if (!buf || !len)        return NULL;    i = calloc(1, sizeof(*i));    if (!i)        return NULL;    io = jose_io_incref(&i->io);    io->feed = malloc_feed;    io->done = malloc_done;    io->free = malloc_free;    i->buf = buf;    i->len = len;    return jose_io_incref(io);}void *jose_io_malloc_steal(void **buf){    if (!buf)        return NULL;    void *out = *buf;    *buf = NULL;    return out;}static boolbuffer_feed(jose_io_t *io, const void *in, size_t len){    io_buffer_t *i = containerof(io, io_buffer_t, io);    if (len > i->max - *i->len)        return false;    memcpy(&i->buf[*i->len], in, len);    *i->len += len;    return true;}static boolbuffer_done(jose_io_t *io){    return true;}static voidbuffer_free(jose_io_t *io){    io_buffer_t *i = containerof(io, io_buffer_t, io);    zero(i, sizeof(*i));    free(i);}jose_io_t *jose_io_buffer(jose_cfg_t *cfg, void *buf, size_t *len){    jose_io_auto_t *io = NULL;    io_buffer_t *i = NULL;    if (!buf || !len)        return NULL;    i = calloc(1, sizeof(*i));    if (!i)        return NULL;    io = jose_io_incref(&i->io);    io->feed = buffer_feed;    io->done = buffer_done;    io->free = buffer_free;    i->buf = buf;    i->max = *len;    i->len = len;    *len = 0;    return jose_io_incref(io);}static boolfile_feed(jose_io_t *io, const void *in, size_t len){    io_file_t *i = containerof(io, io_file_t, io);    return fwrite(in, 1, len, i->file) == len;}static boolfile_done(jose_io_t *io){    return true;}static voidfile_free(jose_io_t *io){    io_file_t *i = containerof(io, io_file_t, io);    zero(i, sizeof(*i));    free(i);}jose_io_t *jose_io_file(jose_cfg_t *cfg, FILE *file){    jose_io_auto_t *io = NULL;    io_file_t *i = NULL;    if (!file)        return NULL;    i = calloc(1, sizeof(*i));    if (!i)        return NULL;    io = jose_io_incref(&i->io);    io->feed = file_feed;    io->done = file_done;    io->free = file_free;    i->file = file;    return jose_io_incref(&i->io);}static boolplex_feed(jose_io_t *io, const void *in, size_t len){    io_plex_t *i = containerof(io, io_plex_t, io);    bool status = false;    for (size_t j = 0; j < i->nnexts; j++) {        bool s = false;        if (!i->nexts[j])            continue;        s = i->nexts[j]->feed(i->nexts[j], in, len);        status |= s;        if (!s) {            jose_io_auto(&i->nexts[j]);            if (i->all)                return false;        }    }    return status;}static boolplex_done(jose_io_t *io){    io_plex_t *i = containerof(io, io_plex_t, io);    bool status = false;    for (size_t j = 0; j < i->nnexts; j++) {        bool s = false;        if (!i->nexts[j])            continue;        s = i->nexts[j]->done(i->nexts[j]);        status |= s;        if (!s) {            jose_io_auto(&i->nexts[j]);            if (i->all)                return false;        }    }    return status;}static voidplex_free(jose_io_t *io){    io_plex_t *i = containerof(io, io_plex_t, io);    for (size_t j = 0; j < i->nnexts; j++)        jose_io_decref(i->nexts[j]);    zero(i, sizeof(*i));    free(i);}jose_io_t *jose_io_multiplex(jose_cfg_t *cfg, jose_io_t **nexts, bool all){    jose_io_auto_t *io = NULL;    io_plex_t *i = NULL;    size_t nnexts = 0;    while (nexts && nexts[nnexts])        nnexts++;    i = calloc(1, sizeof(*i) + sizeof(jose_io_t *) * nnexts);    if (!i)        return NULL;    io = jose_io_incref(&i->io);    io->feed = plex_feed;    io->done = plex_done;    io->free = plex_free;    i->all = all;    i->nnexts = nnexts;    for (size_t j = 0; nexts && j < nnexts; j++)        i->nexts[j] = jose_io_incref(nexts[j]);    return jose_io_incref(&i->io);}
 |