When using relay mechanism, we often encounter the case where new data are
lost or old unconsumed data are overwritten because of slow reader.
Add 'full' field in per-cpu buffer structure to detect if the above case
is happening.  Relay has two modes: 1) non-overwrite mode, 2) overwrite
mode.  So buffer being full here respectively means: 1) relayfs doesn't
intend to accept new data and then simply drop them, or 2) relayfs is
going to start over again and overwrite old unread data with new data.
Note: this counter doesn't need any explicit lock to protect from being
modified by different threads for the better performance consideration.
Writers calling __relay_write/relay_write should consider how to use the
lock and ensure it performs under the lock protection, thus it's not
necessary to add a new small lock here.
Link: https://lkml.kernel.org/r/20250612061201.34272-3-kerneljasonxing@gmail.com
Signed-off-by: Jason Xing <kernelxing@tencent.com>
Reviewed-by: Yushan Zhou <katrinzhou@tencent.com>
Reviewed-by: Jens Axboe <axboe@kernel.dk>
Reviewed-by: Masami Hiramatsu (Google) <mhiramat@kernel.org>
Cc: Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
  */
 #define RELAYFS_CHANNEL_VERSION                7
 
+/*
+ * Relay buffer statistics
+ */
+struct rchan_buf_stats
+{
+       unsigned int full_count;        /* counter for buffer full */
+};
+
 /*
  * Per-cpu relay channel buffer
  */
        struct irq_work wakeup_work;    /* reader wakeup */
        struct dentry *dentry;          /* channel file dentry */
        struct kref kref;               /* channel buffer refcount */
+       struct rchan_buf_stats stats;   /* buffer stats */
        struct page **page_array;       /* array of current buffer pages */
        unsigned int page_count;        /* number of current buffer pages */
        unsigned int finalized;         /* buffer has been finalized */
 
 static int relay_subbuf_start(struct rchan_buf *buf, void *subbuf,
                              void *prev_subbuf)
 {
+       int full = relay_buf_full(buf);
+
+       if (full)
+               buf->stats.full_count++;
+
        if (!buf->chan->cb->subbuf_start)
-               return !relay_buf_full(buf);
+               return !full;
 
        return buf->chan->cb->subbuf_start(buf, subbuf,
                                           prev_subbuf);
        buf->finalized = 0;
        buf->data = buf->start;
        buf->offset = 0;
+       buf->stats.full_count = 0;
 
        for (i = 0; i < buf->chan->n_subbufs; i++)
                buf->padding[i] = 0;