1 /** @file
2 * @brief Bluetooth data buffer API
3 */
4
5 /*
6 * Copyright (c) 2016 Intel Corporation
7 *
8 * SPDX-License-Identifier: Apache-2.0
9 */
10
11 #ifndef ZEPHYR_INCLUDE_BLUETOOTH_BUF_H_
12 #define ZEPHYR_INCLUDE_BLUETOOTH_BUF_H_
13
14 /**
15 * @brief Data buffers
16 * @defgroup bt_buf Data buffers
17 * @ingroup bluetooth
18 * @{
19 */
20
21 #include <stdint.h>
22
23 #include <zephyr/net_buf.h>
24 #include <zephyr/bluetooth/hci.h>
25 #include <zephyr/sys/util.h>
26
27 #ifdef __cplusplus
28 extern "C" {
29 #endif
30
31 /** Possible types of buffers passed around the Bluetooth stack in a form of bitmask. */
32 enum bt_buf_type {
33 /** HCI command */
34 BT_BUF_CMD = BIT(0),
35 /** HCI event */
36 BT_BUF_EVT = BIT(1),
37 /** Outgoing ACL data */
38 BT_BUF_ACL_OUT = BIT(2),
39 /** Incoming ACL data */
40 BT_BUF_ACL_IN = BIT(3),
41 /** Outgoing ISO data */
42 BT_BUF_ISO_OUT = BIT(4),
43 /** Incoming ISO data */
44 BT_BUF_ISO_IN = BIT(5),
45 /** H:4 data */
46 BT_BUF_H4 = BIT(6),
47 };
48
49 /** @brief This is a base type for bt_buf user data. */
50 struct bt_buf_data {
51 uint8_t type;
52 };
53
54 /* Headroom reserved in buffers, primarily for HCI transport encoding purposes */
55 #define BT_BUF_RESERVE 1
56
57 /** Helper to include reserved HCI data in buffer calculations */
58 #define BT_BUF_SIZE(size) (BT_BUF_RESERVE + (size))
59
60 /** Helper to calculate needed buffer size for HCI ACL packets */
61 #define BT_BUF_ACL_SIZE(size) BT_BUF_SIZE(BT_HCI_ACL_HDR_SIZE + (size))
62
63 /** Helper to calculate needed buffer size for HCI Event packets. */
64 #define BT_BUF_EVT_SIZE(size) BT_BUF_SIZE(BT_HCI_EVT_HDR_SIZE + (size))
65
66 /** Helper to calculate needed buffer size for HCI Command packets. */
67 #define BT_BUF_CMD_SIZE(size) BT_BUF_SIZE(BT_HCI_CMD_HDR_SIZE + (size))
68
69 /** Helper to calculate needed buffer size for HCI ISO packets. */
70 #define BT_BUF_ISO_SIZE(size) BT_BUF_SIZE(BT_HCI_ISO_HDR_SIZE + \
71 BT_HCI_ISO_SDU_TS_HDR_SIZE + \
72 (size))
73
74 /** Data size needed for HCI ACL RX buffers */
75 #define BT_BUF_ACL_RX_SIZE BT_BUF_ACL_SIZE(CONFIG_BT_BUF_ACL_RX_SIZE)
76
77 /** Data size needed for HCI Event RX buffers */
78 #define BT_BUF_EVT_RX_SIZE BT_BUF_EVT_SIZE(CONFIG_BT_BUF_EVT_RX_SIZE)
79
80 #if defined(CONFIG_BT_ISO)
81 #define BT_BUF_ISO_RX_SIZE BT_BUF_ISO_SIZE(CONFIG_BT_ISO_RX_MTU)
82 #define BT_BUF_ISO_RX_COUNT CONFIG_BT_ISO_RX_BUF_COUNT
83 #else
84 #define BT_BUF_ISO_RX_SIZE 0
85 #define BT_BUF_ISO_RX_COUNT 0
86 #endif /* CONFIG_BT_ISO */
87
88 /* see Core Spec v6.0 vol.4 part E 7.4.5 */
89 #define BT_BUF_ACL_RX_COUNT_MAX 65535
90
91 #if defined(CONFIG_BT_CONN) && defined(CONFIG_BT_HCI_HOST)
92 /* The host needs more ACL buffers than maximum ACL links. This is because of
93 * the way we re-assemble ACL packets into L2CAP PDUs.
94 *
95 * We keep around the first buffer (that comes from the driver) to do
96 * re-assembly into, and if all links are re-assembling, there will be no buffer
97 * available for the HCI driver to allocate from.
98 *
99 * TODO: When CONFIG_BT_BUF_ACL_RX_COUNT is removed,
100 * remove the MAX and only keep (CONFIG_BT_MAX_CONN + 1)
101 */
102 #define BT_BUF_ACL_RX_COUNT \
103 (MAX(CONFIG_BT_BUF_ACL_RX_COUNT, (CONFIG_BT_MAX_CONN + 1)) + \
104 CONFIG_BT_BUF_ACL_RX_COUNT_EXTRA)
105 #else
106 #define BT_BUF_ACL_RX_COUNT 0
107 #endif /* CONFIG_BT_CONN && CONFIG_BT_HCI_HOST */
108
109 #if defined(CONFIG_BT_BUF_ACL_RX_COUNT) && CONFIG_BT_BUF_ACL_RX_COUNT > 0
110 #warning "CONFIG_BT_BUF_ACL_RX_COUNT is deprecated, see Zephyr 4.1 migration guide"
111 #endif /* CONFIG_BT_BUF_ACL_RX_COUNT && CONFIG_BT_BUF_ACL_RX_COUNT > 0 */
112
113 BUILD_ASSERT(BT_BUF_ACL_RX_COUNT <= BT_BUF_ACL_RX_COUNT_MAX,
114 "Maximum number of ACL RX buffer is 65535, reduce CONFIG_BT_BUF_ACL_RX_COUNT_EXTRA");
115
116 /** Data size needed for HCI ACL, HCI ISO or Event RX buffers */
117 #define BT_BUF_RX_SIZE (MAX(MAX(BT_BUF_ACL_RX_SIZE, BT_BUF_EVT_RX_SIZE), \
118 BT_BUF_ISO_RX_SIZE))
119
120 /** Buffer count needed for HCI ACL, HCI ISO or Event RX buffers */
121 #define BT_BUF_RX_COUNT (MAX(MAX(CONFIG_BT_BUF_EVT_RX_COUNT, \
122 BT_BUF_ACL_RX_COUNT), \
123 BT_BUF_ISO_RX_COUNT))
124
125 /** Data size needed for HCI Command buffers. */
126 #define BT_BUF_CMD_TX_SIZE BT_BUF_CMD_SIZE(CONFIG_BT_BUF_CMD_TX_SIZE)
127
128 /** Allocate a buffer for incoming data
129 *
130 * This will set the buffer type so bt_buf_set_type() does not need to
131 * be explicitly called.
132 *
133 * @param type Type of buffer. Only BT_BUF_EVT, BT_BUF_ACL_IN and BT_BUF_ISO_IN
134 * are allowed.
135 * @param timeout Non-negative waiting period to obtain a buffer or one of the
136 * special values K_NO_WAIT and K_FOREVER.
137 * @return A new buffer.
138 */
139 struct net_buf *bt_buf_get_rx(enum bt_buf_type type, k_timeout_t timeout);
140
141 /** A callback to notify about freed buffer in the incoming data pool.
142 *
143 * This callback is called when a buffer of a given type is freed and can be requested through the
144 * @ref bt_buf_get_rx function. However, this callback is called from the context of the buffer
145 * freeing operation and must not attempt to allocate a new buffer from the same pool.
146 *
147 * @warning When this callback is called, the scheduler is locked and the callee must not perform
148 * any action that makes the current thread unready. This callback must only be used for very
149 * short non-blocking operation (e.g. submitting a work item).
150 *
151 * @param type_mask A bit mask of buffer types that have been freed.
152 */
153 typedef void (*bt_buf_rx_freed_cb_t)(enum bt_buf_type type_mask);
154
155 /** Set the callback to notify about freed buffer in the incoming data pool.
156 *
157 * @param cb Callback to notify about freed buffer in the incoming data pool. If NULL, the callback
158 * is disabled.
159 */
160 void bt_buf_rx_freed_cb_set(bt_buf_rx_freed_cb_t cb);
161
162 /** Allocate a buffer for outgoing data
163 *
164 * This will set the buffer type so bt_buf_set_type() does not need to
165 * be explicitly called.
166 *
167 * @param type Type of buffer. Only BT_BUF_CMD, BT_BUF_ACL_OUT or
168 * BT_BUF_H4, when operating on H:4 mode, are allowed.
169 * @param timeout Non-negative waiting period to obtain a buffer or one of the
170 * special values K_NO_WAIT and K_FOREVER.
171 * @param data Initial data to append to buffer.
172 * @param size Initial data size.
173 * @return A new buffer.
174 */
175 struct net_buf *bt_buf_get_tx(enum bt_buf_type type, k_timeout_t timeout,
176 const void *data, size_t size);
177
178 /** Allocate a buffer for an HCI Event
179 *
180 * This will set the buffer type so bt_buf_set_type() does not need to
181 * be explicitly called.
182 *
183 * @param evt HCI event code
184 * @param discardable Whether the driver considers the event discardable.
185 * @param timeout Non-negative waiting period to obtain a buffer or one of
186 * the special values K_NO_WAIT and K_FOREVER.
187 * @return A new buffer.
188 */
189 struct net_buf *bt_buf_get_evt(uint8_t evt, bool discardable, k_timeout_t timeout);
190
191 /** Set the buffer type
192 *
193 * @param buf Bluetooth buffer
194 * @param type The BT_* type to set the buffer to
195 */
bt_buf_set_type(struct net_buf * buf,enum bt_buf_type type)196 static inline void bt_buf_set_type(struct net_buf *buf, enum bt_buf_type type)
197 {
198 ((struct bt_buf_data *)net_buf_user_data(buf))->type = type;
199 }
200
201 /** Get the buffer type
202 *
203 * @param buf Bluetooth buffer
204 *
205 * @return The BT_* type to of the buffer
206 */
bt_buf_get_type(struct net_buf * buf)207 static inline enum bt_buf_type bt_buf_get_type(struct net_buf *buf)
208 {
209 return (enum bt_buf_type)((struct bt_buf_data *)net_buf_user_data(buf))
210 ->type;
211 }
212
213 /**
214 * @}
215 */
216
217 #ifdef __cplusplus
218 }
219 #endif
220
221 #endif /* ZEPHYR_INCLUDE_BLUETOOTH_BUF_H_ */
222