/* $OpenBSD: sock.c,v 1.20 2009/07/25 10:52:19 ratchov Exp $ */ /* * Copyright (c) 2008 Alexandre Ratchov * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ /* * TODO: * * change f->bufsz to contain only socket-side buffer, * because it's less error prone */ #include #include #include #include "abuf.h" #include "aproc.h" #include "conf.h" #include "dev.h" #include "midi.h" #include "opt.h" #include "sock.h" int sock_attach(struct sock *, int); int sock_read(struct sock *); int sock_write(struct sock *); int sock_execmsg(struct sock *); void sock_reset(struct sock *); struct fileops sock_ops = { "sock", sizeof(struct sock), pipe_close, pipe_read, pipe_write, NULL, /* start */ NULL, /* stop */ pipe_nfds, pipe_pollfd, pipe_revents }; void rsock_done(struct aproc *p) { struct sock *f = (struct sock *)p->u.io.file; DPRINTFN(1, "rsock_done: %p\n", f); if (f == NULL) return; sock_reset(f); f->pipe.file.rproc = NULL; if (f->pipe.file.wproc) { aproc_del(f->pipe.file.wproc); file_del(&f->pipe.file); } p->u.io.file = NULL; } int rsock_in(struct aproc *p, struct abuf *ibuf_dummy) { struct sock *f = (struct sock *)p->u.io.file; struct abuf *obuf; DPRINTFN(4, "rsock_in: %p\n", f); if (!sock_read(f)) return 0; obuf = LIST_FIRST(&p->obuflist); if (obuf) { if (!abuf_flush(obuf)) return 0; } return 1; } int rsock_out(struct aproc *p, struct abuf *obuf) { struct sock *f = (struct sock *)p->u.io.file; if (f->pipe.file.state & FILE_RINUSE) return 0; DPRINTFN(4, "rsock_out: %p\n", f); /* * When calling sock_read(), we may receive a ``STOP'' command, * and detach ``obuf''. In this case, there's no more caller and * we'll stop processing further messages, resulting in a deadlock. * The solution is to iterate over sock_read() in order to * consume all messages(). */ for (;;) { if (!sock_read(f)) return 0; } return 1; } void rsock_eof(struct aproc *p, struct abuf *ibuf_dummy) { DPRINTFN(3, "rsock_eof: %p\n", p->u.io.file); aproc_del(p); } void rsock_hup(struct aproc *p, struct abuf *ibuf) { DPRINTFN(3, "rsock_hup: %p\n", p->u.io.file); aproc_del(p); } void rsock_opos(struct aproc *p, struct abuf *obuf, int delta) { struct sock *f = (struct sock *)p->u.io.file; if (f->mode & AMSG_REC) return; f->delta += delta; DPRINTFN(3, "rsock_opos: %p: delta = %d, f->delta = %d\n", f, delta, f->delta); /* * Negative deltas are xrun notifications for internal uses * only. Don't generate a packet for this, the client will be * notified later. */ if (delta < 0) return; f->tickpending++; for (;;) { if (!sock_write(f)) break; } } struct aproc_ops rsock_ops = { "rsock", rsock_in, rsock_out, rsock_eof, rsock_hup, NULL, /* newin */ NULL, /* newout */ NULL, /* ipos */ rsock_opos, rsock_done }; void wsock_done(struct aproc *p) { struct sock *f = (struct sock *)p->u.io.file; DPRINTFN(1, "wsock_done: %p\n", f); if (f == NULL) return; sock_reset(f); f->pipe.file.wproc = NULL; if (f->pipe.file.rproc) { aproc_del(f->pipe.file.rproc); file_del(&f->pipe.file); } p->u.io.file = NULL; } int wsock_in(struct aproc *p, struct abuf *ibuf) { struct sock *f = (struct sock *)p->u.io.file; if (f->pipe.file.state & FILE_WINUSE) return 0; DPRINTFN(4, "wsock_in: %p\n", f); /* * See remark in rsock_out(). */ for (;;) { if (!sock_write(f)) return 0; } return 1; } int wsock_out(struct aproc *p, struct abuf *obuf_dummy) { struct abuf *ibuf = LIST_FIRST(&p->ibuflist); struct sock *f = (struct sock *)p->u.io.file; DPRINTFN(3, "wsock_out: %p\n", f); if (ibuf) { DPRINTFN(3, "wsock_out: %p, filling ibuf\n", f); if (!abuf_fill(ibuf)) return 0; } if (!sock_write(f)) return 0; return 1; } void wsock_eof(struct aproc *p, struct abuf *obuf) { DPRINTFN(3, "wsock_eof: %p\n", p->u.io.file); aproc_del(p); } void wsock_hup(struct aproc *p, struct abuf *obuf_dummy) { DPRINTFN(3, "wsock_hup: %p\n", p->u.io.file); aproc_del(p); } void wsock_ipos(struct aproc *p, struct abuf *obuf, int delta) { struct sock *f = (struct sock *)p->u.io.file; if (!(f->mode & AMSG_REC)) return; f->delta += delta; DPRINTFN(3, "wsock_ipos: %p, delta = %d, f->delta = %d\n", f, delta, f->delta); /* * Negative deltas are xrun notifications for internal uses * only. Don't generate a packet for this, the client will be * notified later. */ if (delta < 0) return; f->tickpending++; for (;;) { if (!sock_write(f)) break; } } struct aproc_ops wsock_ops = { "wsock", wsock_in, wsock_out, wsock_eof, wsock_hup, NULL, /* newin */ NULL, /* newout */ wsock_ipos, NULL, /* opos */ wsock_done }; /* * Initialise socket in the SOCK_HELLO state with default * parameters. */ struct sock * sock_new(struct fileops *ops, int fd) { struct aproc *rproc, *wproc; struct sock *f; f = (struct sock *)pipe_new(ops, fd, "sock"); if (f == NULL) return NULL; f->pstate = SOCK_HELLO; f->mode = 0; f->opt = opt_byname("default"); if (f->opt) { if (dev_rec) f->wpar = f->opt->wpar; if (dev_play) f->rpar = f->opt->rpar; } f->xrun = AMSG_IGNORE; f->bufsz = dev_bufsz; f->round = dev_round; f->delta = 0; f->tickpending = 0; f->vol = ADATA_UNIT; wproc = aproc_new(&wsock_ops, f->pipe.file.name); wproc->u.io.file = &f->pipe.file; f->pipe.file.wproc = wproc; f->wstate = SOCK_WIDLE; f->wtodo = 0xdeadbeef; rproc = aproc_new(&rsock_ops, f->pipe.file.name); rproc->u.io.file = &f->pipe.file; f->pipe.file.rproc = rproc; f->rstate = SOCK_RMSG; f->rtodo = sizeof(struct amsg); return f; } /* * Free buffers. */ void sock_freebuf(struct sock *f) { struct abuf *rbuf, *wbuf; f->pstate = SOCK_INIT; DPRINTF("sock_freebuf:\n"); rbuf = LIST_FIRST(&f->pipe.file.rproc->obuflist); if (rbuf) abuf_eof(rbuf); wbuf = LIST_FIRST(&f->pipe.file.wproc->ibuflist); if (wbuf) abuf_hup(wbuf); } /* * Allocate buffers, so client can start filling write-end. */ void sock_allocbuf(struct sock *f) { struct abuf *rbuf = NULL, *wbuf = NULL; if (f->mode & AMSG_PLAY) { rbuf = abuf_new(f->bufsz, &f->rpar); aproc_setout(f->pipe.file.rproc, rbuf); } if (f->mode & AMSG_REC) { wbuf = abuf_new(f->bufsz, &f->wpar); aproc_setin(f->pipe.file.wproc, wbuf); } f->delta = 0; f->tickpending = 0; DPRINTF("sock_allocbuf: %p, using %u frames buffer\n", f, f->bufsz); f->pstate = SOCK_START; if (!(f->mode & AMSG_PLAY)) (void)sock_attach(f, 0); } /* * Set volume. */ void sock_setvol(struct sock *f, int vol) { struct abuf *rbuf; f->vol = vol; rbuf = LIST_FIRST(&f->pipe.file.rproc->obuflist); if (!rbuf) { DPRINTF("sock_setvol: no read buffer yet\n"); return; } dev_setvol(rbuf, vol); } /* * Attach play and/or record buffers to dev_mix and/or dev_sub. */ int sock_attach(struct sock *f, int force) { struct abuf *rbuf, *wbuf; rbuf = LIST_FIRST(&f->pipe.file.rproc->obuflist); wbuf = LIST_FIRST(&f->pipe.file.wproc->ibuflist); /* * If in SOCK_START state, dont attach until * the buffer isn't completely filled. */ if (!force && rbuf && ABUF_WOK(rbuf)) return 0; DPRINTF("sock_attach: %p\n", f); f->pstate = SOCK_RUN; /* * Attach them to the device. */ dev_attach(f->pipe.file.name, (f->mode & AMSG_PLAY) ? rbuf : NULL, &f->rpar, f->xrun, (f->mode & AMSG_REC) ? wbuf : NULL, &f->wpar, f->xrun, f->opt->maxweight); if (f->mode & AMSG_PLAY) dev_setvol(rbuf, f->vol); /* * Send the initial position, if needed. */ for (;;) { if (!sock_write(f)) break; } return 1; } void sock_reset(struct sock *f) { switch (f->pstate) { case SOCK_START: (void)sock_attach(f, 1); f->pstate = SOCK_RUN; /* PASSTHROUGH */ case SOCK_RUN: sock_freebuf(f); f->pstate = SOCK_INIT; /* PASSTHROUGH */ case SOCK_INIT: /* nothing yet */ break; } } /* * Read a message from the file descriptor, return 1 if done, 0 * otherwise. The message is stored in f->rmsg. */ int sock_rmsg(struct sock *f) { unsigned count; unsigned char *data; while (f->rtodo > 0) { if (!(f->pipe.file.state & FILE_ROK)) { DPRINTFN(4, "sock_rmsg: blk, rtodo = %u\n", f->rtodo); return 0; } data = (unsigned char *)&f->rmsg; data += sizeof(struct amsg) - f->rtodo; count = file_read(&f->pipe.file, data, f->rtodo); if (count == 0) return 0; f->rtodo -= count; } DPRINTFN(4, "sock_rmsg: %p: done\n", f); return 1; } /* * Write a message to the file descriptor, return 1 if done, 0 * otherwise. The "m" argument is f->rmsg or f->wmsg, and the "ptodo" * points to the f->rtodo or f->wtodo respectively. */ int sock_wmsg(struct sock *f, struct amsg *m, unsigned *ptodo) { unsigned count; unsigned char *data; while (*ptodo > 0) { if (!(f->pipe.file.state & FILE_WOK)) { DPRINTFN(4, "sock_wmsg: blk, *ptodo = %u\n", *ptodo); return 0; } data = (unsigned char *)m; data += sizeof(struct amsg) - *ptodo; count = file_write(&f->pipe.file, data, *ptodo); if (count == 0) return 0; *ptodo -= count; } DPRINTFN(4, "sock_wmsg: %p: done\n", f); return 1; } /* * Read data chunk from the file descriptor, return 1 if at least one * byte was read, 0 if the file blocked. */ int sock_rdata(struct sock *f) { struct aproc *p; struct abuf *obuf; unsigned char *data; unsigned count, n; #ifdef DEBUG if (f->pstate != SOCK_MIDI && f->rtodo == 0) { fprintf(stderr, "sock_rdata: bad call: zero arg\n"); abort(); } #endif p = f->pipe.file.rproc; obuf = LIST_FIRST(&p->obuflist); if (obuf == NULL) return 0; if (ABUF_FULL(obuf) || !(f->pipe.file.state & FILE_ROK)) return 0; data = abuf_wgetblk(obuf, &count, 0); if (f->pstate != SOCK_MIDI && count > f->rtodo) count = f->rtodo; n = file_read(&f->pipe.file, data, count); if (n == 0) return 0; abuf_wcommit(obuf, n); if (f->pstate != SOCK_MIDI) f->rtodo -= n; return 1; } /* * Write data chunk to the file descriptor, return 1 if at least one * byte was written, 0 if the file blocked. */ int sock_wdata(struct sock *f) { struct aproc *p; struct abuf *ibuf; unsigned char *data; unsigned count, n; #define ZERO_MAX 0x1000 static char zero[ZERO_MAX]; #ifdef DEBUG if (f->pstate != SOCK_MIDI && f->wtodo == 0) { fprintf(stderr, "sock_wdata: bad call: zero arg\n"); abort(); } #endif if (!(f->pipe.file.state & FILE_WOK)) return 0; p = f->pipe.file.wproc; ibuf = LIST_FIRST(&p->ibuflist); if (ibuf) { if (ABUF_EMPTY(ibuf)) return 0; data = abuf_rgetblk(ibuf, &count, 0); if (f->pstate != SOCK_MIDI && count > f->wtodo) count = f->wtodo; n = file_write(&f->pipe.file, data, count); if (n == 0) return 0; abuf_rdiscard(ibuf, n); if (f->pstate != SOCK_MIDI) f->wtodo -= n; } else { if (f->pstate == SOCK_MIDI) return 0; /* * There's no dev_detach() routine yet, * so now we abruptly destroy the buffer. * Until we implement dev_detach, complete * the packet with zeros... */ count = ZERO_MAX; if (count > f->wtodo) count = f->wtodo; n = file_write(&f->pipe.file, zero, count); if (n == 0) return 0; f->wtodo -= n; } return 1; } int sock_setpar(struct sock *f) { struct amsg_par *p = &f->rmsg.u.par; unsigned min, max, rate; if (AMSG_ISSET(p->legacy_mode)) { /* * XXX: allow old clients that don't support HELLO * to work */ if ((p->legacy_mode & ~(AMSG_PLAY | AMSG_REC)) || (p->legacy_mode == 0)) { DPRINTF("sock_setpar: bad mode %x\n", p->legacy_mode); return 0; } f->mode = 0; if ((p->legacy_mode & AMSG_PLAY) && dev_mix) f->mode |= AMSG_PLAY; if ((p->legacy_mode & AMSG_REC) && dev_sub) f->mode |= AMSG_REC; DPRINTF("sock_setpar: mode -> %x\n", f->mode); } if (AMSG_ISSET(p->bits)) { if (p->bits < BITS_MIN || p->bits > BITS_MAX) { DPRINTF("sock_setpar: bits out of bounds\n"); return 0; } if (AMSG_ISSET(p->bps)) { if (p->bps < ((p->bits + 7) / 8) || p->bps > 4) { DPRINTF("sock_setpar: bps out of bounds\n"); return 0; } } else p->bps = APARAMS_BPS(p->bits); f->rpar.bits = f->wpar.bits = p->bits; f->rpar.bps = f->wpar.bps = p->bps; DPRINTF("sock_setpar: bits/bps -> %u/%u\n", p->bits, p->bps); } if (AMSG_ISSET(p->sig)) f->rpar.sig = f->wpar.sig = p->sig ? 1 : 0; if (AMSG_ISSET(p->le)) f->rpar.le = f->wpar.le = p->le ? 1 : 0; if (AMSG_ISSET(p->msb)) f->rpar.msb = f->wpar.msb = p->msb ? 1 : 0; if (AMSG_ISSET(p->rchan) && (f->mode & AMSG_REC)) { if (p->rchan < 1) p->rchan = 1; if (p->rchan > NCHAN_MAX) p->rchan = NCHAN_MAX; f->wpar.cmin = f->opt->wpar.cmin; f->wpar.cmax = f->opt->wpar.cmin + p->rchan - 1; if (f->wpar.cmax > f->opt->wpar.cmax) f->wpar.cmax = f->opt->wpar.cmax; DPRINTF("sock_setpar: rchan -> %u:%u\n", f->wpar.cmin, f->wpar.cmax); } if (AMSG_ISSET(p->pchan) && (f->mode & AMSG_PLAY)) { if (p->pchan < 1) p->pchan = 1; if (p->pchan > NCHAN_MAX) p->pchan = NCHAN_MAX; f->rpar.cmin = f->opt->rpar.cmin; f->rpar.cmax = f->opt->rpar.cmin + p->pchan - 1; if (f->rpar.cmax > f->opt->rpar.cmax) f->rpar.cmax = f->opt->rpar.cmax; DPRINTF("sock_setpar: pchan -> %u:%u\n", f->rpar.cmin, f->rpar.cmax); } if (AMSG_ISSET(p->rate)) { if (p->rate < RATE_MIN) p->rate = RATE_MIN; if (p->rate > RATE_MAX) p->rate = RATE_MAX; f->round = dev_roundof(p->rate); f->rpar.rate = f->wpar.rate = p->rate; if (!AMSG_ISSET(p->appbufsz)) { p->appbufsz = dev_bufsz / dev_round * f->round; DPRINTF("sock_setpar: appbufsz -> %u\n", p->appbufsz); } DPRINTF("sock_setpar: rate -> %u, round -> %u\n", p->rate, f->round); } if (AMSG_ISSET(p->xrun)) { if (p->xrun != AMSG_IGNORE && p->xrun != AMSG_SYNC && p->xrun != AMSG_ERROR) { DPRINTF("sock_setpar: bad xrun: %u\n", p->xrun); return 0; } f->xrun = p->xrun; DPRINTF("sock_setpar: xrun -> %u\n", f->xrun); } if (AMSG_ISSET(p->bufsz)) { /* * XXX: bufsz will become read-only, but for now * allow old library to properly work */ DPRINTF("sock_setpar: bufsz: %u\n", p->bufsz); min = (dev_bufsz / dev_round) * f->round; if (p->bufsz < min) p->bufsz = min; p->appbufsz = p->bufsz - min; } if (AMSG_ISSET(p->appbufsz)) { rate = (f->mode & AMSG_PLAY) ? f->rpar.rate : f->wpar.rate; min = 1; max = 1 + rate / dev_round; min *= f->round; max *= f->round; p->appbufsz += f->round - 1; p->appbufsz -= p->appbufsz % f->round; if (p->appbufsz < min) p->appbufsz = min; if (p->appbufsz > max) p->appbufsz = max; f->bufsz = p->appbufsz; DPRINTF("sock_setpar: bufsz -> %u\n", f->bufsz); } #ifdef DEBUG if (debug_level > 0) { fprintf(stderr, "sock_setpar: %p: rpar=", f); aparams_print(&f->rpar); fprintf(stderr, ", wpar="); aparams_print(&f->wpar); fprintf(stderr, ", mode=%u, bufsz=%u\n", f->mode, f->bufsz); } #endif return 1; } /* * allocate buffers, so client can start filling write-end. */ void sock_midiattach(struct sock *f, unsigned mode) { struct abuf *rbuf = NULL, *wbuf = NULL; struct aparams dummy; memset(&dummy, 0, sizeof(dummy)); dummy.bps = 1; if (mode & AMSG_MIDIOUT) { rbuf = abuf_new(3125, &dummy); aproc_setout(f->pipe.file.rproc, rbuf); aproc_setin(thrubox, rbuf); } if (mode & AMSG_MIDIIN) { wbuf = abuf_new(3125, &dummy); aproc_setin(f->pipe.file.wproc, wbuf); aproc_setout(thrubox, wbuf); if (mode & AMSG_MIDIOUT) { rbuf->duplex = wbuf; wbuf->duplex = rbuf; } } } int sock_hello(struct sock *f) { struct amsg_hello *p = &f->rmsg.u.hello; DPRINTF("sock_hello: from <%s>, mode = %x\n", p->who, p->proto); if (thrubox && (p->proto & (AMSG_MIDIIN | AMSG_MIDIOUT))) { if (p->proto & ~(AMSG_MIDIIN | AMSG_MIDIOUT)) { DPRINTF("sock_hello: %x: bad proto\n", p->proto); return 0; } f->mode = p->proto; f->pstate = SOCK_MIDI; sock_midiattach(f, p->proto); return 1; } f->opt = opt_byname(p->opt); if (f->opt == NULL) return 0; if (dev_rec) f->wpar = f->opt->wpar; if (dev_play) f->rpar = f->opt->rpar; if ((p->proto & ~(AMSG_PLAY | AMSG_REC)) != 0 || (p->proto & (AMSG_PLAY | AMSG_REC)) == 0) { DPRINTF("sock_hello: %x: unsupported proto\n", p->proto); return 0; } f->mode = 0; if (p->proto & AMSG_PLAY) { if (!dev_mix) { DPRINTF("sock_hello: playback not supported\n"); return 0; } f->mode |= AMSG_PLAY; } if (p->proto & AMSG_REC) { if (!dev_sub) { DPRINTF("sock_hello: recording not supported\n"); return 0; } f->mode |= AMSG_REC; } f->pstate = SOCK_INIT; return 1; } /* * Execute message in f->rmsg and change the state accordingly; return 1 * on success, and 0 on failure, in which case the socket is destroyed. */ int sock_execmsg(struct sock *f) { struct amsg *m = &f->rmsg; /* * XXX: allow old clients to work without hello on the default socket */ if (f->pstate == SOCK_HELLO && m->cmd != AMSG_HELLO && f->opt != NULL) { DPRINTF("sock_execmsg: legacy client\n"); f->pstate = SOCK_INIT; } switch (m->cmd) { case AMSG_DATA: DPRINTFN(4, "sock_execmsg: %p: DATA\n", f); if (f->pstate != SOCK_RUN && f->pstate != SOCK_START) { DPRINTF("sock_execmsg: %p: DATA, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (!(f->mode & AMSG_PLAY)) { DPRINTF("sock_execmsg: %p: DATA, not allowed\n", f); aproc_del(f->pipe.file.rproc); return 0; } f->rstate = SOCK_RDATA; f->rtodo = m->u.data.size; if (f->rtodo == 0) { DPRINTF("sock_execmsg: zero-length data chunk\n"); aproc_del(f->pipe.file.rproc); return 0; } break; case AMSG_START: DPRINTFN(2, "sock_execmsg: %p: START\n", f); if (f->pstate != SOCK_INIT) { DPRINTF("sock_execmsg: %p: START, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } sock_allocbuf(f); f->rstate = SOCK_RMSG; f->rtodo = sizeof(struct amsg); break; case AMSG_STOP: DPRINTFN(2, "sock_execmsg: %p: STOP\n", f); if (f->pstate != SOCK_RUN && f->pstate != SOCK_START) { DPRINTF("sock_execmsg: %p: STOP, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (f->pstate == SOCK_START) (void)sock_attach(f, 1); sock_freebuf(f); AMSG_INIT(m); m->cmd = AMSG_ACK; f->rstate = SOCK_RRET; f->rtodo = sizeof(struct amsg); break; case AMSG_SETPAR: DPRINTFN(2, "sock_execmsg: %p: SETPAR\n", f); if (f->pstate != SOCK_INIT) { DPRINTF("sock_execmsg: %p: SETPAR, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (!sock_setpar(f)) { aproc_del(f->pipe.file.rproc); return 0; } f->rtodo = sizeof(struct amsg); f->rstate = SOCK_RMSG; break; case AMSG_GETPAR: DPRINTFN(2, "sock_execmsg: %p: GETPAR\n", f); if (f->pstate != SOCK_INIT) { DPRINTF("sock_execmsg: %p: GETPAR, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } AMSG_INIT(m); m->cmd = AMSG_GETPAR; m->u.par.legacy_mode = f->mode; m->u.par.bits = f->rpar.bits; m->u.par.bps = f->rpar.bps; m->u.par.sig = f->rpar.sig; m->u.par.le = f->rpar.le; m->u.par.msb = f->rpar.msb; m->u.par.rate = f->rpar.rate; m->u.par.rchan = f->wpar.cmax - f->wpar.cmin + 1; m->u.par.pchan = f->rpar.cmax - f->rpar.cmin + 1; m->u.par.appbufsz = f->bufsz; m->u.par.bufsz = f->bufsz + (dev_bufsz / dev_round) * f->round; m->u.par.round = f->round; f->rstate = SOCK_RRET; f->rtodo = sizeof(struct amsg); break; case AMSG_GETCAP: DPRINTFN(2, "sock_execmsg: %p: GETCAP\n", f); if (f->pstate != SOCK_INIT) { DPRINTF("sock_execmsg: %p: GETCAP, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } AMSG_INIT(m); m->cmd = AMSG_GETCAP; m->u.cap.rate = dev_rate; m->u.cap.pchan = dev_mix ? (f->opt->rpar.cmax - f->opt->rpar.cmin + 1) : 0; m->u.cap.rchan = dev_sub ? (f->opt->wpar.cmax - f->opt->wpar.cmin + 1) : 0; m->u.cap.bits = sizeof(short) * 8; m->u.cap.bps = sizeof(short); f->rstate = SOCK_RRET; f->rtodo = sizeof(struct amsg); break; case AMSG_SETVOL: DPRINTFN(2, "sock_execmsg: %p: SETVOL\n", f); if (f->pstate != SOCK_RUN && f->pstate != SOCK_START && f->pstate != SOCK_INIT) { DPRINTF("sock_execmsg: %p: SETVOL, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (m->u.vol.ctl > MIDI_MAXCTL) { DPRINTF("sock_execmsg: %p: SETVOL, out of range\n", f); aproc_del(f->pipe.file.rproc); return 0; } DPRINTF("sock_execmsg: SETVOL %u\n", m->u.vol.ctl); sock_setvol(f, MIDI_TO_ADATA(m->u.vol.ctl)); f->rtodo = sizeof(struct amsg); f->rstate = SOCK_RMSG; break; case AMSG_HELLO: DPRINTFN(2, "sock_execmsg: %p: HELLO\n", f); if (f->pstate != SOCK_HELLO) { DPRINTF("sock_execmsg: %p: HELLO, bad state\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (!sock_hello(f)) { aproc_del(f->pipe.file.rproc); return 0; } AMSG_INIT(m); m->cmd = AMSG_ACK; f->rstate = SOCK_RRET; f->rtodo = sizeof(struct amsg); break; default: DPRINTF("sock_execmsg: %p bogus command\n", f); aproc_del(f->pipe.file.rproc); return 0; } if (f->rstate == SOCK_RRET) { if (f->wstate != SOCK_WIDLE || !sock_wmsg(f, &f->rmsg, &f->rtodo)) return 0; DPRINTF("sock_execmsg: %p RRET done\n", f); if (f->pstate == SOCK_MIDI && (f->mode & AMSG_MIDIOUT)) { f->rstate = SOCK_RDATA; f->rtodo = 0; } else { f->rstate = SOCK_RMSG; f->rtodo = sizeof(struct amsg); } } return 1; } /* * Create a new data/pos message. */ int sock_buildmsg(struct sock *f) { struct aproc *p; struct abuf *ibuf; if (f->pstate == SOCK_MIDI) { DPRINTFN(4, "sock_buildmsg: %p: switched to midi\n", f); f->wstate = SOCK_WDATA; f->wtodo = 0; return 1; } /* * If pos changed, build a MOVE message. */ if (f->tickpending && f->delta >= 0) { DPRINTFN(4, "sock_buildmsg: %p: POS: %d\n", f, f->delta); AMSG_INIT(&f->wmsg); f->wmsg.cmd = AMSG_MOVE; f->wmsg.u.ts.delta = f->delta; f->wtodo = sizeof(struct amsg); f->wstate = SOCK_WMSG; f->delta = 0; f->tickpending = 0; return 1; } /* * If data available, build a DATA message. */ p = f->pipe.file.wproc; ibuf = LIST_FIRST(&p->ibuflist); if (ibuf && ABUF_ROK(ibuf)) { AMSG_INIT(&f->wmsg); f->wmsg.cmd = AMSG_DATA; f->wmsg.u.data.size = ibuf->used - (ibuf->used % ibuf->bpf); if (f->wmsg.u.data.size > AMSG_DATAMAX) f->wmsg.u.data.size = AMSG_DATAMAX - (AMSG_DATAMAX % ibuf->bpf); f->wtodo = sizeof(struct amsg); f->wstate = SOCK_WMSG; return 1; } DPRINTFN(4, "sock_buildmsg: %p: idling...\n", f); f->wstate = SOCK_WIDLE; return 0; } /* * Read from the socket file descriptor, fill input buffer and update * the state. Return 1 if at least one message or 1 data byte was * processed, 0 if something blocked. */ int sock_read(struct sock *f) { DPRINTFN(4, "sock_read: %p; rstate = %u, rtodo = %u\n", f, f->rstate, f->rtodo); switch (f->rstate) { case SOCK_RMSG: if (!sock_rmsg(f)) return 0; if (!sock_execmsg(f)) return 0; break; case SOCK_RDATA: if (!sock_rdata(f)) return 0; if (f->pstate != SOCK_MIDI && f->rtodo == 0) { f->rstate = SOCK_RMSG; f->rtodo = sizeof(struct amsg); } if (f->pstate == SOCK_START) (void)sock_attach(f, 0); break; case SOCK_RRET: DPRINTF("sock_read: %p: blocked in RRET\n", f); return 0; } DPRINTFN(4, "sock_read: %p: done, rstate = %u\n", f, f->rstate); return 1; } /* * Process messages to return. */ int sock_return(struct sock *f) { struct aproc *rp; while (f->rstate == SOCK_RRET) { if (!sock_wmsg(f, &f->rmsg, &f->rtodo)) return 0; DPRINTF("sock_return: %p: done\n", f); if (f->pstate == SOCK_MIDI && (f->mode & AMSG_MIDIOUT)) { f->rstate = SOCK_RDATA; f->rtodo = 0; } else { f->rstate = SOCK_RMSG; f->rtodo = sizeof(struct amsg); } if (f->pipe.file.state & FILE_RINUSE) break; f->pipe.file.state |= FILE_RINUSE; for (;;) { /* * in() may trigger rsock_done and destroy the * wsock. */ rp = f->pipe.file.rproc; if (!rp || !rp->ops->in(rp, NULL)) break; } f->pipe.file.state &= ~FILE_RINUSE; if (f->pipe.file.wproc == NULL) return 0; } return 1; } /* * Write messages and data on the socket file descriptor. Return 1 if * at least one message or one data byte was processed, 0 if something * blocked. */ int sock_write(struct sock *f) { DPRINTFN(4, "sock_write: %p: wstate = %u, wtodo = %u\n", f, f->wstate, f->wtodo); switch (f->wstate) { case SOCK_WMSG: if (!sock_wmsg(f, &f->wmsg, &f->wtodo)) return 0; if (f->wmsg.cmd != AMSG_DATA) { f->wstate = SOCK_WIDLE; f->wtodo = 0xdeadbeef; break; } f->wstate = SOCK_WDATA; f->wtodo = f->wmsg.u.data.size; /* PASSTHROUGH */ case SOCK_WDATA: if (!sock_wdata(f)) return 0; if (f->pstate == SOCK_MIDI || f->wtodo > 0) break; f->wstate = SOCK_WIDLE; f->wtodo = 0xdeadbeef; /* PASSTHROUGH */ case SOCK_WIDLE: if (!sock_return(f)) return 0; if (!sock_buildmsg(f)) return 0; break; default: fprintf(stderr, "sock_write: unknown state\n"); abort(); } return 1; }