1 /* $NetBSD: ucom.c,v 1.40 2001/11/13 06:24:54 lukem Exp $ */
4 * Copyright (c) 2001-2003, 2005, 2008
5 * Shunsuke Akiyama <akiyama@jp.FreeBSD.org>.
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 #include <sys/cdefs.h>
31 __FBSDID("$FreeBSD$");
34 * Copyright (c) 1998, 2000 The NetBSD Foundation, Inc.
35 * All rights reserved.
37 * This code is derived from software contributed to The NetBSD Foundation
38 * by Lennart Augustsson (lennart@augustsson.net) at
39 * Carlstedt Research & Technology.
41 * Redistribution and use in source and binary forms, with or without
42 * modification, are permitted provided that the following conditions
44 * 1. Redistributions of source code must retain the above copyright
45 * notice, this list of conditions and the following disclaimer.
46 * 2. Redistributions in binary form must reproduce the above copyright
47 * notice, this list of conditions and the following disclaimer in the
48 * documentation and/or other materials provided with the distribution.
49 * 3. All advertising materials mentioning features or use of this software
50 * must display the following acknowledgement:
51 * This product includes software developed by the NetBSD
52 * Foundation, Inc. and its contributors.
53 * 4. Neither the name of The NetBSD Foundation nor the names of its
54 * contributors may be used to endorse or promote products derived
55 * from this software without specific prior written permission.
57 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
58 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
59 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
60 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
61 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
62 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
63 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
64 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
65 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
66 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
67 * POSSIBILITY OF SUCH DAMAGE.
70 #include <sys/param.h>
71 #include <sys/systm.h>
72 #include <sys/kernel.h>
73 #include <sys/malloc.h>
74 #include <sys/module.h>
76 #include <sys/ioccom.h>
77 #include <sys/fcntl.h>
79 #include <sys/serial.h>
81 #include <sys/clist.h>
83 #include <sys/selinfo.h>
86 #include <sys/sysctl.h>
88 #include <dev/usb/usb.h>
89 #include <dev/usb/usbcdc.h>
91 #include <dev/usb/usbdi.h>
92 #include <dev/usb/usbdi_util.h>
94 #include <dev/usb/usb_quirks.h>
96 #include <dev/usb/ucomvar.h>
99 static int ucomdebug = 0;
100 SYSCTL_NODE(_hw_usb, OID_AUTO, ucom, CTLFLAG_RW, 0, "USB ucom");
101 SYSCTL_INT(_hw_usb_ucom, OID_AUTO, debug, CTLFLAG_RW,
102 &ucomdebug, 0, "ucom debug level");
103 #define DPRINTF(x) do { \
108 #define DPRINTFN(n, x) do { \
109 if (ucomdebug > (n)) \
114 #define DPRINTFN(n, x)
117 static int ucom_modevent(module_t, int, void *);
118 static void ucom_cleanup(struct ucom_softc *);
119 static int ucomparam(struct tty *, struct termios *);
120 static void ucomstart(struct tty *);
121 static void ucomstop(struct tty *, int);
122 static void ucom_shutdown(struct ucom_softc *);
123 static void ucom_dtr(struct ucom_softc *, int);
124 static void ucom_rts(struct ucom_softc *, int);
125 static void ucombreak(struct tty *, int);
126 static usbd_status ucomstartread(struct ucom_softc *);
127 static void ucomreadcb(usbd_xfer_handle, usbd_private_handle, usbd_status);
128 static void ucomwritecb(usbd_xfer_handle, usbd_private_handle, usbd_status);
129 static void ucomstopread(struct ucom_softc *);
131 static t_open_t ucomopen;
132 static t_close_t ucomclose;
133 static t_modem_t ucommodem;
134 static t_ioctl_t ucomioctl;
136 devclass_t ucom_devclass;
138 static moduledata_t ucom_mod = {
144 DECLARE_MODULE(ucom, ucom_mod, SI_SUB_DRIVERS, SI_ORDER_MIDDLE);
145 MODULE_DEPEND(ucom, usb, 1, 1, 1);
146 MODULE_VERSION(ucom, UCOM_MODVER);
149 ucom_modevent(module_t mod, int type, void *data)
164 ucom_attach_tty(struct ucom_softc *sc, int flags, char* fmt, int unit)
168 sc->sc_tty = tp = ttyalloc();
170 tp->t_oproc = ucomstart;
171 tp->t_param = ucomparam;
172 tp->t_stop = ucomstop;
173 tp->t_break = ucombreak;
174 tp->t_open = ucomopen;
175 tp->t_close = ucomclose;
176 tp->t_modem = ucommodem;
177 tp->t_ioctl = ucomioctl;
179 /* Leave the settings of this level to the drivers if needed. */
180 tp->t_ispeedwat = (speed_t)-1;
181 tp->t_ospeedwat = (speed_t)-1;
183 return ttycreate(tp, flags, fmt, unit);
187 ucom_attach(struct ucom_softc *sc)
190 ucom_attach_tty(sc, TS_CALLOUT,
191 "U%d", device_get_unit(sc->sc_dev));
193 DPRINTF(("ucom_attach: ttycreate: tp = %p, %s\n",
194 sc->sc_tty, sc->sc_tty->t_dev->si_name));
200 ucom_detach(struct ucom_softc *sc)
204 DPRINTF(("ucom_detach: sc = %p, tp = %p\n", sc, sc->sc_tty));
208 if (sc->sc_tty->t_state & TS_ISOPEN)
209 ucomclose(sc->sc_tty);
211 if (sc->sc_bulkin_pipe != NULL)
212 usbd_abort_pipe(sc->sc_bulkin_pipe);
213 if (sc->sc_bulkout_pipe != NULL)
214 usbd_abort_pipe(sc->sc_bulkout_pipe);
225 ucom_shutdown(struct ucom_softc *sc)
227 struct tty *tp = sc->sc_tty;
229 DPRINTF(("ucom_shutdown\n"));
231 * Hang up if necessary. Wait a bit, so the other side has time to
232 * notice even if we immediately open the port again.
234 if (ISSET(tp->t_cflag, HUPCL)) {
235 (void)ucommodem(tp, 0, SER_DTR);
236 (void)tsleep(sc, TTIPRI, "ucomsd", hz);
241 ucomopen(struct tty *tp, struct cdev *dev)
243 struct ucom_softc *sc;
252 DPRINTF(("%s: ucomopen: tp = %p\n", device_get_nameunit(sc->sc_dev), tp));
255 sc->sc_lsr = sc->sc_msr = sc->sc_mcr = 0;
257 (void)ucommodem(tp, SER_DTR | SER_RTS, 0);
259 /* Device specific open */
260 if (sc->sc_callback->ucom_open != NULL) {
261 error = sc->sc_callback->ucom_open(sc->sc_parent,
269 DPRINTF(("ucomopen: open pipes in = %d out = %d\n",
270 sc->sc_bulkin_no, sc->sc_bulkout_no));
272 /* Open the bulk pipes */
274 err = usbd_open_pipe(sc->sc_iface, sc->sc_bulkin_no, 0,
275 &sc->sc_bulkin_pipe);
277 printf("%s: open bulk in error (addr %d): %s\n",
278 device_get_nameunit(sc->sc_dev), sc->sc_bulkin_no,
284 err = usbd_open_pipe(sc->sc_iface, sc->sc_bulkout_no,
285 USBD_EXCLUSIVE_USE, &sc->sc_bulkout_pipe);
287 printf("%s: open bulk out error (addr %d): %s\n",
288 device_get_nameunit(sc->sc_dev), sc->sc_bulkout_no,
294 /* Allocate a request and an input buffer and start reading. */
295 sc->sc_ixfer = usbd_alloc_xfer(sc->sc_udev);
296 if (sc->sc_ixfer == NULL) {
301 sc->sc_ibuf = usbd_alloc_buffer(sc->sc_ixfer,
303 if (sc->sc_ibuf == NULL) {
308 sc->sc_oxfer = usbd_alloc_xfer(sc->sc_udev);
309 if (sc->sc_oxfer == NULL) {
314 sc->sc_obuf = usbd_alloc_buffer(sc->sc_oxfer,
317 if (sc->sc_obuf == NULL) {
322 sc->sc_state |= UCS_RXSTOP;
335 ucomclose(struct tty *tp)
337 struct ucom_softc *sc;
341 DPRINTF(("%s: ucomclose \n", device_get_nameunit(sc->sc_dev)));
345 if (sc->sc_callback->ucom_close != NULL)
346 sc->sc_callback->ucom_close(sc->sc_parent, sc->sc_portno);
350 ucomioctl(struct tty *tp, u_long cmd, void *data, int flag, struct thread *p)
352 struct ucom_softc *sc;
359 DPRINTF(("ucomioctl: cmd = 0x%08lx\n", cmd));
362 if (sc->sc_callback->ucom_ioctl != NULL)
363 error = sc->sc_callback->ucom_ioctl(sc->sc_parent,
370 ucommodem(struct tty *tp, int sigon, int sigoff)
372 struct ucom_softc *sc;
379 if (sigon == 0 && sigoff == 0) {
381 if (ISSET(mcr, SER_DTR))
383 if (ISSET(mcr, SER_RTS))
387 if (ISSET(msr, SER_CTS))
389 if (ISSET(msr, SER_DCD))
391 if (ISSET(msr, SER_DSR))
393 if (ISSET(msr, SER_RI))
399 if (ISSET(sigon, SER_DTR))
401 if (ISSET(sigoff, SER_DTR))
403 if (ISSET(sigon, SER_RTS))
405 if (ISSET(sigoff, SER_RTS))
409 onoff = ISSET(sc->sc_mcr, SER_DTR) ? 1 : 0;
412 onoff = ISSET(sc->sc_mcr, SER_RTS) ? 1 : 0;
419 ucombreak(struct tty *tp, int onoff)
421 struct ucom_softc *sc;
424 DPRINTF(("ucombreak: onoff = %d\n", onoff));
425 if (sc->sc_callback->ucom_set == NULL)
427 sc->sc_callback->ucom_set(sc->sc_parent, sc->sc_portno,
428 UCOM_SET_BREAK, onoff);
432 ucom_dtr(struct ucom_softc *sc, int onoff)
434 DPRINTF(("ucom_dtr: onoff = %d\n", onoff));
436 if (sc->sc_callback->ucom_set == NULL)
438 sc->sc_callback->ucom_set(sc->sc_parent, sc->sc_portno,
439 UCOM_SET_DTR, onoff);
443 ucom_rts(struct ucom_softc *sc, int onoff)
445 DPRINTF(("ucom_rts: onoff = %d\n", onoff));
447 if (sc->sc_callback->ucom_set == NULL)
449 sc->sc_callback->ucom_set(sc->sc_parent, sc->sc_portno,
450 UCOM_SET_RTS, onoff);
454 ucom_status_change(struct ucom_softc *sc)
456 struct tty *tp = sc->sc_tty;
460 if (sc->sc_callback->ucom_get_status == NULL) {
466 old_msr = sc->sc_msr;
467 sc->sc_callback->ucom_get_status(sc->sc_parent, sc->sc_portno,
468 &sc->sc_lsr, &sc->sc_msr);
469 if (ISSET((sc->sc_msr ^ old_msr), SER_DCD)) {
470 if (sc->sc_poll == 0)
472 onoff = ISSET(sc->sc_msr, SER_DCD) ? 1 : 0;
473 DPRINTF(("ucom_status_change: DCD changed to %d\n", onoff));
474 ttyld_modem(tp, onoff);
479 ucomparam(struct tty *tp, struct termios *t)
481 struct ucom_softc *sc;
490 DPRINTF(("ucomparam: sc = %p\n", sc));
492 /* Check requested parameters. */
493 if (t->c_ospeed < 0) {
494 DPRINTF(("ucomparam: negative ospeed\n"));
497 if (t->c_ispeed && t->c_ispeed != t->c_ospeed) {
498 DPRINTF(("ucomparam: mismatch ispeed and ospeed\n"));
503 * If there were no changes, don't do anything. This avoids dropping
504 * input and improves performance when all we did was frob things like
507 if (tp->t_ospeed == t->c_ospeed &&
508 tp->t_cflag == t->c_cflag)
511 /* And copy to tty. */
512 tp->t_ispeed = t->c_ospeed;
513 tp->t_ospeed = t->c_ospeed;
514 tp->t_cflag = t->c_cflag;
516 if (sc->sc_callback->ucom_param == NULL)
521 error = sc->sc_callback->ucom_param(sc->sc_parent, sc->sc_portno, t);
523 DPRINTF(("ucomparam: callback: error = %d\n", error));
529 if (t->c_cflag & CRTS_IFLOW) {
530 sc->sc_state |= UCS_RTS_IFLOW;
531 } else if (sc->sc_state & UCS_RTS_IFLOW) {
532 sc->sc_state &= ~UCS_RTS_IFLOW;
533 (void)ucommodem(tp, SER_RTS, 0);
538 uerr = ucomstartread(sc);
539 if (uerr != USBD_NORMAL_COMPLETION)
546 ucomstart(struct tty *tp)
548 struct ucom_softc *sc;
556 DPRINTF(("ucomstart: sc = %p\n", sc));
562 * If there's no sc_oxfer, then ucomclose has removed it. The buffer
563 * has just been flushed in the ttyflush() in ttyclose(). ttyflush()
564 * then calls tt_stop(). ucomstop calls ucomstart, so the right thing
565 * to do here is just abort if sc_oxfer is NULL, as everything else
566 * is cleaned up elsewhere.
568 if (sc->sc_oxfer == NULL)
573 if (tp->t_state & TS_TBLOCK) {
574 if (ISSET(sc->sc_mcr, SER_RTS) &&
575 ISSET(sc->sc_state, UCS_RTS_IFLOW)) {
576 DPRINTF(("ucomstart: clear RTS\n"));
577 (void)ucommodem(tp, 0, SER_RTS);
580 if (!ISSET(sc->sc_mcr, SER_RTS) &&
581 tp->t_rawq.c_cc <= tp->t_ilowat &&
582 ISSET(sc->sc_state, UCS_RTS_IFLOW)) {
583 DPRINTF(("ucomstart: set RTS\n"));
584 (void)ucommodem(tp, SER_RTS, 0);
588 if (ISSET(tp->t_state, TS_BUSY | TS_TIMEOUT | TS_TTSTOP)) {
590 DPRINTF(("ucomstart: stopped\n"));
594 if (tp->t_outq.c_cc <= tp->t_olowat) {
595 if (ISSET(tp->t_state, TS_SO_OLOWAT)) {
596 CLR(tp->t_state, TS_SO_OLOWAT);
597 wakeup(TSA_OLOWAT(tp));
599 selwakeuppri(&tp->t_wsel, TTIPRI);
600 if (tp->t_outq.c_cc == 0) {
601 if (ISSET(tp->t_state, TS_BUSY | TS_SO_OCOMPLETE) ==
602 TS_SO_OCOMPLETE && tp->t_outq.c_cc == 0) {
603 CLR(tp->t_state, TS_SO_OCOMPLETE);
604 wakeup(TSA_OCOMPLETE(tp));
610 /* Grab the first contiguous region of buffer space. */
611 data = tp->t_outq.c_cf;
612 cbp = (struct cblock *) ((intptr_t) tp->t_outq.c_cf & ~CROUND);
613 cnt = min((char *) (cbp+1) - tp->t_outq.c_cf, tp->t_outq.c_cc);
616 DPRINTF(("ucomstart: cnt == 0\n"));
620 SET(tp->t_state, TS_BUSY);
622 if (cnt > sc->sc_obufsize) {
623 DPRINTF(("ucomstart: big buffer %d chars\n", cnt));
624 cnt = sc->sc_obufsize;
626 if (sc->sc_callback->ucom_write != NULL)
627 sc->sc_callback->ucom_write(sc->sc_parent, sc->sc_portno,
628 sc->sc_obuf, data, &cnt);
630 memcpy(sc->sc_obuf, data, cnt);
632 DPRINTF(("ucomstart: %d chars\n", cnt));
633 usbd_setup_xfer(sc->sc_oxfer, sc->sc_bulkout_pipe,
634 (usbd_private_handle)sc, sc->sc_obuf, cnt,
635 USBD_NO_COPY, USBD_NO_TIMEOUT, ucomwritecb);
636 /* What can we do on error? */
637 err = usbd_transfer(sc->sc_oxfer);
638 if (err != USBD_IN_PROGRESS)
639 printf("ucomstart: err=%s\n", usbd_errstr(err));
648 ucomstop(struct tty *tp, int flag)
650 struct ucom_softc *sc;
655 DPRINTF(("ucomstop: %d\n", flag));
657 if ((flag & FREAD) && (sc->sc_state & UCS_RXSTOP) == 0) {
658 DPRINTF(("ucomstop: read\n"));
664 DPRINTF(("ucomstop: write\n"));
666 if (ISSET(tp->t_state, TS_BUSY)) {
668 if (!ISSET(tp->t_state, TS_TTSTOP))
669 SET(tp->t_state, TS_FLUSH);
676 DPRINTF(("ucomstop: done\n"));
680 ucomwritecb(usbd_xfer_handle xfer, usbd_private_handle p, usbd_status status)
682 struct ucom_softc *sc = (struct ucom_softc *)p;
683 struct tty *tp = sc->sc_tty;
687 DPRINTF(("ucomwritecb: status = %d\n", status));
689 if (status == USBD_CANCELLED || sc->sc_dying)
692 if (status != USBD_NORMAL_COMPLETION) {
693 printf("%s: ucomwritecb: %s\n",
694 device_get_nameunit(sc->sc_dev), usbd_errstr(status));
695 if (status == USBD_STALLED)
696 usbd_clear_endpoint_stall_async(sc->sc_bulkin_pipe);
697 /* XXX we should restart after some delay. */
701 usbd_get_xfer_status(xfer, NULL, NULL, &cc, NULL);
702 DPRINTF(("ucomwritecb: cc = %d\n", cc));
703 if (cc <= sc->sc_opkthdrlen) {
704 printf("%s: sent size too small, cc = %d\n",
705 device_get_nameunit(sc->sc_dev), cc);
709 /* convert from USB bytes to tty bytes */
710 cc -= sc->sc_opkthdrlen;
713 CLR(tp->t_state, TS_BUSY);
714 if (ISSET(tp->t_state, TS_FLUSH))
715 CLR(tp->t_state, TS_FLUSH);
717 ndflush(&tp->t_outq, cc);
725 CLR(tp->t_state, TS_BUSY);
731 ucomstartread(struct ucom_softc *sc)
735 DPRINTF(("ucomstartread: start\n"));
737 if (sc->sc_bulkin_pipe == NULL || (sc->sc_state & UCS_RXSTOP) == 0)
738 return (USBD_NORMAL_COMPLETION);
739 sc->sc_state &= ~UCS_RXSTOP;
741 usbd_setup_xfer(sc->sc_ixfer, sc->sc_bulkin_pipe,
742 (usbd_private_handle)sc,
743 sc->sc_ibuf, sc->sc_ibufsize,
744 USBD_SHORT_XFER_OK | USBD_NO_COPY,
745 USBD_NO_TIMEOUT, ucomreadcb);
747 err = usbd_transfer(sc->sc_ixfer);
748 if (err && err != USBD_IN_PROGRESS) {
749 sc->sc_state |= UCS_RXSTOP;
750 DPRINTF(("ucomstartread: err = %s\n", usbd_errstr(err)));
754 return (USBD_NORMAL_COMPLETION);
758 ucomreadcb(usbd_xfer_handle xfer, usbd_private_handle p, usbd_status status)
760 struct ucom_softc *sc = (struct ucom_softc *)p;
761 struct tty *tp = sc->sc_tty;
768 DPRINTF(("ucomreadcb: status = %d\n", status));
770 if (status != USBD_NORMAL_COMPLETION) {
771 if (!(sc->sc_state & UCS_RXSTOP))
772 printf("%s: ucomreadcb: %s\n",
773 device_get_nameunit(sc->sc_dev), usbd_errstr(status));
774 sc->sc_state |= UCS_RXSTOP;
775 if (status == USBD_STALLED)
776 usbd_clear_endpoint_stall_async(sc->sc_bulkin_pipe);
777 /* XXX we should restart after some delay. */
780 sc->sc_state |= UCS_RXSTOP;
782 usbd_get_xfer_status(xfer, NULL, (void **)&cp, &cc, NULL);
783 DPRINTF(("ucomreadcb: got %d chars, tp = %p\n", cc, tp));
787 if (sc->sc_callback->ucom_read != NULL)
788 sc->sc_callback->ucom_read(sc->sc_parent, sc->sc_portno,
791 if (cc > sc->sc_ibufsize) {
792 printf("%s: invalid receive data size, %d chars\n",
793 device_get_nameunit(sc->sc_dev), cc);
800 if (tp->t_state & TS_CAN_BYPASS_L_RINT) {
801 if (tp->t_rawq.c_cc + cc > tp->t_ihiwat
802 && (sc->sc_state & UCS_RTS_IFLOW
803 || tp->t_iflag & IXOFF)
804 && !(tp->t_state & TS_TBLOCK))
806 lostcc = b_to_q((char *)cp, cc, &tp->t_rawq);
809 if (tp->t_state & TS_TTSTOP
810 && (tp->t_iflag & IXANY
811 || tp->t_cc[VSTART] == tp->t_cc[VSTOP])) {
812 tp->t_state &= ~TS_TTSTOP;
813 tp->t_lflag &= ~FLUSHO;
817 printf("%s: lost %d chars (t_rawq)\n", device_get_nameunit(sc->sc_dev),
820 /* Give characters to tty layer. */
822 DPRINTFN(7, ("ucomreadcb: char = 0x%02x\n", *cp));
823 if (ttyld_rint(tp, *cp) == -1) {
824 /* XXX what should we do? */
825 printf("%s: lost %d chars (ttyld_rint)\n",
826 device_get_nameunit(sc->sc_dev), cc);
836 err = ucomstartread(sc);
838 printf("%s: read start failed\n", device_get_nameunit(sc->sc_dev));
839 /* XXX what should we dow now? */
842 if ((sc->sc_state & UCS_RTS_IFLOW) && !ISSET(sc->sc_mcr, SER_RTS)
843 && !(tp->t_state & TS_TBLOCK))
844 ucommodem(tp, SER_RTS, 0);
848 ucom_cleanup(struct ucom_softc *sc)
850 DPRINTF(("ucom_cleanup: closing pipes\n"));
853 if (sc->sc_bulkin_pipe != NULL) {
854 sc->sc_state |= UCS_RXSTOP;
855 usbd_abort_pipe(sc->sc_bulkin_pipe);
856 usbd_close_pipe(sc->sc_bulkin_pipe);
857 sc->sc_bulkin_pipe = NULL;
859 if (sc->sc_bulkout_pipe != NULL) {
860 usbd_abort_pipe(sc->sc_bulkout_pipe);
861 usbd_close_pipe(sc->sc_bulkout_pipe);
862 sc->sc_bulkout_pipe = NULL;
864 if (sc->sc_ixfer != NULL) {
865 usbd_free_xfer(sc->sc_ixfer);
868 if (sc->sc_oxfer != NULL) {
869 usbd_free_xfer(sc->sc_oxfer);
875 ucomstopread(struct ucom_softc *sc)
879 DPRINTF(("ucomstopread: enter\n"));
881 if (!(sc->sc_state & UCS_RXSTOP)) {
882 sc->sc_state |= UCS_RXSTOP;
883 if (sc->sc_bulkin_pipe == NULL) {
884 DPRINTF(("ucomstopread: bulkin pipe NULL\n"));
887 err = usbd_abort_pipe(sc->sc_bulkin_pipe);
889 DPRINTF(("ucomstopread: err = %s\n",
894 DPRINTF(("ucomstopread: leave\n"));