1.. _mpsc_pbuf:
2
3Multi Producer Single Consumer Packet Buffer
4============================================
5
6A :dfn:`Multi Producer Single Consumer Packet Buffer (MPSC_PBUF)` is a circular
7buffer, whose contents are stored in first-in-first-out order. Variable size
8packets are stored in the buffer. Packet buffer works under assumption that there
9is a single context that consumes the data. However, it is possible that another
10context may interfere to flush the data and never come back (panic case).
11Packet is produced in two steps: first requested amount of data is allocated,
12producer fills the data and commits it. Consuming a packet is also performed in
13two steps: consumer claims the packet, gets pointer to it and length and later
14on packet is freed. This approach reduces memory copying.
15
16A :dfn:`MPSC Packet Buffer` has the following key properties:
17
18* Allocate, commit scheme used for packet producing.
19* Claim, free scheme used for packet consuming.
20* Allocator ensures that contiguous memory of requested length is allocated.
21* Following policies can be applied when requested space cannot be allocated:
22
23  * **Overwrite** - oldest entries are dropped until requested amount of memory can
24    be allocated. For each dropped packet user callback is called.
25  * **No overwrite** - When requested amount of space cannot be allocated,
26    allocation fails.
27* Dedicated, optimized API for storing short packets.
28* Allocation with timeout.
29
30Internals
31---------
32
33Each packet in the buffer contains ``MPSC_PBUF`` specific header which is used
34for internal management. Header consists of 2 bit flags. In order to optimize
35memory usage, header can be added on top of the user header using
36:c:macro:`MPSC_PBUF_HDR` and remaining bits in the first word can be application
37specific. Header consists of following flags:
38
39* valid - bit set to one when packet contains valid user packet
40* busy - bit set when packet is being consumed (claimed but not free)
41
42Header state:
43
44+-------+------+----------------------+
45| valid | busy | description          |
46+-------+------+----------------------+
47| 0     | 0    | space is free        |
48+-------+------+----------------------+
49| 1     | 0    | valid packet         |
50+-------+------+----------------------+
51| 1     | 1    | claimed valid packet |
52+-------+------+----------------------+
53| 0     | 1    | internal skip packet |
54+-------+------+----------------------+
55
56Packet buffer space contains free space, valid user packets and internal skip
57packets. Internal skip packets indicates padding, e.g. at the end of the buffer.
58
59Allocation
60^^^^^^^^^^
61
62Using pairs for read and write indexes, available space is determined. If
63space can be allocated, temporary write index is moved and pointer to a space
64within buffer is returned. Packet header is reset. If allocation required
65wrapping of the write index, a skip packet is added to the end of buffer. If
66space cannot be allocated and overwrite is disabled then ``NULL`` pointer is
67returned or context blocks if allocation was with timeout.
68
69Allocation with overwrite
70^^^^^^^^^^^^^^^^^^^^^^^^^
71
72If overwrite is enabled, oldest packets are dropped until requested amount of
73space can be allocated. When packets are dropped ``busy`` flag is checked in the
74header to ensure that currently consumed packet is not overwritten. In that case,
75skip packet is added before busy packet and packets following the busy packet
76are dropped. When busy packet is being freed, such situation is detected and
77packet is converted to skip packet to avoid double processing.
78
79Usage
80-----
81
82Packet header definition
83^^^^^^^^^^^^^^^^^^^^^^^^
84
85Packet header details can be found in :zephyr_file:`include/zephyr/sys/mpsc_packet.h`.
86API functions can be found in :zephyr_file:`include/zephyr/sys/mpsc_pbuf.h`. Headers
87are split to avoid include spam when declaring the packet.
88
89User header structure must start with internal header:
90
91.. code-block:: c
92
93   #include <zephyr/sys/mpsc_packet.h>
94
95   struct foo_header {
96           MPSC_PBUF_HDR;
97           uint32_t length: 32 - MPSC_PBUF_HDR_BITS;
98   };
99
100Packet buffer configuration
101^^^^^^^^^^^^^^^^^^^^^^^^^^^
102
103Configuration structure contains buffer details, configuration flags and
104callbacks. Following callbacks are used by the packet buffer:
105
106* Drop notification - callback called whenever a packet is dropped due to
107  overwrite.
108* Get packet length - callback to determine packet length
109
110Packet producing
111^^^^^^^^^^^^^^^^
112
113Standard, two step method:
114
115.. code-block:: c
116
117   foo_packet *packet = mpsc_pbuf_alloc(buffer, len, K_NO_WAIT);
118
119   fill_data(packet);
120
121   mpsc_pbuf_commit(buffer, packet);
122
123Performance optimized storing of small packets:
124
125* 32 bit word packet
126* 32 bit word with pointer packet
127
128Note that since packets are written by value, they should already contain
129``valid`` bit set in the header.
130
131.. code-block:: c
132
133   mpsc_pbuf_put_word(buffer, data);
134   mpsc_pbuf_put_word_ext(buffer, data, ptr);
135
136Packet consuming
137^^^^^^^^^^^^^^^^
138
139Two step method:
140
141.. code-block:: c
142
143   foo_packet *packet = mpsc_pbuf_claim(buffer);
144
145   process(packet);
146
147   mpsc_pbuf_free(buffer, packet);
148