NetBSD/sys/arch/dreamcast/dev/gdrom.c
jdolecek e0cc03a09b merge kqueue branch into -current
kqueue provides a stateful and efficient event notification framework
currently supported events include socket, file, directory, fifo,
pipe, tty and device changes, and monitoring of processes and signals

kqueue is supported by all writable filesystems in NetBSD tree
(with exception of Coda) and all device drivers supporting poll(2)

based on work done by Jonathan Lemon for FreeBSD
initial NetBSD port done by Luke Mewburn and Jason Thorpe
2002-10-23 09:10:23 +00:00

584 lines
12 KiB
C

/* $NetBSD: gdrom.c,v 1.14 2002/10/23 09:10:59 jdolecek Exp $ */
/*-
* Copyright (c) 2001 Marcus Comstedt
* 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 following acknowledgement:
* This product includes software developed by Marcus Comstedt.
* 4. Neither the name of The NetBSD Foundation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``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 FOUNDATION OR CONTRIBUTORS
* 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.
*/
#include <sys/cdefs.h> /* RCS ID & Copyright macro defns */
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/buf.h>
#include <sys/ioctl.h>
#include <sys/fcntl.h>
#include <sys/disklabel.h>
#include <sys/disk.h>
#include <sys/cdio.h>
#include <sys/proc.h>
#include <sys/conf.h>
#include <machine/sysasicvar.h>
int gdrommatch(struct device *, struct cfdata *, void *);
void gdromattach(struct device *, struct device *, void *);
dev_type_open(gdromopen);
dev_type_close(gdromclose);
dev_type_read(gdromread);
dev_type_write(gdromwrite);
dev_type_ioctl(gdromioctl);
dev_type_strategy(gdromstrategy);
const struct bdevsw gdrom_bdevsw = {
gdromopen, gdromclose, gdromstrategy, gdromioctl, nodump,
nosize, D_DISK
};
const struct cdevsw gdrom_cdevsw = {
gdromopen, gdromclose, gdromread, gdromwrite, gdromioctl,
nostop, notty, nopoll, nommap, nokqfilter, D_DISK
};
struct gdrom_softc {
struct device sc_dv; /* generic device info; must come first */
struct disk dkdev; /* generic disk info */
struct buf curbuf; /* state of current I/O operation */
int is_open, is_busy;
int openpart_start; /* start sector of currently open partition */
int cmd_active;
void *cmd_result_buf; /* where to store result data (16 bit aligned) */
int cmd_result_size; /* number of bytes allocated for buf */
int cmd_actual; /* number of bytes actually read */
int cmd_cond; /* resulting condition of command */
};
CFATTACH_DECL(gdrom, sizeof(struct gdrom_softc),
gdrommatch, gdromattach, NULL, NULL);
struct dkdriver gdromdkdriver = { gdromstrategy };
extern struct cfdriver gdrom_cd;
struct gd_toc {
unsigned int entry[99];
unsigned int first, last;
unsigned int leadout;
};
#define TOC_LBA(n) ((n) & 0xffffff00)
#define TOC_ADR(n) ((n) & 0x0f)
#define TOC_CTRL(n) (((n) & 0xf0) >> 4)
#define TOC_TRACK(n) (((n) & 0x0000ff00) >> 8)
#define GDROM(o) (*(volatile unsigned char *)(0xa05f7000 + (o)))
#define GDSTATSTAT(n) ((n) & 0xf)
#define GDSTATDISK(n) (((n) >> 4) & 0xf)
#define GDROM_BUSY GDROM(0x18)
#define GDROM_DATA (*(volatile short *) & GDROM(0x80))
#define GDROM_REGX GDROM(0x84)
#define GDROM_STAT GDROM(0x8c)
#define GDROM_CNTLO GDROM(0x90)
#define GDROM_CNTHI GDROM(0x94)
#define GDROM_COND GDROM(0x9c)
int gdrom_getstat(void);
int gdrom_do_command(struct gdrom_softc *, void *, void *, unsigned int);
int gdrom_command_sense(struct gdrom_softc *, void *, void *, unsigned int);
int gdrom_read_toc(struct gdrom_softc *, struct gd_toc *);
int gdrom_read_sectors(struct gdrom_softc *, void *, int, int);
int gdrom_mount_disk(struct gdrom_softc *);
int gdrom_intr(void *);
int gdrom_getstat()
{
int s1, s2, s3;
if (GDROM_BUSY & 0x80)
return (-1);
s1 = GDROM_STAT;
s2 = GDROM_STAT;
s3 = GDROM_STAT;
if(GDROM_BUSY & 0x80)
return (-1);
if(s1 == s2)
return (s1);
else if(s2 == s3)
return (s2);
else
return (-1);
}
int
gdrom_intr(void *arg)
{
struct gdrom_softc *sc = arg;
int s, cond;
s = splbio();
cond = GDROM_COND;
#ifdef GDROMDEBUG
printf("GDROM: cond = %x\n", cond);
#endif
if(!sc->cmd_active) {
#ifdef GDROMDEBUG
printf("GDROM: inactive IRQ!?\n");
#endif
splx(s);
return (0);
}
if((cond & 8)) {
int cnt = (GDROM_CNTHI << 8) | GDROM_CNTLO;
#ifdef GDROMDEBUG
printf("GDROM: cnt = %d\n", cnt);
#endif
sc->cmd_actual += cnt;
if(cnt > 0 && sc->cmd_result_size > 0) {
int subcnt = (cnt > sc->cmd_result_size ?
sc->cmd_result_size : cnt);
int16_t *ptr = sc->cmd_result_buf;
sc->cmd_result_buf = ((char *)sc->cmd_result_buf) +
subcnt;
sc->cmd_result_size -= subcnt;
cnt -= subcnt;
while (subcnt > 0) {
*ptr++ = GDROM_DATA;
subcnt -= 2;
}
}
while (cnt > 0) {
__volatile int16_t tmp;
tmp = GDROM_DATA;
cnt -= 2;
}
}
while (GDROM_BUSY & 0x80);
if ((cond & 8) == 0) {
sc->cmd_cond = cond;
sc->cmd_active = 0;
wakeup(&sc->cmd_active);
}
splx(s);
return (0);
}
int gdrom_do_command(struct gdrom_softc *sc, void *req, void *buf,
unsigned int nbyt)
{
int i, s;
short *ptr = req;
while (GDROM_BUSY & 0x88)
;
if (buf != NULL) {
GDROM_CNTLO = nbyt & 0xff;
GDROM_CNTHI = (nbyt >> 8) & 0xff;
GDROM_REGX = 0;
}
sc->cmd_result_buf = buf;
sc->cmd_result_size = nbyt;
if (GDSTATSTAT(GDROM_STAT) == 6)
return (-1);
GDROM_COND = 0xa0;
for (i = 0; i < 64; i++)
;
while ((GDROM_BUSY & 0x88) != 8)
;
s = splbio();
sc->cmd_actual = 0;
sc->cmd_active = 1;
for (i = 0; i< 6; i++)
GDROM_DATA = ptr[i];
while (sc->cmd_active)
tsleep(&sc->cmd_active, PRIBIO, "gdrom", 0);
splx(s);
return (sc->cmd_cond);
}
int gdrom_command_sense(struct gdrom_softc *sc, void *req, void *buf,
unsigned int nbyt)
{
/* 76543210 76543210
0 0x13 -
2 - bufsz(hi)
4 bufsz(lo) -
6 - -
8 - -
10 - - */
unsigned short sense_data[5];
unsigned char cmd[12];
int sense_key, sense_specific;
int cond = gdrom_do_command(sc, req, buf, nbyt);
if (cond < 0) {
#ifdef GDROMDEBUG
printf("GDROM: not ready (2:58)\n");
#endif
return (EIO);
}
if ((cond & 1) == 0) {
#ifdef GDROMDEBUG
printf("GDROM: no sense. 0:0\n");
#endif
return (0);
}
memset(cmd, 0, sizeof(cmd));
cmd[0] = 0x13;
cmd[4] = sizeof(sense_data);
gdrom_do_command(sc, cmd, sense_data, sizeof(sense_data));
sense_key = sense_data[1] & 0xf;
sense_specific = sense_data[4];
if (sense_key == 11 && sense_specific == 0) {
#ifdef GDROMDEBUG
printf("GDROM: aborted (ignored). 0:0\n");
#endif
return (0);
}
#ifdef GDROMDEBUG
printf("GDROM: SENSE %d:", sense_key);
printf("GDROM: %d\n", sense_specific);
#endif
return (sense_key == 0 ? 0 : EIO);
}
int gdrom_read_toc(struct gdrom_softc *sc, struct gd_toc *toc)
{
/* 76543210 76543210
0 0x14 -
2 - bufsz(hi)
4 bufsz(lo) -
6 - -
8 - -
10 - - */
unsigned char cmd[12];
memset(cmd, 0, sizeof(cmd));
cmd[0] = 0x14;
cmd[3] = sizeof(struct gd_toc) >> 8;
cmd[4] = sizeof(struct gd_toc) & 0xff;
return (gdrom_command_sense(sc, cmd, toc, sizeof(struct gd_toc)));
}
int gdrom_read_sectors(struct gdrom_softc *sc, void *buf, int sector, int cnt)
{
/* 76543210 76543210
0 0x30 datafmt
2 sec(hi) sec(mid)
4 sec(lo) -
6 - -
8 cnt(hi) cnt(mid)
10 cnt(lo) - */
unsigned char cmd[12];
memset(cmd, 0, sizeof(cmd));
cmd[0] = 0x30;
cmd[1] = 0x20;
cmd[2] = sector>>16;
cmd[3] = sector>>8;
cmd[4] = sector;
cmd[8] = cnt>>16;
cmd[9] = cnt>>8;
cmd[10] = cnt;
return (gdrom_command_sense(sc, cmd, buf, cnt << 11));
}
int gdrom_mount_disk(struct gdrom_softc *sc)
{
/* 76543210 76543210
0 0x70 -
2 0x1f -
4 - -
6 - -
8 - -
10 - - */
unsigned char cmd[12];
memset(cmd, 0, sizeof(cmd));
cmd[0] = 0x70;
cmd[1] = 0x1f;
return (gdrom_command_sense(sc, cmd, NULL, 0));
}
int
gdrommatch(struct device *parent, struct cfdata *cf, void *aux)
{
static int gdrom_matched = 0;
/* Allow only once instance. */
if (gdrom_matched)
return (0);
gdrom_matched = 1;
return (1);
}
void
gdromattach(struct device *parent, struct device *self, void *aux)
{
struct gdrom_softc *sc;
u_int32_t p, x;
sc = (struct gdrom_softc *)self;
printf(": SH4 IRL 9\n");
/*
* Initialize and attach the disk structure.
*/
sc->dkdev.dk_name = sc->sc_dv.dv_xname;
sc->dkdev.dk_driver = &gdromdkdriver;
disk_attach(&sc->dkdev);
/*
* reenable disabled drive
*/
*((__volatile u_int32_t *)0xa05f74e4) = 0x1fffff;
for (p = 0; p < 0x200000 / 4; p++)
x = ((__volatile u_int32_t *)0xa0000000)[p];
sysasic_intr_establish(SYSASIC_EVENT_GDROM, gdrom_intr, sc);
}
int
gdromopen(dev_t dev, int flags, int devtype, struct proc *p)
{
struct gdrom_softc *sc;
int s, error, unit, cnt;
struct gd_toc toc;
#ifdef GDROMDEBUG
printf("GDROM: open\n");
#endif
unit = DISKUNIT(dev);
if (unit >= gdrom_cd.cd_ndevs)
return (ENXIO);
sc = gdrom_cd.cd_devs[unit];
if (sc == NULL)
return (ENXIO);
if (sc->is_open)
return (EBUSY);
s = splbio();
while(sc->is_busy)
tsleep(&sc->is_busy, PRIBIO, "gdbusy", 0);
sc->is_busy = 1;
splx(s);
for (cnt = 0; cnt < 5; cnt++)
if ((error = gdrom_mount_disk(sc)) == 0)
break;
if (!error)
error = gdrom_read_toc(sc, &toc);
sc->is_busy = 0;
wakeup(&sc->is_busy);
if (error)
return error;
sc->is_open = 1;
sc->openpart_start = 150;
#ifdef GDROMDEBUG
printf("GDROM: open OK\n");
#endif
return (0);
}
int
gdromclose(dev_t dev, int flags, int devtype, struct proc *p)
{
struct gdrom_softc *sc;
int unit;
#ifdef GDROMDEBUG
printf("GDROM: close\n");
#endif
unit = DISKUNIT(dev);
sc = gdrom_cd.cd_devs[unit];
sc->is_open = 0;
return (0);
}
void
gdromstrategy(struct buf *bp)
{
struct gdrom_softc *sc;
int s, unit, error;
#ifdef GDROMDEBUG
printf("GDROM: strategy\n");
#endif
unit = DISKUNIT(bp->b_dev);
sc = gdrom_cd.cd_devs[unit];
if (bp->b_bcount == 0)
goto done;
bp->b_rawblkno = bp->b_blkno / (2048 / DEV_BSIZE) + sc->openpart_start;
#ifdef GDROMDEBUG
printf("GDROM: read_sectors(%p, %d, %ld) [%ld bytes]\n",
bp->b_data, bp->b_rawblkno,
bp->b_bcount>>11, bp->b_bcount);
#endif
s = splbio();
while (sc->is_busy)
tsleep(&sc->is_busy, PRIBIO, "gdbusy", 0);
sc->is_busy = 1;
splx(s);
if ((error = gdrom_read_sectors(sc, bp->b_data, bp->b_rawblkno,
bp->b_bcount >> 11))) {
bp->b_error = error;
bp->b_flags |= B_ERROR;
}
sc->is_busy = 0;
wakeup(&sc->is_busy);
done:
bp->b_resid = bp->b_bcount;
biodone(bp);
}
int
gdromioctl(dev_t dev, u_long cmd, caddr_t addr, int flag, struct proc *p)
{
struct gdrom_softc *sc;
int unit, error;
#ifdef GDROMDEBUG
printf("GDROM: ioctl %lx\n", cmd);
#endif
unit = DISKUNIT(dev);
sc = gdrom_cd.cd_devs[unit];
switch (cmd) {
case CDIOREADMSADDR: {
int s, track, sessno = *(int *)addr;
struct gd_toc toc;
if (sessno != 0)
return (EINVAL);
s = splbio();
while (sc->is_busy)
tsleep(&sc->is_busy, PRIBIO, "gdbusy", 0);
sc->is_busy = 1;
splx(s);
error = gdrom_read_toc(sc, &toc);
sc->is_busy = 0;
wakeup(&sc->is_busy);
if (error)
return error;
for (track = TOC_TRACK(toc.last);
track >= TOC_TRACK(toc.first);
--track)
if (TOC_CTRL(toc.entry[track-1]))
break;
if (track < TOC_TRACK(toc.first) || track > 100)
return (ENXIO);
*(int *)addr = htonl(TOC_LBA(toc.entry[track-1])) -
sc->openpart_start;
return (0);
}
default:
return (EINVAL);
}
#ifdef DIAGNOSTIC
panic("gdromioctl: impossible");
#endif
}
int
gdromread(dev_t dev, struct uio *uio, int flags)
{
#ifdef GDROMDEBUG
printf("GDROM: read\n");
#endif
return (physio(gdromstrategy, NULL, dev, B_READ, minphys, uio));
}
int
gdromwrite(dev_t dev, struct uio *uio, int flags)
{
return (EROFS);
}