1995-05-30 08:16:23 +00:00
|
|
|
|
/*
|
1995-03-04 20:51:10 +00:00
|
|
|
|
* worm: Write Once device driver
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 1995, HD Associates, Inc.
|
|
|
|
|
* PO Box 276
|
|
|
|
|
* Pepperell, MA 01463
|
|
|
|
|
* 508 433 5266
|
|
|
|
|
* dufault@hda.com
|
|
|
|
|
*
|
1997-12-26 14:52:29 +00:00
|
|
|
|
* Copyright (C) 1996-97 interface business GmbH
|
|
|
|
|
* Naumannstr. 1
|
|
|
|
|
* D-01309 Dresden
|
1996-01-27 17:48:12 +00:00
|
|
|
|
* F.R. Germany
|
|
|
|
|
* <joerg_wunsch@interface-business.de>
|
|
|
|
|
*
|
1995-03-04 20:51:10 +00:00
|
|
|
|
* This code is contributed to the University of California at Berkeley:
|
|
|
|
|
*
|
|
|
|
|
* 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 the University of
|
|
|
|
|
* California, Berkeley and its contributors.
|
|
|
|
|
* 4. Neither the name of the University 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 REGENTS 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 REGENTS 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.
|
|
|
|
|
*
|
1998-01-24 02:54:56 +00:00
|
|
|
|
* $Id: worm.c,v 1.50 1997/12/26 14:52:29 joerg Exp $
|
1995-03-04 20:51:10 +00:00
|
|
|
|
*/
|
|
|
|
|
|
1996-01-05 20:12:53 +00:00
|
|
|
|
#include "opt_bounce.h"
|
1998-01-24 02:54:56 +00:00
|
|
|
|
#include "opt_devfs.h"
|
1996-07-14 10:46:56 +00:00
|
|
|
|
#include "opt_scsi.h"
|
1996-01-05 20:12:53 +00:00
|
|
|
|
|
1995-03-04 20:51:10 +00:00
|
|
|
|
#include <sys/param.h>
|
|
|
|
|
#include <sys/systm.h>
|
|
|
|
|
#include <sys/buf.h>
|
1997-05-05 13:35:01 +00:00
|
|
|
|
#include <sys/cdio.h>
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#include <sys/wormio.h>
|
|
|
|
|
#include <sys/fcntl.h>
|
1995-11-29 10:49:16 +00:00
|
|
|
|
#include <sys/conf.h>
|
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
|
#ifdef DEVFS
|
|
|
|
|
#include <sys/devfsext.h>
|
|
|
|
|
#endif /*DEVFS*/
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#include <scsi/scsiconf.h>
|
1997-03-23 06:33:55 +00:00
|
|
|
|
#include <scsi/scsi_debug.h>
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#include <scsi/scsi_disk.h>
|
1997-03-23 06:33:55 +00:00
|
|
|
|
#include <scsi/scsi_driver.h>
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#include <scsi/scsi_worm.h>
|
1997-05-05 13:35:01 +00:00
|
|
|
|
#include <scsi/scsi_cd.h>
|
1997-04-04 22:12:55 +00:00
|
|
|
|
#include <sys/dkstat.h>
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
struct worm_quirks
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* The device-specific functions that need to be called during
|
|
|
|
|
* the several steps.
|
|
|
|
|
*/
|
|
|
|
|
errval (*prepare_disk)(struct scsi_link *, int dummy, int speed);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
errval (*prepare_track)(struct scsi_link *, struct wormio_prepare_track *t);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
errval (*finalize_track)(struct scsi_link *);
|
|
|
|
|
errval (*finalize_disk)(struct scsi_link *, int toc_type, int onp);
|
1997-06-02 20:05:39 +00:00
|
|
|
|
errval (*write_session)(struct scsi_link *, struct wormio_write_session *);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
};
|
1995-11-29 10:49:16 +00:00
|
|
|
|
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_data
|
|
|
|
|
{
|
1995-11-19 22:22:35 +00:00
|
|
|
|
struct buf_queue_head buf_queue;
|
1997-04-04 22:12:55 +00:00
|
|
|
|
int dkunit; /* disk stats unit number */
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t blk_size; /* Size of each blocks */
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#ifdef DEVFS
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
void *b_devfs_token;
|
|
|
|
|
void *c_devfs_token;
|
|
|
|
|
void *ctl_devfs_token;
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#endif
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
struct worm_quirks *quirks; /* model-specific functions */
|
1997-12-26 14:52:29 +00:00
|
|
|
|
struct wormio_prepare_track preptrack; /* scratch region */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1996-03-10 12:52:47 +00:00
|
|
|
|
u_int8_t dummy; /* use dummy writes */
|
|
|
|
|
u_int8_t speed; /* select drive speed */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t worm_flags; /* driver-internal flags */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#define WORMFL_DISK_PREPED 0x01 /* disk parameters have been spec'ed */
|
1997-12-26 14:52:29 +00:00
|
|
|
|
#define WORMFL_TRACK_PREPED 0x02 /* track parameters have been sent */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#define WORMFL_WRITTEN 0x04 /* track has been written */
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
#define WORMFL_IOCTL_ONLY 0x08 /* O_NDELAY, only ioctls allowed */
|
1997-12-26 14:52:29 +00:00
|
|
|
|
#define WORMFL_TRACK_PREP 0x10 /* track parameters have been spec'ed */
|
1997-05-19 17:30:50 +00:00
|
|
|
|
|
|
|
|
|
int error; /* last error */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
int asc;
|
|
|
|
|
int devmode;
|
|
|
|
|
int error;
|
|
|
|
|
int ret;
|
|
|
|
|
} worm_error[] = {
|
|
|
|
|
{0x24, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_ABSORPTION_CONTROL_ERROR, 0},
|
|
|
|
|
{0xb0, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_CALIBRATION_AREA_ALMOST_FULL, 0},
|
|
|
|
|
{0xb4, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_CALIBRATION_AREA_FULL, SCSIRET_CONTINUE},
|
|
|
|
|
{0xb5, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_DUMMY_BLOCKS_ADDED, 0},
|
|
|
|
|
{0xaa, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_END_OF_MEDIUM, SCSIRET_CONTINUE},
|
|
|
|
|
{0xad, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_BUFFER_UNDERRUN, SCSIRET_CONTINUE},
|
|
|
|
|
{0xaf, WORM_Q_PLASMON|WORM_Q_PHILIPS, WORM_OPTIMUM_POWER_CALIBRATION_ERROR, SCSIRET_CONTINUE},
|
|
|
|
|
{0, 0, 0, 0}
|
1995-03-04 20:51:10 +00:00
|
|
|
|
};
|
|
|
|
|
|
1996-03-10 07:13:15 +00:00
|
|
|
|
static void wormstart(u_int32_t unit, u_int32_t flags);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static errval worm_open(dev_t dev, int flags, int fmt, struct proc *p,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_link *sc_link);
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static errval worm_ioctl(dev_t dev, int cmd, caddr_t addr, int flag,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct proc *p, struct scsi_link *sc_link);
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static errval worm_close(dev_t dev, int flag, int fmt, struct proc *p,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_link *sc_link);
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static void worm_strategy(struct buf *bp, struct scsi_link *sc_link);
|
1997-05-05 13:35:01 +00:00
|
|
|
|
static errval worm_read_toc(struct scsi_link *sc_link,
|
|
|
|
|
u_int32_t mode, u_int32_t start,
|
|
|
|
|
struct cd_toc_entry *data, u_int32_t len);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
static errval worm_rezero_unit(struct scsi_link *sc_link);
|
1997-06-02 20:05:39 +00:00
|
|
|
|
static errval worm_read_session_info(struct scsi_link *, struct wormio_session_info *);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static int worm_sense_handler(struct scsi_xfer *);
|
1997-10-27 21:09:27 +00:00
|
|
|
|
static errval worm_set_blksize(struct scsi_link *sc_link, int size);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
/* XXX should be moved out to an LKM */
|
|
|
|
|
static errval rf4100_prepare_disk(struct scsi_link *, int dummy, int speed);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static errval rf4100_prepare_track(struct scsi_link *, struct wormio_prepare_track *);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
static errval rf4100_finalize_track(struct scsi_link *);
|
|
|
|
|
static errval rf4100_finalize_disk(struct scsi_link *, int toc_type, int onp);
|
|
|
|
|
|
1996-02-02 22:59:48 +00:00
|
|
|
|
static errval hp4020i_prepare_disk(struct scsi_link *, int dummy, int speed);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static errval hp4020i_prepare_track(struct scsi_link *, struct wormio_prepare_track *);
|
1996-02-02 22:59:48 +00:00
|
|
|
|
static errval hp4020i_finalize_track(struct scsi_link *);
|
|
|
|
|
static errval hp4020i_finalize_disk(struct scsi_link *, int toc_type, int onp);
|
1997-06-02 20:05:39 +00:00
|
|
|
|
static errval hp4020i_write_session(struct scsi_link *, struct wormio_write_session *);
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
static worm_devsw_installed = 0;
|
|
|
|
|
|
1995-12-08 11:19:42 +00:00
|
|
|
|
static d_open_t wormopen;
|
|
|
|
|
static d_close_t wormclose;
|
|
|
|
|
static d_ioctl_t wormioctl;
|
1995-12-14 09:55:16 +00:00
|
|
|
|
static d_strategy_t wormstrategy;
|
1995-12-08 11:19:42 +00:00
|
|
|
|
|
|
|
|
|
#define CDEV_MAJOR 62
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
#define BDEV_MAJOR 23
|
|
|
|
|
static struct cdevsw worm_cdevsw;
|
|
|
|
|
static struct bdevsw worm_bdevsw =
|
|
|
|
|
{ wormopen, wormclose, wormstrategy, wormioctl, /*23*/
|
1997-05-10 12:13:17 +00:00
|
|
|
|
nodump, nopsize, D_DISK, "worm", &worm_cdevsw, -1 };
|
1995-12-08 11:19:42 +00:00
|
|
|
|
|
|
|
|
|
|
1996-01-02 15:44:00 +00:00
|
|
|
|
static int
|
1996-01-28 09:56:04 +00:00
|
|
|
|
wormunit(dev_t dev)
|
|
|
|
|
{
|
1996-01-02 15:44:00 +00:00
|
|
|
|
return (minor(dev) & ~(SCSI_FIXED_MASK|SCSI_CONTROL_MASK));
|
|
|
|
|
}
|
|
|
|
|
|
1995-03-04 20:51:10 +00:00
|
|
|
|
SCSI_DEVICE_ENTRIES(worm)
|
|
|
|
|
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static struct scsi_device worm_switch =
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{
|
1997-05-19 17:30:50 +00:00
|
|
|
|
worm_sense_handler,
|
1995-12-08 11:19:42 +00:00
|
|
|
|
wormstart, /* we have a queue, and this is how we service it */
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
"worm",
|
|
|
|
|
0,
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{0, 0},
|
|
|
|
|
SDEV_ONCE_ONLY, /* Only one open allowed */
|
|
|
|
|
wormattach,
|
|
|
|
|
"Write-Once",
|
|
|
|
|
wormopen,
|
1995-12-08 11:19:42 +00:00
|
|
|
|
sizeof(struct scsi_data),
|
1995-03-04 20:51:10 +00:00
|
|
|
|
T_WORM,
|
1996-01-02 15:44:00 +00:00
|
|
|
|
wormunit,
|
1995-03-04 20:51:10 +00:00
|
|
|
|
0,
|
|
|
|
|
worm_open,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
worm_ioctl,
|
1995-03-04 20:51:10 +00:00
|
|
|
|
worm_close,
|
|
|
|
|
worm_strategy,
|
|
|
|
|
};
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static struct worm_quirks worm_quirks_plasmon = {
|
|
|
|
|
rf4100_prepare_disk, rf4100_prepare_track,
|
1997-06-02 20:05:39 +00:00
|
|
|
|
rf4100_finalize_track, rf4100_finalize_disk,
|
|
|
|
|
0
|
1997-05-19 17:30:50 +00:00
|
|
|
|
};
|
|
|
|
|
static struct worm_quirks worm_quirks_philips = {
|
|
|
|
|
hp4020i_prepare_disk, hp4020i_prepare_track,
|
1997-06-02 20:05:39 +00:00
|
|
|
|
hp4020i_finalize_track, hp4020i_finalize_disk,
|
|
|
|
|
hp4020i_write_session
|
1996-01-27 17:48:12 +00:00
|
|
|
|
};
|
|
|
|
|
|
1997-04-04 22:12:55 +00:00
|
|
|
|
static inline void
|
|
|
|
|
worm_registerdev(int unit)
|
|
|
|
|
{
|
|
|
|
|
if (dk_ndrive < DK_NDRIVE) {
|
|
|
|
|
sprintf(dk_names[dk_ndrive], "worm%d", unit);
|
|
|
|
|
dk_wpms[dk_ndrive] = (1*1024*1024/2); /* 1MB/sec XXX - fake! */
|
|
|
|
|
SCSI_DATA(&worm_switch, unit)->dkunit = dk_ndrive++;
|
|
|
|
|
} else {
|
|
|
|
|
SCSI_DATA(&worm_switch, unit)->dkunit = -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
static errval
|
|
|
|
|
worm_size(struct scsi_link *sc_link, int flags)
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{
|
1996-01-27 17:48:12 +00:00
|
|
|
|
errval ret;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
struct scsi_data *worm = sc_link->sd;
|
1997-07-26 15:07:42 +00:00
|
|
|
|
int blk_size;
|
1997-12-26 14:52:29 +00:00
|
|
|
|
u_int32_t n_blks;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("worm_size"));
|
|
|
|
|
|
1997-12-26 14:52:29 +00:00
|
|
|
|
n_blks = scsi_read_capacity(sc_link, &blk_size, flags);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1996-11-06 13:33:53 +00:00
|
|
|
|
/*
|
|
|
|
|
* CD-R devices can assume various sizes, depending on the
|
|
|
|
|
* intended purpose of the track. Hence, READ CAPACITY
|
1997-12-26 14:52:29 +00:00
|
|
|
|
* doesn't give us any good results. We make a more educated
|
|
|
|
|
* guess when it comes to prepare a track.
|
1996-11-06 13:33:53 +00:00
|
|
|
|
*/
|
|
|
|
|
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if (n_blks > 0) {
|
1995-03-04 20:51:10 +00:00
|
|
|
|
sc_link->flags |= SDEV_MEDIA_LOADED;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
ret = 0;
|
1997-12-26 14:52:29 +00:00
|
|
|
|
} else {
|
1995-03-04 20:51:10 +00:00
|
|
|
|
sc_link->flags &= ~SDEV_MEDIA_LOADED;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
ret = ENXIO;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static errval
|
1995-03-04 20:51:10 +00:00
|
|
|
|
wormattach(struct scsi_link *sc_link)
|
|
|
|
|
{
|
1997-04-04 22:12:55 +00:00
|
|
|
|
u_int32_t unit = sc_link->dev_unit;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#ifdef DEVFS
|
|
|
|
|
int mynor;
|
|
|
|
|
#endif
|
1995-03-04 20:51:10 +00:00
|
|
|
|
struct scsi_data *worm = sc_link->sd;
|
|
|
|
|
|
1997-12-20 23:03:49 +00:00
|
|
|
|
if (sc_link->devmodes == 0)
|
|
|
|
|
printf(", warning: unknown drive type", unit);
|
|
|
|
|
|
1997-09-21 22:03:22 +00:00
|
|
|
|
bufq_init(&worm->buf_queue);
|
1995-11-19 22:22:35 +00:00
|
|
|
|
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#ifdef DEVFS
|
1996-01-27 17:48:12 +00:00
|
|
|
|
mynor = wormunit(sc_link->dev);
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
worm->b_devfs_token =
|
|
|
|
|
devfs_add_devswf(&worm_bdevsw, mynor,
|
|
|
|
|
DV_BLK, 0, 0, 0444, "worm%d", mynor);
|
|
|
|
|
worm->c_devfs_token =
|
1996-03-28 14:33:59 +00:00
|
|
|
|
devfs_add_devswf(&worm_cdevsw, mynor,
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
DV_CHR, 0, 0, 0644, "rworm%d", mynor);
|
|
|
|
|
worm->ctl_devfs_token =
|
1996-03-28 14:33:59 +00:00
|
|
|
|
devfs_add_devswf(&worm_cdevsw, mynor | SCSI_CONTROL_MASK,
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
DV_CHR, 0, 0, 0600, "rworm%d.ctl", mynor);
|
1995-12-08 11:19:42 +00:00
|
|
|
|
#endif
|
1997-04-04 22:12:55 +00:00
|
|
|
|
worm_registerdev(unit);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* wormstart looks to see if there is a buf waiting for the device
|
|
|
|
|
* and that the device is not already busy. If both are true,
|
|
|
|
|
* It dequeues the buf and creates a scsi command to perform the
|
|
|
|
|
* transfer required. The transfer request will call scsi_done
|
|
|
|
|
* on completion, which will in turn call this routine again
|
|
|
|
|
* so that the next queued transfer is performed.
|
|
|
|
|
* The bufs are queued by the strategy routine (wormstrategy)
|
|
|
|
|
*
|
|
|
|
|
* This routine is also called after other non-queued requests
|
|
|
|
|
* have been made of the scsi driver, to ensure that the queue
|
|
|
|
|
* continues to be drained.
|
|
|
|
|
* wormstart() is called at splbio
|
|
|
|
|
*
|
|
|
|
|
* XXX It looks like we need a "scsistart" to hoist common code up
|
|
|
|
|
* into. In particular, the removable media checking should be
|
|
|
|
|
* handled in one place.
|
1997-05-19 17:30:50 +00:00
|
|
|
|
*
|
|
|
|
|
* Writes will fail if the disk and track not been prepared via the control
|
|
|
|
|
* device.
|
1995-03-04 20:51:10 +00:00
|
|
|
|
*/
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static void
|
1995-04-23 22:07:56 +00:00
|
|
|
|
wormstart(unit, flags)
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t unit;
|
|
|
|
|
u_int32_t flags;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{
|
|
|
|
|
struct scsi_link *sc_link = SCSI_LINK(&worm_switch, unit);
|
|
|
|
|
struct scsi_data *worm = sc_link->sd;
|
|
|
|
|
register struct buf *bp = 0;
|
1995-12-05 07:14:27 +00:00
|
|
|
|
struct scsi_rw_big cmd;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
u_int32_t lba; /* Logical block address */
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t tl; /* Transfer length */
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("wormstart "));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* We should reject all queued entries if SDEV_MEDIA_LOADED is not true.
|
|
|
|
|
*/
|
|
|
|
|
if (!(sc_link->flags & SDEV_MEDIA_LOADED)) {
|
|
|
|
|
goto badnews; /* no I/O.. media changed or something */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* See if there is a buf to do and we are not already
|
|
|
|
|
* doing one
|
|
|
|
|
*/
|
|
|
|
|
while (sc_link->opennings != 0) {
|
|
|
|
|
|
|
|
|
|
/* if a special awaits, let it proceed first */
|
|
|
|
|
if (sc_link->flags & SDEV_WAITING) {
|
|
|
|
|
sc_link->flags &= ~SDEV_WAITING;
|
1995-04-23 07:39:21 +00:00
|
|
|
|
wakeup(sc_link);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
1995-11-19 22:22:35 +00:00
|
|
|
|
|
1997-09-21 22:03:22 +00:00
|
|
|
|
bp = bufq_first(&worm->buf_queue);
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if (bp == NULL)
|
1995-11-19 22:22:35 +00:00
|
|
|
|
return;
|
1997-12-26 14:52:29 +00:00
|
|
|
|
|
1997-09-21 22:03:22 +00:00
|
|
|
|
bufq_remove(&worm->buf_queue, bp);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if ((bp->b_flags & B_READ) == B_WRITE) {
|
|
|
|
|
if ((worm->worm_flags & WORMFL_TRACK_PREPED) == 0) {
|
|
|
|
|
if ((worm->worm_flags & WORMFL_TRACK_PREP) == 0) {
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB3, ("sequence error\n"));
|
|
|
|
|
bp->b_error = EIO;
|
|
|
|
|
bp->b_flags |= B_ERROR;
|
|
|
|
|
worm->error = WORM_SEQUENCE_ERROR;
|
|
|
|
|
biodone(bp);
|
|
|
|
|
goto badnews;
|
|
|
|
|
} else {
|
|
|
|
|
if (worm->quirks->prepare_track(sc_link, &worm->preptrack)
|
|
|
|
|
!= 0) {
|
|
|
|
|
biodone(bp);
|
|
|
|
|
goto badnews;
|
|
|
|
|
}
|
|
|
|
|
worm->worm_flags |= WORMFL_TRACK_PREPED;
|
|
|
|
|
}
|
|
|
|
|
}
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
1995-03-04 20:51:10 +00:00
|
|
|
|
/*
|
|
|
|
|
* Fill out the scsi command
|
|
|
|
|
*/
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
if ((bp->b_flags & B_READ) == B_WRITE) {
|
|
|
|
|
cmd.op_code = WRITE_BIG;
|
1995-04-23 22:07:56 +00:00
|
|
|
|
flags |= SCSI_DATA_OUT;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
} else {
|
|
|
|
|
cmd.op_code = READ_BIG;
|
1995-04-23 22:07:56 +00:00
|
|
|
|
flags |= SCSI_DATA_IN;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
worm->error = 0;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
lba = bp->b_blkno / (worm->blk_size / DEV_BSIZE);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
tl = bp->b_bcount / worm->blk_size;
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
if (bp->b_flags & B_READ)
|
|
|
|
|
/*
|
|
|
|
|
* Leave the LBA as 0 for write operations, it
|
|
|
|
|
* is reserved in this case (and wouldn't make
|
|
|
|
|
* any sense to set it at all, since CD-R write
|
|
|
|
|
* operations are in `streaming' mode anyway.
|
|
|
|
|
*/
|
|
|
|
|
scsi_uto4b(lba, &cmd.addr_3);
|
1995-12-05 07:14:27 +00:00
|
|
|
|
scsi_uto2b(tl, &cmd.length2);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* go ask the adapter to do all this for us
|
|
|
|
|
*/
|
|
|
|
|
if (scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
1997-12-02 21:07:20 +00:00
|
|
|
|
(u_char *) bp->b_data,
|
1995-03-04 20:51:10 +00:00
|
|
|
|
bp->b_bcount,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
0,
|
1995-03-04 20:51:10 +00:00
|
|
|
|
100000,
|
|
|
|
|
bp,
|
|
|
|
|
flags | SCSI_NOSLEEP) == SUCCESSFULLY_QUEUED) {
|
1997-04-04 22:12:55 +00:00
|
|
|
|
if (worm->dkunit >= 0) { /* Cloned from od.c, possibly with same mistakes. :) */
|
|
|
|
|
dk_xfer[worm->dkunit]++;
|
|
|
|
|
dk_seek[worm->dkunit] = 1; /* single track */
|
|
|
|
|
dk_wds[worm->dkunit] += bp->b_bcount >> 6;
|
|
|
|
|
}
|
1996-01-27 17:48:12 +00:00
|
|
|
|
if ((bp->b_flags & B_READ) == B_WRITE)
|
|
|
|
|
worm->worm_flags |= WORMFL_WRITTEN;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
} else {
|
1995-05-03 23:38:20 +00:00
|
|
|
|
printf("worm%ld: oops not queued\n", unit);
|
1995-10-09 15:15:01 +00:00
|
|
|
|
if (bp) {
|
|
|
|
|
bp->b_flags |= B_ERROR;
|
|
|
|
|
bp->b_error = EIO;
|
|
|
|
|
biodone(bp);
|
|
|
|
|
}
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
} /* go back and see if we can cram more work in.. */
|
1997-07-01 00:22:51 +00:00
|
|
|
|
badnews: ;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static void
|
1995-03-04 20:51:10 +00:00
|
|
|
|
worm_strategy(struct buf *bp, struct scsi_link *sc_link)
|
|
|
|
|
{
|
|
|
|
|
unsigned char unit;
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t opri;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
struct scsi_data *worm;
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
unit = wormunit(bp->b_dev);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
worm = sc_link->sd;
|
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
if ((worm->worm_flags & WORMFL_IOCTL_ONLY) != 0) {
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB3,
|
|
|
|
|
("attempted IO on ioctl-only descriptor\n"));
|
|
|
|
|
bp->b_error = EBADF;
|
|
|
|
|
bp->b_flags |= B_ERROR;
|
|
|
|
|
biodone(bp);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
1996-01-28 23:33:23 +00:00
|
|
|
|
/*
|
|
|
|
|
* check it's not too big a transfer for our adapter
|
|
|
|
|
*/
|
|
|
|
|
wormminphys(bp);
|
|
|
|
|
|
1995-03-04 20:51:10 +00:00
|
|
|
|
opri = splbio();
|
|
|
|
|
|
1995-05-30 08:16:23 +00:00
|
|
|
|
/*
|
1995-03-04 20:51:10 +00:00
|
|
|
|
* Use a bounce buffer if necessary
|
|
|
|
|
* XXX: How can we move this up?
|
1995-05-30 08:16:23 +00:00
|
|
|
|
*/
|
1995-03-04 20:51:10 +00:00
|
|
|
|
#ifdef BOUNCE_BUFFERS
|
|
|
|
|
if (sc_link->flags & SDEV_BOUNCE)
|
|
|
|
|
vm_bounce_alloc(bp);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Place it in the queue of activities for this device
|
|
|
|
|
* at the end.
|
|
|
|
|
*/
|
1997-09-21 22:03:22 +00:00
|
|
|
|
bufq_insert_tail(&worm->buf_queue, bp);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
1995-04-23 22:07:56 +00:00
|
|
|
|
wormstart(unit, 0);
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
|
|
|
|
splx(opri);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
1996-01-27 17:48:12 +00:00
|
|
|
|
* Open the device.
|
|
|
|
|
* Only called for the "real" device, not for the control device.
|
1995-03-04 20:51:10 +00:00
|
|
|
|
*/
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static int
|
1995-03-04 20:51:10 +00:00
|
|
|
|
worm_open(dev_t dev, int flags, int fmt, struct proc *p,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_link *sc_link)
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_data *worm;
|
|
|
|
|
errval error;
|
|
|
|
|
|
|
|
|
|
error = 0;
|
|
|
|
|
worm = sc_link->sd;
|
|
|
|
|
|
1995-03-04 20:51:10 +00:00
|
|
|
|
if (sc_link->flags & SDEV_OPEN)
|
|
|
|
|
return EBUSY;
|
|
|
|
|
|
1997-12-20 23:03:49 +00:00
|
|
|
|
/*
|
|
|
|
|
* Unknown drive type: only the control device can be opened
|
|
|
|
|
* in this case, so scsi(8) and things like cdrecord will
|
|
|
|
|
* work.
|
|
|
|
|
*/
|
|
|
|
|
if (sc_link->devmodes == 0)
|
|
|
|
|
return ENXIO;
|
|
|
|
|
|
1995-03-04 20:51:10 +00:00
|
|
|
|
/*
|
|
|
|
|
* Check that it is still responding and ok.
|
|
|
|
|
* if the media has been changed this will result in a
|
|
|
|
|
* "unit attention" error which the error code will
|
|
|
|
|
* disregard because the SDEV_OPEN flag is not yet set
|
|
|
|
|
*
|
|
|
|
|
* XXX This should REALLY be hoisted up. As soon as Bruce
|
|
|
|
|
* finishes that slice stuff. (Add a different flag,
|
|
|
|
|
* and then do a "scsi_test_unit_ready" with the "ignore
|
|
|
|
|
* unit attention" thing set. Then all this replicated
|
|
|
|
|
* test unit ready code can be pulled up.
|
|
|
|
|
*/
|
|
|
|
|
scsi_test_unit_ready(sc_link, SCSI_SILENT);
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* The semantics of the "flags" is as follows:
|
|
|
|
|
*
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
* If the device has been opened with O_NONBLOCK set, no
|
|
|
|
|
* actual IO will be allowed, and the command sequence is only
|
|
|
|
|
* subject to the restrictions as in worm_ioctl() below.
|
1996-01-27 17:48:12 +00:00
|
|
|
|
*/
|
1995-03-04 20:51:10 +00:00
|
|
|
|
/*
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
* Next time actually take notice of error returns,
|
|
|
|
|
* unit attn errors are now errors.
|
1995-03-04 20:51:10 +00:00
|
|
|
|
*/
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
sc_link->flags |= SDEV_OPEN;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
|
|
|
|
|
if (scsi_test_unit_ready(sc_link, SCSI_SILENT) != 0) {
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB3, ("not ready\n"));
|
1997-12-26 14:52:29 +00:00
|
|
|
|
error = ENXIO;
|
|
|
|
|
goto out;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
if ((flags & O_NONBLOCK) == 0) {
|
1996-01-27 17:48:12 +00:00
|
|
|
|
scsi_start_unit(sc_link, SCSI_SILENT);
|
|
|
|
|
scsi_prevent(sc_link, PR_PREVENT, SCSI_SILENT);
|
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
if((flags & FWRITE) != 0) {
|
|
|
|
|
if ((error = worm_rezero_unit(sc_link)) != 0 ||
|
1997-05-19 17:30:50 +00:00
|
|
|
|
(error = worm_size(sc_link, 0)) != 0) {
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB3,
|
1997-05-19 17:30:50 +00:00
|
|
|
|
("rezero, or get size failed\n"));
|
1997-12-26 14:52:29 +00:00
|
|
|
|
error = EIO;
|
|
|
|
|
goto out;
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
/* read/only */
|
|
|
|
|
if ((error = worm_size(sc_link, 0)) != 0) {
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB3,
|
|
|
|
|
("get size failed\n"));
|
1997-12-26 14:52:29 +00:00
|
|
|
|
error = EIO;
|
|
|
|
|
goto out;
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
}
|
1997-08-01 12:48:35 +00:00
|
|
|
|
worm->blk_size = 2048;
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
}
|
|
|
|
|
} else
|
|
|
|
|
worm->worm_flags |= WORMFL_IOCTL_ONLY;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
switch (*(int *) sc_link->devmodes) {
|
|
|
|
|
case WORM_Q_PLASMON:
|
|
|
|
|
worm->quirks = &worm_quirks_plasmon;
|
|
|
|
|
break;
|
|
|
|
|
case WORM_Q_PHILIPS:
|
|
|
|
|
worm->quirks = &worm_quirks_philips;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
1997-12-26 14:52:29 +00:00
|
|
|
|
error = ENXIO;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
worm->error = 0;
|
1997-12-26 14:52:29 +00:00
|
|
|
|
|
|
|
|
|
out:
|
|
|
|
|
if (error) {
|
|
|
|
|
scsi_prevent(sc_link, PR_ALLOW, SCSI_SILENT);
|
|
|
|
|
worm->worm_flags &= ~(WORMFL_TRACK_PREPED| WORMFL_TRACK_PREP);
|
|
|
|
|
sc_link->flags &= ~SDEV_OPEN;
|
|
|
|
|
}
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
return error;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
1995-10-21 23:13:11 +00:00
|
|
|
|
static int
|
1996-01-28 09:56:04 +00:00
|
|
|
|
worm_close(dev_t dev, int flags, int fmt, struct proc *p,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_link *sc_link)
|
1995-03-04 20:51:10 +00:00
|
|
|
|
{
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct scsi_data *worm = sc_link->sd;
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
errval error;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
error = 0;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
if ((worm->worm_flags & WORMFL_IOCTL_ONLY) == 0) {
|
|
|
|
|
if ((flags & FWRITE) != 0) {
|
1997-05-19 17:30:50 +00:00
|
|
|
|
worm->error = 0;
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if ((worm->worm_flags & WORMFL_TRACK_PREPED) != 0) {
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
error = (worm->quirks->finalize_track)(sc_link);
|
1997-12-26 14:52:29 +00:00
|
|
|
|
worm->worm_flags &=
|
|
|
|
|
~(WORMFL_TRACK_PREPED | WORMFL_TRACK_PREP);
|
|
|
|
|
}
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
}
|
1997-05-19 17:30:50 +00:00
|
|
|
|
scsi_prevent(sc_link, PR_ALLOW, SCSI_SILENT);
|
1997-12-26 14:52:29 +00:00
|
|
|
|
} else
|
|
|
|
|
worm->worm_flags &= ~WORMFL_IOCTL_ONLY;
|
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
sc_link->flags &= ~SDEV_OPEN;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
return error;
|
1995-03-04 20:51:10 +00:00
|
|
|
|
}
|
1995-11-29 10:49:16 +00:00
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* Perform special action on behalf of the user.
|
|
|
|
|
* Knows about the internals of this device
|
|
|
|
|
*/
|
|
|
|
|
errval
|
|
|
|
|
worm_ioctl(dev_t dev, int cmd, caddr_t addr, int flag, struct proc *p,
|
|
|
|
|
struct scsi_link *sc_link)
|
|
|
|
|
{
|
|
|
|
|
errval error = 0;
|
1996-03-10 12:52:47 +00:00
|
|
|
|
u_int8_t unit;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
register struct scsi_data *worm;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Find the device that the user is talking about
|
|
|
|
|
*/
|
|
|
|
|
unit = wormunit(dev);
|
|
|
|
|
worm = sc_link->sd;
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("wormioctl 0x%x ", cmd));
|
|
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
|
case WORMIOCPREPDISK:
|
1997-05-19 17:30:50 +00:00
|
|
|
|
{
|
|
|
|
|
struct wormio_prepare_disk *w =
|
|
|
|
|
(struct wormio_prepare_disk *)addr;
|
|
|
|
|
if (w->dummy != 0 && w->dummy != 1)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else {
|
|
|
|
|
error = (worm->quirks->prepare_disk)
|
|
|
|
|
(sc_link, w->dummy, w->speed);
|
|
|
|
|
if (error == 0) {
|
|
|
|
|
worm->worm_flags |= WORMFL_DISK_PREPED;
|
|
|
|
|
worm->dummy = w->dummy;
|
|
|
|
|
worm->speed = w->speed;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
}
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
case WORMIOCPREPTRACK:
|
1997-05-19 17:30:50 +00:00
|
|
|
|
{
|
|
|
|
|
struct wormio_prepare_track *w =
|
|
|
|
|
(struct wormio_prepare_track *)addr;
|
|
|
|
|
if (w->audio != 0 && w->audio != 1)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else if (w->audio == 0 && w->preemp)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else if ((worm->worm_flags & WORMFL_DISK_PREPED)==0) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
worm->error = WORM_SEQUENCE_ERROR;
|
|
|
|
|
} else {
|
1997-12-26 14:52:29 +00:00
|
|
|
|
/*
|
|
|
|
|
* This sets a flag only. Actual preparation of a
|
|
|
|
|
* track will be deferred up to the first write.
|
|
|
|
|
*/
|
|
|
|
|
worm->worm_flags |= WORMFL_TRACK_PREP;
|
|
|
|
|
worm->preptrack = *w;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
case WORMIOCFINISHTRACK:
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if ((worm->worm_flags & WORMFL_TRACK_PREPED) != 0)
|
|
|
|
|
error = (worm->quirks->finalize_track)(sc_link);
|
|
|
|
|
worm->worm_flags &= ~(WORMFL_TRACK_PREPED | WORMFL_TRACK_PREP);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
break;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
case WORMIOCFIXATION:
|
|
|
|
|
{
|
|
|
|
|
struct wormio_fixation *w =
|
|
|
|
|
(struct wormio_fixation *)addr;
|
|
|
|
|
|
|
|
|
|
if ((worm->worm_flags & WORMFL_WRITTEN) == 0)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else if (w->toc_type < WORM_TOC_TYPE_AUDIO ||
|
|
|
|
|
w->toc_type > WORM_TOC_TYPE_CDI)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else if (w->onp != 0 && w->onp != 1)
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
else {
|
|
|
|
|
worm->worm_flags = 0;
|
|
|
|
|
/* no fixation needed if dummy write */
|
|
|
|
|
if (worm->dummy == 0) {
|
|
|
|
|
worm->error = 0;
|
|
|
|
|
error = (worm->quirks->finalize_disk)
|
|
|
|
|
(sc_link, w->toc_type, w->onp);
|
1996-01-27 17:48:12 +00:00
|
|
|
|
}
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
1997-06-02 20:05:39 +00:00
|
|
|
|
case WORMIOCREADSESSIONINFO:
|
|
|
|
|
{
|
|
|
|
|
struct wormio_session_info si;
|
|
|
|
|
error = worm_read_session_info(sc_link, &si);
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
NTOHS(si.lead_in);
|
|
|
|
|
NTOHS(si.lead_out);
|
|
|
|
|
bcopy(&si, addr, sizeof si);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case WORMIOCWRITESESSION:
|
|
|
|
|
if (worm->quirks->write_session) {
|
|
|
|
|
error = (worm->quirks->write_session)
|
|
|
|
|
(sc_link, (struct wormio_write_session *) addr);
|
1997-12-26 14:52:29 +00:00
|
|
|
|
if (!error)
|
|
|
|
|
worm->worm_flags |= WORMFL_TRACK_PREPED | WORMFL_TRACK_PREP;
|
1997-06-02 20:05:39 +00:00
|
|
|
|
} else
|
|
|
|
|
error = ENXIO;
|
|
|
|
|
break;
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
case WORMIOERROR:
|
|
|
|
|
bcopy(&(worm->error), addr, sizeof (int));
|
|
|
|
|
break;
|
1997-05-05 13:35:01 +00:00
|
|
|
|
|
|
|
|
|
case CDIOREADTOCHEADER:
|
|
|
|
|
{
|
|
|
|
|
struct ioc_toc_header th;
|
|
|
|
|
error = worm_read_toc(sc_link, 0, 0,
|
|
|
|
|
(struct cd_toc_entry *)&th,
|
|
|
|
|
sizeof th);
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
NTOHS(th.len);
|
|
|
|
|
bcopy(&th, addr, sizeof th);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CDIOREADTOCENTRYS:
|
|
|
|
|
{
|
|
|
|
|
struct {
|
|
|
|
|
struct ioc_toc_header header;
|
|
|
|
|
struct cd_toc_entry entries[100];
|
|
|
|
|
} data;
|
|
|
|
|
struct {
|
|
|
|
|
struct ioc_toc_header header;
|
|
|
|
|
struct cd_toc_entry entry;
|
|
|
|
|
} lead;
|
|
|
|
|
struct ioc_read_toc_entry *te =
|
|
|
|
|
(struct ioc_read_toc_entry *) addr;
|
|
|
|
|
struct ioc_toc_header *th;
|
|
|
|
|
u_int32_t len, readlen, idx, num;
|
|
|
|
|
u_int32_t starting_track = te->starting_track;
|
|
|
|
|
|
|
|
|
|
if (te->data_len < sizeof(struct cd_toc_entry)
|
|
|
|
|
|| (te->data_len % sizeof(struct cd_toc_entry)) != 0
|
|
|
|
|
|| te->address_format != CD_MSF_FORMAT
|
|
|
|
|
&& te->address_format != CD_LBA_FORMAT) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
th = &data.header;
|
|
|
|
|
error = worm_read_toc(sc_link, 0, 0,
|
|
|
|
|
(struct cd_toc_entry *)th,
|
|
|
|
|
sizeof (*th));
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (starting_track == 0)
|
|
|
|
|
starting_track = th->starting_track;
|
|
|
|
|
else if (starting_track == 0xaa)
|
|
|
|
|
starting_track = th->ending_track + 1;
|
|
|
|
|
else if (starting_track < th->starting_track ||
|
|
|
|
|
starting_track > th->ending_track + 1) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* calculate reading length without leadout entry */
|
|
|
|
|
readlen = (th->ending_track - starting_track + 1) *
|
|
|
|
|
sizeof(struct cd_toc_entry);
|
|
|
|
|
|
|
|
|
|
/* and with leadout entry */
|
|
|
|
|
len = readlen + sizeof(struct cd_toc_entry);
|
|
|
|
|
if (te->data_len < len) {
|
|
|
|
|
len = te->data_len;
|
|
|
|
|
if (readlen > len)
|
|
|
|
|
readlen = len;
|
|
|
|
|
}
|
|
|
|
|
if (len > sizeof(data.entries)) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
num = len / sizeof(struct cd_toc_entry);
|
|
|
|
|
|
|
|
|
|
if (readlen > 0) {
|
|
|
|
|
error = worm_read_toc(sc_link,
|
|
|
|
|
te->address_format,
|
|
|
|
|
starting_track,
|
|
|
|
|
(struct cd_toc_entry *)&data,
|
|
|
|
|
readlen + sizeof (*th));
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* make leadout entry if needed */
|
|
|
|
|
idx = starting_track + num - 1;
|
|
|
|
|
if (idx == th->ending_track + 1) {
|
|
|
|
|
error = worm_read_toc(sc_link,
|
|
|
|
|
te->address_format, 0xaa,
|
|
|
|
|
(struct cd_toc_entry *)&lead,
|
|
|
|
|
sizeof(lead));
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
data.entries[idx - starting_track] = lead.entry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
error = copyout(data.entries, te->data, len);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case CDIOREADTOCENTRY:
|
|
|
|
|
{
|
|
|
|
|
struct {
|
|
|
|
|
struct ioc_toc_header header;
|
|
|
|
|
struct cd_toc_entry entry;
|
|
|
|
|
} data;
|
|
|
|
|
struct ioc_read_toc_single_entry *te =
|
|
|
|
|
(struct ioc_read_toc_single_entry *) addr;
|
|
|
|
|
struct ioc_toc_header *th;
|
|
|
|
|
u_int32_t track;
|
|
|
|
|
|
|
|
|
|
if (te->address_format != CD_MSF_FORMAT
|
|
|
|
|
&& te->address_format != CD_LBA_FORMAT) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
th = &data.header;
|
|
|
|
|
error = worm_read_toc(sc_link, 0, 0,
|
|
|
|
|
(struct cd_toc_entry *)th,
|
|
|
|
|
sizeof (*th));
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
track = te->track;
|
|
|
|
|
if (track == 0)
|
|
|
|
|
track = th->starting_track;
|
|
|
|
|
else if (track == 0xaa)
|
|
|
|
|
/* OK */;
|
|
|
|
|
else if (track < th->starting_track ||
|
|
|
|
|
track > th->ending_track + 1) {
|
|
|
|
|
error = EINVAL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
error = worm_read_toc(sc_link, te->address_format,
|
|
|
|
|
track,
|
|
|
|
|
(struct cd_toc_entry *)&data,
|
|
|
|
|
sizeof data);
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
bcopy(&data.entry, &te->entry,
|
|
|
|
|
sizeof(struct cd_toc_entry));
|
|
|
|
|
}
|
|
|
|
|
break;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
error = ENOTTY;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
worm_rezero_unit(struct scsi_link *sc_link)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_rezero_unit cmd;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("worm_rezero_unit"));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Re-initialize the unit, just to be sure.
|
|
|
|
|
*/
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = REZERO_UNIT;
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
0, /* no data transfer */
|
|
|
|
|
0,
|
|
|
|
|
/*WORMRETRY*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
0);
|
|
|
|
|
}
|
|
|
|
|
|
1997-06-02 20:05:39 +00:00
|
|
|
|
static errval
|
|
|
|
|
worm_read_session_info(struct scsi_link *sc_link, struct wormio_session_info *data)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_read_session_info cmd;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("worm_read_session_info"));
|
|
|
|
|
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = READ_SESSION_INFO;
|
|
|
|
|
cmd.transfer_length = sizeof(struct wormio_session_info);
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
(u_char *) data,
|
|
|
|
|
sizeof(struct wormio_session_info),
|
|
|
|
|
/*WORMRETRY*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_IN);
|
|
|
|
|
}
|
|
|
|
|
|
1997-05-05 13:35:01 +00:00
|
|
|
|
/*
|
|
|
|
|
* Read table of contents
|
|
|
|
|
*
|
|
|
|
|
* Stolen from cd.c
|
|
|
|
|
*/
|
|
|
|
|
static errval
|
|
|
|
|
worm_read_toc(struct scsi_link *sc_link, u_int32_t mode, u_int32_t start,
|
|
|
|
|
struct cd_toc_entry *data, u_int32_t len)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_read_toc scsi_cmd;
|
|
|
|
|
u_int32_t ntoc;
|
|
|
|
|
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
ntoc = len;
|
|
|
|
|
|
|
|
|
|
scsi_cmd.op_code = READ_TOC;
|
|
|
|
|
if (mode == CD_MSF_FORMAT)
|
|
|
|
|
scsi_cmd.byte2 |= CD_MSF;
|
|
|
|
|
scsi_cmd.from_track = start;
|
|
|
|
|
scsi_cmd.data_len[0] = (ntoc) >> 8;
|
|
|
|
|
scsi_cmd.data_len[1] = (ntoc) & 0xff;
|
|
|
|
|
return (scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(struct scsi_read_toc),
|
|
|
|
|
(u_char *) data,
|
|
|
|
|
len,
|
|
|
|
|
/*WORMRETRY*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_IN));
|
|
|
|
|
}
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static int
|
|
|
|
|
worm_sense_handler(struct scsi_xfer *xs)
|
1996-01-27 17:48:12 +00:00
|
|
|
|
{
|
1997-05-19 17:30:50 +00:00
|
|
|
|
struct scsi_data *worm;
|
|
|
|
|
struct scsi_sense_data *sense;
|
|
|
|
|
struct scsi_sense_extended *ext;
|
|
|
|
|
int asc, devmode, i;
|
|
|
|
|
|
|
|
|
|
worm = xs->sc_link->sd;
|
|
|
|
|
sense = &(xs->sense);
|
|
|
|
|
ext = (struct scsi_sense_extended *) &(sense->ext.extended);
|
|
|
|
|
asc = ext->add_sense_code;
|
|
|
|
|
devmode = *(int *) xs->sc_link->devmodes;
|
|
|
|
|
|
|
|
|
|
for (i = 0; worm_error[i].asc; i++)
|
|
|
|
|
if ((asc == worm_error[i].asc) && (devmode & worm_error[i].devmode)) {
|
|
|
|
|
worm->error = worm_error[i].error;
|
|
|
|
|
return worm_error[i].ret;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
}
|
1997-05-19 17:30:50 +00:00
|
|
|
|
worm->error = -1;
|
|
|
|
|
return SCSIRET_CONTINUE;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
1997-10-27 21:09:27 +00:00
|
|
|
|
static errval
|
|
|
|
|
worm_set_blksize(struct scsi_link *sc_link, int size)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_mode_select scsi_cmd;
|
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct blk_desc desc;
|
|
|
|
|
} dat;
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
scsi_cmd.op_code = MODE_SELECT;
|
|
|
|
|
scsi_cmd.length = sizeof(dat);
|
|
|
|
|
dat.header.blk_desc_len = sizeof(struct blk_desc);
|
|
|
|
|
scsi_uto3b(size, dat.desc.blklen);
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(scsi_cmd),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
sizeof(dat),
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
|
1995-12-08 11:19:42 +00:00
|
|
|
|
static void
|
|
|
|
|
worm_drvinit(void *unused)
|
1995-11-29 10:49:16 +00:00
|
|
|
|
{
|
|
|
|
|
|
After so many people have been bugging me :), finally implement
read-mode access to CD-ROM media in the worm(4) driver. No whistles
and bells yet, like all the CDIO* commands, but at least a start.
In order to do this, i had to slightly rearrange the semantics of an
open(2) on the worm driver: now, opening it with O_NONBLOCK set means
no actual IO operations will be intended but only ioctls are to be
processed. This mode is used by wormcontrol(8) to prepare a track
and/or session.
I have only been able to test this on a 2.2-GAMMA system by now, and
only the !DEVFS part is tested yet. Also, i have only done a dummy
burn so far, but wouldn't expect many surprises else. Report bugs to
me ASAP, if there's reasonable demand and i hear no objections, i
might consider merging it into the 2.2 branch as well.
1997-02-06 22:19:44 +00:00
|
|
|
|
if (!worm_devsw_installed) {
|
|
|
|
|
bdevsw_add_generic(BDEV_MAJOR, CDEV_MAJOR, &worm_bdevsw);
|
1995-11-29 10:49:16 +00:00
|
|
|
|
worm_devsw_installed = 1;
|
1995-11-29 14:41:20 +00:00
|
|
|
|
}
|
1995-11-29 10:49:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SYSINIT(wormdev,SI_SUB_DRIVERS,SI_ORDER_MIDDLE+CDEV_MAJOR,worm_drvinit,NULL)
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* Begin device-specific stuff. Subject to being moved out to LKMs.
|
|
|
|
|
*/
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
static u_char
|
|
|
|
|
ascii_to_6bit (char c)
|
|
|
|
|
{
|
|
|
|
|
if (c < '0' || c > 'Z' || (c > '9' && c < 'A'))
|
|
|
|
|
return 0;
|
|
|
|
|
if (c <= '9')
|
|
|
|
|
return c - '0';
|
|
|
|
|
else
|
|
|
|
|
return c - 'A' + 11;
|
|
|
|
|
}
|
1996-01-27 17:48:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* PLASMON RF4100/4102
|
|
|
|
|
* Perhaps other Plasmon's, too.
|
|
|
|
|
*
|
|
|
|
|
* NB: By now, you'll certainly have to compare the SCSI reference
|
|
|
|
|
* manual in order to understand the following.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* The following mode pages might apply to other drives as well. */
|
|
|
|
|
|
|
|
|
|
struct plasmon_rf4100_pages
|
|
|
|
|
{
|
|
|
|
|
u_char page_code;
|
|
|
|
|
#define RF4100_PAGE_CODE_20 0x20
|
|
|
|
|
#define RF4100_PAGE_CODE_21 0x21
|
|
|
|
|
#define RF4100_PAGE_CODE_22 0x22
|
|
|
|
|
#define RF4100_PAGE_CODE_23 0x23
|
|
|
|
|
#define RF4100_PAGE_CODE_24 0x24
|
|
|
|
|
#define RF4100_PAGE_CODE_25 0x25
|
|
|
|
|
u_char param_len;
|
|
|
|
|
union
|
|
|
|
|
{
|
|
|
|
|
/* page 0x20 omitted by now */
|
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
u_char reserved1;
|
|
|
|
|
u_char mode;
|
|
|
|
|
#define RF4100_RAW_MODE 0x10 /* raw mode enabled */
|
1996-02-02 22:59:48 +00:00
|
|
|
|
#define RF4100_MIXED_MODE 0x08 /* mixed mode data enabled */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
#define RF4100_AUDIO_MODE 0x04 /* audio mode data enabled */
|
|
|
|
|
#define RF4100_MODE_1 0x01 /* mode 1 blocks are enabled */
|
1997-05-19 17:30:50 +00:00
|
|
|
|
#define RF4100_MODE_2 0x02 /* mode 2 blocks are enabled */
|
1996-01-27 17:48:12 +00:00
|
|
|
|
u_char track_number;
|
|
|
|
|
u_char isrc_i1; /* country code, ASCII */
|
|
|
|
|
u_char isrc_i2;
|
|
|
|
|
u_char isrc_i3; /* owner code, ASCII */
|
|
|
|
|
u_char isrc_i4;
|
|
|
|
|
u_char isrc_i5;
|
|
|
|
|
u_char isrc_i6_7; /* country code, BCD */
|
|
|
|
|
u_char isrc_i8_9; /* serial number, BCD */
|
|
|
|
|
u_char isrc_i10_11;
|
|
|
|
|
u_char isrc_i12_0;
|
|
|
|
|
u_char reserved2[2];
|
|
|
|
|
}
|
|
|
|
|
page_0x21;
|
|
|
|
|
/* mode page 0x22 omitted by now */
|
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
u_char speed_select;
|
|
|
|
|
#define RF4100_SPEED_AUDIO 0x01
|
|
|
|
|
#define RF4100_SPEED_DOUBLE 0x02
|
|
|
|
|
u_char dummy_write;
|
|
|
|
|
#define RF4100_DUMMY_WRITE 0x01
|
|
|
|
|
u_char reserved[4];
|
|
|
|
|
}
|
|
|
|
|
page_0x23;
|
|
|
|
|
/* pages 0x24 and 0x25 omitted by now */
|
|
|
|
|
}
|
|
|
|
|
pages;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
rf4100_prepare_disk(struct scsi_link *sc_link, int dummy, int speed)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_mode_select scsi_cmd;
|
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct plasmon_rf4100_pages page;
|
|
|
|
|
} dat;
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t pagelen, dat_len;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
pagelen = sizeof(dat.page.pages.page_0x23) + PAGE_HEADERLEN;
|
|
|
|
|
dat_len = sizeof(struct scsi_mode_header) + pagelen;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("rf4100_prepare_disk"));
|
1995-11-29 10:49:16 +00:00
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
if (speed != RF4100_SPEED_AUDIO && speed != RF4100_SPEED_DOUBLE)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Set up a mode page 0x23
|
|
|
|
|
*/
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
scsi_cmd.op_code = MODE_SELECT;
|
|
|
|
|
scsi_cmd.length = dat_len;
|
|
|
|
|
/* dat.header.dev_spec = host application code; (see spec) */
|
|
|
|
|
dat.page.page_code = RF4100_PAGE_CODE_23;
|
|
|
|
|
dat.page.param_len = sizeof(dat.page.pages.page_0x23);
|
|
|
|
|
dat.page.pages.page_0x23.speed_select = speed;
|
|
|
|
|
dat.page.pages.page_0x23.dummy_write = dummy? RF4100_DUMMY_WRITE: 0;
|
|
|
|
|
/*
|
|
|
|
|
* Fire it off.
|
|
|
|
|
*/
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(scsi_cmd),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
dat_len,
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
1997-05-19 17:30:50 +00:00
|
|
|
|
rf4100_prepare_track(struct scsi_link *sc_link, struct wormio_prepare_track *t)
|
1996-01-27 17:48:12 +00:00
|
|
|
|
{
|
|
|
|
|
struct scsi_mode_select scsi_cmd;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
struct scsi_data *worm;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct blk_desc blk_desc;
|
|
|
|
|
struct plasmon_rf4100_pages page;
|
|
|
|
|
} dat;
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t pagelen, dat_len, blk_len;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
int year;
|
|
|
|
|
|
|
|
|
|
worm = sc_link->sd;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
|
|
|
|
pagelen = sizeof(dat.page.pages.page_0x21) + PAGE_HEADERLEN;
|
|
|
|
|
dat_len = sizeof(struct scsi_mode_header)
|
|
|
|
|
+ sizeof(struct blk_desc)
|
|
|
|
|
+ pagelen;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("rf4100_prepare_track"));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Set up a mode page 0x21. Note that the block descriptor is
|
|
|
|
|
* mandatory in at least one of the MODE SELECT commands, in
|
|
|
|
|
* order to select the block length in question. We do this
|
|
|
|
|
* here, just prior to opening the write channel. (Spec:
|
|
|
|
|
* ``All information for the write is included in the MODE
|
|
|
|
|
* SELECT, MODE PAGE 21h, and the write channel can be
|
|
|
|
|
* considered open on receipt of the first WRITE command.'' I
|
|
|
|
|
* didn't have luck with an explicit WRITE TRACK command
|
|
|
|
|
* anyway, this might be different for other CD-R drives. -
|
|
|
|
|
* J<EFBFBD>rg)
|
|
|
|
|
*/
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
scsi_cmd.op_code = MODE_SELECT;
|
|
|
|
|
scsi_cmd.length = dat_len;
|
|
|
|
|
dat.header.blk_desc_len = sizeof(struct blk_desc);
|
|
|
|
|
dat.page.page_code = RF4100_PAGE_CODE_21;
|
|
|
|
|
dat.page.param_len = sizeof(dat.page.pages.page_0x21);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
/* dat.header.dev_spec = host application code; (see spec) */
|
|
|
|
|
if (t->audio) {
|
|
|
|
|
blk_len = 2352;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_AUDIO_MODE +
|
|
|
|
|
(t->preemp? RF4100_MODE_1 : 0);
|
|
|
|
|
} else
|
|
|
|
|
switch (t->track_type) {
|
|
|
|
|
case BLOCK_RAW:
|
|
|
|
|
blk_len = 2352;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_RAW_MODE;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_1:
|
|
|
|
|
blk_len = 2048;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_1;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2:
|
|
|
|
|
blk_len = 2336;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_1:
|
|
|
|
|
blk_len = 2048;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_1b:
|
|
|
|
|
blk_len = 2056;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_2:
|
|
|
|
|
blk_len = 2324;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_2b:
|
|
|
|
|
blk_len = 2332;
|
|
|
|
|
dat.page.pages.page_0x21.mode = RF4100_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
dat.page.pages.page_0x21.mode |= t->copy_bits << 5;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
worm->blk_size = blk_len;
|
|
|
|
|
|
|
|
|
|
dat.page.pages.page_0x21.track_number = t->track_number;
|
|
|
|
|
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i1 = ascii_to_6bit(t->ISRC_country[0]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i2 = ascii_to_6bit(t->ISRC_country[1]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i3 = ascii_to_6bit(t->ISRC_owner[0]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i4 = ascii_to_6bit(t->ISRC_owner[1]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i5 = ascii_to_6bit(t->ISRC_owner[2]);
|
|
|
|
|
year = t->ISRC_year > 1900 ? t->ISRC_year - 1900 : t->ISRC_year;
|
|
|
|
|
if (year > 99 || year < 0)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i6_7 = bin2bcd(year);
|
|
|
|
|
if (t->ISRC_serial[0]) {
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i8_9 = ((t->ISRC_serial[0]-'0') << 4) |
|
1997-05-19 17:30:50 +00:00
|
|
|
|
(t->ISRC_serial[1] - '0');
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i10_11 = ((t->ISRC_serial[2]-'0') << 4) |
|
1997-05-19 17:30:50 +00:00
|
|
|
|
(t->ISRC_serial[3] - '0');
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i12_0 = (t->ISRC_serial[4] - '0' << 4);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
scsi_uto3b(blk_len, dat.blk_desc.blklen);
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
/*
|
|
|
|
|
* Fire it off.
|
|
|
|
|
*/
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(scsi_cmd),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
dat_len,
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
rf4100_finalize_track(struct scsi_link *sc_link)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_synchronize_cache cmd;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
int error;
|
|
|
|
|
|
1996-01-27 17:48:12 +00:00
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("rf4100_finalize_track"));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Only a "synchronize cache" is needed.
|
|
|
|
|
*/
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = SYNCHRONIZE_CACHE;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
error = scsi_scsi_cmd(sc_link,
|
1996-01-27 17:48:12 +00:00
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
0, /* no data transfer */
|
|
|
|
|
0,
|
|
|
|
|
1,
|
|
|
|
|
60000, /* this may take a while */
|
|
|
|
|
NULL,
|
|
|
|
|
0);
|
1997-10-27 21:09:27 +00:00
|
|
|
|
if (!error)
|
|
|
|
|
error = worm_set_blksize(sc_link, 2048);
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
return error;
|
1996-01-27 17:48:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
rf4100_finalize_disk(struct scsi_link *sc_link, int toc_type, int onp)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_fixation cmd;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("rf4100_finalize_disk"));
|
|
|
|
|
|
|
|
|
|
if (toc_type < 0 || toc_type > WORM_TOC_TYPE_CDI)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Fixate this session. Mark the next one as opened if onp
|
|
|
|
|
* is true. Otherwise, the disk will be finalized once and
|
|
|
|
|
* for all. ONP stands for "open next program area".
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = FIXATION;
|
|
|
|
|
cmd.action = (onp? WORM_FIXATION_ONP: 0) + toc_type;
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
0, /* no data transfer */
|
|
|
|
|
0,
|
|
|
|
|
1,
|
|
|
|
|
20*60*1000, /* takes a huge amount of time */
|
|
|
|
|
NULL,
|
|
|
|
|
0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* End Plasmon RF4100/4102 section.
|
|
|
|
|
*/
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* HP C4324/C4325 (This is what the scsi spec. and firmware says)
|
|
|
|
|
* Drive model 4020i
|
|
|
|
|
* This is very similar to the Plasmon above.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* The following mode pages might apply to other drives as well. */
|
|
|
|
|
|
|
|
|
|
struct hp_4020i_pages
|
|
|
|
|
{
|
|
|
|
|
u_char page_code;
|
|
|
|
|
#define HP4020I_PAGE_CODE_20 0x20
|
|
|
|
|
#define HP4020I_PAGE_CODE_21 0x21
|
|
|
|
|
#define HP4020I_PAGE_CODE_22 0x22
|
|
|
|
|
#define HP4020I_PAGE_CODE_23 0x23
|
|
|
|
|
#define HP4020I_PAGE_CODE_24 0x24
|
|
|
|
|
#define HP4020I_PAGE_CODE_25 0x25
|
|
|
|
|
u_char param_len;
|
|
|
|
|
union
|
|
|
|
|
{
|
|
|
|
|
/* page 0x20 omitted by now */
|
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
u_char reserved1;
|
|
|
|
|
u_char mode;
|
|
|
|
|
#define HP4020I_RAW_MODE 0x10 /* raw mode enabled */
|
|
|
|
|
#define HP4020I_MIXED_MODE 0x08 /* mixed mode data enabled */
|
|
|
|
|
#define HP4020I_AUDIO_MODE 0x04 /* audio mode data enabled */
|
|
|
|
|
#define HP4020I_MODE_1 0x01 /* mode 1 blocks are enabled */
|
|
|
|
|
#define HP4020I_MODE_2 0x02 /* mode 2 blocks are enabled */
|
|
|
|
|
u_char track_number;
|
|
|
|
|
u_char isrc_i1; /* country code, ASCII */
|
|
|
|
|
u_char isrc_i2;
|
|
|
|
|
u_char isrc_i3; /* owner code, ASCII */
|
|
|
|
|
u_char isrc_i4;
|
|
|
|
|
u_char isrc_i5;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
u_char isrc_i6_7; /* year code, BCD */
|
1996-02-02 22:59:48 +00:00
|
|
|
|
u_char isrc_i8_9; /* serial number, BCD */
|
|
|
|
|
u_char isrc_i10_11;
|
|
|
|
|
u_char isrc_i12_0;
|
|
|
|
|
u_char reserved2[2];
|
|
|
|
|
}
|
|
|
|
|
page_0x21;
|
1997-06-02 20:05:39 +00:00
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
u_char catalog_valid;
|
|
|
|
|
u_char catalog_c1_c2; /* catalog number, BCD */
|
|
|
|
|
u_char catalog_c3_c4;
|
|
|
|
|
u_char catalog_c5_c6;
|
|
|
|
|
u_char catalog_c7_c8;
|
|
|
|
|
u_char catalog_c9_c10;
|
|
|
|
|
u_char catalog_c11_c12;
|
|
|
|
|
u_char catalog_c13_0;
|
|
|
|
|
} page_0x22;
|
1996-02-02 22:59:48 +00:00
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
u_char speed_select;
|
|
|
|
|
#define HP4020I_SPEED_AUDIO 0x01
|
|
|
|
|
#define HP4020I_SPEED_DOUBLE 0x02
|
|
|
|
|
u_char dummy_write;
|
|
|
|
|
#define HP4020I_DUMMY_WRITE 0x01
|
|
|
|
|
u_char reserved[4];
|
|
|
|
|
}
|
|
|
|
|
page_0x23;
|
|
|
|
|
/* pages 0x24 and 0x25 omitted by now */
|
|
|
|
|
}
|
|
|
|
|
pages;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
hp4020i_prepare_disk(struct scsi_link *sc_link, int dummy, int speed)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_mode_select scsi_cmd;
|
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct hp_4020i_pages page;
|
|
|
|
|
} dat;
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t pagelen, dat_len;
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
|
|
|
|
pagelen = sizeof(dat.page.pages.page_0x23) + PAGE_HEADERLEN;
|
|
|
|
|
dat_len = sizeof(struct scsi_mode_header) + pagelen;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("hp4020i_prepare_disk"));
|
|
|
|
|
|
|
|
|
|
if (speed != HP4020I_SPEED_AUDIO && speed != HP4020I_SPEED_DOUBLE)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Set up a mode page 0x23
|
|
|
|
|
*/
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
scsi_cmd.op_code = MODE_SELECT;
|
|
|
|
|
scsi_cmd.byte2 |= SMS_PF;
|
|
|
|
|
scsi_cmd.length = dat_len;
|
|
|
|
|
/* dat.header.dev_spec = host application code; (see spec) */
|
|
|
|
|
dat.page.page_code = HP4020I_PAGE_CODE_23;
|
|
|
|
|
dat.page.param_len = sizeof(dat.page.pages.page_0x23);
|
|
|
|
|
dat.page.pages.page_0x23.speed_select = speed;
|
|
|
|
|
dat.page.pages.page_0x23.dummy_write = dummy? HP4020I_DUMMY_WRITE: 0;
|
|
|
|
|
/*
|
|
|
|
|
* Fire it off.
|
|
|
|
|
*/
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(scsi_cmd),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
dat_len,
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
1997-05-19 17:30:50 +00:00
|
|
|
|
hp4020i_prepare_track(struct scsi_link *sc_link, struct wormio_prepare_track *t)
|
1996-02-02 22:59:48 +00:00
|
|
|
|
{
|
|
|
|
|
struct scsi_mode_select scsi_cmd;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
struct scsi_data *worm;
|
1996-02-02 22:59:48 +00:00
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct blk_desc blk_desc;
|
|
|
|
|
struct hp_4020i_pages page;
|
|
|
|
|
} dat;
|
1996-03-10 07:13:15 +00:00
|
|
|
|
u_int32_t pagelen, dat_len, blk_len;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
int year;
|
|
|
|
|
|
|
|
|
|
worm = sc_link->sd;
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
|
|
|
|
pagelen = sizeof(dat.page.pages.page_0x21) + PAGE_HEADERLEN;
|
|
|
|
|
dat_len = sizeof(struct scsi_mode_header)
|
|
|
|
|
+ sizeof(struct blk_desc)
|
|
|
|
|
+ pagelen;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("hp4020i_prepare_track"));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Set up a mode page 0x21. Note that the block descriptor is
|
|
|
|
|
* mandatory in at least one of the MODE SELECT commands, in
|
|
|
|
|
* order to select the block length in question. We do this
|
|
|
|
|
* here, just prior to opening the write channel. (Spec:
|
|
|
|
|
* ``All information for the write is included in the MODE
|
|
|
|
|
* SELECT, MODE PAGE 21h, and the write channel can be
|
|
|
|
|
* considered open on receipt of the first WRITE command.'' I
|
|
|
|
|
* didn't have luck with an explicit WRITE TRACK command
|
|
|
|
|
* anyway, this might be different for other CD-R drives. -
|
|
|
|
|
* J<EFBFBD>rg)
|
|
|
|
|
*/
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
bzero(&scsi_cmd, sizeof(scsi_cmd));
|
|
|
|
|
scsi_cmd.op_code = MODE_SELECT;
|
|
|
|
|
scsi_cmd.byte2 |= SMS_PF;
|
|
|
|
|
scsi_cmd.length = dat_len;
|
|
|
|
|
dat.header.blk_desc_len = sizeof(struct blk_desc);
|
|
|
|
|
dat.page.page_code = HP4020I_PAGE_CODE_21;
|
|
|
|
|
dat.page.param_len = sizeof(dat.page.pages.page_0x21);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
/* dat.header.dev_spec = host application code; (see spec) */
|
|
|
|
|
if (t->audio) {
|
|
|
|
|
blk_len = 2352;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_AUDIO_MODE +
|
|
|
|
|
(t->preemp? HP4020I_MODE_1 : 0);
|
|
|
|
|
} else
|
|
|
|
|
switch (t->track_type) {
|
|
|
|
|
case BLOCK_RAW:
|
|
|
|
|
blk_len = 2352;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_RAW_MODE;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_1:
|
|
|
|
|
blk_len = 2048;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_1;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2:
|
|
|
|
|
blk_len = 2336;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_1:
|
|
|
|
|
blk_len = 2048;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_1b:
|
|
|
|
|
blk_len = 2056;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_2:
|
|
|
|
|
blk_len = 2324;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
case BLOCK_MODE_2_FORM_2b:
|
|
|
|
|
blk_len = 2332;
|
|
|
|
|
dat.page.pages.page_0x21.mode = HP4020I_MODE_2;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return EINVAL;
|
|
|
|
|
}
|
|
|
|
|
dat.page.pages.page_0x21.mode |= t->copy_bits << 5;
|
|
|
|
|
|
|
|
|
|
worm->blk_size = blk_len;
|
|
|
|
|
|
|
|
|
|
dat.page.pages.page_0x21.track_number = t->track_number;
|
|
|
|
|
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i1 = ascii_to_6bit(t->ISRC_country[0]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i2 = ascii_to_6bit(t->ISRC_country[1]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i3 = ascii_to_6bit(t->ISRC_owner[0]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i4 = ascii_to_6bit(t->ISRC_owner[1]);
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i5 = ascii_to_6bit(t->ISRC_owner[2]);
|
|
|
|
|
year = t->ISRC_year > 1900 ? t->ISRC_year - 1900 : t->ISRC_year;
|
|
|
|
|
if (year > 99 || year < 0)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
dat.page.pages.page_0x21.isrc_i6_7 = bin2bcd(year);
|
|
|
|
|
if (t->ISRC_serial[0]) {
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i8_9 = ((t->ISRC_serial[0]-'0') << 4) |
|
1997-05-19 17:30:50 +00:00
|
|
|
|
(t->ISRC_serial[1] - '0');
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i10_11 = ((t->ISRC_serial[2]-'0') << 4) |
|
1997-05-19 17:30:50 +00:00
|
|
|
|
(t->ISRC_serial[3] - '0');
|
1997-05-28 21:25:49 +00:00
|
|
|
|
dat.page.pages.page_0x21.isrc_i12_0 = (t->ISRC_serial[4] - '0' << 4);
|
1997-05-19 17:30:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
scsi_uto3b(blk_len, dat.blk_desc.blklen);
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Fire it off.
|
|
|
|
|
*/
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &scsi_cmd,
|
|
|
|
|
sizeof(scsi_cmd),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
dat_len,
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
hp4020i_finalize_track(struct scsi_link *sc_link)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_synchronize_cache cmd;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
int error;
|
|
|
|
|
|
1996-02-02 22:59:48 +00:00
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("hp4020i_finalize_track"));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Only a "synchronize cache" is needed.
|
|
|
|
|
*/
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = SYNCHRONIZE_CACHE;
|
1997-05-19 17:30:50 +00:00
|
|
|
|
error = scsi_scsi_cmd(sc_link,
|
1996-02-02 22:59:48 +00:00
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
0, /* no data transfer */
|
|
|
|
|
0,
|
|
|
|
|
1,
|
|
|
|
|
60000, /* this may take a while */
|
|
|
|
|
NULL,
|
|
|
|
|
0);
|
1997-10-27 21:09:27 +00:00
|
|
|
|
if (!error)
|
|
|
|
|
error = worm_set_blksize(sc_link, 2048);
|
|
|
|
|
|
1997-05-19 17:30:50 +00:00
|
|
|
|
return error;
|
1996-02-02 22:59:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static errval
|
|
|
|
|
hp4020i_finalize_disk(struct scsi_link *sc_link, int toc_type, int onp)
|
|
|
|
|
{
|
|
|
|
|
struct scsi_fixation cmd;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("hp4020i_finalize_disk"));
|
|
|
|
|
|
|
|
|
|
if (toc_type < 0 || toc_type > WORM_TOC_TYPE_CDI)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Fixate this session. Mark the next one as opened if onp
|
|
|
|
|
* is true. Otherwise, the disk will be finalized once and
|
|
|
|
|
* for all. ONP stands for "open next program area".
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
bzero(&cmd, sizeof(cmd));
|
|
|
|
|
cmd.op_code = FIXATION;
|
|
|
|
|
cmd.action = (onp? WORM_FIXATION_ONP: 0) + toc_type;
|
|
|
|
|
return scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd,
|
|
|
|
|
sizeof(cmd),
|
|
|
|
|
0, /* no data transfer */
|
|
|
|
|
0,
|
|
|
|
|
1,
|
|
|
|
|
20*60*1000, /* takes a huge amount of time */
|
|
|
|
|
NULL,
|
|
|
|
|
0);
|
|
|
|
|
}
|
1997-06-02 20:05:39 +00:00
|
|
|
|
static errval
|
|
|
|
|
hp4020i_write_session(struct scsi_link *sc_link, struct wormio_write_session *ws)
|
|
|
|
|
{
|
|
|
|
|
struct {
|
|
|
|
|
struct scsi_mode_header header;
|
|
|
|
|
struct blk_desc blk_desc;
|
|
|
|
|
struct hp_4020i_pages page;
|
|
|
|
|
} dat;
|
|
|
|
|
struct scsi_mode_select cmd1;
|
|
|
|
|
struct scsi_write_session cmd2;
|
|
|
|
|
struct scsi_data *worm;
|
|
|
|
|
u_int32_t pagelen, dat_len, blk_len;
|
|
|
|
|
errval error;
|
|
|
|
|
|
|
|
|
|
SC_DEBUG(sc_link, SDEV_DB2, ("hp4020i_write_session"));
|
|
|
|
|
|
|
|
|
|
if (ws->toc_type < 0 || ws->toc_type > WORM_TOC_TYPE_CDI || ws->lofp & ~3)
|
|
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
|
|
pagelen = sizeof(dat.page.pages.page_0x22) + PAGE_HEADERLEN;
|
|
|
|
|
dat_len = sizeof(struct scsi_mode_header)
|
|
|
|
|
+ sizeof(struct blk_desc)
|
|
|
|
|
+ pagelen;
|
|
|
|
|
|
|
|
|
|
worm = sc_link->sd;
|
|
|
|
|
|
|
|
|
|
/* set the block size to 2352 and the catalog */
|
|
|
|
|
bzero(&dat, sizeof(dat));
|
|
|
|
|
bzero(&cmd1, sizeof(cmd1));
|
|
|
|
|
cmd1.op_code = MODE_SELECT;
|
|
|
|
|
cmd1.byte2 |= SMS_PF;
|
|
|
|
|
cmd1.length = dat_len;
|
|
|
|
|
dat.header.blk_desc_len = sizeof(struct blk_desc);
|
|
|
|
|
dat.page.page_code = HP4020I_PAGE_CODE_22;
|
|
|
|
|
dat.page.param_len = sizeof(dat.page.pages.page_0x22);
|
|
|
|
|
blk_len = 2352;
|
|
|
|
|
if (ws->catalog[0] >= '0' && ws->catalog[0] <= '9') {
|
|
|
|
|
dat.page.pages.page_0x22.catalog_valid = 1;
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c1_c2 = ((ws->catalog[0]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[1]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c3_c4 = ((ws->catalog[2]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[3]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c5_c6 = ((ws->catalog[4]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[5]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c7_c8 = ((ws->catalog[6]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[7]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c9_c10 = ((ws->catalog[8]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[9]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c11_c12 = ((ws->catalog[10]-'0') << 4)
|
|
|
|
|
| ((ws->catalog[11]-'0') << 4);
|
|
|
|
|
dat.page.pages.page_0x22.catalog_c13_0 = ((ws->catalog[12]-'0') << 4);
|
|
|
|
|
}
|
|
|
|
|
scsi_uto3b(blk_len, dat.blk_desc.blklen);
|
|
|
|
|
|
|
|
|
|
error = scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd1,
|
|
|
|
|
sizeof(cmd1),
|
|
|
|
|
(u_char *) &dat,
|
|
|
|
|
dat_len,
|
|
|
|
|
/*WORM_RETRIES*/ 4,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
|
|
|
|
|
if (!error) {
|
|
|
|
|
worm->blk_size = blk_len;
|
|
|
|
|
bzero(&cmd2, sizeof(cmd2));
|
|
|
|
|
cmd2.op_code = WRITE_SESSION;
|
|
|
|
|
cmd2.action = (ws->lofp << 4) | (ws->onp? WORM_FIXATION_ONP: 0) + ws->toc_type;
|
|
|
|
|
scsi_uto2b(ws->length, &cmd2.transfer_length_2);
|
1996-02-02 22:59:48 +00:00
|
|
|
|
|
1997-06-02 20:05:39 +00:00
|
|
|
|
error = scsi_scsi_cmd(sc_link,
|
|
|
|
|
(struct scsi_generic *) &cmd2,
|
|
|
|
|
sizeof(cmd2),
|
|
|
|
|
ws->track_desc,
|
|
|
|
|
ws->length,
|
|
|
|
|
1,
|
|
|
|
|
5000,
|
|
|
|
|
NULL,
|
|
|
|
|
SCSI_DATA_OUT);
|
|
|
|
|
}
|
|
|
|
|
return error;
|
|
|
|
|
}
|
|
|
|
|
|
1996-02-02 22:59:48 +00:00
|
|
|
|
/*
|
|
|
|
|
* End HP C4324/C4325 (4020i) section.
|
|
|
|
|
*/
|