// SPDX-License-Identifier: GPL-2.0 /* * Copyright (C) 2025 - Google LLC * Author: Vincent Donnefort */ #include #include #include #include enum simple_rb_link_type { SIMPLE_RB_LINK_NORMAL = 0, SIMPLE_RB_LINK_HEAD = 1, SIMPLE_RB_LINK_HEAD_MOVING }; #define SIMPLE_RB_LINK_MASK ~(SIMPLE_RB_LINK_HEAD | SIMPLE_RB_LINK_HEAD_MOVING) static void simple_bpage_set_head_link(struct simple_buffer_page *bpage) { unsigned long link = (unsigned long)bpage->link.next; link &= SIMPLE_RB_LINK_MASK; link |= SIMPLE_RB_LINK_HEAD; /* * Paired with simple_rb_find_head() to order access between the head * link and overrun. It ensures we always report an up-to-date value * after swapping the reader page. */ smp_store_release(&bpage->link.next, (struct list_head *)link); } static bool simple_bpage_unset_head_link(struct simple_buffer_page *bpage, struct simple_buffer_page *dst, enum simple_rb_link_type new_type) { unsigned long *link = (unsigned long *)(&bpage->link.next); unsigned long old = (*link & SIMPLE_RB_LINK_MASK) | SIMPLE_RB_LINK_HEAD; unsigned long new = (unsigned long)(&dst->link) | new_type; return try_cmpxchg(link, &old, new); } static void simple_bpage_set_normal_link(struct simple_buffer_page *bpage) { unsigned long link = (unsigned long)bpage->link.next; WRITE_ONCE(bpage->link.next, (struct list_head *)(link & SIMPLE_RB_LINK_MASK)); } static struct simple_buffer_page *simple_bpage_from_link(struct list_head *link) { unsigned long ptr = (unsigned long)link & SIMPLE_RB_LINK_MASK; return container_of((struct list_head *)ptr, struct simple_buffer_page, link); } static struct simple_buffer_page *simple_bpage_next_page(struct simple_buffer_page *bpage) { return simple_bpage_from_link(bpage->link.next); } static void simple_bpage_reset(struct simple_buffer_page *bpage) { bpage->write = 0; bpage->entries = 0; local_set(&bpage->page->commit, 0); } static void simple_bpage_init(struct simple_buffer_page *bpage, void *page) { INIT_LIST_HEAD(&bpage->link); bpage->page = (struct buffer_data_page *)page; simple_bpage_reset(bpage); } #define simple_rb_meta_inc(__meta, __inc) \ WRITE_ONCE((__meta), (__meta + __inc)) static bool simple_rb_loaded(struct simple_rb_per_cpu *cpu_buffer) { return !!cpu_buffer->bpages; } static int simple_rb_find_head(struct simple_rb_per_cpu *cpu_buffer) { int retry = cpu_buffer->nr_pages * 2; struct simple_buffer_page *head; head = cpu_buffer->head_page; while (retry--) { unsigned long link; spin: /* See smp_store_release in simple_bpage_set_head_link() */ link = (unsigned long)smp_load_acquire(&head->link.prev->next); switch (link & ~SIMPLE_RB_LINK_MASK) { /* Found the head */ case SIMPLE_RB_LINK_HEAD: cpu_buffer->head_page = head; return 0; /* The writer caught the head, we can spin, that won't be long */ case SIMPLE_RB_LINK_HEAD_MOVING: goto spin; } head = simple_bpage_next_page(head); } return -EBUSY; } /** * simple_ring_buffer_swap_reader_page - Swap ring-buffer head with the reader * @cpu_buffer: A simple_rb_per_cpu * * This function enables consuming reading. It ensures the current head page will not be overwritten * and can be safely read. * * Returns 0 on success, -ENODEV if @cpu_buffer was unloaded or -EBUSY if we failed to catch the * head page. */ int simple_ring_buffer_swap_reader_page(struct simple_rb_per_cpu *cpu_buffer) { struct simple_buffer_page *last, *head, *reader; unsigned long overrun; int retry = 8; int ret; if (!simple_rb_loaded(cpu_buffer)) return -ENODEV; reader = cpu_buffer->reader_page; do { /* Run after the writer to find the head */ ret = simple_rb_find_head(cpu_buffer); if (ret) return ret; head = cpu_buffer->head_page; /* Connect the reader page around the header page */ reader->link.next = head->link.next; reader->link.prev = head->link.prev; /* The last page before the head */ last = simple_bpage_from_link(head->link.prev); /* The reader page points to the new header page */ simple_bpage_set_head_link(reader); overrun = cpu_buffer->meta->overrun; } while (!simple_bpage_unset_head_link(last, reader, SIMPLE_RB_LINK_NORMAL) && retry--); if (!retry) return -EINVAL; cpu_buffer->head_page = simple_bpage_from_link(reader->link.next); cpu_buffer->head_page->link.prev = &reader->link; cpu_buffer->reader_page = head; cpu_buffer->meta->reader.lost_events = overrun - cpu_buffer->last_overrun; cpu_buffer->meta->reader.id = cpu_buffer->reader_page->id; cpu_buffer->last_overrun = overrun; return 0; } EXPORT_SYMBOL_GPL(simple_ring_buffer_swap_reader_page); static struct simple_buffer_page *simple_rb_move_tail(struct simple_rb_per_cpu *cpu_buffer) { struct simple_buffer_page *tail, *new_tail; tail = cpu_buffer->tail_page; new_tail = simple_bpage_next_page(tail); if (simple_bpage_unset_head_link(tail, new_tail, SIMPLE_RB_LINK_HEAD_MOVING)) { /* * Oh no! we've caught the head. There is none anymore and * swap_reader will spin until we set the new one. Overrun must * be written first, to make sure we report the correct number * of lost events. */ simple_rb_meta_inc(cpu_buffer->meta->overrun, new_tail->entries); simple_rb_meta_inc(cpu_buffer->meta->pages_lost, 1); simple_bpage_set_head_link(new_tail); simple_bpage_set_normal_link(tail); } simple_bpage_reset(new_tail); cpu_buffer->tail_page = new_tail; simple_rb_meta_inc(cpu_buffer->meta->pages_touched, 1); return new_tail; } static unsigned long rb_event_size(unsigned long length) { struct ring_buffer_event *event; return length + RB_EVNT_HDR_SIZE + sizeof(event->array[0]); } static struct ring_buffer_event * rb_event_add_ts_extend(struct ring_buffer_event *event, u64 delta) { event->type_len = RINGBUF_TYPE_TIME_EXTEND; event->time_delta = delta & TS_MASK; event->array[0] = delta >> TS_SHIFT; return (struct ring_buffer_event *)((unsigned long)event + 8); } static struct ring_buffer_event * simple_rb_reserve_next(struct simple_rb_per_cpu *cpu_buffer, unsigned long length, u64 timestamp) { unsigned long ts_ext_size = 0, event_size = rb_event_size(length); struct simple_buffer_page *tail = cpu_buffer->tail_page; struct ring_buffer_event *event; u32 write, prev_write; u64 time_delta; time_delta = timestamp - cpu_buffer->write_stamp; if (test_time_stamp(time_delta)) ts_ext_size = 8; prev_write = tail->write; write = prev_write + event_size + ts_ext_size; if (unlikely(write > (PAGE_SIZE - BUF_PAGE_HDR_SIZE))) tail = simple_rb_move_tail(cpu_buffer); if (!tail->entries) { tail->page->time_stamp = timestamp; time_delta = 0; ts_ext_size = 0; write = event_size; prev_write = 0; } tail->write = write; tail->entries++; cpu_buffer->write_stamp = timestamp; event = (struct ring_buffer_event *)(tail->page->data + prev_write); if (ts_ext_size) { event = rb_event_add_ts_extend(event, time_delta); time_delta = 0; } event->type_len = 0; event->time_delta = time_delta; event->array[0] = event_size - RB_EVNT_HDR_SIZE; return event; } /** * simple_ring_buffer_reserve - Reserve an entry in @cpu_buffer * @cpu_buffer: A simple_rb_per_cpu * @length: Size of the entry in bytes * @timestamp: Timestamp of the entry * * Returns the address of the entry where to write data or NULL */ void *simple_ring_buffer_reserve(struct simple_rb_per_cpu *cpu_buffer, unsigned long length, u64 timestamp) { struct ring_buffer_event *rb_event; if (cmpxchg(&cpu_buffer->status, SIMPLE_RB_READY, SIMPLE_RB_WRITING) != SIMPLE_RB_READY) return NULL; rb_event = simple_rb_reserve_next(cpu_buffer, length, timestamp); return &rb_event->array[1]; } EXPORT_SYMBOL_GPL(simple_ring_buffer_reserve); /** * simple_ring_buffer_commit - Commit the entry reserved with simple_ring_buffer_reserve() * @cpu_buffer: The simple_rb_per_cpu where the entry has been reserved */ void simple_ring_buffer_commit(struct simple_rb_per_cpu *cpu_buffer) { local_set(&cpu_buffer->tail_page->page->commit, cpu_buffer->tail_page->write); simple_rb_meta_inc(cpu_buffer->meta->entries, 1); /* * Paired with simple_rb_enable_tracing() to ensure data is * written to the ring-buffer before teardown. */ smp_store_release(&cpu_buffer->status, SIMPLE_RB_READY); } EXPORT_SYMBOL_GPL(simple_ring_buffer_commit); static u32 simple_rb_enable_tracing(struct simple_rb_per_cpu *cpu_buffer, bool enable) { u32 prev_status; if (enable) return cmpxchg(&cpu_buffer->status, SIMPLE_RB_UNAVAILABLE, SIMPLE_RB_READY); /* Wait for the buffer to be released */ do { prev_status = cmpxchg_acquire(&cpu_buffer->status, SIMPLE_RB_READY, SIMPLE_RB_UNAVAILABLE); } while (prev_status == SIMPLE_RB_WRITING); return prev_status; } /** * simple_ring_buffer_reset - Reset @cpu_buffer * @cpu_buffer: A simple_rb_per_cpu * * This will not clear the content of the data, only reset counters and pointers * * Returns 0 on success or -ENODEV if @cpu_buffer was unloaded. */ int simple_ring_buffer_reset(struct simple_rb_per_cpu *cpu_buffer) { struct simple_buffer_page *bpage; u32 prev_status; int ret; if (!simple_rb_loaded(cpu_buffer)) return -ENODEV; prev_status = simple_rb_enable_tracing(cpu_buffer, false); ret = simple_rb_find_head(cpu_buffer); if (ret) return ret; bpage = cpu_buffer->tail_page = cpu_buffer->head_page; do { simple_bpage_reset(bpage); bpage = simple_bpage_next_page(bpage); } while (bpage != cpu_buffer->head_page); simple_bpage_reset(cpu_buffer->reader_page); cpu_buffer->last_overrun = 0; cpu_buffer->write_stamp = 0; cpu_buffer->meta->reader.read = 0; cpu_buffer->meta->reader.lost_events = 0; cpu_buffer->meta->entries = 0; cpu_buffer->meta->overrun = 0; cpu_buffer->meta->read = 0; cpu_buffer->meta->pages_lost = 0; cpu_buffer->meta->pages_touched = 0; if (prev_status == SIMPLE_RB_READY) simple_rb_enable_tracing(cpu_buffer, true); return 0; } EXPORT_SYMBOL_GPL(simple_ring_buffer_reset); int simple_ring_buffer_init_mm(struct simple_rb_per_cpu *cpu_buffer, struct simple_buffer_page *bpages, const struct ring_buffer_desc *desc, void *(*load_page)(unsigned long va), void (*unload_page)(void *va)) { struct simple_buffer_page *bpage = bpages; int ret = 0; void *page; int i; /* At least 1 reader page and two pages in the ring-buffer */ if (desc->nr_page_va < 3) return -EINVAL; memset(cpu_buffer, 0, sizeof(*cpu_buffer)); cpu_buffer->meta = load_page(desc->meta_va); if (!cpu_buffer->meta) return -EINVAL; memset(cpu_buffer->meta, 0, sizeof(*cpu_buffer->meta)); cpu_buffer->meta->meta_page_size = PAGE_SIZE; cpu_buffer->meta->nr_subbufs = cpu_buffer->nr_pages; /* The reader page is not part of the ring initially */ page = load_page(desc->page_va[0]); if (!page) { unload_page(cpu_buffer->meta); return -EINVAL; } simple_bpage_init(bpage, page); bpage->id = 0; cpu_buffer->nr_pages = 1; cpu_buffer->reader_page = bpage; cpu_buffer->tail_page = bpage + 1; cpu_buffer->head_page = bpage + 1; for (i = 1; i < desc->nr_page_va; i++) { page = load_page(desc->page_va[i]); if (!page) { ret = -EINVAL; break; } simple_bpage_init(++bpage, page); bpage->link.next = &(bpage + 1)->link; bpage->link.prev = &(bpage - 1)->link; bpage->id = i; cpu_buffer->nr_pages = i + 1; } if (ret) { for (i--; i >= 0; i--) unload_page((void *)desc->page_va[i]); unload_page(cpu_buffer->meta); return ret; } /* Close the ring */ bpage->link.next = &cpu_buffer->tail_page->link; cpu_buffer->tail_page->link.prev = &bpage->link; /* The last init'ed page points to the head page */ simple_bpage_set_head_link(bpage); cpu_buffer->bpages = bpages; return 0; } static void *__load_page(unsigned long page) { return (void *)page; } static void __unload_page(void *page) { } /** * simple_ring_buffer_init - Init @cpu_buffer based on @desc * @cpu_buffer: A simple_rb_per_cpu buffer to init, allocated by the caller. * @bpages: Array of simple_buffer_pages, with as many elements as @desc->nr_page_va * @desc: A ring_buffer_desc * * Returns 0 on success or -EINVAL if the content of @desc is invalid */ int simple_ring_buffer_init(struct simple_rb_per_cpu *cpu_buffer, struct simple_buffer_page *bpages, const struct ring_buffer_desc *desc) { return simple_ring_buffer_init_mm(cpu_buffer, bpages, desc, __load_page, __unload_page); } EXPORT_SYMBOL_GPL(simple_ring_buffer_init); void simple_ring_buffer_unload_mm(struct simple_rb_per_cpu *cpu_buffer, void (*unload_page)(void *)) { int p; if (!simple_rb_loaded(cpu_buffer)) return; simple_rb_enable_tracing(cpu_buffer, false); unload_page(cpu_buffer->meta); for (p = 0; p < cpu_buffer->nr_pages; p++) unload_page(cpu_buffer->bpages[p].page); cpu_buffer->bpages = NULL; } /** * simple_ring_buffer_unload - Prepare @cpu_buffer for deletion * @cpu_buffer: A simple_rb_per_cpu that will be deleted. */ void simple_ring_buffer_unload(struct simple_rb_per_cpu *cpu_buffer) { return simple_ring_buffer_unload_mm(cpu_buffer, __unload_page); } EXPORT_SYMBOL_GPL(simple_ring_buffer_unload); /** * simple_ring_buffer_enable_tracing - Enable or disable writing to @cpu_buffer * @cpu_buffer: A simple_rb_per_cpu * @enable: True to enable tracing, False to disable it * * Returns 0 on success or -ENODEV if @cpu_buffer was unloaded */ int simple_ring_buffer_enable_tracing(struct simple_rb_per_cpu *cpu_buffer, bool enable) { if (!simple_rb_loaded(cpu_buffer)) return -ENODEV; simple_rb_enable_tracing(cpu_buffer, enable); return 0; } EXPORT_SYMBOL_GPL(simple_ring_buffer_enable_tracing);