Zephyr API Documentation 4.4.99
A Scalable Open Source RTOS
Loading...
Searching...
No Matches
cmux.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2022 Trackunit Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7/*
8 * This library uses CMUX to create multiple data channels, called DLCIs, on a single serial bus.
9 * Each DLCI has an address from 1 to 63. DLCI address 0 is reserved for control commands.
10 *
11 * Design overview:
12 *
13 * DLCI1 <-----------+ +-------> DLCI1
14 * v v
15 * DLCI2 <---> CMUX instance <--> Serial bus <--> Client <--> DLCI2
16 * ^ ^
17 * DLCI3 <-----------+ +-------> DLCI3
18 *
19 * Writing to and from the CMUX instances is done using the modem_pipe API.
20 */
21
22#include <zephyr/kernel.h>
23#include <zephyr/types.h>
25#include <zephyr/sys/atomic.h>
26
27#include <zephyr/modem/pipe.h>
28#include <zephyr/modem/stats.h>
29
30#ifndef ZEPHYR_MODEM_CMUX_
31#define ZEPHYR_MODEM_CMUX_
32
33#ifdef __cplusplus
34extern "C" {
35#endif
36
45
46struct modem_cmux;
47
52
53typedef void (*modem_cmux_callback)(struct modem_cmux *cmux, enum modem_cmux_event event,
54 void *user_data);
55
81
85
86#if CONFIG_MODEM_CMUX_MTU > 127
87#define MODEM_CMUX_HEADER_SIZE 7
88#else
89#define MODEM_CMUX_HEADER_SIZE 6
90#endif
91
92
93/* Total size of the CMUX work buffers */
94#define MODEM_CMUX_WORK_BUFFER_SIZE (CONFIG_MODEM_CMUX_MTU + MODEM_CMUX_HEADER_SIZE + \
95 CONFIG_MODEM_CMUX_WORK_BUFFER_SIZE_EXTRA)
96
97enum modem_cmux_state {
98 MODEM_CMUX_STATE_DISCONNECTED = 0,
99 MODEM_CMUX_STATE_CONNECTING,
100 MODEM_CMUX_STATE_CONNECTED,
101 MODEM_CMUX_STATE_ENTER_POWERSAVE,
102 MODEM_CMUX_STATE_POWERSAVE,
103 MODEM_CMUX_STATE_CONFIRM_POWERSAVE,
104 MODEM_CMUX_STATE_WAKEUP,
105 MODEM_CMUX_STATE_DISCONNECTING,
106};
107
108enum modem_cmux_receive_state {
109 MODEM_CMUX_RECEIVE_STATE_SOF = 0,
110 MODEM_CMUX_RECEIVE_STATE_RESYNC,
111 MODEM_CMUX_RECEIVE_STATE_ADDRESS,
112 MODEM_CMUX_RECEIVE_STATE_ADDRESS_CONT,
113 MODEM_CMUX_RECEIVE_STATE_CONTROL,
114 MODEM_CMUX_RECEIVE_STATE_LENGTH,
115 MODEM_CMUX_RECEIVE_STATE_LENGTH_CONT,
116 MODEM_CMUX_RECEIVE_STATE_DATA,
117 MODEM_CMUX_RECEIVE_STATE_DATA_CONT,
118 MODEM_CMUX_RECEIVE_STATE_FCS,
119 MODEM_CMUX_RECEIVE_STATE_EOF,
120};
121
122enum modem_cmux_dlci_state {
123 MODEM_CMUX_DLCI_STATE_CLOSED,
124 MODEM_CMUX_DLCI_STATE_OPENING,
125 MODEM_CMUX_DLCI_STATE_OPEN,
126 MODEM_CMUX_DLCI_STATE_CLOSING,
127};
128
129struct modem_cmux_dlci {
130 sys_snode_t node;
131
132 /* Pipe */
133 struct modem_pipe pipe;
134
135 /* Context */
136 uint16_t dlci_address;
137 struct modem_cmux *cmux;
138
139 /* Receive buffer */
140 struct ring_buf receive_rb;
141 struct k_mutex receive_rb_lock;
142
143 /* Work */
144 struct k_work_delayable open_work;
145 struct k_work_delayable close_work;
146
147 /* State */
148 enum modem_cmux_dlci_state state;
149
150 /* Statistics */
151#if CONFIG_MODEM_STATS
152 struct modem_stats_buffer receive_buf_stats;
153#endif
154 /* Flow control */
155 bool flow_control : 1;
156 bool rx_full : 1;
157 bool msc_sent : 1;
158};
159
160struct modem_cmux_frame {
161 uint8_t dlci_address;
162 bool cr;
163 bool pf;
164 uint8_t type;
165 const uint8_t *data;
166 uint16_t data_len;
167};
168
169struct modem_cmux_work {
170 struct k_work_delayable dwork;
171 struct modem_cmux *cmux;
172};
173
174struct modem_cmux {
175 /* Bus pipe */
176 struct modem_pipe *pipe;
177
178 /* DLCI channel contexts */
179 sys_slist_t dlcis;
180
181 /* State */
182 enum modem_cmux_state state;
183 uint8_t retry_count;
184 bool flow_control_on : 1;
185 bool initiator : 1;
186
187 /* Work lock */
188 bool attached : 1;
189 struct k_spinlock work_lock;
190
191 /* Receive state*/
192 enum modem_cmux_receive_state receive_state;
193 int receive_buf_len;
194
195 /* Transmit buffer */
196 struct ring_buf transmit_rb;
197 struct k_mutex transmit_rb_lock;
198
199 /* Received frame */
200 struct modem_cmux_frame frame;
201 int frame_start;
202 uint16_t frame_header_len;
203
204 /* Work */
205 struct k_work_delayable receive_work;
206 struct k_work_delayable transmit_work;
207 struct k_work_delayable connect_work;
208 struct k_work_delayable disconnect_work;
209 struct k_work_delayable runtime_pm_work;
210
211 /* Synchronize actions */
212 struct k_event event;
213 k_timepoint_t t3_timepoint;
214 k_timepoint_t idle_timepoint;
215
216 /* Statistics */
217#if CONFIG_MODEM_STATS
218 struct modem_stats_buffer receive_buf_stats;
219 struct modem_stats_buffer transmit_buf_stats;
220#endif
221 struct modem_cmux_config config;
222};
223
227
233void modem_cmux_init(struct modem_cmux *cmux, const struct modem_cmux_config *config);
234
246
254struct modem_pipe *modem_cmux_dlci_init(struct modem_cmux *cmux, struct modem_cmux_dlci *dlci,
255 const struct modem_cmux_dlci_config *config);
256
263int modem_cmux_attach(struct modem_cmux *cmux, struct modem_pipe *pipe);
264
275int modem_cmux_connect(struct modem_cmux *cmux);
276
287int modem_cmux_connect_async(struct modem_cmux *cmux);
288
299int modem_cmux_disconnect(struct modem_cmux *cmux);
300
311int modem_cmux_disconnect_async(struct modem_cmux *cmux);
312
323void modem_cmux_release(struct modem_cmux *cmux);
324
328
329#ifdef __cplusplus
330}
331#endif
332
333#endif /* ZEPHYR_MODEM_CMUX_ */
int modem_cmux_connect(struct modem_cmux *cmux)
Connect CMUX instance.
int modem_cmux_disconnect(struct modem_cmux *cmux)
Close down and disconnect CMUX instance.
int modem_cmux_disconnect_async(struct modem_cmux *cmux)
Close down and disconnect CMUX instance asynchronously.
void(* modem_cmux_callback)(struct modem_cmux *cmux, enum modem_cmux_event event, void *user_data)
Definition cmux.h:53
int modem_cmux_connect_async(struct modem_cmux *cmux)
Connect CMUX instance asynchronously.
modem_cmux_event
Definition cmux.h:48
int modem_cmux_attach(struct modem_cmux *cmux, struct modem_pipe *pipe)
Attach CMUX instance to pipe.
struct modem_pipe * modem_cmux_dlci_init(struct modem_cmux *cmux, struct modem_cmux_dlci *dlci, const struct modem_cmux_dlci_config *config)
Initialize DLCI instance and register it with CMUX instance.
void modem_cmux_init(struct modem_cmux *cmux, const struct modem_cmux_config *config)
Initialize CMUX instance.
void modem_cmux_release(struct modem_cmux *cmux)
Release CMUX instance from pipe.
@ MODEM_CMUX_EVENT_DISCONNECTED
Definition cmux.h:50
@ MODEM_CMUX_EVENT_CONNECTED
Definition cmux.h:49
struct _slist sys_slist_t
Single-linked list structure.
Definition slist.h:49
struct _snode sys_snode_t
Single-linked list node structure.
Definition slist.h:39
Public kernel APIs.
state
Definition parser_state.h:29
__UINT8_TYPE__ uint8_t
Definition stdint.h:88
__UINT16_TYPE__ uint16_t
Definition stdint.h:89
Kernel timeout type.
Definition clock.h:65
Contains CMUX instance configuration data.
Definition cmux.h:59
bool no_powersave_handshake
Skip the in-band power-save handshake on both entry and exit.
Definition cmux.h:77
bool enable_runtime_power_management
Enable runtime power management.
Definition cmux.h:73
uint8_t * receive_buf
Receive buffer.
Definition cmux.h:65
modem_cmux_callback callback
Invoked when event occurs.
Definition cmux.h:61
uint16_t receive_buf_size
Size of receive buffer in bytes [127, ...].
Definition cmux.h:67
void * user_data
Free to use pointer passed to event handler when invoked.
Definition cmux.h:63
uint8_t * transmit_buf
Transmit buffer.
Definition cmux.h:69
k_timeout_t idle_timeout
Idle timeout for power save.
Definition cmux.h:79
bool close_pipe_on_power_save
Close pipe on power save.
Definition cmux.h:75
uint16_t transmit_buf_size
Size of transmit buffer in bytes [149, ...].
Definition cmux.h:71
CMUX DLCI configuration.
Definition cmux.h:238
uint8_t dlci_address
DLCI channel address.
Definition cmux.h:240
uint8_t * receive_buf
Receive buffer used by pipe.
Definition cmux.h:242
uint16_t receive_buf_size
Size of receive buffer used by pipe [127, ...].
Definition cmux.h:244