2002-11-12 15:22:19 +00:00
|
|
|
/*
|
2003-04-17 03:38:03 +00:00
|
|
|
* Copyright (c) 2003 Hidetoshi Shimokawa
|
2002-11-12 15:22:19 +00:00
|
|
|
* Copyright (c) 1998-2002 Katsushi Kobayashi and Hidetoshi Shimokawa
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the acknowledgement as bellow:
|
|
|
|
*
|
|
|
|
* This product includes software developed by K. Kobayashi and H. Shimokawa
|
|
|
|
*
|
|
|
|
* 4. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
|
|
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* $FreeBSD$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/mbuf.h>
|
|
|
|
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/conf.h>
|
|
|
|
#include <sys/poll.h>
|
|
|
|
|
|
|
|
#include <sys/bus.h>
|
2003-04-17 03:38:03 +00:00
|
|
|
#include <machine/bus.h>
|
2002-11-12 15:22:19 +00:00
|
|
|
|
|
|
|
#include <sys/ioccom.h>
|
|
|
|
|
|
|
|
#include <dev/firewire/firewire.h>
|
|
|
|
#include <dev/firewire/firewirereg.h>
|
2003-04-17 03:38:03 +00:00
|
|
|
#include <dev/firewire/fwdma.h>
|
2002-11-12 15:22:19 +00:00
|
|
|
#include <dev/firewire/fwmem.h>
|
2003-01-15 05:26:23 +00:00
|
|
|
#include <dev/firewire/iec68113.h>
|
2002-11-12 15:22:19 +00:00
|
|
|
|
|
|
|
#define CDEV_MAJOR 127
|
|
|
|
#define FWNODE_INVAL 0xffff
|
|
|
|
|
|
|
|
static d_open_t fw_open;
|
|
|
|
static d_close_t fw_close;
|
|
|
|
static d_ioctl_t fw_ioctl;
|
|
|
|
static d_poll_t fw_poll;
|
|
|
|
static d_read_t fw_read; /* for Isochronous packet */
|
|
|
|
static d_write_t fw_write;
|
|
|
|
static d_mmap_t fw_mmap;
|
|
|
|
|
|
|
|
struct cdevsw firewire_cdevsw =
|
|
|
|
{
|
2003-03-06 05:06:44 +00:00
|
|
|
#if __FreeBSD_version >= 500104
|
2003-03-03 12:15:54 +00:00
|
|
|
.d_open = fw_open,
|
|
|
|
.d_close = fw_close,
|
|
|
|
.d_read = fw_read,
|
|
|
|
.d_write = fw_write,
|
|
|
|
.d_ioctl = fw_ioctl,
|
|
|
|
.d_poll = fw_poll,
|
|
|
|
.d_mmap = fw_mmap,
|
|
|
|
.d_name = "fw",
|
|
|
|
.d_maj = CDEV_MAJOR,
|
|
|
|
.d_flags = D_MEM
|
2003-03-06 05:06:44 +00:00
|
|
|
#else
|
|
|
|
fw_open, fw_close, fw_read, fw_write, fw_ioctl,
|
|
|
|
fw_poll, fw_mmap, nostrategy, "fw", CDEV_MAJOR, nodump, nopsize, D_MEM
|
|
|
|
#endif
|
2002-11-12 15:22:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
fw_open (dev_t dev, int flags, int fmt, fw_proc *td)
|
|
|
|
{
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
|
|
|
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_open(dev, flags, fmt, td);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
if(sc->fc->ir[sub]->flag & FWXFERQ_OPEN){
|
|
|
|
err = EBUSY;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if(sc->fc->it[sub]->flag & FWXFERQ_OPEN){
|
|
|
|
err = EBUSY;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
if(sc->fc->ir[sub]->flag & FWXFERQ_MODEMASK){
|
|
|
|
err = EBUSY;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
/* Default is per packet mode */
|
|
|
|
sc->fc->ir[sub]->flag |= FWXFERQ_OPEN;
|
|
|
|
sc->fc->it[sub]->flag |= FWXFERQ_OPEN;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
fw_close (dev_t dev, int flags, int fmt, fw_proc *td)
|
|
|
|
{
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
|
|
|
struct fw_xfer *xfer;
|
|
|
|
struct fw_bind *fwb;
|
|
|
|
int err = 0;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_close(dev, flags, fmt, td);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
if(!(sc->fc->ir[sub]->flag & FWXFERQ_OPEN)){
|
|
|
|
err = EINVAL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
sc->fc->ir[sub]->flag &= ~FWXFERQ_OPEN;
|
|
|
|
if(!(sc->fc->it[sub]->flag & FWXFERQ_OPEN)){
|
|
|
|
err = EINVAL;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
sc->fc->it[sub]->flag &= ~FWXFERQ_OPEN;
|
|
|
|
|
|
|
|
if(sc->fc->ir[sub]->flag & FWXFERQ_RUNNING){
|
|
|
|
sc->fc->irx_disable(sc->fc, sub);
|
|
|
|
}
|
|
|
|
if(sc->fc->it[sub]->flag & FWXFERQ_RUNNING){
|
|
|
|
sc->fc->it[sub]->flag &= ~FWXFERQ_RUNNING;
|
|
|
|
sc->fc->itx_disable(sc->fc, sub);
|
|
|
|
}
|
|
|
|
if(sc->fc->ir[sub]->flag & FWXFERQ_EXTBUF){
|
2003-04-17 03:38:03 +00:00
|
|
|
if (sc->fc->ir[sub]->buf != NULL)
|
|
|
|
fwdma_free_multiseg(sc->fc->ir[sub]->buf);
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->ir[sub]->buf = NULL;
|
2003-02-01 15:04:33 +00:00
|
|
|
free(sc->fc->ir[sub]->bulkxfer, M_FW);
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->ir[sub]->bulkxfer = NULL;
|
|
|
|
sc->fc->ir[sub]->flag &= ~FWXFERQ_EXTBUF;
|
2003-01-16 13:09:33 +00:00
|
|
|
sc->fc->ir[sub]->psize = PAGE_SIZE;
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->ir[sub]->maxq = FWMAXQUEUE;
|
|
|
|
}
|
|
|
|
if(sc->fc->it[sub]->flag & FWXFERQ_EXTBUF){
|
2003-04-17 03:38:03 +00:00
|
|
|
if (sc->fc->it[sub]->buf != NULL)
|
|
|
|
fwdma_free_multiseg(sc->fc->it[sub]->buf);
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->it[sub]->buf = NULL;
|
2003-02-01 15:04:33 +00:00
|
|
|
free(sc->fc->it[sub]->bulkxfer, M_FW);
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->it[sub]->bulkxfer = NULL;
|
|
|
|
sc->fc->it[sub]->flag &= ~FWXFERQ_EXTBUF;
|
2003-01-16 13:09:33 +00:00
|
|
|
sc->fc->it[sub]->psize = 0;
|
2002-11-12 15:22:19 +00:00
|
|
|
sc->fc->it[sub]->maxq = FWMAXQUEUE;
|
|
|
|
}
|
|
|
|
for(xfer = STAILQ_FIRST(&sc->fc->ir[sub]->q);
|
|
|
|
xfer != NULL; xfer = STAILQ_FIRST(&sc->fc->ir[sub]->q)){
|
|
|
|
sc->fc->ir[sub]->queued--;
|
|
|
|
STAILQ_REMOVE_HEAD(&sc->fc->ir[sub]->q, link);
|
|
|
|
|
|
|
|
xfer->resp = 0;
|
2003-04-17 03:38:03 +00:00
|
|
|
fw_xfer_done(xfer);
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
|
|
|
for(fwb = STAILQ_FIRST(&sc->fc->ir[sub]->binds); fwb != NULL;
|
|
|
|
fwb = STAILQ_FIRST(&sc->fc->ir[sub]->binds)){
|
|
|
|
STAILQ_REMOVE(&sc->fc->binds, fwb, fw_bind, fclist);
|
|
|
|
STAILQ_REMOVE_HEAD(&sc->fc->ir[sub]->binds, chlist);
|
2003-02-01 15:04:33 +00:00
|
|
|
free(fwb, M_FW);
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
sc->fc->ir[sub]->flag &= ~(FWXFERQ_MODEMASK | FWXFERQ_CHTAGMASK);
|
|
|
|
sc->fc->it[sub]->flag &= ~(FWXFERQ_MODEMASK | FWXFERQ_CHTAGMASK);
|
2002-11-12 15:22:19 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* read request.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
fw_read (dev_t dev, struct uio *uio, int ioflag)
|
|
|
|
{
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
struct fw_xferq *ir;
|
|
|
|
struct fw_xfer *xfer;
|
|
|
|
int err = 0, s, slept = 0;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
|
|
|
struct fw_pkt *fp;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_read(dev, uio, ioflag);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
|
|
|
|
ir = sc->fc->ir[sub];
|
|
|
|
|
|
|
|
readloop:
|
|
|
|
xfer = STAILQ_FIRST(&ir->q);
|
2003-04-17 03:38:03 +00:00
|
|
|
if (ir->stproc == NULL) {
|
2003-01-28 15:09:07 +00:00
|
|
|
/* iso bulkxfer */
|
2002-11-12 15:22:19 +00:00
|
|
|
ir->stproc = STAILQ_FIRST(&ir->stvalid);
|
2003-01-28 15:09:07 +00:00
|
|
|
if (ir->stproc != NULL) {
|
2002-11-12 15:22:19 +00:00
|
|
|
s = splfw();
|
|
|
|
STAILQ_REMOVE_HEAD(&ir->stvalid, link);
|
|
|
|
splx(s);
|
|
|
|
ir->queued = 0;
|
|
|
|
}
|
|
|
|
}
|
2003-01-28 15:09:07 +00:00
|
|
|
if (xfer == NULL && ir->stproc == NULL) {
|
|
|
|
/* no data avaliable */
|
|
|
|
if (slept == 0) {
|
2002-11-12 15:22:19 +00:00
|
|
|
slept = 1;
|
|
|
|
ir->flag |= FWXFERQ_WAKEUP;
|
2003-03-02 16:54:40 +00:00
|
|
|
err = tsleep(ir, FWPRI, "fw_read", hz);
|
2003-01-28 15:09:07 +00:00
|
|
|
ir->flag &= ~FWXFERQ_WAKEUP;
|
|
|
|
if (err == 0)
|
|
|
|
goto readloop;
|
|
|
|
} else if (slept == 1)
|
2002-11-12 15:22:19 +00:00
|
|
|
err = EIO;
|
2003-01-28 15:09:07 +00:00
|
|
|
return err;
|
|
|
|
} else if(xfer != NULL) {
|
2003-04-17 03:38:03 +00:00
|
|
|
/* per packet mode or FWACT_CH bind?*/
|
2002-11-12 15:22:19 +00:00
|
|
|
s = splfw();
|
|
|
|
ir->queued --;
|
|
|
|
STAILQ_REMOVE_HEAD(&ir->q, link);
|
|
|
|
splx(s);
|
2003-04-17 03:38:03 +00:00
|
|
|
fp = (struct fw_pkt *)xfer->recv.buf;
|
2002-11-12 15:22:19 +00:00
|
|
|
if(sc->fc->irx_post != NULL)
|
|
|
|
sc->fc->irx_post(sc->fc, fp->mode.ld);
|
2003-04-17 03:38:03 +00:00
|
|
|
err = uiomove(xfer->recv.buf, xfer->recv.len, uio);
|
|
|
|
/* XXX we should recycle this xfer */
|
2002-11-12 15:22:19 +00:00
|
|
|
fw_xfer_free( xfer);
|
2003-01-28 15:09:07 +00:00
|
|
|
} else if(ir->stproc != NULL) {
|
|
|
|
/* iso bulkxfer */
|
2003-04-17 03:38:03 +00:00
|
|
|
fp = (struct fw_pkt *)fwdma_v_addr(ir->buf,
|
|
|
|
ir->stproc->poffset + ir->queued);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(sc->fc->irx_post != NULL)
|
|
|
|
sc->fc->irx_post(sc->fc, fp->mode.ld);
|
2003-04-17 03:38:03 +00:00
|
|
|
if(fp->mode.stream.len == 0){
|
2002-11-12 15:22:19 +00:00
|
|
|
err = EIO;
|
|
|
|
return err;
|
|
|
|
}
|
2003-01-28 15:09:07 +00:00
|
|
|
err = uiomove((caddr_t)fp,
|
2003-04-17 03:38:03 +00:00
|
|
|
fp->mode.stream.len + sizeof(u_int32_t), uio);
|
2002-11-12 15:22:19 +00:00
|
|
|
ir->queued ++;
|
|
|
|
if(ir->queued >= ir->bnpacket){
|
|
|
|
s = splfw();
|
|
|
|
STAILQ_INSERT_TAIL(&ir->stfree, ir->stproc, link);
|
|
|
|
splx(s);
|
2003-01-26 15:39:04 +00:00
|
|
|
sc->fc->irx_enable(sc->fc, sub);
|
2002-11-12 15:22:19 +00:00
|
|
|
ir->stproc = NULL;
|
|
|
|
}
|
2003-01-28 15:09:07 +00:00
|
|
|
if (uio->uio_resid >= ir->psize) {
|
|
|
|
slept = -1;
|
|
|
|
goto readloop;
|
|
|
|
}
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
fw_write (dev_t dev, struct uio *uio, int ioflag)
|
|
|
|
{
|
|
|
|
int err = 0;
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
|
|
|
int s, slept = 0;
|
|
|
|
struct fw_pkt *fp;
|
|
|
|
struct fw_xfer *xfer;
|
|
|
|
struct fw_xferq *xferq;
|
|
|
|
struct firewire_comm *fc;
|
|
|
|
struct fw_xferq *it;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_write(dev, uio, ioflag);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
fc = sc->fc;
|
|
|
|
it = sc->fc->it[sub];
|
|
|
|
|
2003-04-17 03:38:03 +00:00
|
|
|
xferq = NULL;
|
2002-11-12 15:22:19 +00:00
|
|
|
/* Discard unsent buffered stream packet, when sending Asyrequrst */
|
|
|
|
if(xferq != NULL && it->stproc != NULL){
|
|
|
|
s = splfw();
|
|
|
|
STAILQ_INSERT_TAIL(&it->stfree, it->stproc, link);
|
|
|
|
splx(s);
|
|
|
|
it->stproc = NULL;
|
|
|
|
}
|
2003-01-24 13:03:19 +00:00
|
|
|
if (xferq == NULL) {
|
2002-11-12 15:22:19 +00:00
|
|
|
isoloop:
|
2003-01-26 15:39:04 +00:00
|
|
|
if (it->stproc == NULL) {
|
2002-11-12 15:22:19 +00:00
|
|
|
it->stproc = STAILQ_FIRST(&it->stfree);
|
2003-01-26 15:39:04 +00:00
|
|
|
if (it->stproc != NULL) {
|
2002-11-12 15:22:19 +00:00
|
|
|
s = splfw();
|
|
|
|
STAILQ_REMOVE_HEAD(&it->stfree, link);
|
|
|
|
splx(s);
|
|
|
|
it->queued = 0;
|
2003-01-26 15:39:04 +00:00
|
|
|
} else if (slept == 0) {
|
2002-11-12 15:22:19 +00:00
|
|
|
slept = 1;
|
|
|
|
err = sc->fc->itx_enable(sc->fc, sub);
|
2003-01-28 15:09:07 +00:00
|
|
|
if (err)
|
2002-11-12 15:22:19 +00:00
|
|
|
return err;
|
2003-03-02 16:54:40 +00:00
|
|
|
err = tsleep(it, FWPRI,
|
2003-01-28 15:09:07 +00:00
|
|
|
"fw_write", hz);
|
|
|
|
if (err)
|
2002-11-12 15:22:19 +00:00
|
|
|
return err;
|
|
|
|
goto isoloop;
|
2003-01-28 15:09:07 +00:00
|
|
|
} else {
|
2002-11-12 15:22:19 +00:00
|
|
|
err = EIO;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
fp = (struct fw_pkt *)fwdma_v_addr(it->buf,
|
|
|
|
it->stproc->poffset + it->queued);
|
2003-01-28 15:09:07 +00:00
|
|
|
err = uiomove((caddr_t)fp, sizeof(struct fw_isohdr), uio);
|
|
|
|
err = uiomove((caddr_t)fp->mode.stream.payload,
|
2003-04-17 03:38:03 +00:00
|
|
|
fp->mode.stream.len, uio);
|
2002-11-12 15:22:19 +00:00
|
|
|
it->queued ++;
|
2003-01-28 15:09:07 +00:00
|
|
|
if (it->queued >= it->bnpacket) {
|
2002-11-12 15:22:19 +00:00
|
|
|
s = splfw();
|
|
|
|
STAILQ_INSERT_TAIL(&it->stvalid, it->stproc, link);
|
|
|
|
splx(s);
|
|
|
|
it->stproc = NULL;
|
|
|
|
err = sc->fc->itx_enable(sc->fc, sub);
|
|
|
|
}
|
2003-01-28 15:09:07 +00:00
|
|
|
if (uio->uio_resid >= sizeof(struct fw_isohdr)) {
|
|
|
|
slept = 0;
|
|
|
|
goto isoloop;
|
|
|
|
}
|
2002-11-12 15:22:19 +00:00
|
|
|
return err;
|
2003-01-24 13:03:19 +00:00
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
if (xferq != NULL) {
|
|
|
|
xfer = fw_xfer_alloc_buf(M_FWXFER, uio->uio_resid, 12);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(xfer == NULL){
|
|
|
|
err = ENOMEM;
|
|
|
|
return err;
|
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
xfer->dst = fp->mode.hdr.dst;
|
2002-11-12 15:22:19 +00:00
|
|
|
xfer->send.len = uio->uio_resid;
|
|
|
|
xfer->spd = 0;/* XXX: how to setup it */
|
|
|
|
xfer->act.hand = fw_asy_callback;
|
|
|
|
|
|
|
|
err = uiomove(xfer->send.buf, uio->uio_resid, uio);
|
|
|
|
if(err){
|
|
|
|
fw_xfer_free( xfer);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
fw_asyreq(fc, -1, xfer);
|
2003-03-02 16:54:40 +00:00
|
|
|
err = tsleep(xfer, FWPRI, "fw_write", hz);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(xfer->resp == EBUSY)
|
|
|
|
return EBUSY;
|
|
|
|
fw_xfer_free( xfer);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ioctl support.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
fw_ioctl (dev_t dev, u_long cmd, caddr_t data, int flag, fw_proc *td)
|
|
|
|
{
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
2003-04-17 03:38:03 +00:00
|
|
|
int s, i, len, err = 0;
|
2002-11-12 15:22:19 +00:00
|
|
|
struct fw_device *fwdev;
|
|
|
|
struct fw_bind *fwb;
|
|
|
|
struct fw_xferq *ir, *it;
|
|
|
|
struct fw_xfer *xfer;
|
|
|
|
struct fw_pkt *fp;
|
2003-01-25 14:47:33 +00:00
|
|
|
struct fw_devinfo *devinfo;
|
2002-11-12 15:22:19 +00:00
|
|
|
|
|
|
|
struct fw_devlstreq *fwdevlst = (struct fw_devlstreq *)data;
|
|
|
|
struct fw_asyreq *asyreq = (struct fw_asyreq *)data;
|
|
|
|
struct fw_isochreq *ichreq = (struct fw_isochreq *)data;
|
|
|
|
struct fw_isobufreq *ibufreq = (struct fw_isobufreq *)data;
|
|
|
|
struct fw_asybindreq *bindreq = (struct fw_asybindreq *)data;
|
|
|
|
struct fw_crom_buf *crom_buf = (struct fw_crom_buf *)data;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_ioctl(dev, cmd, data, flag, td);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
if (!data)
|
|
|
|
return(EINVAL);
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case FW_STSTREAM:
|
|
|
|
sc->fc->it[sub]->flag &= ~0xff;
|
|
|
|
sc->fc->it[sub]->flag |= (0x3f & ichreq->ch);
|
|
|
|
sc->fc->it[sub]->flag |= ((0x3 & ichreq->tag) << 6);
|
|
|
|
err = 0;
|
|
|
|
break;
|
|
|
|
case FW_GTSTREAM:
|
|
|
|
ichreq->ch = sc->fc->it[sub]->flag & 0x3f;
|
|
|
|
ichreq->tag =(sc->fc->it[sub]->flag) >> 2 & 0x3;
|
|
|
|
err = 0;
|
|
|
|
break;
|
|
|
|
case FW_SRSTREAM:
|
|
|
|
sc->fc->ir[sub]->flag &= ~0xff;
|
|
|
|
sc->fc->ir[sub]->flag |= (0x3f & ichreq->ch);
|
|
|
|
sc->fc->ir[sub]->flag |= ((0x3 & ichreq->tag) << 6);
|
|
|
|
err = sc->fc->irx_enable(sc->fc, sub);
|
|
|
|
break;
|
|
|
|
case FW_GRSTREAM:
|
|
|
|
ichreq->ch = sc->fc->ir[sub]->flag & 0x3f;
|
|
|
|
ichreq->tag =(sc->fc->ir[sub]->flag) >> 2 & 0x3;
|
|
|
|
err = 0;
|
|
|
|
break;
|
|
|
|
case FW_SSTBUF:
|
|
|
|
ir = sc->fc->ir[sub];
|
|
|
|
it = sc->fc->it[sub];
|
|
|
|
|
|
|
|
if(ir->flag & FWXFERQ_RUNNING || it->flag & FWXFERQ_RUNNING){
|
|
|
|
return(EBUSY);
|
|
|
|
}
|
|
|
|
if((ir->flag & FWXFERQ_EXTBUF) || (it->flag & FWXFERQ_EXTBUF)){
|
|
|
|
return(EBUSY);
|
|
|
|
}
|
|
|
|
if((ibufreq->rx.nchunk *
|
|
|
|
ibufreq->rx.psize * ibufreq->rx.npacket) +
|
|
|
|
(ibufreq->tx.nchunk *
|
|
|
|
ibufreq->tx.psize * ibufreq->tx.npacket) <= 0){
|
|
|
|
return(EINVAL);
|
|
|
|
}
|
|
|
|
ir->bulkxfer
|
2003-04-17 03:38:03 +00:00
|
|
|
= (struct fw_bulkxfer *)malloc(sizeof(struct fw_bulkxfer) * ibufreq->rx.nchunk, M_FW, M_WAITOK);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(ir->bulkxfer == NULL){
|
|
|
|
return(ENOMEM);
|
|
|
|
}
|
|
|
|
it->bulkxfer
|
2003-04-17 03:38:03 +00:00
|
|
|
= (struct fw_bulkxfer *)malloc(sizeof(struct fw_bulkxfer) * ibufreq->tx.nchunk, M_FW, M_WAITOK);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(it->bulkxfer == NULL){
|
|
|
|
return(ENOMEM);
|
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
if (ibufreq->rx.psize > 0) {
|
|
|
|
ibufreq->rx.psize = roundup2(ibufreq->rx.psize,
|
|
|
|
sizeof(u_int32_t));
|
|
|
|
ir->buf = fwdma_malloc_multiseg(
|
|
|
|
sc->fc, sizeof(u_int32_t),
|
|
|
|
ibufreq->rx.psize,
|
|
|
|
ibufreq->rx.nchunk * ibufreq->rx.npacket,
|
|
|
|
BUS_DMA_WAITOK);
|
|
|
|
|
|
|
|
if(ir->buf == NULL){
|
|
|
|
free(ir->bulkxfer, M_FW);
|
|
|
|
free(it->bulkxfer, M_FW);
|
|
|
|
ir->bulkxfer = NULL;
|
|
|
|
it->bulkxfer = NULL;
|
|
|
|
it->buf = NULL;
|
|
|
|
return(ENOMEM);
|
|
|
|
}
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
if (ibufreq->tx.psize > 0) {
|
|
|
|
ibufreq->tx.psize = roundup2(ibufreq->tx.psize,
|
|
|
|
sizeof(u_int32_t));
|
|
|
|
it->buf = fwdma_malloc_multiseg(
|
|
|
|
sc->fc, sizeof(u_int32_t),
|
|
|
|
ibufreq->tx.psize,
|
|
|
|
ibufreq->tx.nchunk * ibufreq->tx.npacket,
|
|
|
|
BUS_DMA_WAITOK);
|
|
|
|
|
|
|
|
if(it->buf == NULL){
|
|
|
|
free(ir->bulkxfer, M_FW);
|
|
|
|
free(it->bulkxfer, M_FW);
|
|
|
|
fwdma_free_multiseg(ir->buf);
|
|
|
|
ir->bulkxfer = NULL;
|
|
|
|
it->bulkxfer = NULL;
|
|
|
|
it->buf = NULL;
|
|
|
|
return(ENOMEM);
|
|
|
|
}
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ir->bnchunk = ibufreq->rx.nchunk;
|
|
|
|
ir->bnpacket = ibufreq->rx.npacket;
|
|
|
|
ir->psize = (ibufreq->rx.psize + 3) & ~3;
|
|
|
|
ir->queued = 0;
|
|
|
|
|
|
|
|
it->bnchunk = ibufreq->tx.nchunk;
|
|
|
|
it->bnpacket = ibufreq->tx.npacket;
|
|
|
|
it->psize = (ibufreq->tx.psize + 3) & ~3;
|
2003-01-26 15:39:04 +00:00
|
|
|
it->queued = 0;
|
|
|
|
|
2002-11-12 15:22:19 +00:00
|
|
|
STAILQ_INIT(&ir->stvalid);
|
|
|
|
STAILQ_INIT(&ir->stfree);
|
2003-01-26 15:39:04 +00:00
|
|
|
STAILQ_INIT(&ir->stdma);
|
2002-11-12 15:22:19 +00:00
|
|
|
ir->stproc = NULL;
|
|
|
|
|
|
|
|
STAILQ_INIT(&it->stvalid);
|
|
|
|
STAILQ_INIT(&it->stfree);
|
2003-01-26 15:39:04 +00:00
|
|
|
STAILQ_INIT(&it->stdma);
|
2002-11-12 15:22:19 +00:00
|
|
|
it->stproc = NULL;
|
|
|
|
|
|
|
|
for(i = 0 ; i < sc->fc->ir[sub]->bnchunk; i++){
|
2003-04-17 03:38:03 +00:00
|
|
|
ir->bulkxfer[i].poffset = i * ir->bnpacket;
|
|
|
|
ir->bulkxfer[i].mbuf = NULL;
|
2002-11-12 15:22:19 +00:00
|
|
|
STAILQ_INSERT_TAIL(&ir->stfree,
|
|
|
|
&ir->bulkxfer[i], link);
|
|
|
|
}
|
|
|
|
for(i = 0 ; i < sc->fc->it[sub]->bnchunk; i++){
|
2003-04-17 03:38:03 +00:00
|
|
|
it->bulkxfer[i].poffset = i * it->bnpacket;
|
|
|
|
it->bulkxfer[i].mbuf = NULL;
|
2002-11-12 15:22:19 +00:00
|
|
|
STAILQ_INSERT_TAIL(&it->stfree,
|
|
|
|
&it->bulkxfer[i], link);
|
|
|
|
}
|
|
|
|
ir->flag &= ~FWXFERQ_MODEMASK;
|
|
|
|
ir->flag |= FWXFERQ_STREAM;
|
|
|
|
ir->flag |= FWXFERQ_EXTBUF;
|
|
|
|
|
|
|
|
it->flag &= ~FWXFERQ_MODEMASK;
|
|
|
|
it->flag |= FWXFERQ_STREAM;
|
|
|
|
it->flag |= FWXFERQ_EXTBUF;
|
|
|
|
err = 0;
|
|
|
|
break;
|
|
|
|
case FW_GSTBUF:
|
|
|
|
ibufreq->rx.nchunk = sc->fc->ir[sub]->bnchunk;
|
|
|
|
ibufreq->rx.npacket = sc->fc->ir[sub]->bnpacket;
|
|
|
|
ibufreq->rx.psize = sc->fc->ir[sub]->psize;
|
|
|
|
|
|
|
|
ibufreq->tx.nchunk = sc->fc->it[sub]->bnchunk;
|
|
|
|
ibufreq->tx.npacket = sc->fc->it[sub]->bnpacket;
|
|
|
|
ibufreq->tx.psize = sc->fc->it[sub]->psize;
|
|
|
|
break;
|
|
|
|
case FW_ASYREQ:
|
2003-04-17 03:38:03 +00:00
|
|
|
xfer = fw_xfer_alloc_buf(M_FWXFER, asyreq->req.len,
|
|
|
|
PAGE_SIZE /* XXX */);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(xfer == NULL){
|
|
|
|
err = ENOMEM;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
fp = &asyreq->pkt;
|
|
|
|
switch (asyreq->req.type) {
|
|
|
|
case FWASREQNODE:
|
2003-04-17 03:38:03 +00:00
|
|
|
xfer->dst = fp->mode.hdr.dst;
|
2002-11-12 15:22:19 +00:00
|
|
|
break;
|
|
|
|
case FWASREQEUI:
|
2003-01-30 05:18:35 +00:00
|
|
|
fwdev = fw_noderesolve_eui64(sc->fc,
|
2003-02-09 10:14:22 +00:00
|
|
|
&asyreq->req.dst.eui);
|
2002-11-12 15:22:19 +00:00
|
|
|
if (fwdev == NULL) {
|
2003-01-06 08:07:20 +00:00
|
|
|
device_printf(sc->fc->bdev,
|
|
|
|
"cannot find node\n");
|
2002-11-12 15:22:19 +00:00
|
|
|
err = EINVAL;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
xfer->dst = fwdev->dst;
|
2003-04-17 03:38:03 +00:00
|
|
|
fp->mode.hdr.dst = FWLOCALBUS | xfer->dst;
|
2002-11-12 15:22:19 +00:00
|
|
|
break;
|
|
|
|
case FWASRESTL:
|
|
|
|
/* XXX what's this? */
|
|
|
|
break;
|
|
|
|
case FWASREQSTREAM:
|
|
|
|
/* nothing to do */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
xfer->spd = asyreq->req.sped;
|
|
|
|
bcopy(fp, xfer->send.buf, xfer->send.len);
|
|
|
|
xfer->act.hand = fw_asy_callback;
|
|
|
|
err = fw_asyreq(sc->fc, sub, xfer);
|
|
|
|
if(err){
|
|
|
|
fw_xfer_free( xfer);
|
|
|
|
return err;
|
|
|
|
}
|
2003-03-02 16:54:40 +00:00
|
|
|
err = tsleep(xfer, FWPRI, "asyreq", hz);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(err == 0){
|
|
|
|
if(asyreq->req.len >= xfer->recv.len){
|
|
|
|
asyreq->req.len = xfer->recv.len;
|
|
|
|
}else{
|
|
|
|
err = EINVAL;
|
|
|
|
}
|
2003-04-17 03:38:03 +00:00
|
|
|
bcopy(xfer->recv.buf, fp, asyreq->req.len);
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
|
|
|
error:
|
|
|
|
fw_xfer_free( xfer);
|
|
|
|
break;
|
|
|
|
case FW_IBUSRST:
|
|
|
|
sc->fc->ibr(sc->fc);
|
|
|
|
break;
|
|
|
|
case FW_CBINDADDR:
|
|
|
|
fwb = fw_bindlookup(sc->fc,
|
|
|
|
bindreq->start.hi, bindreq->start.lo);
|
|
|
|
if(fwb == NULL){
|
|
|
|
err = EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
STAILQ_REMOVE(&sc->fc->binds, fwb, fw_bind, fclist);
|
|
|
|
STAILQ_REMOVE(&sc->fc->ir[sub]->binds, fwb, fw_bind, chlist);
|
2003-02-01 15:04:33 +00:00
|
|
|
free(fwb, M_FW);
|
2002-11-12 15:22:19 +00:00
|
|
|
break;
|
|
|
|
case FW_SBINDADDR:
|
|
|
|
if(bindreq->len <= 0 ){
|
|
|
|
err = EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if(bindreq->start.hi > 0xffff ){
|
|
|
|
err = EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
2003-02-01 15:04:33 +00:00
|
|
|
fwb = (struct fw_bind *)malloc(sizeof (struct fw_bind), M_FW, M_NOWAIT);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(fwb == NULL){
|
|
|
|
err = ENOMEM;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
fwb->start_hi = bindreq->start.hi;
|
|
|
|
fwb->start_lo = bindreq->start.lo;
|
|
|
|
fwb->addrlen = bindreq->len;
|
2003-04-17 03:38:03 +00:00
|
|
|
fwb->sub = sub;
|
|
|
|
fwb->act_type = FWACT_CH;
|
2002-11-12 15:22:19 +00:00
|
|
|
|
2003-02-03 07:33:31 +00:00
|
|
|
xfer = fw_xfer_alloc(M_FWXFER);
|
2002-11-12 15:22:19 +00:00
|
|
|
if(xfer == NULL){
|
|
|
|
err = ENOMEM;
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
xfer->fc = sc->fc;
|
|
|
|
|
2003-04-17 03:38:03 +00:00
|
|
|
s = splfw();
|
|
|
|
/* XXX broken. need multiple xfer */
|
|
|
|
STAILQ_INIT(&fwb->xferlist);
|
|
|
|
STAILQ_INSERT_TAIL(&fwb->xferlist, xfer, link);
|
|
|
|
splx(s);
|
2002-11-12 15:22:19 +00:00
|
|
|
err = fw_bindadd(sc->fc, fwb);
|
|
|
|
break;
|
|
|
|
case FW_GDEVLST:
|
2003-01-25 14:47:33 +00:00
|
|
|
i = len = 1;
|
|
|
|
/* myself */
|
|
|
|
devinfo = &fwdevlst->dev[0];
|
|
|
|
devinfo->dst = sc->fc->nodeid;
|
|
|
|
devinfo->status = 0; /* XXX */
|
|
|
|
devinfo->eui.hi = sc->fc->eui.hi;
|
|
|
|
devinfo->eui.lo = sc->fc->eui.lo;
|
2003-02-01 14:42:49 +00:00
|
|
|
STAILQ_FOREACH(fwdev, &sc->fc->devices, link) {
|
2003-01-25 14:47:33 +00:00
|
|
|
if(len < FW_MAX_DEVLST){
|
|
|
|
devinfo = &fwdevlst->dev[len++];
|
|
|
|
devinfo->dst = fwdev->dst;
|
|
|
|
devinfo->status =
|
|
|
|
(fwdev->status == FWDEVINVAL)?0:1;
|
|
|
|
devinfo->eui.hi = fwdev->eui.hi;
|
|
|
|
devinfo->eui.lo = fwdev->eui.lo;
|
2002-11-12 15:22:19 +00:00
|
|
|
}
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
fwdevlst->n = i;
|
2003-01-25 14:47:33 +00:00
|
|
|
fwdevlst->info_len = len;
|
2002-11-12 15:22:19 +00:00
|
|
|
break;
|
|
|
|
case FW_GTPMAP:
|
|
|
|
bcopy(sc->fc->topology_map, data,
|
|
|
|
(sc->fc->topology_map->crc_len + 1) * 4);
|
|
|
|
break;
|
|
|
|
case FW_GCROM:
|
2003-02-01 14:42:49 +00:00
|
|
|
STAILQ_FOREACH(fwdev, &sc->fc->devices, link)
|
|
|
|
if (FW_EUI64_EQUAL(fwdev->eui, crom_buf->eui))
|
2002-11-12 15:22:19 +00:00
|
|
|
break;
|
|
|
|
if (fwdev == NULL) {
|
|
|
|
err = FWNODE_INVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (fwdev->rommax < CSRROMOFF)
|
|
|
|
len = 0;
|
|
|
|
else
|
|
|
|
len = fwdev->rommax - CSRROMOFF + 4;
|
|
|
|
if (crom_buf->len < len)
|
|
|
|
len = crom_buf->len;
|
|
|
|
else
|
|
|
|
crom_buf->len = len;
|
|
|
|
err = copyout(&fwdev->csrrom[0], crom_buf->ptr, len);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
sc->fc->ioctl (dev, cmd, data, flag, td);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
int
|
|
|
|
fw_poll(dev_t dev, int events, fw_proc *td)
|
|
|
|
{
|
|
|
|
int revents;
|
|
|
|
int tmp;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
int sub = DEV2DMACH(dev);
|
|
|
|
struct firewire_softc *sc;
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
|
|
|
return fwmem_poll(dev, events, td);
|
|
|
|
|
|
|
|
sc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
revents = 0;
|
|
|
|
tmp = POLLIN | POLLRDNORM;
|
|
|
|
if (events & tmp) {
|
|
|
|
if (STAILQ_FIRST(&sc->fc->ir[sub]->q) != NULL)
|
|
|
|
revents |= tmp;
|
|
|
|
else
|
|
|
|
selrecord(td, &sc->fc->ir[sub]->rsel);
|
|
|
|
}
|
|
|
|
tmp = POLLOUT | POLLWRNORM;
|
|
|
|
if (events & tmp) {
|
|
|
|
/* XXX should be fixed */
|
|
|
|
revents |= tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return revents;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2003-04-17 03:38:03 +00:00
|
|
|
#if __FreeBSD_version < 500102
|
2003-02-27 12:51:24 +00:00
|
|
|
fw_mmap (dev_t dev, vm_offset_t offset, int nproto)
|
|
|
|
#else
|
2003-04-17 03:38:03 +00:00
|
|
|
fw_mmap (dev_t dev, vm_offset_t offset, vm_paddr_t *paddr, int nproto)
|
2003-02-27 12:51:24 +00:00
|
|
|
#endif
|
2002-11-12 15:22:19 +00:00
|
|
|
{
|
|
|
|
struct firewire_softc *fc;
|
|
|
|
int unit = DEV2UNIT(dev);
|
|
|
|
|
|
|
|
if (DEV_FWMEM(dev))
|
2003-04-17 03:38:03 +00:00
|
|
|
#if __FreeBSD_version < 500102
|
2003-02-27 12:51:24 +00:00
|
|
|
return fwmem_mmap(dev, offset, nproto);
|
|
|
|
#else
|
2003-02-25 03:21:22 +00:00
|
|
|
return fwmem_mmap(dev, offset, paddr, nproto);
|
2003-02-27 12:51:24 +00:00
|
|
|
#endif
|
2002-11-12 15:22:19 +00:00
|
|
|
|
|
|
|
fc = devclass_get_softc(firewire_devclass, unit);
|
|
|
|
|
|
|
|
return EINVAL;
|
|
|
|
}
|