diff options
Diffstat (limited to 'sys/dev/usb/umidi.c')
-rw-r--r-- | sys/dev/usb/umidi.c | 489 |
1 files changed, 233 insertions, 256 deletions
diff --git a/sys/dev/usb/umidi.c b/sys/dev/usb/umidi.c index 266b719820d..9188f8fb55d 100644 --- a/sys/dev/usb/umidi.c +++ b/sys/dev/usb/umidi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: umidi.c,v 1.9 2003/05/19 00:36:33 nate Exp $ */ +/* $OpenBSD: umidi.c,v 1.10 2004/06/27 19:44:48 deraadt Exp $ */ /* $NetBSD: umidi.c,v 1.16 2002/07/11 21:14:32 augustss Exp $ */ /* * Copyright (c) 2001 The NetBSD Foundation, Inc. @@ -123,7 +123,7 @@ static usbd_status start_output_transfer(struct umidi_endpoint *); static int out_jack_output(struct umidi_jack *, int); static void in_intr(usbd_xfer_handle, usbd_private_handle, usbd_status); static void out_intr(usbd_xfer_handle, usbd_private_handle, usbd_status); -static void out_build_packet(int, struct umidi_packet *, uByte); +static int out_build_packet(int, struct umidi_packet *, uByte); struct midi_hw_if umidi_hw_if = { @@ -163,6 +163,7 @@ USB_ATTACH(umidi) usbd_status err; USB_ATTACH_START(umidi, sc, uaa); char devinfo[1024]; + int i; DPRINTFN(1,("umidi_attach\n")); @@ -180,15 +181,11 @@ USB_ATTACH(umidi) err = alloc_all_endpoints(sc); if (err!=USBD_NORMAL_COMPLETION) { - printf("%s: alloc_all_endpoints failed. (err=%d)\n", - USBDEVNAME(sc->sc_dev), err); goto error; } err = alloc_all_jacks(sc); if (err!=USBD_NORMAL_COMPLETION) { free_all_endpoints(sc); - printf("%s: alloc_all_jacks failed. (err=%d)\n", - USBDEVNAME(sc->sc_dev), err); goto error; } printf("%s: out=%d, in=%d\n", @@ -200,22 +197,22 @@ USB_ATTACH(umidi) unbind_all_jacks(sc); free_all_jacks(sc); free_all_endpoints(sc); - printf("%s: assign_all_jacks_automatically failed. (err=%d)\n", - USBDEVNAME(sc->sc_dev), err); goto error; } err = attach_all_mididevs(sc); if (err!=USBD_NORMAL_COMPLETION) { free_all_jacks(sc); free_all_endpoints(sc); - printf("%s: attach_all_mididevs failed. (err=%d)\n", - USBDEVNAME(sc->sc_dev), err); } #ifdef UMIDI_DEBUG dump_sc(sc); #endif + for (i = 0; i < sc->sc_in_num_endpoints; i++) { + (void)start_input_transfer(&sc->sc_in_ep[i]); + } + usbd_add_drv_event(USB_EVENT_DRIVER_ATTACH, sc->sc_udev, USBDEV(sc->sc_dev)); @@ -328,7 +325,6 @@ void umidi_getinfo(void *addr, struct midi_info *mi) { struct umidi_mididev *mididev = addr; -/* struct umidi_softc *sc = mididev->sc; */ mi->name = "USB MIDI I/F"; /* XXX: model name */ mi->props = MIDI_PROP_OUT_INTR; @@ -349,23 +345,22 @@ alloc_pipe(struct umidi_endpoint *ep) usbd_status err; DPRINTF(("%s: alloc_pipe %p\n", USBDEVNAME(sc->sc_dev), ep)); - LIST_INIT(&ep->queue_head); + TAILQ_INIT(&ep->queue_head); ep->xfer = usbd_alloc_xfer(sc->sc_udev); if (ep->xfer == NULL) { - err = USBD_NOMEM; - goto quit; + return USBD_NOMEM; } - ep->buffer = usbd_alloc_buffer(ep->xfer, UMIDI_PACKET_SIZE); + ep->buffer = usbd_alloc_buffer(ep->xfer, ep->packetsize); if (ep->buffer == NULL) { - usbd_free_xfer(ep->xfer); - err = USBD_NOMEM; - goto quit; + usbd_free_xfer(ep->xfer); + return USBD_NOMEM; } err = usbd_open_pipe(sc->sc_iface, ep->addr, 0, &ep->pipe); - if (err) - usbd_free_xfer(ep->xfer); -quit: - return err; + if (err != USBD_NORMAL_COMPLETION) { + usbd_free_xfer(ep->xfer); + return err; + } + return USBD_NORMAL_COMPLETION; } static void @@ -403,14 +398,17 @@ alloc_all_endpoints(struct umidi_softc *sc) ep = sc->sc_endpoints; for (i=sc->sc_out_num_endpoints+sc->sc_in_num_endpoints; i>0; i--) { - err = alloc_pipe(ep++); + err = alloc_pipe(ep); if (err!=USBD_NORMAL_COMPLETION) { - for (; ep!=sc->sc_endpoints; ep--) - free_pipe(ep-1); + while(ep != sc->sc_endpoints) { + ep--; + free_pipe(ep); + } free(sc->sc_endpoints, M_USBDEV); sc->sc_endpoints = sc->sc_out_ep = sc->sc_in_ep = NULL; break; } + ep++; } return err; } @@ -459,8 +457,8 @@ alloc_all_endpoints_fixed_ep(struct umidi_softc *sc) sc->sc_iface, fp->out_ep[i].ep); if (!epd) { - printf("%s: cannot get endpoint descriptor(out:%d)\n", - USBDEVNAME(sc->sc_dev), fp->out_ep[i].ep); + DPRINTF(("%s: cannot get endpoint descriptor(out:%d)\n", + USBDEVNAME(sc->sc_dev), fp->out_ep[i].ep)); err = USBD_INVAL; goto error; } @@ -472,12 +470,13 @@ alloc_all_endpoints_fixed_ep(struct umidi_softc *sc) goto error; } ep->sc = sc; + ep->packetsize = UGETW(epd->wMaxPacketSize); ep->addr = epd->bEndpointAddress; ep->num_jacks = fp->out_ep[i].num_jacks; sc->sc_out_num_jacks += fp->out_ep[i].num_jacks; ep->num_open = 0; memset(ep->jacks, 0, sizeof(ep->jacks)); - LIST_INIT(&ep->queue_head); + TAILQ_INIT(&ep->queue_head); ep++; } ep = &sc->sc_in_ep[0]; @@ -486,8 +485,8 @@ alloc_all_endpoints_fixed_ep(struct umidi_softc *sc) sc->sc_iface, fp->in_ep[i].ep); if (!epd) { - printf("%s: cannot get endpoint descriptor(in:%d)\n", - USBDEVNAME(sc->sc_dev), fp->in_ep[i].ep); + DPRINTF(("%s: cannot get endpoint descriptor(in:%d)\n", + USBDEVNAME(sc->sc_dev), fp->in_ep[i].ep)); err = USBD_INVAL; goto error; } @@ -500,6 +499,7 @@ alloc_all_endpoints_fixed_ep(struct umidi_softc *sc) } ep->sc = sc; ep->addr = epd->bEndpointAddress; + ep->packetsize = UGETW(epd->wMaxPacketSize); ep->num_jacks = fp->in_ep[i].num_jacks; sc->sc_in_num_jacks += fp->in_ep[i].num_jacks; ep->num_open = 0; @@ -520,7 +520,7 @@ alloc_all_endpoints_yamaha(struct umidi_softc *sc) /* This driver currently supports max 1in/1out bulk endpoints */ usb_descriptor_t *desc; usb_endpoint_descriptor_t *epd; - int out_addr, in_addr, i; + int out_addr, in_addr, in_packetsize, i; int dir; size_t remain, descsize; @@ -535,8 +535,10 @@ alloc_all_endpoints_yamaha(struct umidi_softc *sc) dir = UE_GET_DIR(epd->bEndpointAddress); if (dir==UE_DIR_OUT && !out_addr) out_addr = epd->bEndpointAddress; - else if (dir==UE_DIR_IN && !in_addr) + else if (dir==UE_DIR_IN && !in_addr) { in_addr = epd->bEndpointAddress; + in_packetsize = UGETW(epd->wMaxPacketSize); + } } } desc = NEXT_D(desc); @@ -591,6 +593,7 @@ alloc_all_endpoints_yamaha(struct umidi_softc *sc) sc->sc_out_ep = sc->sc_endpoints; sc->sc_out_ep->sc = sc; sc->sc_out_ep->addr = out_addr; + sc->sc_out_ep->packetsize = UMIDI_PACKET_SIZE; sc->sc_out_ep->num_jacks = sc->sc_out_num_jacks; sc->sc_out_ep->num_open = 0; memset(sc->sc_out_ep->jacks, 0, sizeof(sc->sc_out_ep->jacks)); @@ -601,6 +604,7 @@ alloc_all_endpoints_yamaha(struct umidi_softc *sc) sc->sc_in_ep = sc->sc_endpoints+sc->sc_out_num_endpoints; sc->sc_in_ep->sc = sc; sc->sc_in_ep->addr = in_addr; + sc->sc_in_ep->packetsize = in_packetsize; sc->sc_in_ep->num_jacks = sc->sc_in_num_jacks; sc->sc_in_ep->num_open = 0; memset(sc->sc_in_ep->jacks, 0, sizeof(sc->sc_in_ep->jacks)); @@ -613,20 +617,17 @@ alloc_all_endpoints_yamaha(struct umidi_softc *sc) static usbd_status alloc_all_endpoints_genuine(struct umidi_softc *sc) { + usb_interface_descriptor_t *interface_desc; + usb_config_descriptor_t *config_desc; usb_descriptor_t *desc; int num_ep; size_t remain, descsize; struct umidi_endpoint *p, *q, *lowest, *endep, tmpep; - int epaddr; + int epaddr, eppacketsize; - desc = TO_D(usbd_get_interface_descriptor(sc->sc_iface)); - num_ep = TO_IFD(desc)->bNumEndpoints; - desc = NEXT_D(desc); /* ifd -> csifd */ - remain = ((size_t)UGETW(TO_CSIFD(desc)->wTotalLength) - - (size_t)desc->bLength); - desc = NEXT_D(desc); - - sc->sc_endpoints = p = malloc(sizeof(struct umidi_endpoint)*num_ep, + interface_desc = usbd_get_interface_descriptor(sc->sc_iface); + num_ep = interface_desc->bNumEndpoints; + sc->sc_endpoints = p = malloc(sizeof(struct umidi_endpoint) * num_ep, M_USBDEV, M_WAITOK); if (!p) return USBD_NOMEM; @@ -636,6 +637,9 @@ alloc_all_endpoints_genuine(struct umidi_softc *sc) epaddr = -1; /* get the list of endpoints for midi stream */ + config_desc = usbd_get_config_descriptor(sc->sc_udev); + desc = (usb_descriptor_t *) config_desc; + remain = (size_t)UGETW(config_desc->wTotalLength); while (remain>=sizeof(usb_descriptor_t)) { descsize = desc->bLength; if (descsize>remain || descsize==0) @@ -644,6 +648,7 @@ alloc_all_endpoints_genuine(struct umidi_softc *sc) remain>=USB_ENDPOINT_DESCRIPTOR_SIZE && UE_GET_XFERTYPE(TO_EPD(desc)->bmAttributes) == UE_BULK) { epaddr = TO_EPD(desc)->bEndpointAddress; + eppacketsize = UGETW(TO_EPD(desc)->wMaxPacketSize); } else if (desc->bDescriptorType==UDESC_CS_ENDPOINT && remain>=UMIDI_CS_ENDPOINT_DESCRIPTOR_SIZE && epaddr!=-1) { @@ -651,6 +656,7 @@ alloc_all_endpoints_genuine(struct umidi_softc *sc) num_ep--; p->sc = sc; p->addr = epaddr; + p->packetsize = eppacketsize; p->num_jacks = TO_CSEPD(desc)->bNumEmbMIDIJack; if (UE_GET_DIR(epaddr)==UE_DIR_OUT) { sc->sc_out_num_endpoints++; @@ -860,8 +866,6 @@ assign_all_jacks_automatically(struct umidi_softc *sc) static usbd_status open_out_jack(struct umidi_jack *jack, void *arg, void (*intr)(void *)) { - struct umidi_endpoint *ep = jack->endpoint; - if (jack->opened) return USBD_IN_USE; @@ -869,59 +873,34 @@ open_out_jack(struct umidi_jack *jack, void *arg, void (*intr)(void *)) jack->u.out.intr = intr; init_packet(&jack->packet); jack->opened = 1; - ep->num_open++; - + jack->endpoint->num_open++; + return USBD_NORMAL_COMPLETION; } static usbd_status open_in_jack(struct umidi_jack *jack, void *arg, void (*intr)(void *, int)) { - usbd_status err = USBD_NORMAL_COMPLETION; - struct umidi_endpoint *ep = jack->endpoint; - if (jack->opened) return USBD_IN_USE; jack->arg = arg; jack->u.in.intr = intr; jack->opened = 1; - if (ep->num_open++==0 && UE_GET_DIR(ep->addr)==UE_DIR_IN) { - err = start_input_transfer(ep); - if (err!=USBD_NORMAL_COMPLETION) { - ep->num_open--; - } - } - - return err; + jack->endpoint->num_open++; + + return USBD_NORMAL_COMPLETION; } static void close_out_jack(struct umidi_jack *jack) { - struct umidi_jack *tail; - int s; - if (jack->opened) { - s = splusb(); - LIST_FOREACH(tail, - &jack->endpoint->queue_head, - u.out.queue_entry) - if (tail == jack) { - LIST_REMOVE(jack, u.out.queue_entry); - break; - } - if (jack == jack->endpoint->queue_tail) { - /* find tail */ - LIST_FOREACH(tail, - &jack->endpoint->queue_head, - u.out.queue_entry) { - if (!LIST_NEXT(tail, u.out.queue_entry)) { - jack->endpoint->queue_tail = tail; - } - } +#ifdef UMIDI_DEBUG + if (!TAILQ_EMPTY(&jack->endpoint->queue_head)) { + printf("close_out_jack: queue_head still not empty\n"); } - splx(s); +#endif jack->opened = 0; jack->endpoint->num_open--; } @@ -932,7 +911,7 @@ close_in_jack(struct umidi_jack *jack) { if (jack->opened) { jack->opened = 0; - jack->endpoint->num_open--; + jack->endpoint->num_open--; } } @@ -1109,28 +1088,6 @@ static const int packet_length[16] = { /*F*/ 1, }; -static const struct { - int cin; - packet_state_t next; -} packet_0xFX[16] = { - /*F0: SysEx */ { 0x04, PS_EXCL_1 }, - /*F1: MTC */ { 0x02, PS_NORMAL_1OF2 }, - /*F2: S.POS */ { 0x03, PS_NORMAL_1OF3 }, - /*F3: S.SEL */ { 0x02, PS_NORMAL_1OF2 }, - /*F4: UNDEF */ { 0x00, PS_INITIAL }, - /*F5: UNDEF */ { 0x00, PS_INITIAL }, - /*F6: Tune */ { 0x0F, PS_END }, - /*F7: EofEx */ { 0x00, PS_INITIAL }, - /*F8: Timing */ { 0x0F, PS_END }, - /*F9: UNDEF */ { 0x00, PS_INITIAL }, - /*FA: Start */ { 0x0F, PS_END }, - /*FB: Cont */ { 0x0F, PS_END }, - /*FC: Stop */ { 0x0F, PS_END }, - /*FD: UNDEF */ { 0x00, PS_INITIAL }, - /*FE: ActS */ { 0x0F, PS_END }, - /*FF: Reset */ { 0x0F, PS_END }, -}; - #define GET_CN(p) (((unsigned char)(p)>>4)&0x0F) #define GET_CIN(p) ((unsigned char)(p)&0x0F) #define MIX_CN_CIN(cn, cin) \ @@ -1140,33 +1097,46 @@ static const struct { static void init_packet(struct umidi_packet *packet) { - memset(packet->buffer, 0, UMIDI_PACKET_SIZE); - packet->state = PS_INITIAL; + packet->status = 0; + packet->index = 0; } static usbd_status start_input_transfer(struct umidi_endpoint *ep) { + usbd_status err; usbd_setup_xfer(ep->xfer, ep->pipe, (usbd_private_handle)ep, - ep->buffer, UMIDI_PACKET_SIZE, - USBD_NO_COPY, USBD_NO_TIMEOUT, in_intr); - return usbd_transfer(ep->xfer); + ep->buffer, ep->packetsize, + USBD_SHORT_XFER_OK | USBD_NO_COPY, USBD_NO_TIMEOUT, in_intr); + err = usbd_transfer(ep->xfer); + if (err != USBD_NORMAL_COMPLETION && err != USBD_IN_PROGRESS) { + DPRINTF(("%s: start_input_transfer: usbd_transfer() failed err=%s\n", + USBDEVNAME(ep->sc->sc_dev), usbd_errstr(err))); + return err; + } + return USBD_NORMAL_COMPLETION; } static usbd_status start_output_transfer(struct umidi_endpoint *ep) { + usbd_status err; usbd_setup_xfer(ep->xfer, ep->pipe, (usbd_private_handle)ep, ep->buffer, UMIDI_PACKET_SIZE, USBD_NO_COPY, USBD_NO_TIMEOUT, out_intr); - return usbd_transfer(ep->xfer); + err = usbd_transfer(ep->xfer); + if (err != USBD_NORMAL_COMPLETION && err != USBD_IN_PROGRESS) { + DPRINTF(("%s: start_output_transfer: usbd_transfer() failed err=%s\n", + USBDEVNAME(ep->sc->sc_dev), usbd_errstr(err))); + return err; + } + return USBD_NORMAL_COMPLETION; } #ifdef UMIDI_DEBUG #define DPR_PACKET(dir, sc, p) \ -if ((unsigned char)(p)->buffer[1]!=0xFE) \ DPRINTFN(500, \ ("%s: umidi packet(" #dir "): %02X %02X %02X %02X\n", \ USBDEVNAME(sc->sc_dev), \ @@ -1183,76 +1153,74 @@ out_jack_output(struct umidi_jack *out_jack, int d) { struct umidi_endpoint *ep = out_jack->endpoint; struct umidi_softc *sc = ep->sc; - int error; int s; if (sc->sc_dying) return EIO; - error = 0; - if (out_jack->opened) { - DPRINTFN(1000, ("umidi_output: ep=%p 0x%02x\n", ep, d)); - out_build_packet(out_jack->cable_number, &out_jack->packet, d); - switch (out_jack->packet.state) { - case PS_EXCL_0: - case PS_END: - DPR_PACKET(out, sc, &out_jack->packet); - s = splusb(); - if (LIST_EMPTY(&ep->queue_head)) { - memcpy(ep->buffer, - out_jack->packet.buffer, - UMIDI_PACKET_SIZE); - start_output_transfer(ep); - } - if (LIST_EMPTY(&ep->queue_head)) - LIST_INSERT_HEAD(&ep->queue_head, - out_jack, u.out.queue_entry); - else - LIST_INSERT_AFTER(ep->queue_tail, - out_jack, u.out.queue_entry); - ep->queue_tail = out_jack; - splx(s); - break; - default: - error = EINPROGRESS; + if (!out_jack->opened) { + return ENODEV; + } + + if (out_build_packet(out_jack->cable_number, &out_jack->packet, d)) { + DPR_PACKET(out, sc, &out_jack->packet); + s = splusb(); + if (TAILQ_EMPTY(&ep->queue_head)) { + memcpy(ep->buffer, + out_jack->packet.buffer, + UMIDI_PACKET_SIZE); + TAILQ_INSERT_TAIL(&ep->queue_head, + out_jack, u.out.queue_entry); + start_output_transfer(ep); + } else { + DPRINTF(("%s: out_jack_output: packet ignored\n", USBDEVNAME(sc->sc_dev))); } - } else - error = ENODEV; - - return error; + splx(s); + return 0; + } + + return EINPROGRESS; } static void in_intr(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status) { - int cn, len, i; + int cn, evlen, remain, i; + unsigned char *buf; struct umidi_endpoint *ep = (struct umidi_endpoint *)priv; struct umidi_jack *jack; - if (ep->sc->sc_dying || !ep->num_open) + if (ep->sc->sc_dying) return; - cn = GET_CN(ep->buffer[0]); - len = packet_length[GET_CIN(ep->buffer[0])]; - jack = ep->jacks[cn]; - if (cn>=ep->num_jacks || !jack) { - DPRINTF(("%s: stray umidi packet (in): %02X %02X %02X %02X\n", - USBDEVNAME(ep->sc->sc_dev), - (unsigned)ep->buffer[0], - (unsigned)ep->buffer[1], - (unsigned)ep->buffer[2], - (unsigned)ep->buffer[3])); - return; - } - if (!jack->binded || !jack->opened) - return; - DPR_PACKET(in, ep->sc, &jack->packet); - if (jack->u.in.intr) { - for (i=0; i<len; i++) { - (*jack->u.in.intr)(jack->arg, ep->buffer[i+1]); - } + usbd_get_xfer_status(xfer, NULL, NULL, &remain, NULL); + if (status != USBD_NORMAL_COMPLETION) { + DPRINTF(("umidi: in_intr: abnormal status: %s\n", usbd_errstr(status))); + goto quit; + } + buf = ep->buffer; + while (remain >= UMIDI_PACKET_SIZE) { + cn = GET_CN(buf[0]); + if (cn < ep->num_jacks && (jack = ep->jacks[cn]) && + jack->binded && jack->opened && jack->u.in.intr) { + evlen = packet_length[GET_CIN(buf[0])]; + for (i=0; i<evlen; i++) + (*jack->u.in.intr)(jack->arg, buf[i+1]); + } else + DPRINTFN(10, ("in_intr: unused packet %02x %02x %02x %02x\n", + buf[0], buf[1], buf[2], buf[3])); + + buf += UMIDI_PACKET_SIZE; + remain -= UMIDI_PACKET_SIZE; } +#ifdef UMIDI_DEBUG + if (remain != 0) { + DPRINTF(("umidi: in_intr: remain != 0\n")); + } +#endif + +quit: (void)start_input_transfer(ep); } @@ -1262,120 +1230,129 @@ out_intr(usbd_xfer_handle xfer, usbd_private_handle priv, usbd_status status) struct umidi_endpoint *ep = (struct umidi_endpoint *)priv; struct umidi_softc *sc = ep->sc; struct umidi_jack *jack; - - if (sc->sc_dying || !ep->num_open) + int s; + + if (sc->sc_dying) return; - jack = LIST_FIRST(&ep->queue_head); - if (jack && jack->opened) { - LIST_REMOVE(jack, u.out.queue_entry); - if (!LIST_EMPTY(&ep->queue_head)) { + s = splusb(); + jack = TAILQ_FIRST(&ep->queue_head); + if (jack) { + TAILQ_REMOVE(&ep->queue_head, jack, u.out.queue_entry); + if (!TAILQ_EMPTY(&ep->queue_head)) { memcpy(ep->buffer, - LIST_FIRST(&ep->queue_head)->packet.buffer, + TAILQ_FIRST(&ep->queue_head)->packet.buffer, UMIDI_PACKET_SIZE); (void)start_output_transfer(ep); } - if (jack->u.out.intr) { + if (jack->opened && jack->u.out.intr) (*jack->u.out.intr)(jack->arg); - } } + splx(s); } -static void -out_build_packet(int cable_number, struct umidi_packet *packet, uByte in) +#define UMIDI_VOICELEN(status) (umidi_evlen[((status) >> 4) & 7]) +unsigned umidi_evlen[] = { 4, 4, 4, 4, 3, 3, 4 }; + +#define EV_SYSEX 0xf0 +#define EV_MTC 0xf1 +#define EV_SPP 0xf2 +#define EV_SONGSEL 0xf3 +#define EV_TUNE_REQ 0xf6 +#define EV_SYSEX_STOP 0xf7 + +static int +out_build_packet(int cable_number, struct umidi_packet *packet, uByte data) { - int cin; - uByte prev; - -retry: - switch (packet->state) { - case PS_END: - case PS_INITIAL: - prev = packet->buffer[1]; - memset(packet->buffer, 0, UMIDI_PACKET_SIZE); - if (in<0x80) { - if (prev>=0x80 && prev<0xf0) { - /* running status */ - out_build_packet(cable_number, packet, prev); - goto retry; - } - /* ??? */ + if (data >= 0xf8) { /* is it a realtime message ? */ + packet->buffer[0] = data >> 4 | cable_number << 4; + packet->buffer[1] = data; + packet->buffer[2] = 0; + packet->buffer[3] = 0; + return 1; + } + if (data >= 0xf0) { /* is it a common message ? */ + switch(data) { + case EV_SYSEX: + packet->buffer[1] = packet->status = data; + packet->index = 2; + break; + case EV_SYSEX_STOP: + if (packet->status != EV_SYSEX) break; + packet->status = data; + packet->buffer[packet->index++] = data; + packet->buffer[0] = (0x4 - 1 + packet->index) | cable_number << 4; + goto packetready; + case EV_TUNE_REQ: + packet->status = data; + packet->buffer[0] = 0x5 | cable_number << 4; + packet->index = 1; + goto packetready; + default: + packet->status = data; break; } - if (in>=0xf0) { - cin=packet_0xFX[in&0x0F].cin; - packet->state=packet_0xFX[in&0x0F].next; - } else { - cin=(unsigned char)in>>4; - switch (packet_length[cin]) { - case 2: - packet->state = PS_NORMAL_1OF2; - break; - case 3: - packet->state = PS_NORMAL_1OF3; - break; - default: - /* ??? */ - packet->state = PS_INITIAL; + return 0; + } + if (data >= 0x80) { /* is it a voice message ? */ + packet->status = data; + packet->index = 0; + return 0; + } + + /* else it is a data byte */ + if (packet->status >= 0xf0) { + switch(packet->status) { + case EV_SYSEX: /* sysex starts or continues */ + if (packet->index == 0) + packet->index = 1; + + packet->buffer[packet->index++] = data; + if (packet->index >= UMIDI_PACKET_SIZE) { + packet->buffer[0] = 0x4 | cable_number << 4; + goto packetready; } - } - packet->buffer[0] = MIX_CN_CIN(cable_number, cin); - packet->buffer[1] = in; - break; - case PS_NORMAL_1OF3: - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[2] = in; - packet->state = PS_NORMAL_2OF3; - break; - case PS_NORMAL_2OF3: - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[3] = in; - packet->state = PS_END; - break; - case PS_NORMAL_1OF2: - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[2] = in; - packet->state = PS_END; - break; - case PS_EXCL_0: - memset(packet->buffer, 0, UMIDI_PACKET_SIZE); - if (in==0xF7) { - packet->buffer[0] = MIX_CN_CIN(cable_number, 0x05); - packet->buffer[1] = 0xF7; - packet->state = PS_END; break; - } - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[1] = in; - packet->state = PS_EXCL_1; - break; - case PS_EXCL_1: - if (in==0xF7) { - packet->buffer[0] = MIX_CN_CIN(cable_number, 0x06); - packet->buffer[2] = 0xF7; - packet->state = PS_END; + case EV_MTC: /* messages with 1 data byte */ + case EV_SONGSEL: + packet->buffer[0] = 0x2 | cable_number << 4; + packet->buffer[1] = packet->status; + packet->buffer[2] = data; + packet->index = 3; + goto packetready; + case EV_SPP: /* messages with 2 data bytes */ + if (packet->index == 0) { + packet->buffer[0] = 0x3 | cable_number << 4; + packet->index = 1; + } + packet->buffer[packet->index++] = data; + if (packet->index >= UMIDI_PACKET_SIZE) { + packet->buffer[1] = packet->status; + goto packetready; + } break; - } - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[2] = in; - packet->state = PS_EXCL_2; - break; - case PS_EXCL_2: - if (in==0xF7) { - packet->buffer[0] = MIX_CN_CIN(cable_number, 0x07); - packet->buffer[3] = 0xF7; - packet->state = PS_END; + default: /* ignore data with unknown status */ break; } - if (in>=0x80) { /* ??? */ packet->state = PS_END; break; } - packet->buffer[0] = MIX_CN_CIN(cable_number, 0x04); - packet->buffer[3] = in; - packet->state = PS_EXCL_0; - break; - default: - printf("umidi: ambiguous state.\n"); - packet->state = PS_INITIAL; - goto retry; + return 0; + } + if (packet->status >= 0x80) { /* is it a voice message ? */ + if (packet->index == 0) { + packet->buffer[0] = packet->status >> 4 | cable_number << 4; + packet->buffer[1] = packet->status; + packet->index = 2; + } + packet->buffer[packet->index++] = data; + if (packet->index >= UMIDI_VOICELEN(packet->status)) + goto packetready; } + /* ignore data with unknown status */ + return 0; + +packetready: + while (packet->index < UMIDI_PACKET_SIZE) + packet->buffer[packet->index++] = 0; + + packet->index = 0; + return 1; } - |