/* SPDX-License-Identifier: GPL-2.0-only */ /* * Copyright 2023 Red Hat */ #ifndef VDO_FUNNEL_QUEUE_H #define VDO_FUNNEL_QUEUE_H #include <linux/atomic.h> #include <linux/cache.h> /* * A funnel queue is a simple (almost) lock-free queue that accepts entries from multiple threads * (multi-producer) and delivers them to a single thread (single-consumer). "Funnel" is an attempt * to evoke the image of requests from more than one producer being "funneled down" to a single * consumer. * * This is an unsynchronized but thread-safe data structure when used as intended. There is no * mechanism to ensure that only one thread is consuming from the queue. If more than one thread * attempts to consume from the queue, the resulting behavior is undefined. Clients must not * directly access or manipulate the internals of the queue, which are only exposed for the purpose * of allowing the very simple enqueue operation to be inlined. * * The implementation requires that a funnel_queue_entry structure (a link pointer) is embedded in * the queue entries, and pointers to those structures are used exclusively by the queue. No macros * are defined to template the queue, so the offset of the funnel_queue_entry in the records placed * in the queue must all be the same so the client can derive their structure pointer from the * entry pointer returned by vdo_funnel_queue_poll(). * * Callers are wholly responsible for allocating and freeing the entries. Entries may be freed as * soon as they are returned since this queue is not susceptible to the "ABA problem" present in * many lock-free data structures. The queue is dynamically allocated to ensure cache-line * alignment, but no other dynamic allocation is used. * * The algorithm is not actually 100% lock-free. There is a single point in vdo_funnel_queue_put() * at which a preempted producer will prevent the consumers from seeing items added to the queue by * later producers, and only if the queue is short enough or the consumer fast enough for it to * reach what was the end of the queue at the time of the preemption. * * The consumer function, vdo_funnel_queue_poll(), will return NULL when the queue is empty. To * wait for data to consume, spin (if safe) or combine the queue with a struct event_count to * signal the presence of new entries. */ /* This queue link structure must be embedded in client entries. */ struct funnel_queue_entry { … }; /* * The dynamically allocated queue structure, which is allocated on a cache line boundary so the * producer and consumer fields in the structure will land on separate cache lines. This should be * consider opaque but it is exposed here so vdo_funnel_queue_put() can be inlined. */ struct __aligned(L1_CACHE_BYTES) funnel_queue { … }; int __must_check vdo_make_funnel_queue(struct funnel_queue **queue_ptr); void vdo_free_funnel_queue(struct funnel_queue *queue); /* * Put an entry on the end of the queue. * * The entry pointer must be to the struct funnel_queue_entry embedded in the caller's data * structure. The caller must be able to derive the address of the start of their data structure * from the pointer that passed in here, so every entry in the queue must have the struct * funnel_queue_entry at the same offset within the client's structure. */ static inline void vdo_funnel_queue_put(struct funnel_queue *queue, struct funnel_queue_entry *entry) { … } struct funnel_queue_entry *__must_check vdo_funnel_queue_poll(struct funnel_queue *queue); bool __must_check vdo_is_funnel_queue_empty(struct funnel_queue *queue); bool __must_check vdo_is_funnel_queue_idle(struct funnel_queue *queue); #endif /* VDO_FUNNEL_QUEUE_H */