]>
Commit | Line | Data |
---|---|---|
e3b3d0f5 | 1 | // SPDX-License-Identifier: GPL-1.0+ |
1da177e4 LT |
2 | /* generic HDLC line discipline for Linux |
3 | * | |
4 | * Written by Paul Fulghum [email protected] | |
5 | * for Microgate Corporation | |
6 | * | |
7 | * Microgate and SyncLink are registered trademarks of Microgate Corporation | |
8 | * | |
9 | * Adapted from ppp.c, written by Michael Callahan <[email protected]>, | |
10 | * Al Longyear <[email protected]>, | |
11 | * Paul Mackerras <[email protected]> | |
12 | * | |
13 | * Original release 01/11/99 | |
1da177e4 | 14 | * |
1da177e4 LT |
15 | * This module implements the tty line discipline N_HDLC for use with |
16 | * tty device drivers that support bit-synchronous HDLC communications. | |
17 | * | |
18 | * All HDLC data is frame oriented which means: | |
19 | * | |
20 | * 1. tty write calls represent one complete transmit frame of data | |
43741e9b | 21 | * The device driver should accept the complete frame or none of |
1da177e4 LT |
22 | * the frame (busy) in the write method. Each write call should have |
23 | * a byte count in the range of 2-65535 bytes (2 is min HDLC frame | |
24 | * with 1 addr byte and 1 ctrl byte). The max byte count of 65535 | |
25 | * should include any crc bytes required. For example, when using | |
26 | * CCITT CRC32, 4 crc bytes are required, so the maximum size frame | |
27 | * the application may transmit is limited to 65531 bytes. For CCITT | |
28 | * CRC16, the maximum application frame size would be 65533. | |
29 | * | |
30 | * | |
31 | * 2. receive callbacks from the device driver represents | |
32 | * one received frame. The device driver should bypass | |
33 | * the tty flip buffer and call the line discipline receive | |
34 | * callback directly to avoid fragmenting or concatenating | |
35 | * multiple frames into a single receive callback. | |
36 | * | |
37 | * The HDLC line discipline queues the receive frames in separate | |
38 | * buffers so complete receive frames can be returned by the | |
39 | * tty read calls. | |
40 | * | |
41 | * 3. tty read calls returns an entire frame of data or nothing. | |
43741e9b | 42 | * |
1da177e4 LT |
43 | * 4. all send and receive data is considered raw. No processing |
44 | * or translation is performed by the line discipline, regardless | |
45 | * of the tty flags | |
46 | * | |
47 | * 5. When line discipline is queried for the amount of receive | |
48 | * data available (FIOC), 0 is returned if no data available, | |
49 | * otherwise the count of the next available frame is returned. | |
50 | * (instead of the sum of all received frame counts). | |
51 | * | |
52 | * These conventions allow the standard tty programming interface | |
53 | * to be used for synchronous HDLC applications when used with | |
54 | * this line discipline (or another line discipline that is frame | |
55 | * oriented such as N_PPP). | |
56 | * | |
57 | * The SyncLink driver (synclink.c) implements both asynchronous | |
58 | * (using standard line discipline N_TTY) and synchronous HDLC | |
59 | * (using N_HDLC) communications, with the latter using the above | |
60 | * conventions. | |
61 | * | |
62 | * This implementation is very basic and does not maintain | |
63 | * any statistics. The main point is to enforce the raw data | |
64 | * and frame orientation of HDLC communications. | |
65 | * | |
66 | * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED | |
67 | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | |
68 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | |
69 | * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, | |
70 | * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | |
71 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | |
72 | * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) | |
73 | * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, | |
74 | * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
75 | * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED | |
76 | * OF THE POSSIBILITY OF SUCH DAMAGE. | |
77 | */ | |
78 | ||
1da177e4 LT |
79 | #include <linux/module.h> |
80 | #include <linux/init.h> | |
81 | #include <linux/kernel.h> | |
82 | #include <linux/sched.h> | |
83 | #include <linux/types.h> | |
84 | #include <linux/fcntl.h> | |
85 | #include <linux/interrupt.h> | |
86 | #include <linux/ptrace.h> | |
87 | ||
1da177e4 LT |
88 | #include <linux/poll.h> |
89 | #include <linux/in.h> | |
90 | #include <linux/ioctl.h> | |
91 | #include <linux/slab.h> | |
92 | #include <linux/tty.h> | |
93 | #include <linux/errno.h> | |
94 | #include <linux/string.h> /* used in new tty drivers */ | |
95 | #include <linux/signal.h> /* used in new tty drivers */ | |
96 | #include <linux/if.h> | |
97 | #include <linux/bitops.h> | |
98 | ||
7c0f6ba6 | 99 | #include <linux/uaccess.h> |
5ffa6e34 | 100 | #include "tty.h" |
1da177e4 LT |
101 | |
102 | /* | |
103 | * Buffers for individual HDLC frames | |
104 | */ | |
43741e9b | 105 | #define MAX_HDLC_FRAME_SIZE 65535 |
1da177e4 LT |
106 | #define DEFAULT_RX_BUF_COUNT 10 |
107 | #define MAX_RX_BUF_COUNT 60 | |
be10eb75 | 108 | #define DEFAULT_TX_BUF_COUNT 3 |
1da177e4 LT |
109 | |
110 | struct n_hdlc_buf { | |
82f2341c | 111 | struct list_head list_item; |
1da177e4 | 112 | int count; |
85f4c951 | 113 | char buf[]; |
1da177e4 LT |
114 | }; |
115 | ||
1da177e4 | 116 | struct n_hdlc_buf_list { |
82f2341c | 117 | struct list_head list; |
1da177e4 LT |
118 | int count; |
119 | spinlock_t spinlock; | |
120 | }; | |
121 | ||
122 | /** | |
123 | * struct n_hdlc - per device instance data structure | |
724ac070 JS |
124 | * @tbusy: reentrancy flag for tx wakeup code |
125 | * @woke_up: tx wakeup needs to be run again as it was called while @tbusy | |
126 | * @tx_buf_list: list of pending transmit frame buffers | |
127 | * @rx_buf_list: list of received frame buffers | |
128 | * @tx_free_buf_list: list unused transmit frame buffers | |
129 | * @rx_free_buf_list: list unused received frame buffers | |
1da177e4 LT |
130 | */ |
131 | struct n_hdlc { | |
0f238298 JS |
132 | bool tbusy; |
133 | bool woke_up; | |
1da177e4 LT |
134 | struct n_hdlc_buf_list tx_buf_list; |
135 | struct n_hdlc_buf_list rx_buf_list; | |
136 | struct n_hdlc_buf_list tx_free_buf_list; | |
137 | struct n_hdlc_buf_list rx_free_buf_list; | |
1ee33b1c TH |
138 | struct work_struct write_work; |
139 | struct tty_struct *tty_for_write_work; | |
1da177e4 LT |
140 | }; |
141 | ||
142 | /* | |
143 | * HDLC buffer list manipulation functions | |
144 | */ | |
82f2341c AP |
145 | static void n_hdlc_buf_return(struct n_hdlc_buf_list *buf_list, |
146 | struct n_hdlc_buf *buf); | |
1da177e4 LT |
147 | static void n_hdlc_buf_put(struct n_hdlc_buf_list *list, |
148 | struct n_hdlc_buf *buf); | |
149 | static struct n_hdlc_buf *n_hdlc_buf_get(struct n_hdlc_buf_list *list); | |
150 | ||
151 | /* Local functions */ | |
152 | ||
2bfb2b75 | 153 | static struct n_hdlc *n_hdlc_alloc(void); |
1ee33b1c | 154 | static void n_hdlc_tty_write_work(struct work_struct *work); |
1da177e4 | 155 | |
1da177e4 LT |
156 | /* max frame size for memory allocations */ |
157 | static int maxframe = 4096; | |
158 | ||
be10eb75 PF |
159 | static void flush_rx_queue(struct tty_struct *tty) |
160 | { | |
75011682 | 161 | struct n_hdlc *n_hdlc = tty->disc_data; |
be10eb75 PF |
162 | struct n_hdlc_buf *buf; |
163 | ||
164 | while ((buf = n_hdlc_buf_get(&n_hdlc->rx_buf_list))) | |
165 | n_hdlc_buf_put(&n_hdlc->rx_free_buf_list, buf); | |
166 | } | |
167 | ||
168 | static void flush_tx_queue(struct tty_struct *tty) | |
169 | { | |
75011682 | 170 | struct n_hdlc *n_hdlc = tty->disc_data; |
be10eb75 | 171 | struct n_hdlc_buf *buf; |
be10eb75 PF |
172 | |
173 | while ((buf = n_hdlc_buf_get(&n_hdlc->tx_buf_list))) | |
174 | n_hdlc_buf_put(&n_hdlc->tx_free_buf_list, buf); | |
be10eb75 PF |
175 | } |
176 | ||
30fafd92 JS |
177 | static void n_hdlc_free_buf_list(struct n_hdlc_buf_list *list) |
178 | { | |
179 | struct n_hdlc_buf *buf; | |
180 | ||
181 | do { | |
182 | buf = n_hdlc_buf_get(list); | |
183 | kfree(buf); | |
184 | } while (buf); | |
185 | } | |
186 | ||
1da177e4 LT |
187 | /** |
188 | * n_hdlc_tty_close - line discipline close | |
724ac070 | 189 | * @tty: pointer to tty info structure |
1da177e4 LT |
190 | * |
191 | * Called when the line discipline is changed to something | |
192 | * else, the tty is closed, or the tty detects a hangup. | |
193 | */ | |
194 | static void n_hdlc_tty_close(struct tty_struct *tty) | |
195 | { | |
75011682 | 196 | struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 | 197 | |
1da177e4 | 198 | #if defined(TTY_NO_WRITE_SPLIT) |
8d79bb5c | 199 | clear_bit(TTY_NO_WRITE_SPLIT, &tty->flags); |
1da177e4 | 200 | #endif |
5f289514 | 201 | tty->disc_data = NULL; |
43e784ec JS |
202 | |
203 | /* Ensure that the n_hdlcd process is not hanging on select()/poll() */ | |
204 | wake_up_interruptible(&tty->read_wait); | |
205 | wake_up_interruptible(&tty->write_wait); | |
206 | ||
1ee33b1c TH |
207 | cancel_work_sync(&n_hdlc->write_work); |
208 | ||
43e784ec JS |
209 | n_hdlc_free_buf_list(&n_hdlc->rx_free_buf_list); |
210 | n_hdlc_free_buf_list(&n_hdlc->tx_free_buf_list); | |
211 | n_hdlc_free_buf_list(&n_hdlc->rx_buf_list); | |
212 | n_hdlc_free_buf_list(&n_hdlc->tx_buf_list); | |
213 | kfree(n_hdlc); | |
1da177e4 LT |
214 | } /* end of n_hdlc_tty_close() */ |
215 | ||
216 | /** | |
217 | * n_hdlc_tty_open - called when line discipline changed to n_hdlc | |
724ac070 | 218 | * @tty: pointer to tty info structure |
1da177e4 LT |
219 | * |
220 | * Returns 0 if success, otherwise error code | |
221 | */ | |
2bfb2b75 | 222 | static int n_hdlc_tty_open(struct tty_struct *tty) |
1da177e4 | 223 | { |
75011682 | 224 | struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 | 225 | |
b18d1c2e | 226 | pr_debug("%s() called (device=%s)\n", __func__, tty->name); |
f3c2e277 | 227 | |
1da177e4 LT |
228 | /* There should not be an existing table for this slot. */ |
229 | if (n_hdlc) { | |
d86b05cb | 230 | pr_err("%s: tty already associated!\n", __func__); |
1da177e4 LT |
231 | return -EEXIST; |
232 | } | |
43741e9b | 233 | |
1da177e4 LT |
234 | n_hdlc = n_hdlc_alloc(); |
235 | if (!n_hdlc) { | |
d86b05cb | 236 | pr_err("%s: n_hdlc_alloc failed\n", __func__); |
1da177e4 LT |
237 | return -ENFILE; |
238 | } | |
43741e9b | 239 | |
1ee33b1c TH |
240 | INIT_WORK(&n_hdlc->write_work, n_hdlc_tty_write_work); |
241 | n_hdlc->tty_for_write_work = tty; | |
1da177e4 | 242 | tty->disc_data = n_hdlc; |
33f0f88f | 243 | tty->receive_room = 65536; |
43741e9b | 244 | |
1da177e4 | 245 | /* change tty_io write() to not split large writes into 8K chunks */ |
8d79bb5c | 246 | set_bit(TTY_NO_WRITE_SPLIT, &tty->flags); |
43741e9b | 247 | |
be10eb75 | 248 | /* flush receive data from driver */ |
f34d7a5b | 249 | tty_driver_flush_buffer(tty); |
66c3bdf1 | 250 | |
1da177e4 | 251 | return 0; |
43741e9b | 252 | |
1da177e4 LT |
253 | } /* end of n_tty_hdlc_open() */ |
254 | ||
255 | /** | |
256 | * n_hdlc_send_frames - send frames on pending send buffer list | |
724ac070 JS |
257 | * @n_hdlc: pointer to ldisc instance data |
258 | * @tty: pointer to tty instance data | |
1da177e4 LT |
259 | * |
260 | * Send frames on pending send buffer list until the driver does not accept a | |
261 | * frame (busy) this function is called after adding a frame to the send buffer | |
262 | * list and by the tty wakeup callback. | |
263 | */ | |
264 | static void n_hdlc_send_frames(struct n_hdlc *n_hdlc, struct tty_struct *tty) | |
265 | { | |
266 | register int actual; | |
267 | unsigned long flags; | |
268 | struct n_hdlc_buf *tbuf; | |
269 | ||
43741e9b JS |
270 | check_again: |
271 | ||
272 | spin_lock_irqsave(&n_hdlc->tx_buf_list.spinlock, flags); | |
1da177e4 | 273 | if (n_hdlc->tbusy) { |
0f238298 JS |
274 | n_hdlc->woke_up = true; |
275 | spin_unlock_irqrestore(&n_hdlc->tx_buf_list.spinlock, flags); | |
1da177e4 LT |
276 | return; |
277 | } | |
0f238298 JS |
278 | n_hdlc->tbusy = true; |
279 | n_hdlc->woke_up = false; | |
1da177e4 LT |
280 | spin_unlock_irqrestore(&n_hdlc->tx_buf_list.spinlock, flags); |
281 | ||
82f2341c | 282 | tbuf = n_hdlc_buf_get(&n_hdlc->tx_buf_list); |
1da177e4 | 283 | while (tbuf) { |
b18d1c2e | 284 | pr_debug("sending frame %p, count=%d\n", tbuf, tbuf->count); |
f3c2e277 | 285 | |
1da177e4 | 286 | /* Send the next block of data to device */ |
7962fce9 | 287 | set_bit(TTY_DO_WRITE_WAKEUP, &tty->flags); |
f34d7a5b | 288 | actual = tty->ops->write(tty, tbuf->buf, tbuf->count); |
b0fed314 JS |
289 | |
290 | /* rollback was possible and has been done */ | |
291 | if (actual == -ERESTARTSYS) { | |
82f2341c | 292 | n_hdlc_buf_return(&n_hdlc->tx_buf_list, tbuf); |
b0fed314 JS |
293 | break; |
294 | } | |
1da177e4 LT |
295 | /* if transmit error, throw frame away by */ |
296 | /* pretending it was accepted by driver */ | |
297 | if (actual < 0) | |
298 | actual = tbuf->count; | |
43741e9b | 299 | |
1da177e4 | 300 | if (actual == tbuf->count) { |
b18d1c2e | 301 | pr_debug("frame %p completed\n", tbuf); |
f3c2e277 | 302 | |
1da177e4 LT |
303 | /* free current transmit buffer */ |
304 | n_hdlc_buf_put(&n_hdlc->tx_free_buf_list, tbuf); | |
82f2341c | 305 | |
1da177e4 LT |
306 | /* wait up sleeping writers */ |
307 | wake_up_interruptible(&tty->write_wait); | |
43741e9b | 308 | |
1da177e4 LT |
309 | /* get next pending transmit buffer */ |
310 | tbuf = n_hdlc_buf_get(&n_hdlc->tx_buf_list); | |
311 | } else { | |
b18d1c2e | 312 | pr_debug("frame %p pending\n", tbuf); |
82f2341c AP |
313 | |
314 | /* | |
315 | * the buffer was not accepted by driver, | |
316 | * return it back into tx queue | |
317 | */ | |
318 | n_hdlc_buf_return(&n_hdlc->tx_buf_list, tbuf); | |
1da177e4 LT |
319 | break; |
320 | } | |
321 | } | |
43741e9b | 322 | |
1da177e4 | 323 | if (!tbuf) |
7962fce9 | 324 | clear_bit(TTY_DO_WRITE_WAKEUP, &tty->flags); |
43741e9b | 325 | |
1da177e4 LT |
326 | /* Clear the re-entry flag */ |
327 | spin_lock_irqsave(&n_hdlc->tx_buf_list.spinlock, flags); | |
0f238298 | 328 | n_hdlc->tbusy = false; |
43741e9b JS |
329 | spin_unlock_irqrestore(&n_hdlc->tx_buf_list.spinlock, flags); |
330 | ||
331 | if (n_hdlc->woke_up) | |
332 | goto check_again; | |
1da177e4 LT |
333 | } /* end of n_hdlc_send_frames() */ |
334 | ||
1ee33b1c TH |
335 | /** |
336 | * n_hdlc_tty_write_work - Asynchronous callback for transmit wakeup | |
337 | * @work: pointer to work_struct | |
338 | * | |
339 | * Called when low level device driver can accept more send data. | |
340 | */ | |
341 | static void n_hdlc_tty_write_work(struct work_struct *work) | |
342 | { | |
343 | struct n_hdlc *n_hdlc = container_of(work, struct n_hdlc, write_work); | |
344 | struct tty_struct *tty = n_hdlc->tty_for_write_work; | |
345 | ||
346 | n_hdlc_send_frames(n_hdlc, tty); | |
347 | } /* end of n_hdlc_tty_write_work() */ | |
348 | ||
1da177e4 LT |
349 | /** |
350 | * n_hdlc_tty_wakeup - Callback for transmit wakeup | |
724ac070 | 351 | * @tty: pointer to associated tty instance data |
1da177e4 LT |
352 | * |
353 | * Called when low level device driver can accept more send data. | |
354 | */ | |
355 | static void n_hdlc_tty_wakeup(struct tty_struct *tty) | |
356 | { | |
75011682 | 357 | struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 | 358 | |
1ee33b1c | 359 | schedule_work(&n_hdlc->write_work); |
1da177e4 LT |
360 | } /* end of n_hdlc_tty_wakeup() */ |
361 | ||
1da177e4 LT |
362 | /** |
363 | * n_hdlc_tty_receive - Called by tty driver when receive data is available | |
724ac070 JS |
364 | * @tty: pointer to tty instance data |
365 | * @data: pointer to received data | |
366 | * @flags: pointer to flags for data | |
367 | * @count: count of received data in bytes | |
1da177e4 LT |
368 | * |
369 | * Called by tty low level driver when receive data is available. Data is | |
370 | * interpreted as one HDLC frame. | |
371 | */ | |
a8d9cd23 | 372 | static void n_hdlc_tty_receive(struct tty_struct *tty, const u8 *data, |
892bc209 | 373 | const u8 *flags, size_t count) |
1da177e4 | 374 | { |
75011682 | 375 | register struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 LT |
376 | register struct n_hdlc_buf *buf; |
377 | ||
e8161447 | 378 | pr_debug("%s() called count=%zu\n", __func__, count); |
f3c2e277 | 379 | |
1283c721 | 380 | if (count > maxframe) { |
b18d1c2e | 381 | pr_debug("rx count>maxframesize, data discarded\n"); |
55db4c64 | 382 | return; |
1da177e4 LT |
383 | } |
384 | ||
43741e9b | 385 | /* get a free HDLC buffer */ |
1da177e4 LT |
386 | buf = n_hdlc_buf_get(&n_hdlc->rx_free_buf_list); |
387 | if (!buf) { | |
80967ff2 JS |
388 | /* |
389 | * no buffers in free list, attempt to allocate another rx | |
390 | * buffer unless the maximum count has been reached | |
391 | */ | |
1da177e4 | 392 | if (n_hdlc->rx_buf_list.count < MAX_RX_BUF_COUNT) |
85f4c951 GS |
393 | buf = kmalloc(struct_size(buf, buf, maxframe), |
394 | GFP_ATOMIC); | |
1da177e4 | 395 | } |
43741e9b | 396 | |
1da177e4 | 397 | if (!buf) { |
b18d1c2e | 398 | pr_debug("no more rx buffers, data discarded\n"); |
55db4c64 | 399 | return; |
1da177e4 | 400 | } |
43741e9b | 401 | |
1da177e4 | 402 | /* copy received data to HDLC buffer */ |
8d79bb5c | 403 | memcpy(buf->buf, data, count); |
1283c721 | 404 | buf->count = count; |
1da177e4 LT |
405 | |
406 | /* add HDLC buffer to list of received frames */ | |
407 | n_hdlc_buf_put(&n_hdlc->rx_buf_list, buf); | |
43741e9b | 408 | |
1da177e4 | 409 | /* wake up any blocked reads and perform async signalling */ |
2bfb2b75 | 410 | wake_up_interruptible(&tty->read_wait); |
df6de639 JS |
411 | if (tty->fasync != NULL) |
412 | kill_fasync(&tty->fasync, SIGIO, POLL_IN); | |
1da177e4 LT |
413 | |
414 | } /* end of n_hdlc_tty_receive() */ | |
415 | ||
416 | /** | |
4a4efbde | 417 | * n_hdlc_tty_read - Called to retrieve one frame of data (if available) |
724ac070 JS |
418 | * @tty: pointer to tty instance data |
419 | * @file: pointer to open file object | |
ef80f77b | 420 | * @kbuf: pointer to returned data buffer |
724ac070 | 421 | * @nr: size of returned data buffer |
ef80f77b LJ |
422 | * @cookie: stored rbuf from previous run |
423 | * @offset: offset into the data buffer | |
43741e9b | 424 | * |
1da177e4 LT |
425 | * Returns the number of bytes returned or error code. |
426 | */ | |
427 | static ssize_t n_hdlc_tty_read(struct tty_struct *tty, struct file *file, | |
49b8220c JSS |
428 | u8 *kbuf, size_t nr, void **cookie, |
429 | unsigned long offset) | |
1da177e4 | 430 | { |
75011682 | 431 | struct n_hdlc *n_hdlc = tty->disc_data; |
1035b63d | 432 | int ret = 0; |
1da177e4 | 433 | struct n_hdlc_buf *rbuf; |
1035b63d | 434 | DECLARE_WAITQUEUE(wait, current); |
1da177e4 | 435 | |
3b830a9c LT |
436 | /* Is this a repeated call for an rbuf we already found earlier? */ |
437 | rbuf = *cookie; | |
438 | if (rbuf) | |
439 | goto have_rbuf; | |
440 | ||
1035b63d | 441 | add_wait_queue(&tty->read_wait, &wait); |
04f378b1 | 442 | |
1da177e4 | 443 | for (;;) { |
0f40fbbc | 444 | if (test_bit(TTY_OTHER_CLOSED, &tty->flags)) { |
1035b63d PF |
445 | ret = -EIO; |
446 | break; | |
04f378b1 | 447 | } |
1035b63d PF |
448 | if (tty_hung_up_p(file)) |
449 | break; | |
1da177e4 | 450 | |
1035b63d | 451 | set_current_state(TASK_INTERRUPTIBLE); |
1da177e4 LT |
452 | |
453 | rbuf = n_hdlc_buf_get(&n_hdlc->rx_buf_list); | |
3b830a9c | 454 | if (rbuf) |
1da177e4 | 455 | break; |
43741e9b | 456 | |
1da177e4 | 457 | /* no data */ |
c96cf923 | 458 | if (tty_io_nonblock(tty, file)) { |
1035b63d PF |
459 | ret = -EAGAIN; |
460 | break; | |
04f378b1 | 461 | } |
1035b63d PF |
462 | |
463 | schedule(); | |
464 | ||
04f378b1 | 465 | if (signal_pending(current)) { |
1035b63d PF |
466 | ret = -EINTR; |
467 | break; | |
04f378b1 | 468 | } |
1da177e4 | 469 | } |
1035b63d PF |
470 | |
471 | remove_wait_queue(&tty->read_wait, &wait); | |
472 | __set_current_state(TASK_RUNNING); | |
473 | ||
3b830a9c LT |
474 | if (!rbuf) |
475 | return ret; | |
476 | *cookie = rbuf; | |
477 | ||
478 | have_rbuf: | |
479 | /* Have we used it up entirely? */ | |
480 | if (offset >= rbuf->count) | |
481 | goto done_with_rbuf; | |
482 | ||
483 | /* More data to go, but can't copy any more? EOVERFLOW */ | |
484 | ret = -EOVERFLOW; | |
485 | if (!nr) | |
486 | goto done_with_rbuf; | |
487 | ||
488 | /* Copy as much data as possible */ | |
489 | ret = rbuf->count - offset; | |
490 | if (ret > nr) | |
491 | ret = nr; | |
492 | memcpy(kbuf, rbuf->buf+offset, ret); | |
493 | offset += ret; | |
494 | ||
495 | /* If we still have data left, we leave the rbuf in the cookie */ | |
496 | if (offset < rbuf->count) | |
497 | return ret; | |
498 | ||
499 | done_with_rbuf: | |
500 | *cookie = NULL; | |
501 | ||
502 | if (n_hdlc->rx_free_buf_list.count > DEFAULT_RX_BUF_COUNT) | |
503 | kfree(rbuf); | |
504 | else | |
505 | n_hdlc_buf_put(&n_hdlc->rx_free_buf_list, rbuf); | |
506 | ||
1da177e4 | 507 | return ret; |
43741e9b | 508 | |
1da177e4 LT |
509 | } /* end of n_hdlc_tty_read() */ |
510 | ||
511 | /** | |
512 | * n_hdlc_tty_write - write a single frame of data to device | |
724ac070 JS |
513 | * @tty: pointer to associated tty device instance data |
514 | * @file: pointer to file object data | |
515 | * @data: pointer to transmit data (one frame) | |
516 | * @count: size of transmit frame in bytes | |
43741e9b | 517 | * |
1da177e4 LT |
518 | * Returns the number of bytes written (or error code). |
519 | */ | |
520 | static ssize_t n_hdlc_tty_write(struct tty_struct *tty, struct file *file, | |
49b8220c | 521 | const u8 *data, size_t count) |
1da177e4 | 522 | { |
75011682 | 523 | struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 LT |
524 | int error = 0; |
525 | DECLARE_WAITQUEUE(wait, current); | |
526 | struct n_hdlc_buf *tbuf; | |
527 | ||
b18d1c2e | 528 | pr_debug("%s() called count=%zd\n", __func__, count); |
f3c2e277 | 529 | |
1da177e4 | 530 | /* verify frame size */ |
1283c721 | 531 | if (count > maxframe) { |
f3c2e277 JS |
532 | pr_debug("%s: truncating user packet from %zu to %d\n", |
533 | __func__, count, maxframe); | |
1da177e4 LT |
534 | count = maxframe; |
535 | } | |
43741e9b | 536 | |
1da177e4 | 537 | add_wait_queue(&tty->write_wait, &wait); |
1035b63d PF |
538 | |
539 | for (;;) { | |
540 | set_current_state(TASK_INTERRUPTIBLE); | |
43741e9b | 541 | |
1035b63d PF |
542 | tbuf = n_hdlc_buf_get(&n_hdlc->tx_free_buf_list); |
543 | if (tbuf) | |
544 | break; | |
545 | ||
c96cf923 | 546 | if (tty_io_nonblock(tty, file)) { |
c72f527c PF |
547 | error = -EAGAIN; |
548 | break; | |
549 | } | |
1da177e4 | 550 | schedule(); |
844cc5f9 | 551 | |
1da177e4 LT |
552 | if (signal_pending(current)) { |
553 | error = -EINTR; | |
554 | break; | |
555 | } | |
556 | } | |
557 | ||
1035b63d | 558 | __set_current_state(TASK_RUNNING); |
1da177e4 LT |
559 | remove_wait_queue(&tty->write_wait, &wait); |
560 | ||
43741e9b | 561 | if (!error) { |
1da177e4 LT |
562 | /* Retrieve the user's buffer */ |
563 | memcpy(tbuf->buf, data, count); | |
564 | ||
565 | /* Send the data */ | |
566 | tbuf->count = error = count; | |
8d79bb5c JS |
567 | n_hdlc_buf_put(&n_hdlc->tx_buf_list, tbuf); |
568 | n_hdlc_send_frames(n_hdlc, tty); | |
1da177e4 | 569 | } |
1035b63d | 570 | |
1da177e4 | 571 | return error; |
43741e9b | 572 | |
1da177e4 LT |
573 | } /* end of n_hdlc_tty_write() */ |
574 | ||
575 | /** | |
576 | * n_hdlc_tty_ioctl - process IOCTL system call for the tty device. | |
724ac070 | 577 | * @tty: pointer to tty instance data |
724ac070 JS |
578 | * @cmd: IOCTL command code |
579 | * @arg: argument for IOCTL call (cmd dependent) | |
1da177e4 LT |
580 | * |
581 | * Returns command dependent result. | |
582 | */ | |
d78328bc JS |
583 | static int n_hdlc_tty_ioctl(struct tty_struct *tty, unsigned int cmd, |
584 | unsigned long arg) | |
1da177e4 | 585 | { |
75011682 | 586 | struct n_hdlc *n_hdlc = tty->disc_data; |
1da177e4 LT |
587 | int error = 0; |
588 | int count; | |
589 | unsigned long flags; | |
82f2341c AP |
590 | struct n_hdlc_buf *buf = NULL; |
591 | ||
b18d1c2e | 592 | pr_debug("%s() called %d\n", __func__, cmd); |
f3c2e277 | 593 | |
1da177e4 LT |
594 | switch (cmd) { |
595 | case FIONREAD: | |
596 | /* report count of read data available */ | |
597 | /* in next available frame (if any) */ | |
8d79bb5c | 598 | spin_lock_irqsave(&n_hdlc->rx_buf_list.spinlock, flags); |
82f2341c AP |
599 | buf = list_first_entry_or_null(&n_hdlc->rx_buf_list.list, |
600 | struct n_hdlc_buf, list_item); | |
601 | if (buf) | |
602 | count = buf->count; | |
1da177e4 LT |
603 | else |
604 | count = 0; | |
8d79bb5c | 605 | spin_unlock_irqrestore(&n_hdlc->rx_buf_list.spinlock, flags); |
1da177e4 LT |
606 | error = put_user(count, (int __user *)arg); |
607 | break; | |
608 | ||
609 | case TIOCOUTQ: | |
610 | /* get the pending tx byte count in the driver */ | |
f34d7a5b | 611 | count = tty_chars_in_buffer(tty); |
1da177e4 | 612 | /* add size of next output frame in queue */ |
8d79bb5c | 613 | spin_lock_irqsave(&n_hdlc->tx_buf_list.spinlock, flags); |
82f2341c AP |
614 | buf = list_first_entry_or_null(&n_hdlc->tx_buf_list.list, |
615 | struct n_hdlc_buf, list_item); | |
616 | if (buf) | |
617 | count += buf->count; | |
8d79bb5c | 618 | spin_unlock_irqrestore(&n_hdlc->tx_buf_list.spinlock, flags); |
1da177e4 LT |
619 | error = put_user(count, (int __user *)arg); |
620 | break; | |
621 | ||
be10eb75 PF |
622 | case TCFLSH: |
623 | switch (arg) { | |
624 | case TCIOFLUSH: | |
625 | case TCOFLUSH: | |
626 | flush_tx_queue(tty); | |
627 | } | |
df561f66 | 628 | fallthrough; /* to default */ |
be10eb75 | 629 | |
1da177e4 | 630 | default: |
7c783601 | 631 | error = n_tty_ioctl_helper(tty, cmd, arg); |
1da177e4 LT |
632 | break; |
633 | } | |
634 | return error; | |
43741e9b | 635 | |
1da177e4 LT |
636 | } /* end of n_hdlc_tty_ioctl() */ |
637 | ||
638 | /** | |
639 | * n_hdlc_tty_poll - TTY callback for poll system call | |
724ac070 JS |
640 | * @tty: pointer to tty instance data |
641 | * @filp: pointer to open file object for device | |
642 | * @wait: wait queue for operations | |
43741e9b | 643 | * |
1da177e4 LT |
644 | * Determine which operations (read/write) will not block and return info |
645 | * to caller. | |
646 | * Returns a bit mask containing info on which ops will not block. | |
647 | */ | |
afc9a42b | 648 | static __poll_t n_hdlc_tty_poll(struct tty_struct *tty, struct file *filp, |
1da177e4 LT |
649 | poll_table *wait) |
650 | { | |
75011682 | 651 | struct n_hdlc *n_hdlc = tty->disc_data; |
afc9a42b | 652 | __poll_t mask = 0; |
1da177e4 | 653 | |
5f289514 JS |
654 | /* |
655 | * queue the current process into any wait queue that may awaken in the | |
656 | * future (read and write) | |
657 | */ | |
658 | poll_wait(filp, &tty->read_wait, wait); | |
659 | poll_wait(filp, &tty->write_wait, wait); | |
660 | ||
661 | /* set bits for operations that won't block */ | |
662 | if (!list_empty(&n_hdlc->rx_buf_list.list)) | |
663 | mask |= EPOLLIN | EPOLLRDNORM; /* readable */ | |
664 | if (test_bit(TTY_OTHER_CLOSED, &tty->flags)) | |
665 | mask |= EPOLLHUP; | |
666 | if (tty_hung_up_p(filp)) | |
667 | mask |= EPOLLHUP; | |
668 | if (!tty_is_writelocked(tty) && | |
669 | !list_empty(&n_hdlc->tx_free_buf_list.list)) | |
670 | mask |= EPOLLOUT | EPOLLWRNORM; /* writable */ | |
671 | ||
1da177e4 LT |
672 | return mask; |
673 | } /* end of n_hdlc_tty_poll() */ | |
674 | ||
740708ab JS |
675 | static void n_hdlc_alloc_buf(struct n_hdlc_buf_list *list, unsigned int count, |
676 | const char *name) | |
677 | { | |
678 | struct n_hdlc_buf *buf; | |
679 | unsigned int i; | |
680 | ||
681 | for (i = 0; i < count; i++) { | |
682 | buf = kmalloc(struct_size(buf, buf, maxframe), GFP_KERNEL); | |
683 | if (!buf) { | |
b18d1c2e JS |
684 | pr_debug("%s(), kmalloc() failed for %s buffer %u\n", |
685 | __func__, name, i); | |
740708ab JS |
686 | return; |
687 | } | |
688 | n_hdlc_buf_put(list, buf); | |
689 | } | |
690 | } | |
691 | ||
1da177e4 LT |
692 | /** |
693 | * n_hdlc_alloc - allocate an n_hdlc instance data structure | |
694 | * | |
695 | * Returns a pointer to newly created structure if success, otherwise %NULL | |
696 | */ | |
697 | static struct n_hdlc *n_hdlc_alloc(void) | |
698 | { | |
8e25f8ce | 699 | struct n_hdlc *n_hdlc = kzalloc(sizeof(*n_hdlc), GFP_KERNEL); |
1da177e4 LT |
700 | |
701 | if (!n_hdlc) | |
702 | return NULL; | |
703 | ||
e9b736d8 JS |
704 | spin_lock_init(&n_hdlc->rx_free_buf_list.spinlock); |
705 | spin_lock_init(&n_hdlc->tx_free_buf_list.spinlock); | |
706 | spin_lock_init(&n_hdlc->rx_buf_list.spinlock); | |
707 | spin_lock_init(&n_hdlc->tx_buf_list.spinlock); | |
82f2341c AP |
708 | |
709 | INIT_LIST_HEAD(&n_hdlc->rx_free_buf_list.list); | |
710 | INIT_LIST_HEAD(&n_hdlc->tx_free_buf_list.list); | |
711 | INIT_LIST_HEAD(&n_hdlc->rx_buf_list.list); | |
712 | INIT_LIST_HEAD(&n_hdlc->tx_buf_list.list); | |
713 | ||
740708ab JS |
714 | n_hdlc_alloc_buf(&n_hdlc->rx_free_buf_list, DEFAULT_RX_BUF_COUNT, "rx"); |
715 | n_hdlc_alloc_buf(&n_hdlc->tx_free_buf_list, DEFAULT_TX_BUF_COUNT, "tx"); | |
716 | ||
1da177e4 | 717 | return n_hdlc; |
43741e9b | 718 | |
1da177e4 LT |
719 | } /* end of n_hdlc_alloc() */ |
720 | ||
82f2341c AP |
721 | /** |
722 | * n_hdlc_buf_return - put the HDLC buffer after the head of the specified list | |
724ac070 JS |
723 | * @buf_list: pointer to the buffer list |
724 | * @buf: pointer to the buffer | |
82f2341c AP |
725 | */ |
726 | static void n_hdlc_buf_return(struct n_hdlc_buf_list *buf_list, | |
727 | struct n_hdlc_buf *buf) | |
728 | { | |
729 | unsigned long flags; | |
730 | ||
731 | spin_lock_irqsave(&buf_list->spinlock, flags); | |
732 | ||
733 | list_add(&buf->list_item, &buf_list->list); | |
734 | buf_list->count++; | |
735 | ||
736 | spin_unlock_irqrestore(&buf_list->spinlock, flags); | |
737 | } | |
738 | ||
1da177e4 LT |
739 | /** |
740 | * n_hdlc_buf_put - add specified HDLC buffer to tail of specified list | |
724ac070 JS |
741 | * @buf_list: pointer to buffer list |
742 | * @buf: pointer to buffer | |
1da177e4 | 743 | */ |
82f2341c | 744 | static void n_hdlc_buf_put(struct n_hdlc_buf_list *buf_list, |
1da177e4 LT |
745 | struct n_hdlc_buf *buf) |
746 | { | |
747 | unsigned long flags; | |
82f2341c AP |
748 | |
749 | spin_lock_irqsave(&buf_list->spinlock, flags); | |
750 | ||
751 | list_add_tail(&buf->list_item, &buf_list->list); | |
752 | buf_list->count++; | |
753 | ||
754 | spin_unlock_irqrestore(&buf_list->spinlock, flags); | |
1da177e4 LT |
755 | } /* end of n_hdlc_buf_put() */ |
756 | ||
757 | /** | |
758 | * n_hdlc_buf_get - remove and return an HDLC buffer from list | |
724ac070 | 759 | * @buf_list: pointer to HDLC buffer list |
43741e9b | 760 | * |
1da177e4 LT |
761 | * Remove and return an HDLC buffer from the head of the specified HDLC buffer |
762 | * list. | |
763 | * Returns a pointer to HDLC buffer if available, otherwise %NULL. | |
764 | */ | |
82f2341c | 765 | static struct n_hdlc_buf *n_hdlc_buf_get(struct n_hdlc_buf_list *buf_list) |
1da177e4 LT |
766 | { |
767 | unsigned long flags; | |
768 | struct n_hdlc_buf *buf; | |
82f2341c AP |
769 | |
770 | spin_lock_irqsave(&buf_list->spinlock, flags); | |
771 | ||
772 | buf = list_first_entry_or_null(&buf_list->list, | |
773 | struct n_hdlc_buf, list_item); | |
1da177e4 | 774 | if (buf) { |
82f2341c AP |
775 | list_del(&buf->list_item); |
776 | buf_list->count--; | |
1da177e4 | 777 | } |
82f2341c AP |
778 | |
779 | spin_unlock_irqrestore(&buf_list->spinlock, flags); | |
1da177e4 | 780 | return buf; |
1da177e4 LT |
781 | } /* end of n_hdlc_buf_get() */ |
782 | ||
edee649f JS |
783 | static struct tty_ldisc_ops n_hdlc_ldisc = { |
784 | .owner = THIS_MODULE, | |
fbadf70a | 785 | .num = N_HDLC, |
edee649f JS |
786 | .name = "hdlc", |
787 | .open = n_hdlc_tty_open, | |
788 | .close = n_hdlc_tty_close, | |
789 | .read = n_hdlc_tty_read, | |
790 | .write = n_hdlc_tty_write, | |
791 | .ioctl = n_hdlc_tty_ioctl, | |
792 | .poll = n_hdlc_tty_poll, | |
793 | .receive_buf = n_hdlc_tty_receive, | |
794 | .write_wakeup = n_hdlc_tty_wakeup, | |
795 | .flush_buffer = flush_rx_queue, | |
796 | }; | |
797 | ||
1da177e4 LT |
798 | static int __init n_hdlc_init(void) |
799 | { | |
800 | int status; | |
801 | ||
802 | /* range check maxframe arg */ | |
c549725f | 803 | maxframe = clamp(maxframe, 4096, MAX_HDLC_FRAME_SIZE); |
1da177e4 | 804 | |
fbadf70a | 805 | status = tty_register_ldisc(&n_hdlc_ldisc); |
1da177e4 | 806 | if (!status) |
cda3756c JS |
807 | pr_info("N_HDLC line discipline registered with maxframe=%d\n", |
808 | maxframe); | |
1da177e4 | 809 | else |
cda3756c JS |
810 | pr_err("N_HDLC: error registering line discipline: %d\n", |
811 | status); | |
1da177e4 | 812 | |
1da177e4 | 813 | return status; |
43741e9b | 814 | |
1da177e4 LT |
815 | } /* end of init_module() */ |
816 | ||
1da177e4 LT |
817 | static void __exit n_hdlc_exit(void) |
818 | { | |
357a6a87 | 819 | tty_unregister_ldisc(&n_hdlc_ldisc); |
1da177e4 LT |
820 | } |
821 | ||
822 | module_init(n_hdlc_init); | |
823 | module_exit(n_hdlc_exit); | |
824 | ||
825 | MODULE_LICENSE("GPL"); | |
826 | MODULE_AUTHOR("Paul Fulghum [email protected]"); | |
1da177e4 LT |
827 | module_param(maxframe, int, 0); |
828 | MODULE_ALIAS_LDISC(N_HDLC); |