From c83cf42cc3298131ca428774409401b6545ab75f Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 10:29:35 +0000 Subject: [PATCH 1/7] Feature: Outlined basic IO interface --- Makefile.am | 4 + include/igloo/interface.h | 54 ++++++++ include/igloo/io.h | 255 ++++++++++++++++++++++++++++++++++++++ include/igloo/types.h | 2 + src/interface.c | 32 +++++ src/io.c | 25 ++++ src/private.h | 21 ++++ 7 files changed, 393 insertions(+) create mode 100644 include/igloo/interface.h create mode 100644 include/igloo/io.h create mode 100644 src/interface.c create mode 100644 src/io.c diff --git a/Makefile.am b/Makefile.am index 9e70d15..2af9cb7 100644 --- a/Makefile.am +++ b/Makefile.am @@ -20,13 +20,17 @@ pkginclude_HEADERS = \ include/igloo/ro.h \ include/igloo/types.h \ include/igloo/typedef.h \ + include/igloo/interface.h \ + include/igloo/io.h \ include/igloo/buffer.h \ include/igloo/list.h \ include/igloo/reportxml.h libigloo_la_SOURCES = \ src/libigloo.c \ + src/interface.c \ src/ro.c \ + src/io.c \ src/buffer.c \ src/list.c \ src/reportxml.c diff --git a/include/igloo/interface.h b/include/igloo/interface.h new file mode 100644 index 0000000..59292b1 --- /dev/null +++ b/include/igloo/interface.h @@ -0,0 +1,54 @@ +/* Copyright (C) 2019 Philipp "ph3-der-loewe" Schafft + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, + * Boston, MA 02110-1301, USA. + */ + +#ifndef _LIBIGLOO__INTERFACE_H_ +#define _LIBIGLOO__INTERFACE_H_ +/** + * @file + * Put a good description of this file here + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Put stuff here */ + +#include "ro.h" + +/* Basic arguments that should be passed to all interface functions */ +#define igloo_INTERFACE_BASIC_ARGS igloo_ro_t self, igloo_ro_t *backend_object, void **backend_userdata + +/* Basic free callback. + * If this does not set *backend_object to igloo_RO_NULL + * backend_object will be freed by calling igloo_ro_unref(). + * If this does not set *backend_userdata to NULL + * + */ +typedef int (*igloo_interface_free_t)(igloo_INTERFACE_BASIC_ARGS); + + +typedef struct { + igloo_interface_free_t free; +} igloo_interface_base_ifdesc_t; + +#ifdef __cplusplus +} +#endif + +#endif /* ! _LIBIGLOO__INTERFACE_H_ */ diff --git a/include/igloo/io.h b/include/igloo/io.h new file mode 100644 index 0000000..e3314ec --- /dev/null +++ b/include/igloo/io.h @@ -0,0 +1,255 @@ +/* Copyright (C) 2019 Philipp "ph3-der-loewe" Schafft + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, + * Boston, MA 02110-1301, USA. + */ + +#ifndef _LIBIGLOO__IO_H_ +#define _LIBIGLOO__IO_H_ +/** + * @file + * Put a good description of this file here + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Put stuff here */ + +#include + +#ifdef IGLOO_CTC_HAVE_SYS_SELECT_H +#include +#endif +#ifdef IGLOO_CTC_HAVE_POLL +#include +#endif +#ifdef IGLOO_CTC_HAVE_STDINT_H +#include +#endif +#ifdef IGLOO_CTC_STDC_HEADERS +#include +#endif + +#include "ro.h" +#include "interface.h" + +/* About thread safety: + * This set of functions is thread safe. + */ + +igloo_RO_FORWARD_TYPE(igloo_io_t); + +/* Type of blocking. + */ +typedef enum { + /* Error state, used as return value only. */ + igloo_IO_BLOCKINGMODE_ERROR = -1, + /* No blocking is done. This is the eqivalent of setting O_NONBLOCK on a POSIX system. */ + igloo_IO_BLOCKINGMODE_NONE = 0, + /* Full blocking is done. This is the eqivalent of unsetting O_NONBLOCK on a POSIX system. */ + igloo_IO_BLOCKINGMODE_FULL +} libigloo_io_blockingmode_t; + +/* Type for operaton flags. + */ +#ifdef IGLOO_CTC_HAVE_STDINT_H +typedef uint_least32_t igloo_io_opflag_t; +#else +typedef unsigned long int igloo_io_opflag_t; +#endif + +/* No operation flags set. Usefull for variable initialization. */ +#define igloo_IO_OPFLAG_NONE ((igloo_io_opflag_t)0x0000) +/* Use default flags. Other flags can be set in addition. */ +#define igloo_IO_OPFLAG_DEFAULTS ((igloo_io_opflag_t)0x8000) + +/* Operate on actual data. */ +#define igloo_IO_OPFLAG_DATA ((igloo_io_opflag_t)0x0001) +/* Operate on metadata. */ +#define igloo_IO_OPFLAG_METADATA ((igloo_io_opflag_t)0x0002) +/* Operate on data and metadata. */ +#define igloo_IO_OPFLAG_FULL (igloo_IO_OPFLAG_DATAONLY|igloo_IO_OPFLAG_METADATAONLY) +/* Instructs the operation that it should ignore the output state of the object. + * This may improve performance as buffer flushes may be skipped. + * However with this set any external software interacting with this object + * MUST NOT interact with the output side of this object. + */ +#define igloo_IO_OPFLAG_NOWRITE ((igloo_io_opflag_t)0x0010) + +/* Get the referenced value */ +#define igloo_IO_OPFLAG_GET ((igloo_io_opflag_t)0x0100) +/* Set the referenced value */ +#define igloo_IO_OPFLAG_SET ((igloo_io_opflag_t)0x0200) + +/* Advanced control functions. + */ +typedef enum { + igloo_IO_CONTROL_NONE = 0 +} igloo_io_control_t; + +/* Interface description */ +typedef struct { + igloo_interface_base_ifdesc_t __base; + + ssize_t (*read)(igloo_INTERFACE_BASIC_ARGS, void *buffer, size_t len); + ssize_t (*write)(igloo_INTERFACE_BASIC_ARGS, const void *buffer, size_t len); + int (*flush)(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags); + int (*sync)(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags); + int (*set_blockingmode)(igloo_INTERFACE_BASIC_ARGS, libigloo_io_blockingmode_t mode); + libigloo_io_blockingmode_t (*get_blockingmode)(igloo_INTERFACE_BASIC_ARGS); + int (*get_fd_for_systemcall)(igloo_INTERFACE_BASIC_ARGS); +#ifdef IGLOO_CTC_STDC_HEADERS + int (*control)(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags, igloo_io_control_t control, va_list ap); +#else + int (*control)(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags, igloo_io_control_t control, ...); +#endif +} igloo_io_ifdesc_t; + +/* Read data from a IO handle. + * Parameters: + * io + * The IO handle to operate on. + * buffer + * The buffer to read to. + * len + * The maximum amount of bytes to read. + * Returns: + * The actual amount of bytes read. + */ +ssize_t igloo_io_read(igloo_io_t *io, void *buffer, size_t len); +/* Write data to a IO handle. + * Parameters: + * io + * The IO handle to operate on. + * buffer + * The buffer to write from. + * len + * The maximum amount of bytes to write. + * Returns: + * The actual amount of bytes written. + */ +ssize_t igloo_io_write(igloo_io_t *io, const void *buffer, size_t len); + +/* Flush internal buffers to the underlying object. + * This does not guarantee that all data has been written to the physical level + * on return. It just queues the flush. + * Parameters: + * io + * The IO handle to operate on. + * flags + * Flags for this operation. + */ +int igloo_io_flush(igloo_io_t *io, igloo_io_opflag_t flags); +/* Sync object with physical state. This is used to get the object into a state + * that allows passing the underlying object to other software. + * This may also flush internal buffers. + * Parameters: + * io + * The IO handle to operate on. + * flags + * Flags for this operation. + */ +int igloo_io_sync(igloo_io_t *io, igloo_io_opflag_t flags); + +/* Set and get the blocking state of the object. + * Parameters: + * io + * The IO handle to operate on. + * mode + * The new blocking mode. + */ +int igloo_io_set_blockingmode(igloo_io_t *io, libigloo_io_blockingmode_t mode); +libigloo_io_blockingmode_t igloo_io_get_blockingmode(igloo_io_t *io); + +#ifdef IGLOO_CTC_HAVE_SYS_SELECT_H +/* Those functions act as igloo's replacement for FD_SET(), FD_CLR(), and FD_SET() and + * allow the caller to use the select() system call. + * Parameters: + * io + * The IO handle to operate on. + * set + * The fd_set to operate on. + * maxfd + * This parameter is updated to reflect the added handle(s) + * in select()'s nfds parameter. + * flags + * Flags for this operation. + * It may be useful to igloo_IO_OPFLAG_NOWRITE in some cases. See it's definition. + * Notes: + * This may call igloo_io_sync(). If the object is touched between calls to igloo_io_select_set(), + * select() the user MUST call igloo_io_sync() with the correct flags. It is RECOMMENDED + * to avoid this, as finding the correct parameters for igloo_io_sync() might be tricky. + * The object MUST NOT be touched between select() and igloo_io_select_isset(). + */ +int igloo_io_select_set(igloo_io_t *io, fd_set *set, int *maxfd, igloo_io_opflag_t flags); +int igloo_io_select_clear(igloo_io_t *io, fd_set *set); +int igloo_io_select_isset(igloo_io_t *io, fd_set *set); +#endif + +#ifdef IGLOO_CTC_HAVE_POLL +/* This function initializes the pollfd structure to use this handle + * with the poll() system call. + * Parameters: + * io + * The IO handle to operate on. + * fd + * The structure to initialize. + * events + * The events to set. libigloo uses different internal parameters based + * on this parameter's value. The parameter MUST NOT be changed in the structre + * after initialization. + * Notes: + * This may call igloo_io_sync(). If the object is touched between calls to igloo_io_poll_fill(), + * poll() the user MUST call igloo_io_sync() with the correct flags. It is RECOMMENDED + * to avoid this, as finding the correct parameters for igloo_io_sync() might be tricky. + */ +int igloo_io_poll_fill(igloo_io_t *io, struct pollfd *fd, short events); +#endif + +/* On a listen socket accept a new connection. + * Parameters: + * io + * The IO handle to operate on. + * Returns: + * The new connection or igloo_RO_NULL. + */ +/* TODO: Allow accept to accept()'s and accept4()'s additional parameters. +igloo_io_t *igloo_io_accept(igloo_io_t *io); +*/ + +/* Advanced control interface. + * Parameters: + * io + * The IO handle to operate on. + * flags + * Flags for this operation. + * Most controls accept igloo_IO_OPFLAG_GET and igloo_IO_OPFLAG_SET. + * If both are set the value is first set as with igloo_IO_OPFLAG_SET and + * the OLD value is returned as with igloo_IO_OPFLAG_GET. + * control + * The control command to use. + * ... + * Additional parameters if any. + * Values are always passed as pointers. + */ +int igloo_io_control(igloo_io_t *io, igloo_io_opflag_t flags, igloo_io_control_t control, ...); + +#ifdef __cplusplus +} +#endif + +#endif /* ! _LIBIGLOO__IO_H_ */ diff --git a/include/igloo/types.h b/include/igloo/types.h index 9ea33bf..5a1e7c1 100644 --- a/include/igloo/types.h +++ b/include/igloo/types.h @@ -32,6 +32,7 @@ extern "C" { /* Included in case is not yet included */ #include "typedef.h" +typedef struct igloo_io_tag igloo_io_t; typedef struct igloo_buffer_tag igloo_buffer_t; typedef struct igloo_list_tag igloo_list_t; @@ -50,6 +51,7 @@ igloo_RO_FORWARD_TYPE(igloo_ro_base_t); typedef union __attribute__ ((__transparent_union__)) { /* Those are libigloo's own types */ igloo_RO_TYPE(igloo_ro_base_t) + igloo_RO_TYPE(igloo_io_t) igloo_RO_TYPE(igloo_buffer_t) igloo_RO_TYPE(igloo_list_t) igloo_RO_TYPE(igloo_reportxml_t) diff --git a/src/interface.c b/src/interface.c new file mode 100644 index 0000000..48c09f8 --- /dev/null +++ b/src/interface.c @@ -0,0 +1,32 @@ +/* Icecast + * + * This program is distributed under the GNU General Public License, version 2. + * A copy of this license is included with this source. + * + * Copyright 2019, Philipp "ph3-der-loewe" Schafft , + */ + +#ifdef HAVE_CONFIG_H +#include +#endif + +#include + +#include +#include +#include "private.h" + +void igloo_interface_base_free(igloo_ro_t self) +{ + igloo__interface_base_t *iface = igloo_INTERFACE_CAST(self); + + if (iface->ifdesc->free) + iface->ifdesc->free(igloo_INTERFACE_BASIC_CALL(self)); + + if (!igloo_RO_IS_NULL(iface->backend_object)) + igloo_ro_unref(iface->backend_object); + + if (iface->backend_userdata) + free(iface->backend_userdata); +} + diff --git a/src/io.c b/src/io.c new file mode 100644 index 0000000..88166fe --- /dev/null +++ b/src/io.c @@ -0,0 +1,25 @@ +/* Icecast + * + * This program is distributed under the GNU General Public License, version 2. + * A copy of this license is included with this source. + * + * Copyright 2019, Philipp "ph3-der-loewe" Schafft , + */ + +#ifdef HAVE_CONFIG_H +#include +#endif + +#include + +#include +#include +#include "private.h" + +struct igloo_io_tag { + igloo_interface_base(io) +}; + +igloo_RO_PUBLIC_TYPE(igloo_io_t, + igloo_RO_TYPEDECL_FREE(igloo_interface_base_free) + ); diff --git a/src/private.h b/src/private.h index ed7ed5a..113519f 100644 --- a/src/private.h +++ b/src/private.h @@ -19,6 +19,8 @@ #ifndef _LIBIGLOO__PRIVATE_H_ #define _LIBIGLOO__PRIVATE_H_ +#include + /* init/shutdown of the library */ void igloo_thread_initialize(void); void igloo_thread_initialize_with_log_id(int log_id); @@ -30,5 +32,24 @@ void igloo_resolver_shutdown(void); void igloo_log_initialize(void); void igloo_log_shutdown(void); +/* Basic interface */ +#define igloo_interface_base(type) \ + /* The base object. */ \ + igloo_ro_base_t __base; \ + /* The interface description */ \ + const igloo_ ## type ## _ifdesc_t *ifdesc; \ + /* Backend object */ \ + igloo_ro_t backend_object; \ + /* Backend userdata */ \ + void *backend_userdata; + +typedef struct { + igloo_interface_base(interface_base) +} igloo__interface_base_t; + +void igloo_interface_base_free(igloo_ro_t self); + +#define igloo_INTERFACE_CAST(obj) ((igloo__interface_base_t*)igloo_RO__GETBASE(obj)) +#define igloo_INTERFACE_BASIC_CALL(obj) (obj), &(igloo_INTERFACE_CAST(obj)->backend_object), &(igloo_INTERFACE_CAST(obj)->backend_userdata) #endif From 71176a6fc5b54529b3abde1d714f7047a759bc1a Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 11:02:18 +0000 Subject: [PATCH 2/7] Feature: Added some basic implementation --- src/io.c | 218 +++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 218 insertions(+) diff --git a/src/io.c b/src/io.c index 88166fe..2a92261 100644 --- a/src/io.c +++ b/src/io.c @@ -11,6 +11,7 @@ #endif #include +#include #include #include @@ -18,8 +19,225 @@ struct igloo_io_tag { igloo_interface_base(io) + int touched; +#if defined(IGLOO_CTC_HAVE_SYS_SELECT_H) || defined(IGLOO_CTC_HAVE_POLL) + int fd; +#endif }; igloo_RO_PUBLIC_TYPE(igloo_io_t, igloo_RO_TYPEDECL_FREE(igloo_interface_base_free) ); + +ssize_t igloo_io_read(igloo_io_t *io, void *buffer, size_t len) +{ + if (!io || !buffer) + return -1; + + if (!len) + return 0; + + io->touched = 1; + + if (io->ifdesc->read) + return io->ifdesc->read(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + + return -1; +} + +ssize_t igloo_io_write(igloo_io_t *io, const void *buffer, size_t len) +{ + if (!io || !buffer) + return -1; + + if (!len) + return 0; + + io->touched = 1; + + if (io->ifdesc->write) + return io->ifdesc->write(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + + return -1; +} + +int igloo_io_flush(igloo_io_t *io, igloo_io_opflag_t flags) +{ + if (!io) + return -1; + + io->touched = 1; + + if (io->ifdesc->flush) + return io->ifdesc->flush(igloo_INTERFACE_BASIC_CALL(io), flags); + + return -1; +} + +int igloo_io_sync(igloo_io_t *io, igloo_io_opflag_t flags) +{ + int ret; + + if (!io) + return -1; + + if (io->ifdesc->flush) + io->ifdesc->flush(igloo_INTERFACE_BASIC_CALL(io), flags); + + if (io->ifdesc->sync) { + ret = io->ifdesc->sync(igloo_INTERFACE_BASIC_CALL(io), flags); + if (ret != 0) + return ret; + + io->touched = 0; + + return ret; + } + + return -1; +} + +int igloo_io_set_blockingmode(igloo_io_t *io, libigloo_io_blockingmode_t mode) +{ + if (!io) + return -1; + + if (mode != igloo_IO_BLOCKINGMODE_NONE && mode != igloo_IO_BLOCKINGMODE_FULL) + return -1; + + io->touched = 1; + + if (io->ifdesc->set_blockingmode) + return io->ifdesc->set_blockingmode(igloo_INTERFACE_BASIC_CALL(io), mode); + + return -1; +} +libigloo_io_blockingmode_t igloo_io_get_blockingmode(igloo_io_t *io) +{ + if (!io) + return igloo_IO_BLOCKINGMODE_ERROR; + + if (io->ifdesc->get_blockingmode) + return io->ifdesc->get_blockingmode(igloo_INTERFACE_BASIC_CALL(io)); + + return igloo_IO_BLOCKINGMODE_ERROR; +} + +#ifdef IGLOO_CTC_HAVE_SYS_SELECT_H +int igloo_io_select_set(igloo_io_t *io, fd_set *set, int *maxfd, igloo_io_opflag_t flags) +{ + int ret; + + if (!io || !set || !maxfd) + return -1; + + if (!io->ifdesc->get_fd_for_systemcall) + return -1; + + if (io->touched || !(flags & igloo_IO_OPFLAG_NOWRITE)) { + ret = igloo_io_sync(io, igloo_IO_OPFLAG_DEFAULTS|(flags & igloo_IO_OPFLAG_NOWRITE)); + if (ret != 0) + return ret; + if (io->touched) + return -1; + } + + io->fd = io->ifdesc->get_fd_for_systemcall(igloo_INTERFACE_BASIC_CALL(io)); + if (io->fd < 0) + return -1; + + FD_SET(io->fd, set); + if (*maxfd < io->fd) + *maxfd = io->fd; + + return 0; +} + +int igloo_io_select_clear(igloo_io_t *io, fd_set *set) +{ + if (!io || !set) + return -1; + + if (io->touched || io->fd < 0) + return -1; + + FD_CLR(io->fd, set); + + return 0; +} + +int igloo_io_select_isset(igloo_io_t *io, fd_set *set) +{ + if (!io || !set) + return -1; + + if (io->touched || io->fd < 0) + return -1; + + return FD_ISSET(io->fd, set) ? 1 : 0; +} +#endif + +#ifdef IGLOO_CTC_HAVE_POLL +int igloo_io_poll_fill(igloo_io_t *io, struct pollfd *fd, short events) +{ + static const short safe_events = POLLIN|POLLPRI +#ifdef POLLRDHUP + |POLLRDHUP +#endif +#ifdef POLLRDNORM + |POLLRDNORM +#endif + ; + int is_safe; + int ret; + + if (!io || !fd) + return -1; + + if (!io->ifdesc->get_fd_for_systemcall) + return -1; + + is_safe = !((events|safe_events) - safe_events); + + if (io->touched || !is_safe) { + ret = igloo_io_sync(io, igloo_IO_OPFLAG_DEFAULTS|(is_safe ? 0 : igloo_IO_OPFLAG_NOWRITE)); + if (ret != 0) + return ret; + if (io->touched) + return -1; + } + + io->fd = io->ifdesc->get_fd_for_systemcall(igloo_INTERFACE_BASIC_CALL(io)); + if (io->fd < 0) + return -1; + + memset(fd, 0, sizeof(*fd)); + fd->fd = io->fd; + fd->events = events; + + return 0; +} +#endif + +int igloo_io_control(igloo_io_t *io, igloo_io_opflag_t flags, igloo_io_control_t control, ...) +{ +#ifdef IGLOO_CTC_STDC_HEADERS + int ret; + va_list ap; + + if (!io) + return -1; + + if (!io->ifdesc->control) + return -1; + + va_start(ap, control); + ret = io->ifdesc->control(igloo_INTERFACE_BASIC_CALL(io), flags, control, ap); + va_end(ap); + + return ret; +#else + return -1; +#endif +} From 812a2ccaaf1222e87cff00bef374428f84bf2e83 Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 11:26:21 +0000 Subject: [PATCH 3/7] Feature: Added igloo_io_new() --- include/igloo/io.h | 13 +++++++++++++ src/interface.c | 2 +- src/io.c | 28 ++++++++++++++++++++++++++++ 3 files changed, 42 insertions(+), 1 deletion(-) diff --git a/include/igloo/io.h b/include/igloo/io.h index e3314ec..9ad54d7 100644 --- a/include/igloo/io.h +++ b/include/igloo/io.h @@ -119,6 +119,19 @@ typedef struct { #endif } igloo_io_ifdesc_t; +/* This creates a new IO handle from a interface description and state. + * Parameters: + * ifdesc + * The interface description to use. + * backend_object + * A object used by the backend or igloo_RO_NULL. + * backend_userdata + * A userdata pointer used by the backend or NULL. + * name, associated + * See refobject_new(). + */ +igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated); + /* Read data from a IO handle. * Parameters: * io diff --git a/src/interface.c b/src/interface.c index 48c09f8..5271d65 100644 --- a/src/interface.c +++ b/src/interface.c @@ -20,7 +20,7 @@ void igloo_interface_base_free(igloo_ro_t self) { igloo__interface_base_t *iface = igloo_INTERFACE_CAST(self); - if (iface->ifdesc->free) + if (iface->ifdesc && iface->ifdesc->free) iface->ifdesc->free(igloo_INTERFACE_BASIC_CALL(self)); if (!igloo_RO_IS_NULL(iface->backend_object)) diff --git a/src/io.c b/src/io.c index 2a92261..2b54b62 100644 --- a/src/io.c +++ b/src/io.c @@ -29,6 +29,34 @@ igloo_RO_PUBLIC_TYPE(igloo_io_t, igloo_RO_TYPEDECL_FREE(igloo_interface_base_free) ); + +igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated) +{ + igloo_io_t *io; + + if (!ifdesc) + return NULL; + + io = igloo_ro_new_raw(igloo_io_t, name, associated); + + if (!io) + return NULL; + + if (!igloo_RO_IS_NULL(backend_object)) { + if (igloo_ro_ref(backend_object) != 0) { + igloo_ro_unref(io); + return NULL; + } + } + + io->ifdesc = ifdesc; + io->backend_object = backend_object; + io->backend_userdata = backend_userdata; + + return io; +} + + ssize_t igloo_io_read(igloo_io_t *io, void *buffer, size_t len) { if (!io || !buffer) From f5bc4e77c450616b6368c1c9fbdd7094ae3056eb Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 11:55:34 +0000 Subject: [PATCH 4/7] Feature: Added igloo_stdio_new_file() --- Makefile.am | 2 ++ include/igloo/stdio.h | 40 +++++++++++++++++++++++ src/stdio.c | 76 +++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 118 insertions(+) create mode 100644 include/igloo/stdio.h create mode 100644 src/stdio.c diff --git a/Makefile.am b/Makefile.am index 2af9cb7..6475978 100644 --- a/Makefile.am +++ b/Makefile.am @@ -22,6 +22,7 @@ pkginclude_HEADERS = \ include/igloo/typedef.h \ include/igloo/interface.h \ include/igloo/io.h \ + include/igloo/stdio.h \ include/igloo/buffer.h \ include/igloo/list.h \ include/igloo/reportxml.h @@ -31,6 +32,7 @@ libigloo_la_SOURCES = \ src/interface.c \ src/ro.c \ src/io.c \ + src/stdio.c \ src/buffer.c \ src/list.c \ src/reportxml.c diff --git a/include/igloo/stdio.h b/include/igloo/stdio.h new file mode 100644 index 0000000..cb00917 --- /dev/null +++ b/include/igloo/stdio.h @@ -0,0 +1,40 @@ +/* Copyright (C) 2019 Philipp "ph3-der-loewe" Schafft + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, + * Boston, MA 02110-1301, USA. + */ + +#ifndef _LIBIGLOO__STDIO_H_ +#define _LIBIGLOO__STDIO_H_ +/** + * @file + * Put a good description of this file here + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/* Put stuff here */ + +#include "io.h" + +igloo_io_t * igloo_stdio_new_file(const char *filename, const char *mode, const char *name, igloo_ro_t associated); + +#ifdef __cplusplus +} +#endif + +#endif /* ! _LIBIGLOO__STDIO_H_ */ diff --git a/src/stdio.c b/src/stdio.c new file mode 100644 index 0000000..1078f0a --- /dev/null +++ b/src/stdio.c @@ -0,0 +1,76 @@ +/* Icecast + * + * This program is distributed under the GNU General Public License, version 2. + * A copy of this license is included with this source. + * + * Copyright 2019, Philipp "ph3-der-loewe" Schafft , + */ + +#ifdef HAVE_CONFIG_H +#include +#endif + +#include + +#include + +static int __free(igloo_INTERFACE_BASIC_ARGS) +{ + fclose(*backend_userdata); + *backend_userdata = NULL; + return 0; +} + + +static ssize_t __read(igloo_INTERFACE_BASIC_ARGS, void *buffer, size_t len) +{ + return fread(buffer, 1, len, *backend_userdata); +} +static ssize_t __write(igloo_INTERFACE_BASIC_ARGS, const void *buffer, size_t len) +{ + return fwrite(buffer, 1, len, *backend_userdata); +} + +static int __flush(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags) +{ + return fflush(*backend_userdata) == 0 ? 0 : -1; +} +static int __sync(igloo_INTERFACE_BASIC_ARGS, igloo_io_opflag_t flags) +{ + return 0; +} +static libigloo_io_blockingmode_t __get_blockingmode(igloo_INTERFACE_BASIC_ARGS) +{ + return igloo_IO_BLOCKINGMODE_FULL; +} +static int __get_fd_for_systemcall(igloo_INTERFACE_BASIC_ARGS) +{ + return fileno(*backend_userdata); +} + +static const igloo_io_ifdesc_t igloo_stdio_ifdesc = { + .__base = {.free = __free}, + .read = __read, + .write = __write, + .flush = __flush, + .sync = __sync, + .get_blockingmode = __get_blockingmode, + .get_fd_for_systemcall = __get_fd_for_systemcall +}; + +igloo_io_t * igloo_stdio_new_file(const char *filename, const char *mode, const char *name, igloo_ro_t associated) +{ + FILE *file = fopen(filename, mode); + igloo_io_t *io; + + if (!file) + return NULL; + + io = igloo_io_new(&igloo_stdio_ifdesc, igloo_RO_NULL, file, name, associated); + if (!io) { + fclose(file); + return io; + } + + return io; +} From 129bdeeef39ba8dac2b6215ce95279dec8c39a8d Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 12:36:16 +0000 Subject: [PATCH 5/7] Feature: Make interface decleration much nicer --- include/igloo/interface.h | 12 ++++++++++++ src/interface.c | 30 ++++++++++++++++++++++++++++++ src/io.c | 23 +---------------------- src/private.h | 3 +++ src/stdio.c | 4 +++- 5 files changed, 49 insertions(+), 23 deletions(-) diff --git a/include/igloo/interface.h b/include/igloo/interface.h index 59292b1..45febc3 100644 --- a/include/igloo/interface.h +++ b/include/igloo/interface.h @@ -42,11 +42,23 @@ extern "C" { */ typedef int (*igloo_interface_free_t)(igloo_INTERFACE_BASIC_ARGS); +#define igloo_INTERFACE_DESCRIPTION_BASE__VERSION 1 typedef struct { + size_t base_length; + int base_version; + size_t description_length; igloo_interface_free_t free; } igloo_interface_base_ifdesc_t; +#define igloo_INTERFACE_DESCRIPTION_BASE(type, ...) \ + .__base = { \ + .base_length = sizeof(igloo_interface_base_ifdesc_t), \ + .base_version = igloo_INTERFACE_DESCRIPTION_BASE__VERSION, \ + .description_length = sizeof(type) \ + , ## __VA_ARGS__ \ + } + #ifdef __cplusplus } #endif diff --git a/src/interface.c b/src/interface.c index 5271d65..0a57aef 100644 --- a/src/interface.c +++ b/src/interface.c @@ -30,3 +30,33 @@ void igloo_interface_base_free(igloo_ro_t self) free(iface->backend_userdata); } +igloo_ro_t igloo_interface_base_new_real(const igloo_ro_type_t *type, size_t description_length, const igloo_interface_base_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated) +{ + igloo_ro_t self; + igloo__interface_base_t *base; + + if (!ifdesc) + return igloo_RO_NULL; + + if (ifdesc->base_length != sizeof(igloo_interface_base_ifdesc_t) || ifdesc->base_version != igloo_INTERFACE_DESCRIPTION_BASE__VERSION || ifdesc->description_length != description_length) + return igloo_RO_NULL; + + self = igloo_ro_new__raw(type, name, associated); + base = igloo_INTERFACE_CAST(self); + + if (igloo_RO_IS_NULL(self)) + return igloo_RO_NULL; + + if (!igloo_RO_IS_NULL(backend_object)) { + if (igloo_ro_ref(backend_object) != 0) { + igloo_ro_unref(self); + return igloo_RO_NULL; + } + } + + base->ifdesc = ifdesc; + base->backend_object = backend_object; + base->backend_userdata = backend_userdata; + + return self; +} diff --git a/src/io.c b/src/io.c index 2b54b62..21c6a49 100644 --- a/src/io.c +++ b/src/io.c @@ -32,28 +32,7 @@ igloo_RO_PUBLIC_TYPE(igloo_io_t, igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated) { - igloo_io_t *io; - - if (!ifdesc) - return NULL; - - io = igloo_ro_new_raw(igloo_io_t, name, associated); - - if (!io) - return NULL; - - if (!igloo_RO_IS_NULL(backend_object)) { - if (igloo_ro_ref(backend_object) != 0) { - igloo_ro_unref(io); - return NULL; - } - } - - io->ifdesc = ifdesc; - io->backend_object = backend_object; - io->backend_userdata = backend_userdata; - - return io; + return igloo_interface_base_new(igloo_io_t, ifdesc, backend_object, backend_userdata, name, associated); } diff --git a/src/private.h b/src/private.h index 113519f..35d0791 100644 --- a/src/private.h +++ b/src/private.h @@ -52,4 +52,7 @@ void igloo_interface_base_free(igloo_ro_t self); #define igloo_INTERFACE_CAST(obj) ((igloo__interface_base_t*)igloo_RO__GETBASE(obj)) #define igloo_INTERFACE_BASIC_CALL(obj) (obj), &(igloo_INTERFACE_CAST(obj)->backend_object), &(igloo_INTERFACE_CAST(obj)->backend_userdata) +igloo_ro_t igloo_interface_base_new_real(const igloo_ro_type_t *type, size_t description_length, const igloo_interface_base_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated); +#define igloo_interface_base_new(type, ifdesc, backend_object, backend_userdata, name, associated) igloo_RO_TO_TYPE(igloo_interface_base_new_real(igloo_ro__type__ ## type, sizeof(*(ifdesc)), (const igloo_interface_base_ifdesc_t*)(ifdesc), (backend_object), (backend_userdata), (name), (associated)), type) + #endif diff --git a/src/stdio.c b/src/stdio.c index 1078f0a..27f9ced 100644 --- a/src/stdio.c +++ b/src/stdio.c @@ -49,7 +49,9 @@ static int __get_fd_for_systemcall(igloo_INTERFACE_BASIC_ARGS) } static const igloo_io_ifdesc_t igloo_stdio_ifdesc = { - .__base = {.free = __free}, + igloo_INTERFACE_DESCRIPTION_BASE(igloo_io_ifdesc_t, + .free = __free + ), .read = __read, .write = __write, .flush = __flush, From 02e0375711e0c5631dca6b545b324d63b55299b4 Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Thu, 27 Jun 2019 12:38:33 +0000 Subject: [PATCH 6/7] Fix: Mark fresh IO objects as touched. --- src/io.c | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/src/io.c b/src/io.c index 21c6a49..aad86ee 100644 --- a/src/io.c +++ b/src/io.c @@ -32,7 +32,13 @@ igloo_RO_PUBLIC_TYPE(igloo_io_t, igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_object, void *backend_userdata, const char *name, igloo_ro_t associated) { - return igloo_interface_base_new(igloo_io_t, ifdesc, backend_object, backend_userdata, name, associated); + igloo_io_t *io = igloo_interface_base_new(igloo_io_t, ifdesc, backend_object, backend_userdata, name, associated); + if (!io) + return NULL; + + io->touched = 1; + + return io; } From efc054cc89ad3a9ccbdd06a58de202024fa802d4 Mon Sep 17 00:00:00 2001 From: Philipp Schafft Date: Sun, 30 Jun 2019 09:51:49 +0000 Subject: [PATCH 7/7] Feature: Make all IO functions thread safe. --- src/io.c | 134 ++++++++++++++++++++++++++++++++++++++++++------------- 1 file changed, 104 insertions(+), 30 deletions(-) diff --git a/src/io.c b/src/io.c index aad86ee..1ae625d 100644 --- a/src/io.c +++ b/src/io.c @@ -19,14 +19,28 @@ struct igloo_io_tag { igloo_interface_base(io) + + /* Mutex for igloo_ro_*(). */ + igloo_mutex_t lock; + int touched; #if defined(IGLOO_CTC_HAVE_SYS_SELECT_H) || defined(IGLOO_CTC_HAVE_POLL) int fd; #endif }; +static void __free(igloo_ro_t self) +{ + igloo_io_t *io = igloo_RO_TO_TYPE(self, igloo_io_t); + + igloo_thread_mutex_lock(&(io->lock)); + igloo_thread_mutex_unlock(&(io->lock)); + igloo_thread_mutex_destroy(&(io->lock)); + igloo_interface_base_free(self); +} + igloo_RO_PUBLIC_TYPE(igloo_io_t, - igloo_RO_TYPEDECL_FREE(igloo_interface_base_free) + igloo_RO_TYPEDECL_FREE(__free) ); @@ -36,6 +50,8 @@ igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_ob if (!io) return NULL; + igloo_thread_mutex_create(&(io->lock)); + io->touched = 1; return io; @@ -44,47 +60,59 @@ igloo_io_t * igloo_io_new(const igloo_io_ifdesc_t *ifdesc, igloo_ro_t backend_ob ssize_t igloo_io_read(igloo_io_t *io, void *buffer, size_t len) { + ssize_t ret = -1; + if (!io || !buffer) return -1; if (!len) return 0; + igloo_thread_mutex_lock(&(io->lock)); io->touched = 1; if (io->ifdesc->read) - return io->ifdesc->read(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + ret = io->ifdesc->read(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + igloo_thread_mutex_unlock(&(io->lock)); - return -1; + return ret; } ssize_t igloo_io_write(igloo_io_t *io, const void *buffer, size_t len) { + ssize_t ret = -1; + if (!io || !buffer) return -1; if (!len) return 0; + igloo_thread_mutex_lock(&(io->lock)); io->touched = 1; if (io->ifdesc->write) - return io->ifdesc->write(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + ret = io->ifdesc->write(igloo_INTERFACE_BASIC_CALL(io), buffer, len); + igloo_thread_mutex_unlock(&(io->lock)); - return -1; + return ret; } int igloo_io_flush(igloo_io_t *io, igloo_io_opflag_t flags) { + int ret = -1; + if (!io) return -1; + igloo_thread_mutex_lock(&(io->lock)); io->touched = 1; if (io->ifdesc->flush) - return io->ifdesc->flush(igloo_INTERFACE_BASIC_CALL(io), flags); + ret = io->ifdesc->flush(igloo_INTERFACE_BASIC_CALL(io), flags); + igloo_thread_mutex_unlock(&(io->lock)); - return -1; + return ret; } int igloo_io_sync(igloo_io_t *io, igloo_io_opflag_t flags) @@ -94,46 +122,59 @@ int igloo_io_sync(igloo_io_t *io, igloo_io_opflag_t flags) if (!io) return -1; + igloo_thread_mutex_lock(&(io->lock)); if (io->ifdesc->flush) io->ifdesc->flush(igloo_INTERFACE_BASIC_CALL(io), flags); if (io->ifdesc->sync) { ret = io->ifdesc->sync(igloo_INTERFACE_BASIC_CALL(io), flags); - if (ret != 0) + if (ret != 0) { + igloo_thread_mutex_unlock(&(io->lock)); return ret; + } io->touched = 0; + igloo_thread_mutex_unlock(&(io->lock)); return ret; } + igloo_thread_mutex_unlock(&(io->lock)); return -1; } int igloo_io_set_blockingmode(igloo_io_t *io, libigloo_io_blockingmode_t mode) { + int ret = -1; + if (!io) return -1; if (mode != igloo_IO_BLOCKINGMODE_NONE && mode != igloo_IO_BLOCKINGMODE_FULL) return -1; + igloo_thread_mutex_lock(&(io->lock)); io->touched = 1; if (io->ifdesc->set_blockingmode) - return io->ifdesc->set_blockingmode(igloo_INTERFACE_BASIC_CALL(io), mode); + ret = io->ifdesc->set_blockingmode(igloo_INTERFACE_BASIC_CALL(io), mode); + igloo_thread_mutex_unlock(&(io->lock)); - return -1; + return ret; } libigloo_io_blockingmode_t igloo_io_get_blockingmode(igloo_io_t *io) { + libigloo_io_blockingmode_t ret = igloo_IO_BLOCKINGMODE_ERROR; + if (!io) return igloo_IO_BLOCKINGMODE_ERROR; + igloo_thread_mutex_lock(&(io->lock)); if (io->ifdesc->get_blockingmode) - return io->ifdesc->get_blockingmode(igloo_INTERFACE_BASIC_CALL(io)); + ret = io->ifdesc->get_blockingmode(igloo_INTERFACE_BASIC_CALL(io)); + igloo_thread_mutex_unlock(&(io->lock)); - return igloo_IO_BLOCKINGMODE_ERROR; + return ret; } #ifdef IGLOO_CTC_HAVE_SYS_SELECT_H @@ -144,15 +185,24 @@ int igloo_io_select_set(igloo_io_t *io, fd_set *set, int *maxfd, igloo_io_opflag if (!io || !set || !maxfd) return -1; - if (!io->ifdesc->get_fd_for_systemcall) + igloo_thread_mutex_lock(&(io->lock)); + if (!io->ifdesc->get_fd_for_systemcall) { + igloo_thread_mutex_unlock(&(io->lock)); return -1; + } if (io->touched || !(flags & igloo_IO_OPFLAG_NOWRITE)) { + igloo_thread_mutex_unlock(&(io->lock)); ret = igloo_io_sync(io, igloo_IO_OPFLAG_DEFAULTS|(flags & igloo_IO_OPFLAG_NOWRITE)); + if (ret != 0) return ret; - if (io->touched) + + igloo_thread_mutex_lock(&(io->lock)); + if (io->touched) { + igloo_thread_mutex_unlock(&(io->lock)); return -1; + } } io->fd = io->ifdesc->get_fd_for_systemcall(igloo_INTERFACE_BASIC_CALL(io)); @@ -163,6 +213,8 @@ int igloo_io_select_set(igloo_io_t *io, fd_set *set, int *maxfd, igloo_io_opflag if (*maxfd < io->fd) *maxfd = io->fd; + igloo_thread_mutex_unlock(&(io->lock)); + return 0; } @@ -171,23 +223,31 @@ int igloo_io_select_clear(igloo_io_t *io, fd_set *set) if (!io || !set) return -1; - if (io->touched || io->fd < 0) + igloo_thread_mutex_lock(&(io->lock)); + if (io->touched || io->fd < 0) { + igloo_thread_mutex_unlock(&(io->lock)); return -1; + } FD_CLR(io->fd, set); + igloo_thread_mutex_unlock(&(io->lock)); return 0; } int igloo_io_select_isset(igloo_io_t *io, fd_set *set) { + int ret = -1; + if (!io || !set) return -1; - if (io->touched || io->fd < 0) - return -1; + igloo_thread_mutex_lock(&(io->lock)); + if (!io->touched && io->fd >= 0) + ret = FD_ISSET(io->fd, set) ? 1 : 0; + igloo_thread_mutex_unlock(&(io->lock)); - return FD_ISSET(io->fd, set) ? 1 : 0; + return ret; } #endif @@ -208,27 +268,40 @@ int igloo_io_poll_fill(igloo_io_t *io, struct pollfd *fd, short events) if (!io || !fd) return -1; - if (!io->ifdesc->get_fd_for_systemcall) - return -1; - is_safe = !((events|safe_events) - safe_events); + igloo_thread_mutex_lock(&(io->lock)); + if (!io->ifdesc->get_fd_for_systemcall) { + igloo_thread_mutex_unlock(&(io->lock)); + return -1; + } + if (io->touched || !is_safe) { + igloo_thread_mutex_unlock(&(io->lock)); ret = igloo_io_sync(io, igloo_IO_OPFLAG_DEFAULTS|(is_safe ? 0 : igloo_IO_OPFLAG_NOWRITE)); + if (ret != 0) return ret; - if (io->touched) + + igloo_thread_mutex_lock(&(io->lock)); + if (io->touched) { + igloo_thread_mutex_unlock(&(io->lock)); return -1; + } } io->fd = io->ifdesc->get_fd_for_systemcall(igloo_INTERFACE_BASIC_CALL(io)); - if (io->fd < 0) + if (io->fd < 0) { + igloo_thread_mutex_unlock(&(io->lock)); return -1; + } memset(fd, 0, sizeof(*fd)); fd->fd = io->fd; fd->events = events; + igloo_thread_mutex_unlock(&(io->lock)); + return 0; } #endif @@ -236,18 +309,19 @@ int igloo_io_poll_fill(igloo_io_t *io, struct pollfd *fd, short events) int igloo_io_control(igloo_io_t *io, igloo_io_opflag_t flags, igloo_io_control_t control, ...) { #ifdef IGLOO_CTC_STDC_HEADERS - int ret; + int ret = -1; va_list ap; if (!io) return -1; - if (!io->ifdesc->control) - return -1; - - va_start(ap, control); - ret = io->ifdesc->control(igloo_INTERFACE_BASIC_CALL(io), flags, control, ap); - va_end(ap); + igloo_thread_mutex_lock(&(io->lock)); + if (io->ifdesc->control) { + va_start(ap, control); + ret = io->ifdesc->control(igloo_INTERFACE_BASIC_CALL(io), flags, control, ap); + va_end(ap); + } + igloo_thread_mutex_unlock(&(io->lock)); return ret; #else