Zephyr API Documentation 4.2.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
59
60/* Total size of the CMUX work buffers */
61#define MODEM_CMUX_WORK_BUFFER_SIZE (CONFIG_MODEM_CMUX_MTU + 7 + \
62 CONFIG_MODEM_CMUX_WORK_BUFFER_SIZE_EXTRA)
63
64enum modem_cmux_state {
65 MODEM_CMUX_STATE_DISCONNECTED = 0,
66 MODEM_CMUX_STATE_CONNECTING,
67 MODEM_CMUX_STATE_CONNECTED,
68 MODEM_CMUX_STATE_DISCONNECTING,
69};
70
71enum modem_cmux_receive_state {
72 MODEM_CMUX_RECEIVE_STATE_SOF = 0,
73 MODEM_CMUX_RECEIVE_STATE_RESYNC,
74 MODEM_CMUX_RECEIVE_STATE_ADDRESS,
75 MODEM_CMUX_RECEIVE_STATE_ADDRESS_CONT,
76 MODEM_CMUX_RECEIVE_STATE_CONTROL,
77 MODEM_CMUX_RECEIVE_STATE_LENGTH,
78 MODEM_CMUX_RECEIVE_STATE_LENGTH_CONT,
79 MODEM_CMUX_RECEIVE_STATE_DATA,
80 MODEM_CMUX_RECEIVE_STATE_FCS,
81 MODEM_CMUX_RECEIVE_STATE_EOF,
82};
83
84enum modem_cmux_dlci_state {
85 MODEM_CMUX_DLCI_STATE_CLOSED,
86 MODEM_CMUX_DLCI_STATE_OPENING,
87 MODEM_CMUX_DLCI_STATE_OPEN,
88 MODEM_CMUX_DLCI_STATE_CLOSING,
89};
90
91struct modem_cmux_dlci {
92 sys_snode_t node;
93
94 /* Pipe */
95 struct modem_pipe pipe;
96
97 /* Context */
98 uint16_t dlci_address;
99 struct modem_cmux *cmux;
100
101 /* Receive buffer */
102 struct ring_buf receive_rb;
103 struct k_mutex receive_rb_lock;
104
105 /* Work */
106 struct k_work_delayable open_work;
107 struct k_work_delayable close_work;
108
109 /* State */
110 enum modem_cmux_dlci_state state;
111
112 /* Statistics */
113#if CONFIG_MODEM_STATS
114 struct modem_stats_buffer receive_buf_stats;
115#endif
116};
117
118struct modem_cmux_frame {
119 uint8_t dlci_address;
120 bool cr;
121 bool pf;
122 uint8_t type;
123 const uint8_t *data;
124 uint16_t data_len;
125};
126
127struct modem_cmux_work {
128 struct k_work_delayable dwork;
129 struct modem_cmux *cmux;
130};
131
132struct modem_cmux {
133 /* Bus pipe */
134 struct modem_pipe *pipe;
135
136 /* Event handler */
137 modem_cmux_callback callback;
138 void *user_data;
139
140 /* DLCI channel contexts */
141 sys_slist_t dlcis;
142
143 /* State */
144 enum modem_cmux_state state;
145 bool flow_control_on;
146
147 /* Work lock */
148 bool attached;
149 struct k_spinlock work_lock;
150
151 /* Receive state*/
152 enum modem_cmux_receive_state receive_state;
153
154 /* Receive buffer */
155 uint8_t *receive_buf;
156 uint16_t receive_buf_size;
157 uint16_t receive_buf_len;
158
159 uint8_t work_buf[MODEM_CMUX_WORK_BUFFER_SIZE];
160
161 /* Transmit buffer */
162 struct ring_buf transmit_rb;
163 struct k_mutex transmit_rb_lock;
164
165 /* Received frame */
166 struct modem_cmux_frame frame;
167 uint8_t frame_header[5];
168 uint16_t frame_header_len;
169
170 /* Work */
171 struct k_work_delayable receive_work;
172 struct k_work_delayable transmit_work;
173 struct k_work_delayable connect_work;
174 struct k_work_delayable disconnect_work;
175
176 /* Synchronize actions */
177 struct k_event event;
178
179 /* Statistics */
180#if CONFIG_MODEM_STATS
181 struct modem_stats_buffer receive_buf_stats;
182 struct modem_stats_buffer transmit_buf_stats;
183#endif
184};
185
189
207
213void modem_cmux_init(struct modem_cmux *cmux, const struct modem_cmux_config *config);
214
226
234struct modem_pipe *modem_cmux_dlci_init(struct modem_cmux *cmux, struct modem_cmux_dlci *dlci,
235 const struct modem_cmux_dlci_config *config);
236
243int modem_cmux_attach(struct modem_cmux *cmux, struct modem_pipe *pipe);
244
255int modem_cmux_connect(struct modem_cmux *cmux);
256
267int modem_cmux_connect_async(struct modem_cmux *cmux);
268
279int modem_cmux_disconnect(struct modem_cmux *cmux);
280
291int modem_cmux_disconnect_async(struct modem_cmux *cmux);
292
303void modem_cmux_release(struct modem_cmux *cmux);
304
308
309#ifdef __cplusplus
310}
311#endif
312
313#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
Contains CMUX instance configuration data.
Definition cmux.h:193
uint8_t * receive_buf
Receive buffer.
Definition cmux.h:199
modem_cmux_callback callback
Invoked when event occurs.
Definition cmux.h:195
uint16_t receive_buf_size
Size of receive buffer in bytes [127, ...].
Definition cmux.h:201
void * user_data
Free to use pointer passed to event handler when invoked.
Definition cmux.h:197
uint8_t * transmit_buf
Transmit buffer.
Definition cmux.h:203
uint16_t transmit_buf_size
Size of transmit buffer in bytes [149, ...].
Definition cmux.h:205
CMUX DLCI configuration.
Definition cmux.h:218
uint8_t dlci_address
DLCI channel address.
Definition cmux.h:220
uint8_t * receive_buf
Receive buffer used by pipe.
Definition cmux.h:222
uint16_t receive_buf_size
Size of receive buffer used by pipe [127, ...].
Definition cmux.h:224