Zephyr API Documentation 4.3.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
79
83
84#if CONFIG_MODEM_CMUX_MTU > 127
85#define MODEM_CMUX_HEADER_SIZE 7
86#else
87#define MODEM_CMUX_HEADER_SIZE 6
88#endif
89
90
91/* Total size of the CMUX work buffers */
92#define MODEM_CMUX_WORK_BUFFER_SIZE (CONFIG_MODEM_CMUX_MTU + MODEM_CMUX_HEADER_SIZE + \
93 CONFIG_MODEM_CMUX_WORK_BUFFER_SIZE_EXTRA)
94
95enum modem_cmux_state {
96 MODEM_CMUX_STATE_DISCONNECTED = 0,
97 MODEM_CMUX_STATE_CONNECTING,
98 MODEM_CMUX_STATE_CONNECTED,
99 MODEM_CMUX_STATE_ENTER_POWERSAVE,
100 MODEM_CMUX_STATE_POWERSAVE,
101 MODEM_CMUX_STATE_CONFIRM_POWERSAVE,
102 MODEM_CMUX_STATE_WAKEUP,
103 MODEM_CMUX_STATE_DISCONNECTING,
104};
105
106enum modem_cmux_receive_state {
107 MODEM_CMUX_RECEIVE_STATE_SOF = 0,
108 MODEM_CMUX_RECEIVE_STATE_RESYNC,
109 MODEM_CMUX_RECEIVE_STATE_ADDRESS,
110 MODEM_CMUX_RECEIVE_STATE_ADDRESS_CONT,
111 MODEM_CMUX_RECEIVE_STATE_CONTROL,
112 MODEM_CMUX_RECEIVE_STATE_LENGTH,
113 MODEM_CMUX_RECEIVE_STATE_LENGTH_CONT,
114 MODEM_CMUX_RECEIVE_STATE_DATA,
115 MODEM_CMUX_RECEIVE_STATE_FCS,
116 MODEM_CMUX_RECEIVE_STATE_EOF,
117};
118
119enum modem_cmux_dlci_state {
120 MODEM_CMUX_DLCI_STATE_CLOSED,
121 MODEM_CMUX_DLCI_STATE_OPENING,
122 MODEM_CMUX_DLCI_STATE_OPEN,
123 MODEM_CMUX_DLCI_STATE_CLOSING,
124};
125
126struct modem_cmux_dlci {
127 sys_snode_t node;
128
129 /* Pipe */
130 struct modem_pipe pipe;
131
132 /* Context */
133 uint16_t dlci_address;
134 struct modem_cmux *cmux;
135
136 /* Receive buffer */
137 struct ring_buf receive_rb;
138 struct k_mutex receive_rb_lock;
139
140 /* Work */
141 struct k_work_delayable open_work;
142 struct k_work_delayable close_work;
143
144 /* State */
145 enum modem_cmux_dlci_state state;
146
147 /* Statistics */
148#if CONFIG_MODEM_STATS
149 struct modem_stats_buffer receive_buf_stats;
150#endif
151 /* Flow control */
152 bool flow_control : 1;
153 bool rx_full : 1;
154 bool msc_sent : 1;
155};
156
157struct modem_cmux_frame {
158 uint8_t dlci_address;
159 bool cr;
160 bool pf;
161 uint8_t type;
162 const uint8_t *data;
163 uint16_t data_len;
164};
165
166struct modem_cmux_work {
167 struct k_work_delayable dwork;
168 struct modem_cmux *cmux;
169};
170
171struct modem_cmux {
172 /* Bus pipe */
173 struct modem_pipe *pipe;
174
175 /* DLCI channel contexts */
176 sys_slist_t dlcis;
177
178 /* State */
179 enum modem_cmux_state state;
180 bool flow_control_on : 1;
181 bool initiator : 1;
182
183 /* Work lock */
184 bool attached : 1;
185 struct k_spinlock work_lock;
186
187 /* Receive state*/
188 enum modem_cmux_receive_state receive_state;
189 uint16_t receive_buf_len;
190
191 uint8_t work_buf[MODEM_CMUX_WORK_BUFFER_SIZE];
192
193 /* Transmit buffer */
194 struct ring_buf transmit_rb;
195 struct k_mutex transmit_rb_lock;
196
197 /* Received frame */
198 struct modem_cmux_frame frame;
199 uint8_t frame_header[5];
200 uint16_t frame_header_len;
201
202 /* Work */
203 struct k_work_delayable receive_work;
204 struct k_work_delayable transmit_work;
205 struct k_work_delayable connect_work;
206 struct k_work_delayable disconnect_work;
207 struct k_work_delayable runtime_pm_work;
208
209 /* Synchronize actions */
210 struct k_event event;
211 k_timepoint_t t3_timepoint;
212 k_timepoint_t idle_timepoint;
213
214 /* Statistics */
215#if CONFIG_MODEM_STATS
216 struct modem_stats_buffer receive_buf_stats;
217 struct modem_stats_buffer transmit_buf_stats;
218#endif
219 struct modem_cmux_config config;
220};
221
225
231void modem_cmux_init(struct modem_cmux *cmux, const struct modem_cmux_config *config);
232
244
252struct modem_pipe *modem_cmux_dlci_init(struct modem_cmux *cmux, struct modem_cmux_dlci *dlci,
253 const struct modem_cmux_dlci_config *config);
254
261int modem_cmux_attach(struct modem_cmux *cmux, struct modem_pipe *pipe);
262
273int modem_cmux_connect(struct modem_cmux *cmux);
274
285int modem_cmux_connect_async(struct modem_cmux *cmux);
286
297int modem_cmux_disconnect(struct modem_cmux *cmux);
298
309int modem_cmux_disconnect_async(struct modem_cmux *cmux);
310
321void modem_cmux_release(struct modem_cmux *cmux);
322
326
327#ifdef __cplusplus
328}
329#endif
330
331#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 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:77
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:236
uint8_t dlci_address
DLCI channel address.
Definition cmux.h:238
uint8_t * receive_buf
Receive buffer used by pipe.
Definition cmux.h:240
uint16_t receive_buf_size
Size of receive buffer used by pipe [127, ...].
Definition cmux.h:242