Files
dvdisaster/scsi-layer.h
2009-11-21 16:29:02 +09:00

287 lines
8.2 KiB
C

/* dvdisaster: Additional error correction for optical media.
* Copyright (C) 2004-2009 Carsten Gnoerlich.
* Project home page: http://www.dvdisaster.com
* Email: carsten@dvdisaster.com -or- cgnoerlich@fsfe.org
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA,
* or direct your browser at http://www.gnu.org.
*/
#ifndef SCSI_LAYER_H
#define SCSI_LAYER_H
#ifdef SYS_LINUX
#include <sys/ioctl.h>
#include <linux/cdrom.h>
#endif
#ifdef SYS_MINGW
#include <windows.h>
#include <winioctl.h>
#endif
#ifdef SYS_FREEBSD
#include <camlib.h>
#endif
#ifdef SYS_DARWIN
#define REAL_VERSION VERSION
#undef VERSION
#include <CoreFoundation/CoreFoundation.h>
#include <IOKit/IOKitLib.h>
#include <IOKit/scsi-commands/SCSITaskLib.h>
#include <IOKit/storage/IODVDTypes.h>
#include <mach/mach.h>
#include <string.h>
#include <stdlib.h>
#define VERSION REAL_VERSION
#endif
/***
*** Define the Sense data structure.
***/
/*
* Linux already has one
*/
#ifdef SYS_LINUX
#define MAX_CDB_SIZE CDROM_PACKET_SIZE
/* Now globally defined for all OSes here */
//typedef struct request_sense Sense;
#endif
#ifdef SYS_FREEBSD
#define MAX_CDB_SIZE SCSI_MAX_CDBLEN
#endif
#if defined(SYS_UNKNOWN) || defined(SYS_MINGW) || defined(SYS_NETBSD) || defined(SYS_SOLARIS) || defined(SYS_DARWIN)
#define MAX_CDB_SIZE 16 /* longest possible SCSI command */
#endif
/*
* The sense struct is named differently on various OSes,
* Some have subtle differences in the used data types.
* To avoid typecasting mayhem we simply reproduce the Linux
* version here and use it on all OS versions.
*/
typedef struct _Sense {
guint8 error_code : 7;
guint8 valid : 1;
guint8 segment_number;
guint8 sense_key : 4;
guint8 reserved2 : 1;
guint8 ili : 1;
guint8 reserved1 : 2;
guint8 information[4];
guint8 add_sense_len;
guint8 command_info[4];
guint8 asc;
guint8 ascq;
guint8 fruc;
guint8 sks[3];
guint8 asb[46];
} Sense;
/***
*** The DeviceHandle is pretty much our device abstraction layer.
***
* It contains info about the opened device and the inserted medium.
*/
typedef struct _DeviceHandle
{ /*
* OS-specific data for device access
*/
#if defined(SYS_LINUX) || defined(SYS_NETBSD) || defined(SYS_SOLARIS)
int fd; /* device file descriptor */
#endif
#ifdef SYS_FREEBSD
struct cam_device *camdev; /* camlib device handle */
union ccb *ccb;
#endif
#ifdef SYS_MINGW
HANDLE fd; /* Windows file handle for the device (SPTI case) */
int aspiUsed; /* TRUE is device is accessed via ASPI */
int ha,target,lun; /* ASPI way of describing drives */
#endif
#ifdef SYS_DARWIN
IOCFPlugInInterface **plugInInterface;
MMCDeviceInterface **mmcDeviceInterface;
SCSITaskDeviceInterface **scsiTaskDeviceInterface;
SCSITaskInterface **taskInterface;
IOVirtualRange *range;
#endif
/*
* OS-independent data about the device
*/
char *device; /* /dev/foo or whatever the OS uses to name it */
char devinfo[34]; /* whole device info string from INQUIRY */
char vendor[34]; /* vendor and product info only */
Sense sense; /* sense data from last operation */
double singleRate; /* supposed KB/sec @ single speed */
int maxRate; /* guessed maximum transfer rate */
int clusterSize; /* number of sectors per cluster */
/*
* Raw reading support
*/
int canReadDefective; /* TRUE if drive claims to raw read uncorrectable sectors */
int canC2Scan; /* TRUE if drive supports C2 error scanning */
int c2[MAX_CLUSTER_SIZE]; /* C2 errors per sector */
unsigned char previousReadMode;/* read mode prior to switching to raw reads */
unsigned char previousRetries; /* retries prior to switching */
unsigned char currentReadMode; /* current raw read mode */
RawBuffer *rawBuffer; /* for performing raw read analysis */
int (*read)(struct _DeviceHandle*, unsigned char*, int, int);
int (*readRaw)(struct _DeviceHandle*, unsigned char*, int, int);
/*
* Information about currently inserted medium
*/
gint64 sectors; /* actually used number of sectors */
int sessions; /* number of sessions */
int layers; /* and layers */
char manuID[20]; /* Manufacturer info from ADIP/lead-in */
int mainType; /* CD or DVD */
int subType; /* see enum below */
char *typeDescr; /* human readable form of subType */
int bookType; /* book type */
char *bookDescr; /* human readable of above */
int profile; /* profile selected by drive */
char *profileDescr; /* human readable form of above */
char *shortProfile; /* short version of above */
int isDash; /* DVD- */
int isPlus; /* DVD+ */
int incomplete; /* disc is not finalized or otherwise broken */
int discStatus; /* obtained from READ DISC INFORMATION query */
int rewriteable;
char *mediumDescr; /* textual description of medium */
guint8 mediumFP[16]; /* Medium fingerprint */
gint64 fpSector; /* Sector used for calculating the fingerprint */
int fpState; /* 0=unknown; 1=unreadable; 2=present */
/*
* size alternatives from different sources
*/
gint64 readCapacity; /* value returned by READ CAPACITY */
gint64 userAreaSize; /* size of user area according to DVD Info struct */
gint64 blankCapacity; /* blank capacity (maybe 0 if query failed) */
gint64 rs02Size; /* size reported in RS02 header */
/*
* file system(s) found on medium
*/
EccHeader *rs02Header; /* copy of RS02 header */
struct _IsoInfo *isoInfo; /* Information gathered from ISO filesystem */
/*
* debugging stuff
*/
Bitmap *defects; /* for defect simulation */
} DeviceHandle;
/*
* Media types seem not to be standardized anywhere,
* so we make up our own here.
*/
#define MAIN_TYPE_MASK 0xf0
#define CD 0x10
#define DATA1 0x11
#define XA21 0x12
#define DVD 0x20
#define DVD_RAM 0x21
#define DVD_DASH_R 0x22
#define DVD_DASH_RW 0x23
#define DVD_PLUS_R 0x24
#define DVD_PLUS_RW 0x25
#define DVD_DASH_R_DL 0x26
#define DVD_DASH_RW_DL 0x27
#define DVD_PLUS_R_DL 0x28
#define DVD_PLUS_RW_DL 0x29
#define BD 0x40
#define BD_R 0x41
#define BD_RE 0x42
#define UNSUPPORTED 0x00
/* transport io modes */
#define DATA_WRITE 0
#define DATA_READ 1
#define DATA_NONE 2
/***
*** Exported functions
***/
/*
* OS-dependent wrappers from scsi-<os>.c
*/
DeviceHandle* OpenDevice(char*);
#ifdef SYS_MINGW
DeviceHandle* open_aspi_device(char*, int);
DeviceHandle* open_spti_device(char*);
#endif
int SendPacket(DeviceHandle*, unsigned char*, int, unsigned char*, int, Sense*, int);
/***
*** scsi-layer.c
***
* The really user-visible stuff
*/
enum
{ MODE_PAGE_UNSET,
MODE_PAGE_SET
};
DeviceHandle* OpenAndQueryDevice(char*);
DeviceHandle* QueryMediumInfo(char*);
gint64 CurrentMediumSize(int);
int GetMediumFingerprint(DeviceHandle*, guint8*, gint64);
void CloseDevice(DeviceHandle*);
int InquireDevice(DeviceHandle*, int);
void SetRawMode(DeviceHandle*, int);
void SpinupDevice(DeviceHandle*);
void LoadMedium(struct _DeviceHandle*, int);
int TestUnitReady(DeviceHandle*);
int ReadSectors(DeviceHandle*, unsigned char*, gint64, int);
int ReadSectorsFast(DeviceHandle*, unsigned char*, gint64, int);
#endif /* SCSI_LAYER_H */