X-Git-Url: http://git.lttng.org/?a=blobdiff_plain;f=libringbuffer%2Ffrontend_internal.h;h=f6d91bde2bf63f0917b9bd27b90e73ed54429bcc;hb=35cbacdb34948e328f5d360f2fd76618dd678e2f;hp=1a9c2d4b78e82e7645e1267430b41e6f0c3218c2;hpb=730be651d21f3950980fd61e1c65953017c6322e;p=lttng-ust.git diff --git a/libringbuffer/frontend_internal.h b/libringbuffer/frontend_internal.h index 1a9c2d4b..f6d91bde 100644 --- a/libringbuffer/frontend_internal.h +++ b/libringbuffer/frontend_internal.h @@ -34,6 +34,7 @@ #include #include #include +#include #include #include @@ -205,6 +206,15 @@ void lib_ring_buffer_reserve_push_reader(struct lttng_ust_lib_ring_buffer *buf, consumed_new) != consumed_old)); } +/* + * Move consumed position to the beginning of subbuffer in which the + * write offset is. Should only be used on ring buffers that are not + * actively being written into, because clear_reader does not take into + * account the commit counters when moving the consumed position, which + * can make concurrent trace producers or consumers observe consumed + * position further than the write offset, which breaks ring buffer + * algorithm guarantees. + */ static inline void lib_ring_buffer_clear_reader(struct lttng_ust_lib_ring_buffer *buf, struct lttng_ust_shm_handle *handle) @@ -221,12 +231,10 @@ void lib_ring_buffer_clear_reader(struct lttng_ust_lib_ring_buffer *buf, do { offset = v_read(config, &buf->offset); consumed_old = uatomic_read(&buf->consumed); - if (caa_unlikely(subbuf_trunc(offset, chan) - - subbuf_trunc(consumed_old, chan) - > 0)) - consumed_new = subbuf_trunc(offset, chan); - else - return; + CHAN_WARN_ON(chan, (long) (subbuf_trunc(offset, chan) + - subbuf_trunc(consumed_old, chan)) + < 0); + consumed_new = subbuf_trunc(offset, chan); } while (caa_unlikely(uatomic_cmpxchg(&buf->consumed, consumed_old, consumed_new) != consumed_old)); }