2 * Copyright (c) 1997, 2001 Hellmuth Michaelis. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 *---------------------------------------------------------------------------
27 * i4b_rbch.c - device driver for raw B channel data
28 * ---------------------------------------------------
30 * $FreeBSD: src/sys/i4b/driver/i4b_rbch.c,v 1.10.2.3 2001/08/12 16:22:48 hm Exp $
31 * $DragonFly: src/sys/net/i4b/driver/i4b_rbch.c,v 1.22 2006/12/22 23:44:55 swildner Exp $
33 * last edit-date: [Sat Aug 11 18:06:57 2001]
35 *---------------------------------------------------------------------------*/
37 #include "use_i4brbch.h"
41 #include <sys/param.h>
42 #include <sys/systm.h>
46 #include <sys/kernel.h>
48 #include <sys/socket.h>
51 #include <sys/thread2.h>
52 #include <sys/vnode.h>
54 #include <net/i4b/include/machine/i4b_ioctl.h>
55 #include <net/i4b/include/machine/i4b_rbch_ioctl.h>
56 #include <net/i4b/include/machine/i4b_debug.h>
58 #include "../include/i4b_global.h"
59 #include "../include/i4b_mbuf.h"
60 #include "../include/i4b_l3l4.h"
61 #include "../layer4/i4b_l4.h"
64 #include <sys/event.h>
65 #include <sys/filio.h>
67 static drvr_link_t rbch_drvr_linktab[NI4BRBCH];
68 static isdn_link_t *isdn_linktab[NI4BRBCH];
70 #define I4BRBCHACCT 1 /* enable accounting messages */
71 #define I4BRBCHACCTINTVL 2 /* accounting msg interval in secs */
73 static struct rbch_softc {
75 int sc_unit; /* unit number */
77 int sc_devstate; /* state of driver */
79 #define ST_CONNECTED 0x01
80 #define ST_ISOPEN 0x02
81 #define ST_RDWAITDATA 0x04
82 #define ST_WRWAITEMPTY 0x08
83 #define ST_NOBLOCK 0x10
85 int sc_bprot; /* B-ch protocol used */
87 call_desc_t *sc_cd; /* Call Descriptor */
91 struct ifqueue sc_hdlcq; /* hdlc read queue */
92 #define I4BRBCHMAXQLEN 10
94 struct selinfo selp; /* select / poll */
97 struct callout sc_timeout;
99 int sc_iinb; /* isdn driver # of inbytes */
100 int sc_ioutb; /* isdn driver # of outbytes */
101 int sc_linb; /* last # of bytes rx'd */
102 int sc_loutb; /* last # of bytes tx'd */
103 int sc_fn; /* flag, first null acct */
105 } rbch_softc[NI4BRBCH];
107 static void rbch_rx_data_rdy(int unit);
108 static void rbch_tx_queue_empty(int unit);
109 static void rbch_connect(int unit, void *cdp);
110 static void rbch_disconnect(int unit, void *cdp);
111 static void rbch_init_linktab(int unit);
112 static void rbch_clrq(int unit);
114 #define PDEVSTATIC static
115 #define IOCTL_CMD_T u_long
117 PDEVSTATIC d_open_t i4brbchopen;
118 PDEVSTATIC d_close_t i4brbchclose;
119 PDEVSTATIC d_read_t i4brbchread;
120 PDEVSTATIC d_write_t i4brbchwrite;
121 PDEVSTATIC d_ioctl_t i4brbchioctl;
122 PDEVSTATIC d_kqfilter_t i4brbchkqfilter;
124 PDEVSTATIC void i4brbchkfilt_detach(struct knote *);
125 PDEVSTATIC int i4brbchkfilt_read(struct knote *, long);
126 PDEVSTATIC int i4brbchkfilt_write(struct knote *, long);
128 PDEVSTATIC d_poll_t i4brbchpoll;
129 #define POLLFIELD i4brbchpoll
131 #define CDEV_MAJOR 57
133 static struct dev_ops i4brbch_ops = {
134 { "i4brbch", CDEV_MAJOR, D_KQFILTER },
135 .d_open = i4brbchopen,
136 .d_close = i4brbchclose,
137 .d_read = i4brbchread,
138 .d_write = i4brbchwrite,
139 .d_ioctl = i4brbchioctl,
141 .d_kqfilter = i4brbchkqfilter
144 static void i4brbchattach(void *);
145 PSEUDO_SET(i4brbchattach, i4b_rbch);
147 /*===========================================================================*
148 * DEVICE DRIVER ROUTINES
149 *===========================================================================*/
151 /*---------------------------------------------------------------------------*
152 * interface attach routine
153 *---------------------------------------------------------------------------*/
155 i4brbchattach(void *dummy)
159 #ifndef HACK_NO_PSEUDO_ATTACH_MSG
160 kprintf("i4brbch: %d raw B channel access device(s) attached\n", NI4BRBCH);
163 for(i=0; i < NI4BRBCH; i++)
165 make_dev(&i4brbch_ops, i,
166 UID_ROOT, GID_WHEEL, 0600, "i4brbch%d", i);
169 callout_init(&rbch_softc[i].sc_timeout);
170 rbch_softc[i].sc_fn = 1;
172 rbch_softc[i].sc_unit = i;
173 rbch_softc[i].sc_devstate = ST_IDLE;
174 rbch_softc[i].sc_hdlcq.ifq_maxlen = I4BRBCHMAXQLEN;
175 rbch_softc[i].it_in.c_ispeed = rbch_softc[i].it_in.c_ospeed = 64000;
176 termioschars(&rbch_softc[i].it_in);
177 rbch_init_linktab(i);
181 /*---------------------------------------------------------------------------*
183 *---------------------------------------------------------------------------*/
185 i4brbchopen(struct dev_open_args *ap)
187 cdev_t dev = ap->a_head.a_dev;
188 int unit = minor(dev);
193 if(rbch_softc[unit].sc_devstate & ST_ISOPEN)
200 rbch_softc[unit].sc_devstate |= ST_ISOPEN;
202 NDBGL4(L4_RBCHDBG, "unit %d, open", unit);
207 /*---------------------------------------------------------------------------*
209 *---------------------------------------------------------------------------*/
211 i4brbchclose(struct dev_close_args *ap)
213 cdev_t dev = ap->a_head.a_dev;
214 int unit = minor(dev);
215 struct rbch_softc *sc = &rbch_softc[unit];
217 if(sc->sc_devstate & ST_CONNECTED)
218 i4b_l4_drvrdisc(BDRV_RBCH, unit);
220 sc->sc_devstate &= ~ST_ISOPEN;
224 NDBGL4(L4_RBCHDBG, "unit %d, closed", unit);
229 /*---------------------------------------------------------------------------*
230 * read from rbch device
231 *---------------------------------------------------------------------------*/
233 i4brbchread(struct dev_read_args *ap)
235 cdev_t dev = ap->a_head.a_dev;
236 struct uio *uio = ap->a_uio;
239 int unit = minor(dev);
241 struct rbch_softc *sc = &rbch_softc[unit];
245 NDBGL4(L4_RBCHDBG, "unit %d, enter read", unit);
248 if(!(sc->sc_devstate & ST_ISOPEN))
251 NDBGL4(L4_RBCHDBG, "unit %d, read while not open", unit);
255 if((sc->sc_devstate & ST_NOBLOCK) || (ap->a_ioflag & IO_NDELAY))
257 if(!(sc->sc_devstate & ST_CONNECTED)) {
262 if(sc->sc_bprot == BPROT_RHDLC)
265 iqp = isdn_linktab[unit]->rx_queue;
267 if(IF_QEMPTY(iqp) && (sc->sc_devstate & ST_ISOPEN)) {
274 while(!(sc->sc_devstate & ST_CONNECTED))
276 NDBGL4(L4_RBCHDBG, "unit %d, wait read init", unit);
278 if((error = tsleep((caddr_t) &rbch_softc[unit],
279 PCATCH, "rrrbch", 0 )) != 0)
282 NDBGL4(L4_RBCHDBG, "unit %d, error %d tsleep", unit, error);
287 if(sc->sc_bprot == BPROT_RHDLC)
290 iqp = isdn_linktab[unit]->rx_queue;
292 while(IF_QEMPTY(iqp) && (sc->sc_devstate & ST_ISOPEN))
294 sc->sc_devstate |= ST_RDWAITDATA;
296 NDBGL4(L4_RBCHDBG, "unit %d, wait read data", unit);
298 if((error = tsleep((caddr_t) &isdn_linktab[unit]->rx_queue,
299 PCATCH, "rrbch", 0 )) != 0)
302 NDBGL4(L4_RBCHDBG, "unit %d, error %d tsleep read", unit, error);
303 sc->sc_devstate &= ~ST_RDWAITDATA;
311 NDBGL4(L4_RBCHDBG, "unit %d, read %d bytes", unit, m->m_len);
315 error = uiomove(m->m_data, m->m_len, uio);
319 NDBGL4(L4_RBCHDBG, "unit %d, error %d uiomove", unit, error);
331 /*---------------------------------------------------------------------------*
332 * write to rbch device
333 *---------------------------------------------------------------------------*/
335 i4brbchwrite(struct dev_write_args *ap)
337 cdev_t dev = ap->a_head.a_dev;
338 struct uio *uio = ap->a_uio;
341 int unit = minor(dev);
342 struct rbch_softc *sc = &rbch_softc[unit];
346 NDBGL4(L4_RBCHDBG, "unit %d, write", unit);
349 if(!(sc->sc_devstate & ST_ISOPEN))
351 NDBGL4(L4_RBCHDBG, "unit %d, write while not open", unit);
356 if((sc->sc_devstate & ST_NOBLOCK) || (ap->a_ioflag & IO_NDELAY))
358 if(!(sc->sc_devstate & ST_CONNECTED)) {
362 if(IF_QFULL(isdn_linktab[unit]->tx_queue) && (sc->sc_devstate & ST_ISOPEN)) {
369 while(!(sc->sc_devstate & ST_CONNECTED))
371 NDBGL4(L4_RBCHDBG, "unit %d, write wait init", unit);
373 error = tsleep((caddr_t) &rbch_softc[unit],
374 PCATCH, "wrrbch", 0 );
375 if(error == ERESTART) {
379 else if(error == EINTR)
382 NDBGL4(L4_RBCHDBG, "unit %d, EINTR during wait init", unit);
388 NDBGL4(L4_RBCHDBG, "unit %d, error %d tsleep init", unit, error);
391 tsleep((caddr_t) &rbch_softc[unit], PCATCH, "xrbch", (hz*1));
394 while(IF_QFULL(isdn_linktab[unit]->tx_queue) && (sc->sc_devstate & ST_ISOPEN))
396 sc->sc_devstate |= ST_WRWAITEMPTY;
398 NDBGL4(L4_RBCHDBG, "unit %d, write queue full", unit);
400 if ((error = tsleep((caddr_t) &isdn_linktab[unit]->tx_queue,
401 PCATCH, "wrbch", 0)) != 0) {
402 sc->sc_devstate &= ~ST_WRWAITEMPTY;
403 if(error == ERESTART)
408 else if(error == EINTR)
411 NDBGL4(L4_RBCHDBG, "unit %d, EINTR during wait write", unit);
417 NDBGL4(L4_RBCHDBG, "unit %d, error %d tsleep write", unit, error);
424 if(!(sc->sc_devstate & ST_ISOPEN))
426 NDBGL4(L4_RBCHDBG, "unit %d, not open anymore", unit);
431 if((m = i4b_Bgetmbuf(BCH_MAX_DATALEN)) != NULL)
433 m->m_len = (int)szmin(BCH_MAX_DATALEN, uio->uio_resid);
435 NDBGL4(L4_RBCHDBG, "unit %d, write %d bytes", unit, m->m_len);
437 error = uiomove(m->m_data, (size_t)m->m_len, uio);
439 if(IF_QFULL(isdn_linktab[unit]->tx_queue))
442 IF_ENQUEUE(isdn_linktab[unit]->tx_queue, m);
443 (*isdn_linktab[unit]->bch_tx_start)(isdn_linktab[unit]->unit, isdn_linktab[unit]->channel);
451 /*---------------------------------------------------------------------------*
452 * rbch device ioctl handlibg
453 *---------------------------------------------------------------------------*/
455 i4brbchioctl(struct dev_ioctl_args *ap)
457 cdev_t dev = ap->a_head.a_dev;
459 int unit = minor(dev);
460 struct rbch_softc *sc = &rbch_softc[unit];
464 case FIOASYNC: /* Set async mode */
465 if (*(int *)ap->a_data)
467 NDBGL4(L4_RBCHDBG, "unit %d, setting async mode", unit);
471 NDBGL4(L4_RBCHDBG, "unit %d, clearing async mode", unit);
475 case TIOCCDTR: /* Clear DTR */
476 if(sc->sc_devstate & ST_CONNECTED)
478 NDBGL4(L4_RBCHDBG, "unit %d, disconnecting for DTR down", unit);
479 i4b_l4_drvrdisc(BDRV_RBCH, unit);
483 case I4B_RBCH_DIALOUT:
487 for (l = 0; l < TELNO_MAX && ((char *)ap->a_data)[l]; l++)
491 NDBGL4(L4_RBCHDBG, "unit %d, attempting dialout to %s", unit, (char *)ap->a_data);
492 i4b_l4_dialoutnumber(BDRV_RBCH, unit, l, (char *)ap->a_data);
495 /* fall through to SDTR */
498 case TIOCSDTR: /* Set DTR */
499 NDBGL4(L4_RBCHDBG, "unit %d, attempting dialout (DTR)", unit);
500 i4b_l4_dialout(BDRV_RBCH, unit);
503 case TIOCSETA: /* Set termios struct */
506 case TIOCGETA: /* Get termios struct */
507 *(struct termios *)ap->a_data = sc->it_in;
511 *(int *)ap->a_data = TIOCM_LE|TIOCM_DTR|TIOCM_RTS|TIOCM_CTS|TIOCM_DSR;
512 if (sc->sc_devstate & ST_CONNECTED)
513 *(int *)ap->a_data |= TIOCM_CD;
516 case I4B_RBCH_VR_REQ:
520 mvr = (msg_vr_req_t *)ap->a_data;
522 mvr->version = VERSION;
528 default: /* Unknown stuff */
529 NDBGL4(L4_RBCHDBG, "unit %d, ioctl, unknown cmd %lx", unit, ap->a_cmd);
536 /*---------------------------------------------------------------------------*
538 *---------------------------------------------------------------------------*/
540 i4brbchpoll(struct dev_poll_args *ap)
542 cdev_t dev = ap->a_head.a_dev;
543 int revents = 0; /* Events we found */
544 int unit = minor(dev);
545 struct rbch_softc *sc = &rbch_softc[unit];
547 /* We can't check for anything but IN or OUT */
550 if(!(sc->sc_devstate & ST_ISOPEN))
557 * Writes are OK if we are connected and the
558 * transmit queue can take them
561 if((ap->a_events & (POLLOUT|POLLWRNORM)) &&
562 (sc->sc_devstate & ST_CONNECTED) &&
563 !IF_QFULL(isdn_linktab[unit]->tx_queue))
565 revents |= (ap->a_events & (POLLOUT|POLLWRNORM));
568 /* ... while reads are OK if we have any data */
570 if((ap->a_events & (POLLIN|POLLRDNORM)) &&
571 (sc->sc_devstate & ST_CONNECTED))
575 if(sc->sc_bprot == BPROT_RHDLC)
578 iqp = isdn_linktab[unit]->rx_queue;
581 revents |= (ap->a_events & (POLLIN|POLLRDNORM));
585 selrecord(curthread, &sc->selp);
588 ap->a_events = revents;
592 static struct filterops i4brbchkfiltops_read =
593 { 1, NULL, i4brbchkfilt_detach, i4brbchkfilt_read };
594 static struct filterops i4brbchkfiltops_write =
595 { 1, NULL, i4brbchkfilt_detach, i4brbchkfilt_write };
598 i4brbchkqfilter(struct dev_kqfilter_args *ap)
600 cdev_t dev = ap->a_head.a_dev;
601 int unit = minor(dev);
602 struct rbch_softc *sc = &rbch_softc[unit];
603 struct knote *kn = ap->a_kn;
608 switch (kn->kn_filter) {
610 kn->kn_fop = &i4brbchkfiltops_read;
611 kn->kn_hook = (caddr_t)dev;
614 kn->kn_fop = &i4brbchkfiltops_write;
615 kn->kn_hook = (caddr_t)dev;
618 ap->a_result = EOPNOTSUPP;
623 klist = &sc->selp.si_note;
624 SLIST_INSERT_HEAD(klist, kn, kn_selnext);
631 i4brbchkfilt_detach(struct knote *kn)
633 cdev_t dev = (cdev_t)kn->kn_hook;
634 int unit = minor(dev);
635 struct rbch_softc *sc = &rbch_softc[unit];
639 klist = &sc->selp.si_note;
640 SLIST_REMOVE(klist, kn, knote, kn_selnext);
645 i4brbchkfilt_read(struct knote *kn, long hint)
647 cdev_t dev = (cdev_t)kn->kn_hook;
648 int unit = minor(dev);
649 struct rbch_softc *sc = &rbch_softc[unit];
654 if (!(sc->sc_devstate & ST_ISOPEN)) {
656 kn->kn_flags |= EV_ERROR;
661 if (sc->sc_devstate & ST_CONNECTED) {
664 if(sc->sc_bprot == BPROT_RHDLC)
667 iqp = isdn_linktab[unit]->rx_queue;
679 i4brbchkfilt_write(struct knote *kn, long hint)
681 cdev_t dev = (cdev_t)kn->kn_hook;
682 int unit = minor(dev);
683 struct rbch_softc *sc = &rbch_softc[unit];
688 if (!(sc->sc_devstate & ST_ISOPEN)) {
690 kn->kn_flags |= EV_ERROR;
696 * Writes are OK if we are connected and the
697 * transmit queue can take them
699 if ((sc->sc_devstate & ST_CONNECTED) &&
700 !IF_QFULL(isdn_linktab[unit]->tx_queue))
711 /*---------------------------------------------------------------------------*
713 *---------------------------------------------------------------------------*/
715 rbch_timeout(struct rbch_softc *sc)
717 bchan_statistics_t bs;
718 int unit = sc->sc_unit;
720 /* get # of bytes in and out from the HSCX driver */
722 (*isdn_linktab[unit]->bch_stat)
723 (isdn_linktab[unit]->unit, isdn_linktab[unit]->channel, &bs);
725 sc->sc_ioutb += bs.outbytes;
726 sc->sc_iinb += bs.inbytes;
728 if((sc->sc_iinb != sc->sc_linb) || (sc->sc_ioutb != sc->sc_loutb) || sc->sc_fn)
730 int ri = (sc->sc_iinb - sc->sc_linb)/I4BRBCHACCTINTVL;
731 int ro = (sc->sc_ioutb - sc->sc_loutb)/I4BRBCHACCTINTVL;
733 if((sc->sc_iinb == sc->sc_linb) && (sc->sc_ioutb == sc->sc_loutb))
738 sc->sc_linb = sc->sc_iinb;
739 sc->sc_loutb = sc->sc_ioutb;
741 i4b_l4_accounting(BDRV_RBCH, unit, ACCT_DURING,
742 sc->sc_ioutb, sc->sc_iinb, ro, ri, sc->sc_ioutb, sc->sc_iinb);
744 callout_reset(&sc->sc_timeout, I4BRBCHACCTINTVL * hz,
745 (void *)rbch_timeout, sc);
747 #endif /* I4BRBCHACCT */
749 /*===========================================================================*
750 * ISDN INTERFACE ROUTINES
751 *===========================================================================*/
753 /*---------------------------------------------------------------------------*
754 * this routine is called from L4 handler at connect time
755 *---------------------------------------------------------------------------*/
757 rbch_connect(int unit, void *cdp)
759 call_desc_t *cd = (call_desc_t *)cdp;
760 struct rbch_softc *sc = &rbch_softc[unit];
762 sc->sc_bprot = cd->bprot;
765 if(sc->sc_bprot == BPROT_RHDLC)
772 callout_reset(&sc->sc_timeout, I4BRBCHACCTINTVL * hz,
773 (void *)rbch_timeout, sc);
776 if(!(sc->sc_devstate & ST_CONNECTED))
778 NDBGL4(L4_RBCHDBG, "unit %d, wakeup", unit);
779 sc->sc_devstate |= ST_CONNECTED;
785 /*---------------------------------------------------------------------------*
786 * this routine is called from L4 handler at disconnect time
787 *---------------------------------------------------------------------------*/
789 rbch_disconnect(int unit, void *cdp)
791 call_desc_t *cd = (call_desc_t *)cdp;
792 struct rbch_softc *sc = &rbch_softc[unit];
798 NDBGL4(L4_RBCHDBG, "rbch%d: channel %d not active",
799 cd->driver_unit, cd->channelid);
805 NDBGL4(L4_RBCHDBG, "unit %d, disconnect", unit);
807 sc->sc_devstate &= ~ST_CONNECTED;
812 i4b_l4_accounting(BDRV_RBCH, unit, ACCT_FINAL,
813 sc->sc_ioutb, sc->sc_iinb, 0, 0, sc->sc_ioutb, sc->sc_iinb);
814 callout_stop(&sc->sc_timeout);
819 /*---------------------------------------------------------------------------*
820 * feedback from daemon in case of dial problems
821 *---------------------------------------------------------------------------*/
823 rbch_dialresponse(int unit, int status, cause_t cause)
827 /*---------------------------------------------------------------------------*
829 *---------------------------------------------------------------------------*/
831 rbch_updown(int unit, int updown)
835 /*---------------------------------------------------------------------------*
836 * this routine is called from the HSCX interrupt handler
837 * when a new frame (mbuf) has been received and is to be put on
839 *---------------------------------------------------------------------------*/
841 rbch_rx_data_rdy(int unit)
843 if(rbch_softc[unit].sc_bprot == BPROT_RHDLC)
847 if((m = *isdn_linktab[unit]->rx_mbuf) == NULL)
850 m->m_pkthdr.len = m->m_len;
852 if(IF_QFULL(&(rbch_softc[unit].sc_hdlcq)))
854 NDBGL4(L4_RBCHDBG, "unit %d: hdlc rx queue full!", unit);
859 IF_ENQUEUE(&(rbch_softc[unit].sc_hdlcq), m);
863 if(rbch_softc[unit].sc_devstate & ST_RDWAITDATA)
865 NDBGL4(L4_RBCHDBG, "unit %d, wakeup", unit);
866 rbch_softc[unit].sc_devstate &= ~ST_RDWAITDATA;
867 wakeup((caddr_t) &isdn_linktab[unit]->rx_queue);
871 NDBGL4(L4_RBCHDBG, "unit %d, NO wakeup", unit);
873 selwakeup(&rbch_softc[unit].selp);
876 /*---------------------------------------------------------------------------*
877 * this routine is called from the HSCX interrupt handler
878 * when the last frame has been sent out and there is no
879 * further frame (mbuf) in the tx queue.
880 *---------------------------------------------------------------------------*/
882 rbch_tx_queue_empty(int unit)
884 if(rbch_softc[unit].sc_devstate & ST_WRWAITEMPTY)
886 NDBGL4(L4_RBCHDBG, "unit %d, wakeup", unit);
887 rbch_softc[unit].sc_devstate &= ~ST_WRWAITEMPTY;
888 wakeup((caddr_t) &isdn_linktab[unit]->tx_queue);
892 NDBGL4(L4_RBCHDBG, "unit %d, NO wakeup", unit);
894 selwakeup(&rbch_softc[unit].selp);
897 /*---------------------------------------------------------------------------*
898 * this routine is called from the HSCX interrupt handler
899 * each time a packet is received or transmitted
900 *---------------------------------------------------------------------------*/
902 rbch_activity(int unit, int rxtx)
904 if (rbch_softc[unit].sc_cd)
905 rbch_softc[unit].sc_cd->last_active_time = SECOND;
906 selwakeup(&rbch_softc[unit].selp);
909 /*---------------------------------------------------------------------------*
910 * clear an hdlc rx queue for a rbch unit
911 *---------------------------------------------------------------------------*/
918 IF_DRAIN(&rbch_softc[unit].sc_hdlcq);
922 /*---------------------------------------------------------------------------*
923 * return this drivers linktab address
924 *---------------------------------------------------------------------------*/
926 rbch_ret_linktab(int unit)
928 rbch_init_linktab(unit);
929 return(&rbch_drvr_linktab[unit]);
932 /*---------------------------------------------------------------------------*
933 * setup the isdn_linktab for this driver
934 *---------------------------------------------------------------------------*/
936 rbch_set_linktab(int unit, isdn_link_t *ilt)
938 isdn_linktab[unit] = ilt;
941 /*---------------------------------------------------------------------------*
942 * initialize this drivers linktab
943 *---------------------------------------------------------------------------*/
945 rbch_init_linktab(int unit)
947 rbch_drvr_linktab[unit].unit = unit;
948 rbch_drvr_linktab[unit].bch_rx_data_ready = rbch_rx_data_rdy;
949 rbch_drvr_linktab[unit].bch_tx_queue_empty = rbch_tx_queue_empty;
950 rbch_drvr_linktab[unit].bch_activity = rbch_activity;
951 rbch_drvr_linktab[unit].line_connected = rbch_connect;
952 rbch_drvr_linktab[unit].line_disconnected = rbch_disconnect;
953 rbch_drvr_linktab[unit].dial_response = rbch_dialresponse;
954 rbch_drvr_linktab[unit].updown_ind = rbch_updown;
957 /*===========================================================================*/
959 #endif /* NI4BRBCH > 0 */