// Copyright 2011-2022 David Robillard // SPDX-License-Identifier: ISC #ifndef ZIX_RING_H #define ZIX_RING_H #include "zix/allocator.h" #include "zix/attributes.h" #include #ifdef __cplusplus extern "C" { #endif /** @addtogroup zix @{ @name Ring @{ */ /** A lock-free ring buffer. Thread-safe (with a few noted exceptions) for a single reader and single writer, and realtime-safe on both ends. */ typedef struct ZixRingImpl ZixRing; /** Create a new ring. @param size Size in bytes (note this may be rounded up). At most `size` - 1 bytes may be stored in the ring at once. */ ZIX_MALLOC_API ZixRing* ZIX_ALLOCATED zix_ring_new(ZixAllocator* ZIX_NULLABLE allocator, uint32_t size); /// Destroy a ring ZIX_API void zix_ring_free(ZixRing* ZIX_NULLABLE ring); /** Lock the ring data into physical memory. This function is NOT thread safe or real-time safe, but it should be called after zix_ring_new() to lock all ring memory to avoid page faults while using the ring. */ ZIX_API void zix_ring_mlock(ZixRing* ZIX_NONNULL ring); /** Reset (empty) a ring. This function is NOT thread-safe, it may only be called when there is no reader or writer. */ ZIX_API void zix_ring_reset(ZixRing* ZIX_NONNULL ring); /** Return the number of bytes of space available for reading. Reader only. */ ZIX_PURE_API uint32_t zix_ring_read_space(const ZixRing* ZIX_NONNULL ring); /** Return the number of bytes of space available for writing. Writer only. */ ZIX_PURE_API uint32_t zix_ring_write_space(const ZixRing* ZIX_NONNULL ring); /** Return the capacity (the total write space when empty). This function returns a constant for any given ring, and may (but usually shouldn't) be called anywhere. */ ZIX_PURE_API uint32_t zix_ring_capacity(const ZixRing* ZIX_NONNULL ring); /** Read from the ring without advancing the read head. Reader only. */ ZIX_API uint32_t zix_ring_peek(ZixRing* ZIX_NONNULL ring, void* ZIX_NONNULL dst, uint32_t size); /** Read from the ring and advance the read head. Reader only. */ ZIX_API uint32_t zix_ring_read(ZixRing* ZIX_NONNULL ring, void* ZIX_NONNULL dst, uint32_t size); /** Skip data in the ring (advance read head without reading). Reader only. */ ZIX_API uint32_t zix_ring_skip(ZixRing* ZIX_NONNULL ring, uint32_t size); /** Write data to the ring. Writer only. */ ZIX_API uint32_t zix_ring_write(ZixRing* ZIX_NONNULL ring, const void* ZIX_NONNULL src, uint32_t size); /** @} @} */ #ifdef __cplusplus } /* extern "C" */ #endif #endif /* ZIX_RING_H */