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
60enum modem_cmux_state {
61 MODEM_CMUX_STATE_DISCONNECTED = 0,
62 MODEM_CMUX_STATE_CONNECTING,
63 MODEM_CMUX_STATE_CONNECTED,
64 MODEM_CMUX_STATE_DISCONNECTING,
65};
66
67enum modem_cmux_receive_state {
68 MODEM_CMUX_RECEIVE_STATE_SOF = 0,
69 MODEM_CMUX_RECEIVE_STATE_RESYNC,
70 MODEM_CMUX_RECEIVE_STATE_ADDRESS,
71 MODEM_CMUX_RECEIVE_STATE_ADDRESS_CONT,
72 MODEM_CMUX_RECEIVE_STATE_CONTROL,
73 MODEM_CMUX_RECEIVE_STATE_LENGTH,
74 MODEM_CMUX_RECEIVE_STATE_LENGTH_CONT,
75 MODEM_CMUX_RECEIVE_STATE_DATA,
76 MODEM_CMUX_RECEIVE_STATE_FCS,
77 MODEM_CMUX_RECEIVE_STATE_DROP,
78 MODEM_CMUX_RECEIVE_STATE_EOF,
79};
80
81enum modem_cmux_dlci_state {
82 MODEM_CMUX_DLCI_STATE_CLOSED,
83 MODEM_CMUX_DLCI_STATE_OPENING,
84 MODEM_CMUX_DLCI_STATE_OPEN,
85 MODEM_CMUX_DLCI_STATE_CLOSING,
86};
87
88struct modem_cmux_dlci {
89 sys_snode_t node;
90
91 /* Pipe */
92 struct modem_pipe pipe;
93
94 /* Context */
95 uint16_t dlci_address;
96 struct modem_cmux *cmux;
97
98 /* Receive buffer */
99 struct ring_buf receive_rb;
100 struct k_mutex receive_rb_lock;
101
102 /* Work */
103 struct k_work_delayable open_work;
104 struct k_work_delayable close_work;
105
106 /* State */
107 enum modem_cmux_dlci_state state;
108
109 /* Statistics */
110#if CONFIG_MODEM_STATS
111 struct modem_stats_buffer receive_buf_stats;
112#endif
113};
114
115struct modem_cmux_frame {
116 uint8_t dlci_address;
117 bool cr;
118 bool pf;
119 uint8_t type;
120 const uint8_t *data;
121 uint16_t data_len;
122};
123
124struct modem_cmux_work {
125 struct k_work_delayable dwork;
126 struct modem_cmux *cmux;
127};
128
129struct modem_cmux {
130 /* Bus pipe */
131 struct modem_pipe *pipe;
132
133 /* Event handler */
134 modem_cmux_callback callback;
135 void *user_data;
136
137 /* DLCI channel contexts */
138 sys_slist_t dlcis;
139
140 /* State */
141 enum modem_cmux_state state;
142 bool flow_control_on;
143
144 /* Work lock */
145 bool attached;
146 struct k_spinlock work_lock;
147
148 /* Receive state*/
149 enum modem_cmux_receive_state receive_state;
150
151 /* Receive buffer */
152 uint8_t *receive_buf;
153 uint16_t receive_buf_size;
154 uint16_t receive_buf_len;
155
156 uint8_t work_buf[CONFIG_MODEM_CMUX_WORK_BUFFER_SIZE];
157
158 /* Transmit buffer */
159 struct ring_buf transmit_rb;
160 struct k_mutex transmit_rb_lock;
161
162 /* Received frame */
163 struct modem_cmux_frame frame;
164 uint8_t frame_header[5];
165 uint16_t frame_header_len;
166
167 /* Work */
168 struct k_work_delayable receive_work;
169 struct k_work_delayable transmit_work;
170 struct k_work_delayable connect_work;
171 struct k_work_delayable disconnect_work;
172
173 /* Synchronize actions */
174 struct k_event event;
175
176 /* Statistics */
177#if CONFIG_MODEM_STATS
178 struct modem_stats_buffer receive_buf_stats;
179 struct modem_stats_buffer transmit_buf_stats;
180#endif
181};
182
186
204
210void modem_cmux_init(struct modem_cmux *cmux, const struct modem_cmux_config *config);
211
223
231struct modem_pipe *modem_cmux_dlci_init(struct modem_cmux *cmux, struct modem_cmux_dlci *dlci,
232 const struct modem_cmux_dlci_config *config);
233
240int modem_cmux_attach(struct modem_cmux *cmux, struct modem_pipe *pipe);
241
252int modem_cmux_connect(struct modem_cmux *cmux);
253
264int modem_cmux_connect_async(struct modem_cmux *cmux);
265
276int modem_cmux_disconnect(struct modem_cmux *cmux);
277
288int modem_cmux_disconnect_async(struct modem_cmux *cmux);
289
300void modem_cmux_release(struct modem_cmux *cmux);
301
305
306#ifdef __cplusplus
307}
308#endif
309
310#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:190
uint8_t * receive_buf
Receive buffer.
Definition cmux.h:196
modem_cmux_callback callback
Invoked when event occurs.
Definition cmux.h:192
uint16_t receive_buf_size
Size of receive buffer in bytes [127, ...].
Definition cmux.h:198
void * user_data
Free to use pointer passed to event handler when invoked.
Definition cmux.h:194
uint8_t * transmit_buf
Transmit buffer.
Definition cmux.h:200
uint16_t transmit_buf_size
Size of transmit buffer in bytes [149, ...].
Definition cmux.h:202
CMUX DLCI configuration.
Definition cmux.h:215
uint8_t dlci_address
DLCI channel address.
Definition cmux.h:217
uint8_t * receive_buf
Receive buffer used by pipe.
Definition cmux.h:219
uint16_t receive_buf_size
Size of receive buffer used by pipe [127, ...].
Definition cmux.h:221