Add some special cases for the Sitecom adapter.

Make endian safe.
Fix some minor bugs.
This commit is contained in:
augustss 2001-12-12 23:59:48 +00:00
parent 1b6c4afd64
commit 8e98b4a185

View File

@ -1,4 +1,4 @@
/* $NetBSD: umct.c,v 1.4 2001/12/03 01:47:12 augustss Exp $ */ /* $NetBSD: umct.c,v 1.5 2001/12/12 23:59:48 augustss Exp $ */
/* /*
* Copyright (c) 2001 The NetBSD Foundation, Inc. * Copyright (c) 2001 The NetBSD Foundation, Inc.
* All rights reserved. * All rights reserved.
@ -43,7 +43,7 @@
*/ */
#include <sys/cdefs.h> #include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: umct.c,v 1.4 2001/12/03 01:47:12 augustss Exp $"); __KERNEL_RCSID(0, "$NetBSD: umct.c,v 1.5 2001/12/12 23:59:48 augustss Exp $");
#include <sys/param.h> #include <sys/param.h>
#include <sys/systm.h> #include <sys/systm.h>
@ -88,6 +88,7 @@ struct umct_softc {
usbd_device_handle sc_udev; /* USB device */ usbd_device_handle sc_udev; /* USB device */
usbd_interface_handle sc_iface; /* interface */ usbd_interface_handle sc_iface; /* interface */
int sc_iface_number; /* interface number */ int sc_iface_number; /* interface number */
u_int16_t sc_product;
int sc_intr_number; /* interrupt number */ int sc_intr_number; /* interrupt number */
usbd_pipe_handle sc_intr_pipe; /* interrupt pipe */ usbd_pipe_handle sc_intr_pipe; /* interrupt pipe */
@ -117,8 +118,8 @@ struct umct_softc {
#define UMCTOBUFSIZE 256 #define UMCTOBUFSIZE 256
Static void umct_init(struct umct_softc *); Static void umct_init(struct umct_softc *);
Static void umct_set_baudrate(struct umct_softc *, void *); Static void umct_set_baudrate(struct umct_softc *, u_int);
Static void umct_set_lcr(struct umct_softc *, void *); Static void umct_set_lcr(struct umct_softc *, u_int);
Static void umct_intr(usbd_xfer_handle, usbd_private_handle, usbd_status); Static void umct_intr(usbd_xfer_handle, usbd_private_handle, usbd_status);
Static void umct_set(void *, int, int, int); Static void umct_set(void *, int, int, int);
@ -184,6 +185,7 @@ USB_ATTACH(umct)
printf("%s: %s\n", devname, devinfo); printf("%s: %s\n", devname, devinfo);
sc->sc_udev = dev; sc->sc_udev = dev;
sc->sc_product = uaa->product;
DPRINTF(("\n\numct attach: sc=%p\n", sc)); DPRINTF(("\n\numct attach: sc=%p\n", sc));
@ -276,6 +278,9 @@ USB_ATTACH(umct)
uca.portno = UCOM_UNK_PORTNO; uca.portno = UCOM_UNK_PORTNO;
/* bulkin, bulkout set above */ /* bulkin, bulkout set above */
uca.ibufsize = UMCTIBUFSIZE; uca.ibufsize = UMCTIBUFSIZE;
if (sc->sc_product == USB_PRODUCT_MCT_SITECOM_USB232)
uca.obufsize = 16; /* device is broken */
else
uca.obufsize = UMCTOBUFSIZE; uca.obufsize = UMCTOBUFSIZE;
uca.ibufsizepad = UMCTIBUFSIZE; uca.ibufsizepad = UMCTIBUFSIZE;
uca.opkthdrlen = 0; uca.opkthdrlen = 0;
@ -347,7 +352,7 @@ void
umct_set_line_state(struct umct_softc *sc) umct_set_line_state(struct umct_softc *sc)
{ {
usb_device_request_t req; usb_device_request_t req;
int ls = MCR_NONE; uByte ls;
ls = (sc->sc_dtr ? MCR_DTR : 0) | ls = (sc->sc_dtr ? MCR_DTR : 0) |
(sc->sc_rts ? MCR_RTS : 0); (sc->sc_rts ? MCR_RTS : 0);
@ -412,33 +417,52 @@ umct_rts(struct umct_softc *sc, int onoff)
void void
umct_break(struct umct_softc *sc, int onoff) umct_break(struct umct_softc *sc, int onoff)
{ {
u_char data;
DPRINTF(("umct_break: onoff=%d\n", onoff)); DPRINTF(("umct_break: onoff=%d\n", onoff));
data = onoff ? LCR_SET_BREAK : 0; umct_set_lcr(sc, onoff ? LCR_SET_BREAK : 0);
umct_set_lcr(sc, &data);
} }
void void
umct_set_lcr(struct umct_softc *sc, void *data) umct_set_lcr(struct umct_softc *sc, u_int data)
{ {
usb_device_request_t req; usb_device_request_t req;
uByte adata;
adata = data;
req.bmRequestType = UMCT_SET_REQUEST; req.bmRequestType = UMCT_SET_REQUEST;
req.bRequest = REQ_SET_LCR; req.bRequest = REQ_SET_LCR;
USETW(req.wValue, 0); USETW(req.wValue, 0);
USETW(req.wIndex, sc->sc_iface_number); USETW(req.wIndex, sc->sc_iface_number);
USETW(req.wLength, LENGTH_SET_LCR); USETW(req.wLength, LENGTH_SET_LCR);
(void)usbd_do_request(sc->sc_udev, &req, data); (void)usbd_do_request(sc->sc_udev, &req, &adata); /* XXX should check */
} }
void void
umct_set_baudrate(struct umct_softc *sc, void *rate) umct_set_baudrate(struct umct_softc *sc, u_int rate)
{ {
usb_device_request_t req; usb_device_request_t req;
uDWord arate;
u_int val;
if (sc->sc_product == USB_PRODUCT_MCT_SITECOM_USB232) {
switch (rate) {
case 300: val = 0x01; break;
case 600: val = 0x02; break;
case 1200: val = 0x03; break;
case 2400: val = 0x04; break;
case 4800: val = 0x06; break;
case 9600: val = 0x08; break;
case 19200: val = 0x09; break;
case 38400: val = 0x0a; break;
case 57600: val = 0x0b; break;
case 115200: val = 0x0c; break;
default: val = -1; break;
}
} else {
val = UMCT_BAUD_RATE(rate);
}
USETDW(arate, val);
req.bmRequestType = UMCT_SET_REQUEST; req.bmRequestType = UMCT_SET_REQUEST;
req.bRequest = REQ_SET_BAUD_RATE; req.bRequest = REQ_SET_BAUD_RATE;
@ -446,32 +470,24 @@ umct_set_baudrate(struct umct_softc *sc, void *rate)
USETW(req.wIndex, sc->sc_iface_number); USETW(req.wIndex, sc->sc_iface_number);
USETW(req.wLength, LENGTH_BAUD_RATE); USETW(req.wLength, LENGTH_BAUD_RATE);
(void)usbd_do_request(sc->sc_udev, &req, rate); (void)usbd_do_request(sc->sc_udev, &req, arate); /* XXX should check */
} }
void void
umct_init(struct umct_softc *sc) umct_init(struct umct_softc *sc)
{ {
int brate, data; umct_set_baudrate(sc, 9600);
umct_set_lcr(sc, LCR_DATA_BITS_8 | LCR_PARITY_NONE | LCR_STOP_BITS_1);
brate = UMCT_BAUD_RATE(9600);
data |= LCR_DATA_BITS_8 | LCR_PARITY_NONE |
LCR_STOP_BITS_1;
umct_set_baudrate(sc, &brate);
umct_set_lcr(sc, &data);
} }
int int
umct_param(void *addr, int portno, struct termios *t) umct_param(void *addr, int portno, struct termios *t)
{ {
struct umct_softc *sc = addr; struct umct_softc *sc = addr;
int divisor = 0; u_int data = 0;
u_char data = NULL;
DPRINTF(("umct_param: sc=%p\n", sc)); DPRINTF(("umct_param: sc=%p\n", sc));
divisor = UMCT_BAUD_RATE(t->c_ospeed);
DPRINTF(("umct_param: BAUDRATE=%d\n", t->c_ospeed)); DPRINTF(("umct_param: BAUDRATE=%d\n", t->c_ospeed));
if (ISSET(t->c_cflag, CSTOPB)) if (ISSET(t->c_cflag, CSTOPB))
@ -500,9 +516,9 @@ umct_param(void *addr, int portno, struct termios *t)
break; break;
} }
umct_set_baudrate(sc, &divisor); umct_set_baudrate(sc, t->c_ospeed);
umct_set_lcr(sc, &data); umct_set_lcr(sc, data);
return (0); return (0);
} }
@ -519,9 +535,9 @@ umct_open(void *addr, int portno)
DPRINTF(("umct_open: sc=%p\n", sc)); DPRINTF(("umct_open: sc=%p\n", sc));
/* initialize LCR */ /* initialize LCR */
lcr_data |= LCR_DATA_BITS_8 | LCR_PARITY_NONE | lcr_data = LCR_DATA_BITS_8 | LCR_PARITY_NONE |
LCR_STOP_BITS_1; LCR_STOP_BITS_1;
umct_set_lcr(sc, &lcr_data); umct_set_lcr(sc, lcr_data);
if (sc->sc_intr_number != -1 && sc->sc_intr_pipe == NULL) { if (sc->sc_intr_number != -1 && sc->sc_intr_pipe == NULL) {
sc->sc_status = 0; /* clear status bit */ sc->sc_status = 0; /* clear status bit */