Zephyr Project API 4.1.0
A Scalable Open Source RTOS
Loading...
Searching...
No Matches
pbuf.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2023 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7#ifndef ZEPHYR_INCLUDE_IPC_PBUF_H_
8#define ZEPHYR_INCLUDE_IPC_PBUF_H_
9
10#include <zephyr/cache.h>
11#include <zephyr/devicetree.h>
12
13#ifdef __cplusplus
14extern "C" {
15#endif
16
25#define PBUF_PACKET_LEN_SZ sizeof(uint32_t)
26
27/* Amount of data that is left unused to distinguish between empty and full. */
28#define _PBUF_IDX_SIZE sizeof(uint32_t)
29
30/* Minimal length of the data field in the buffer to store the smalest packet
31 * possible.
32 * (+1) for at least one byte of data.
33 * (+_PBUF_IDX_SIZE) to distinguish buffer full and buffer empty.
34 * Rounded up to keep wr/rd indexes pointing to aligned address.
35 */
36#define _PBUF_MIN_DATA_LEN ROUND_UP(PBUF_PACKET_LEN_SZ + 1 + _PBUF_IDX_SIZE, _PBUF_IDX_SIZE)
37
38#if defined(CONFIG_ARCH_POSIX)
39/* For the native simulated boards we need to modify some pointers at init */
40#define PBUF_MAYBE_CONST
41#else
42#define PBUF_MAYBE_CONST const
43#endif
44
49struct pbuf_cfg {
50 volatile uint32_t *rd_idx_loc; /* Address of the variable holding
51 * index value of the first valid byte
52 * in data[].
53 */
54 volatile uint32_t *handshake_loc;/* Address of the variable holding
55 * handshake information.
56 */
57 volatile uint32_t *wr_idx_loc; /* Address of the variable holding
58 * index value of the first free byte
59 * in data[].
60 */
61 uint32_t dcache_alignment; /* CPU data cache line size in bytes.
62 * Used for validation - TODO: To be
63 * replaced by flags.
64 */
65 uint32_t len; /* Length of data[] in bytes. */
66 uint8_t *data_loc; /* Location of the data[]. */
67};
68
75struct pbuf_data {
76 volatile uint32_t wr_idx; /* Index of the first holding first
77 * free byte in data[]. Used for
78 * writing.
79 */
80 volatile uint32_t rd_idx; /* Index of the first holding first
81 * valid byte in data[]. Used for
82 * reading.
83 */
84};
85
86
99struct pbuf {
100 PBUF_MAYBE_CONST struct pbuf_cfg *const cfg; /* Configuration of the
101 * buffer.
102 */
103 struct pbuf_data data; /* Data used to read and write
104 * to the buffer
105 */
106};
107
120#define PBUF_CFG_INIT(mem_addr, size, dcache_align, use_handshake) \
121{ \
122 .rd_idx_loc = (uint32_t *)(mem_addr), \
123 .handshake_loc = use_handshake ? (uint32_t *)((uint8_t *)(mem_addr) + \
124 _PBUF_IDX_SIZE) : NULL, \
125 .wr_idx_loc = (uint32_t *)((uint8_t *)(mem_addr) + MAX(dcache_align, \
126 (use_handshake ? 2 : 1) * _PBUF_IDX_SIZE)), \
127 .data_loc = (uint8_t *)((uint8_t *)(mem_addr) + \
128 MAX(dcache_align, (use_handshake ? 2 : 1) * \
129 _PBUF_IDX_SIZE) + _PBUF_IDX_SIZE), \
130 .len = (uint32_t)((uint32_t)(size) - MAX(dcache_align, \
131 (use_handshake ? 2 : 1) * _PBUF_IDX_SIZE) - _PBUF_IDX_SIZE), \
132 .dcache_alignment = (dcache_align), \
133}
134
142#define PBUF_HEADER_OVERHEAD(dcache_align) \
143 (MAX(dcache_align, _PBUF_IDX_SIZE) + _PBUF_IDX_SIZE)
144
155#define PBUF_DEFINE(name, mem_addr, size, dcache_align, use_handshake, compatibility) \
156 BUILD_ASSERT(dcache_align >= 0, \
157 "Cache line size must be non negative."); \
158 BUILD_ASSERT((size) > 0 && IS_PTR_ALIGNED_BYTES(size, _PBUF_IDX_SIZE), \
159 "Incorrect size."); \
160 BUILD_ASSERT(IS_PTR_ALIGNED_BYTES(mem_addr, MAX(dcache_align, _PBUF_IDX_SIZE)), \
161 "Misaligned memory."); \
162 BUILD_ASSERT(size >= (MAX(dcache_align, _PBUF_IDX_SIZE) + _PBUF_IDX_SIZE + \
163 _PBUF_MIN_DATA_LEN), "Insufficient size."); \
164 BUILD_ASSERT(!(compatibility) || (dcache_align) >= 8, \
165 "Data cache alignment must be at least 8 if compatibility is enabled.");\
166 static PBUF_MAYBE_CONST struct pbuf_cfg cfg_##name = \
167 PBUF_CFG_INIT(mem_addr, size, dcache_align, use_handshake); \
168 static struct pbuf name = { \
169 .cfg = &cfg_##name, \
170 }
171
186int pbuf_tx_init(struct pbuf *pb);
187
202int pbuf_rx_init(struct pbuf *pb);
203
218int pbuf_write(struct pbuf *pb, const char *buf, uint16_t len);
219
236int pbuf_read(struct pbuf *pb, char *buf, uint16_t len);
237
247
256void pbuf_handshake_write(struct pbuf *pb, uint32_t value);
257
270int pbuf_get_initial_buf(struct pbuf *pb, volatile char **buf, uint16_t *len);
271
276#ifdef __cplusplus
277}
278#endif
279
280#endif /* ZEPHYR_INCLUDE_IPC_PBUF_H_ */
cache API interface
Devicetree main header.
int pbuf_tx_init(struct pbuf *pb)
Initialize the Tx packet buffer.
int pbuf_read(struct pbuf *pb, char *buf, uint16_t len)
Read specified amount of data from the packet buffer.
void pbuf_handshake_write(struct pbuf *pb, uint32_t value)
Write handshake word to pbuf.
int pbuf_rx_init(struct pbuf *pb)
Initialize the Rx packet buffer.
int pbuf_write(struct pbuf *pb, const char *buf, uint16_t len)
Write specified amount of data to the packet buffer.
uint32_t pbuf_handshake_read(struct pbuf *pb)
Read handshake word from pbuf.
int pbuf_get_initial_buf(struct pbuf *pb, volatile char **buf, uint16_t *len)
Get first buffer from pbuf.
#define PBUF_MAYBE_CONST
Definition pbuf.h:42
__UINT32_TYPE__ uint32_t
Definition stdint.h:90
__UINT8_TYPE__ uint8_t
Definition stdint.h:88
__UINT16_TYPE__ uint16_t
Definition stdint.h:89
Control block of packet buffer.
Definition pbuf.h:49
uint8_t * data_loc
Definition pbuf.h:66
volatile uint32_t * handshake_loc
Definition pbuf.h:54
uint32_t dcache_alignment
Definition pbuf.h:61
uint32_t len
Definition pbuf.h:65
volatile uint32_t * rd_idx_loc
Definition pbuf.h:50
volatile uint32_t * wr_idx_loc
Definition pbuf.h:57
Data block of the packed buffer.
Definition pbuf.h:75
volatile uint32_t wr_idx
Definition pbuf.h:76
volatile uint32_t rd_idx
Definition pbuf.h:80
Scure packed buffer.
Definition pbuf.h:99
const struct pbuf_cfg *const cfg
Definition pbuf.h:100
struct pbuf_data data
Definition pbuf.h:103