750 lines
18 KiB
C
750 lines
18 KiB
C
|
/*
|
||
|
* Io to a plain file or device
|
||
|
*
|
||
|
* written by:
|
||
|
*
|
||
|
* Alain L. Knaff
|
||
|
* alain@linux.lu
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include "sysincludes.h"
|
||
|
#include "stream.h"
|
||
|
#include "mtools.h"
|
||
|
#include "msdos.h"
|
||
|
#include "plain_io.h"
|
||
|
#include "scsi.h"
|
||
|
#include "partition.h"
|
||
|
#include "llong.h"
|
||
|
|
||
|
typedef struct SimpleFile_t {
|
||
|
Class_t *Class;
|
||
|
int refs;
|
||
|
Stream_t *Next;
|
||
|
Stream_t *Buffer;
|
||
|
struct stat stat;
|
||
|
int fd;
|
||
|
mt_off_t offset;
|
||
|
mt_off_t lastwhere;
|
||
|
int seekable;
|
||
|
int privileged;
|
||
|
#ifdef OS_hpux
|
||
|
int size_limited;
|
||
|
#endif
|
||
|
int scsi_sector_size;
|
||
|
void *extra_data; /* extra system dependant information for scsi */
|
||
|
} SimpleFile_t;
|
||
|
|
||
|
|
||
|
/*
|
||
|
* Create an advisory lock on the device to prevent concurrent writes.
|
||
|
* Uses either lockf, flock, or fcntl locking methods. See the Makefile
|
||
|
* and the Configure files for how to specify the proper method.
|
||
|
*/
|
||
|
|
||
|
int lock_dev(int fd, int mode, struct device *dev)
|
||
|
{
|
||
|
#if (defined(HAVE_FLOCK) && defined (LOCK_EX) && defined(LOCK_NB))
|
||
|
/**/
|
||
|
#else /* FLOCK */
|
||
|
|
||
|
#if (defined(HAVE_LOCKF) && defined(F_TLOCK))
|
||
|
/**/
|
||
|
#else /* LOCKF */
|
||
|
|
||
|
#if (defined(F_SETLK) && defined(F_WRLCK))
|
||
|
struct flock flk;
|
||
|
|
||
|
#endif /* FCNTL */
|
||
|
#endif /* LOCKF */
|
||
|
#endif /* FLOCK */
|
||
|
|
||
|
if(IS_NOLOCK(dev))
|
||
|
return 0;
|
||
|
|
||
|
#if (defined(HAVE_FLOCK) && defined (LOCK_EX) && defined(LOCK_NB))
|
||
|
if (flock(fd, (mode ? LOCK_EX : LOCK_SH)|LOCK_NB) < 0)
|
||
|
#else /* FLOCK */
|
||
|
|
||
|
#if (defined(HAVE_LOCKF) && defined(F_TLOCK))
|
||
|
if (mode && lockf(fd, F_TLOCK, 0) < 0)
|
||
|
#else /* LOCKF */
|
||
|
|
||
|
#if (defined(F_SETLK) && defined(F_WRLCK))
|
||
|
flk.l_type = mode ? F_WRLCK : F_RDLCK;
|
||
|
flk.l_whence = 0;
|
||
|
flk.l_start = 0L;
|
||
|
flk.l_len = 0L;
|
||
|
|
||
|
if (fcntl(fd, F_SETLK, &flk) < 0)
|
||
|
#endif /* FCNTL */
|
||
|
#endif /* LOCKF */
|
||
|
#endif /* FLOCK */
|
||
|
{
|
||
|
if(errno == EINVAL
|
||
|
#ifdef EOPNOTSUPP
|
||
|
|| errno == EOPNOTSUPP
|
||
|
#endif
|
||
|
)
|
||
|
return 0;
|
||
|
else
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
typedef int (*iofn) (int, char *, int);
|
||
|
|
||
|
|
||
|
|
||
|
static int file_io(Stream_t *Stream, char *buf, mt_off_t where, int len,
|
||
|
iofn io)
|
||
|
{
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
int ret;
|
||
|
|
||
|
where += This->offset;
|
||
|
|
||
|
if (This->seekable && where != This->lastwhere ){
|
||
|
if(mt_lseek( This->fd, where, SEEK_SET) < 0 ){
|
||
|
perror("seek");
|
||
|
This->lastwhere = (mt_off_t) -1;
|
||
|
return -1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef OS_hpux
|
||
|
/*
|
||
|
* On HP/UX, we can not write more than MAX_LEN bytes in one go.
|
||
|
* If more are written, the write fails with EINVAL
|
||
|
*/
|
||
|
#define MAX_SCSI_LEN (127*1024)
|
||
|
if(This->size_limited && len > MAX_SCSI_LEN)
|
||
|
len = MAX_SCSI_LEN;
|
||
|
#endif
|
||
|
ret = io(This->fd, buf, len);
|
||
|
|
||
|
#ifdef OS_hpux
|
||
|
if (ret == -1 &&
|
||
|
errno == EINVAL && /* if we got EINVAL */
|
||
|
len > MAX_SCSI_LEN) {
|
||
|
This->size_limited = 1;
|
||
|
len = MAX_SCSI_LEN;
|
||
|
ret = io(This->fd, buf, len);
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if ( ret == -1 ){
|
||
|
perror("plain_io");
|
||
|
This->lastwhere = (mt_off_t) -1;
|
||
|
return -1;
|
||
|
}
|
||
|
This->lastwhere = where + ret;
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
static int file_read(Stream_t *Stream, char *buf, mt_off_t where, size_t len)
|
||
|
{
|
||
|
return file_io(Stream, buf, where, len, (iofn) read);
|
||
|
}
|
||
|
|
||
|
static int file_write(Stream_t *Stream, char *buf, mt_off_t where, size_t len)
|
||
|
{
|
||
|
return file_io(Stream, buf, where, len, (iofn) write);
|
||
|
}
|
||
|
|
||
|
static int file_flush(Stream_t *Stream)
|
||
|
{
|
||
|
#if 0
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
return fsync(This->fd);
|
||
|
#endif
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int file_free(Stream_t *Stream)
|
||
|
{
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
if (This->fd > 2)
|
||
|
return close(This->fd);
|
||
|
else
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int file_geom(Stream_t *Stream, struct device *dev,
|
||
|
struct device *orig_dev,
|
||
|
int media, struct bootsector *boot)
|
||
|
{
|
||
|
int ret;
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
size_t tot_sectors;
|
||
|
int BootP, Infp0, InfpX, InfTm;
|
||
|
int sectors, j;
|
||
|
unsigned char sum;
|
||
|
int sect_per_track;
|
||
|
struct label_blk_t *labelBlock;
|
||
|
|
||
|
dev->ssize = 2; /* allow for init_geom to change it */
|
||
|
dev->use_2m = 0x80; /* disable 2m mode to begin */
|
||
|
|
||
|
if(media == 0xf0 || media >= 0x100){
|
||
|
dev->heads = WORD(nheads);
|
||
|
dev->sectors = WORD(nsect);
|
||
|
tot_sectors = DWORD(bigsect);
|
||
|
SET_INT(tot_sectors, WORD(psect));
|
||
|
sect_per_track = dev->heads * dev->sectors;
|
||
|
tot_sectors += sect_per_track - 1; /* round size up */
|
||
|
dev->tracks = tot_sectors / sect_per_track;
|
||
|
|
||
|
BootP = WORD(ext.old.BootP);
|
||
|
Infp0 = WORD(ext.old.Infp0);
|
||
|
InfpX = WORD(ext.old.InfpX);
|
||
|
InfTm = WORD(ext.old.InfTm);
|
||
|
|
||
|
if(WORD(fatlen)) {
|
||
|
labelBlock = &boot->ext.old.labelBlock;
|
||
|
} else {
|
||
|
labelBlock = &boot->ext.fat32.labelBlock;
|
||
|
}
|
||
|
|
||
|
if (boot->descr >= 0xf0 &&
|
||
|
labelBlock->dos4 == 0x29 &&
|
||
|
strncmp( boot->banner,"2M", 2 ) == 0 &&
|
||
|
BootP < 512 && Infp0 < 512 && InfpX < 512 && InfTm < 512 &&
|
||
|
BootP >= InfTm + 2 && InfTm >= InfpX && InfpX >= Infp0 &&
|
||
|
Infp0 >= 76 ){
|
||
|
for (sum=0, j=63; j < BootP; j++)
|
||
|
sum += boot->jump[j];/* checksum */
|
||
|
dev->ssize = boot->jump[InfTm];
|
||
|
if (!sum && dev->ssize <= 7){
|
||
|
dev->use_2m = 0xff;
|
||
|
dev->ssize |= 0x80; /* is set */
|
||
|
}
|
||
|
}
|
||
|
} else if (media >= 0xf8){
|
||
|
media &= 3;
|
||
|
dev->heads = old_dos[media].heads;
|
||
|
dev->tracks = old_dos[media].tracks;
|
||
|
dev->sectors = old_dos[media].sectors;
|
||
|
dev->ssize = 0x80;
|
||
|
dev->use_2m = ~1;
|
||
|
} else {
|
||
|
fprintf(stderr,"Unknown media type\n");
|
||
|
exit(1);
|
||
|
}
|
||
|
|
||
|
sectors = dev->sectors;
|
||
|
dev->sectors = dev->sectors * WORD(secsiz) / 512;
|
||
|
|
||
|
#ifdef JPD
|
||
|
printf("file_geom:media=%0X=>cyl=%d,heads=%d,sects=%d,ssize=%d,use2m=%X\n",
|
||
|
media, dev->tracks, dev->heads, dev->sectors, dev->ssize,
|
||
|
dev->use_2m);
|
||
|
#endif
|
||
|
ret = init_geom(This->fd,dev, orig_dev, &This->stat);
|
||
|
dev->sectors = sectors;
|
||
|
#ifdef JPD
|
||
|
printf("f_geom: after init_geom(), sects=%d\n", dev->sectors);
|
||
|
#endif
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int file_data(Stream_t *Stream, time_t *date, mt_size_t *size,
|
||
|
int *type, int *address)
|
||
|
{
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
if(date)
|
||
|
*date = This->stat.st_mtime;
|
||
|
if(size)
|
||
|
*size = This->stat.st_size;
|
||
|
if(type)
|
||
|
*type = S_ISDIR(This->stat.st_mode);
|
||
|
if(address)
|
||
|
*address = 0;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* ZIP or other scsi device on Solaris or SunOS system.
|
||
|
Since Sun won't accept a non-Sun label on a scsi disk, we must
|
||
|
bypass Sun's disk interface and use low-level SCSI commands to read
|
||
|
or write the ZIP drive. We thus replace the file_read and file_write
|
||
|
routines with our own scsi_read and scsi_write routines, that use the
|
||
|
uscsi ioctl interface. By James Dugal, jpd@usl.edu, 11-96. Tested
|
||
|
under Solaris 2.5 and SunOS 4.3.1_u1 using GCC.
|
||
|
|
||
|
Note: the mtools.conf entry for a ZIP drive would look like this:
|
||
|
(solaris) drive C: file="/dev/rdsk/c0t5d0s2" partition=4 FAT=16 nodelay exclusive scsi=&
|
||
|
(sunos) drive C: file="/dev/rsd5c" partition=4 FAT=16 nodelay exclusive scsi=1
|
||
|
|
||
|
Note 2: Sol 2.5 wants mtools to be suid-root, to use the ioctl. SunOS is
|
||
|
happy if we just have access to the device, so making mtools sgid to a
|
||
|
group called, say, "ziprw" which has rw permission on /dev/rsd5c, is fine.
|
||
|
*/
|
||
|
|
||
|
#define MAXBLKSPERCMD 255
|
||
|
|
||
|
static void scsi_init(SimpleFile_t *This)
|
||
|
{
|
||
|
int fd = This->fd;
|
||
|
unsigned char cdb[10],buf[8];
|
||
|
|
||
|
memset(cdb, 0, sizeof cdb);
|
||
|
memset(buf,0, sizeof(buf));
|
||
|
cdb[0]=SCSI_READ_CAPACITY;
|
||
|
if (scsi_cmd(fd, (unsigned char *)cdb,
|
||
|
sizeof(cdb), SCSI_IO_READ, buf, sizeof(buf), This->extra_data)==0)
|
||
|
{
|
||
|
This->scsi_sector_size=
|
||
|
((unsigned)buf[5]<<16)|((unsigned)buf[6]<<8)|(unsigned)buf[7];
|
||
|
if (This->scsi_sector_size != 512)
|
||
|
fprintf(stderr," (scsi_sector_size=%d)\n",This->scsi_sector_size);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int scsi_io(Stream_t *Stream, char *buf, mt_off_t where, size_t len, int rwcmd)
|
||
|
{
|
||
|
unsigned int firstblock, nsect;
|
||
|
int clen,r,max;
|
||
|
off_t offset;
|
||
|
unsigned char cdb[10];
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
firstblock=truncBytes32((where + This->offset)/This->scsi_sector_size);
|
||
|
/* 512,1024,2048,... bytes/sector supported */
|
||
|
offset=truncBytes32(where + This->offset -
|
||
|
firstblock*This->scsi_sector_size);
|
||
|
nsect=(offset+len+This->scsi_sector_size-1)/ This->scsi_sector_size;
|
||
|
#if defined(OS_sun) && defined(OS_i386)
|
||
|
if (This->scsi_sector_size>512)
|
||
|
firstblock*=This->scsi_sector_size/512; /* work around a uscsi bug */
|
||
|
#endif /* sun && i386 */
|
||
|
|
||
|
if (len>512) {
|
||
|
/* avoid buffer overruns. The transfer MUST be smaller or
|
||
|
* equal to the requested size! */
|
||
|
while (nsect*This->scsi_sector_size>len)
|
||
|
--nsect;
|
||
|
if(!nsect) {
|
||
|
fprintf(stderr,"Scsi buffer too small\n");
|
||
|
exit(1);
|
||
|
}
|
||
|
if(rwcmd == SCSI_IO_WRITE && offset) {
|
||
|
/* there seems to be no memmove before a write */
|
||
|
fprintf(stderr,"Unaligned write\n");
|
||
|
exit(1);
|
||
|
}
|
||
|
/* a better implementation should use bounce buffers.
|
||
|
* However, in normal operation no buffer overruns or
|
||
|
* unaligned writes should happen anyways, as the logical
|
||
|
* sector size is (hopefully!) equal to the physical one
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
|
||
|
max = scsi_max_length();
|
||
|
|
||
|
if (nsect > max)
|
||
|
nsect=max;
|
||
|
|
||
|
/* set up SCSI READ/WRITE command */
|
||
|
memset(cdb, 0, sizeof cdb);
|
||
|
|
||
|
switch(rwcmd) {
|
||
|
case SCSI_IO_READ:
|
||
|
cdb[0] = SCSI_READ;
|
||
|
break;
|
||
|
case SCSI_IO_WRITE:
|
||
|
cdb[0] = SCSI_WRITE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
cdb[1] = 0;
|
||
|
|
||
|
if (firstblock > 0x1fffff || nsect > 0xff) {
|
||
|
/* I suspect that the ZIP drive also understands Group 1
|
||
|
* commands. If that is indeed true, we may chose Group 1
|
||
|
* more agressively in the future */
|
||
|
|
||
|
cdb[0] |= SCSI_GROUP1;
|
||
|
clen=10; /* SCSI Group 1 cmd */
|
||
|
|
||
|
/* this is one of the rare case where explicit coding is
|
||
|
* more portable than macros... The meaning of scsi command
|
||
|
* bytes is standardised, whereas the preprocessor macros
|
||
|
* handling it might be not... */
|
||
|
|
||
|
cdb[2] = (unsigned char) (firstblock >> 24) & 0xff;
|
||
|
cdb[3] = (unsigned char) (firstblock >> 16) & 0xff;
|
||
|
cdb[4] = (unsigned char) (firstblock >> 8) & 0xff;
|
||
|
cdb[5] = (unsigned char) firstblock & 0xff;
|
||
|
cdb[6] = 0;
|
||
|
cdb[7] = (unsigned char) (nsect >> 8) & 0xff;
|
||
|
cdb[8] = (unsigned char) nsect & 0xff;
|
||
|
cdb[9] = 0;
|
||
|
} else {
|
||
|
clen = 6; /* SCSI Group 0 cmd */
|
||
|
cdb[1] |= (unsigned char) ((firstblock >> 16) & 0x1f);
|
||
|
cdb[2] = (unsigned char) ((firstblock >> 8) & 0xff);
|
||
|
cdb[3] = (unsigned char) firstblock & 0xff;
|
||
|
cdb[4] = (unsigned char) nsect;
|
||
|
cdb[5] = 0;
|
||
|
}
|
||
|
|
||
|
if(This->privileged)
|
||
|
reclaim_privs();
|
||
|
|
||
|
r=scsi_cmd(This->fd, (unsigned char *)cdb, clen, rwcmd, buf,
|
||
|
nsect*This->scsi_sector_size, This->extra_data);
|
||
|
|
||
|
if(This->privileged)
|
||
|
drop_privs();
|
||
|
|
||
|
if(r) {
|
||
|
perror(rwcmd == SCSI_IO_READ ? "SCMD_READ" : "SCMD_WRITE");
|
||
|
return -1;
|
||
|
}
|
||
|
#ifdef JPD
|
||
|
printf("finished %u for %u\n", firstblock, nsect);
|
||
|
#endif
|
||
|
|
||
|
#ifdef JPD
|
||
|
printf("zip: read or write OK\n");
|
||
|
#endif
|
||
|
if (offset>0) memmove(buf,buf+offset,nsect*This->scsi_sector_size-offset);
|
||
|
if (len==256) return 256;
|
||
|
else if (len==512) return 512;
|
||
|
else return nsect*This->scsi_sector_size-offset;
|
||
|
}
|
||
|
|
||
|
int scsi_read(Stream_t *Stream, char *buf, mt_off_t where, size_t len)
|
||
|
{
|
||
|
|
||
|
#ifdef JPD
|
||
|
printf("zip: to read %d bytes at %d\n", len, where);
|
||
|
#endif
|
||
|
return scsi_io(Stream, buf, where, len, SCSI_IO_READ);
|
||
|
}
|
||
|
|
||
|
int scsi_write(Stream_t *Stream, char *buf, mt_off_t where, size_t len)
|
||
|
{
|
||
|
#ifdef JPD
|
||
|
Printf("zip: to write %d bytes at %d\n", len, where);
|
||
|
#endif
|
||
|
return scsi_io(Stream, buf, where, len, SCSI_IO_WRITE);
|
||
|
}
|
||
|
|
||
|
static Class_t ScsiClass = {
|
||
|
scsi_read,
|
||
|
scsi_write,
|
||
|
file_flush,
|
||
|
file_free,
|
||
|
file_geom,
|
||
|
file_data,
|
||
|
0 /* pre-allocate */
|
||
|
};
|
||
|
|
||
|
|
||
|
static Class_t SimpleFileClass = {
|
||
|
file_read,
|
||
|
file_write,
|
||
|
file_flush,
|
||
|
file_free,
|
||
|
file_geom,
|
||
|
file_data,
|
||
|
0 /* pre_allocate */
|
||
|
};
|
||
|
|
||
|
|
||
|
Stream_t *SimpleFileOpen(struct device *dev, struct device *orig_dev,
|
||
|
const char *name, int mode, char *errmsg,
|
||
|
int mode2, int locked, mt_size_t *maxSize)
|
||
|
{
|
||
|
SimpleFile_t *This;
|
||
|
#ifdef __EMX__
|
||
|
HFILE FileHandle;
|
||
|
ULONG Action;
|
||
|
APIRET rc;
|
||
|
#endif
|
||
|
This = New(SimpleFile_t);
|
||
|
if (!This){
|
||
|
printOom();
|
||
|
return 0;
|
||
|
}
|
||
|
This->scsi_sector_size = 512;
|
||
|
This->seekable = 1;
|
||
|
#ifdef OS_hpux
|
||
|
This->size_limited = 0;
|
||
|
#endif
|
||
|
This->Class = &SimpleFileClass;
|
||
|
if (!name || strcmp(name,"-") == 0 ){
|
||
|
if (mode == O_RDONLY)
|
||
|
This->fd = 0;
|
||
|
else
|
||
|
This->fd = 1;
|
||
|
This->seekable = 0;
|
||
|
This->refs = 1;
|
||
|
This->Next = 0;
|
||
|
This->Buffer = 0;
|
||
|
if (fstat(This->fd, &This->stat) < 0) {
|
||
|
Free(This);
|
||
|
if(errmsg)
|
||
|
#ifdef HAVE_SNPRINTF
|
||
|
snprintf(errmsg,199,"Can't stat -: %s",
|
||
|
strerror(errno));
|
||
|
#else
|
||
|
sprintf(errmsg,"Can't stat -: %s",
|
||
|
strerror(errno));
|
||
|
#endif
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
return (Stream_t *) This;
|
||
|
}
|
||
|
|
||
|
|
||
|
if(dev) {
|
||
|
if(!(mode2 & NO_PRIV))
|
||
|
This->privileged = IS_PRIVILEGED(dev);
|
||
|
mode |= dev->mode;
|
||
|
}
|
||
|
|
||
|
precmd(dev);
|
||
|
if(IS_PRIVILEGED(dev) && !(mode2 & NO_PRIV))
|
||
|
reclaim_privs();
|
||
|
|
||
|
#ifdef __EMX__
|
||
|
#define DOSOPEN_FLAGS (OPEN_FLAGS_DASD | OPEN_FLAGS_WRITE_THROUGH | \
|
||
|
OPEN_FLAGS_NOINHERIT | OPEN_FLAGS_RANDOM | \
|
||
|
OPEN_FLAGS_NO_CACHE)
|
||
|
#define DOSOPEN_FD_ACCESS (OPEN_SHARE_DENYREADWRITE | OPEN_ACCESS_READWRITE)
|
||
|
#define DOSOPEN_HD_ACCESS (OPEN_SHARE_DENYNONE | OPEN_ACCESS_READONLY)
|
||
|
|
||
|
if (skip_drive(name) > name) {
|
||
|
rc = DosOpen(
|
||
|
name, &FileHandle, &Action, 0L, FILE_NORMAL,
|
||
|
OPEN_ACTION_OPEN_IF_EXISTS, DOSOPEN_FLAGS |
|
||
|
(IS_NOLOCK(dev)?DOSOPEN_HD_ACCESS:DOSOPEN_FD_ACCESS),
|
||
|
0L);
|
||
|
#ifdef DEBUG
|
||
|
if (rc != NO_ERROR) fprintf (stderr, "DosOpen() returned %d\n", rc);
|
||
|
#endif
|
||
|
if (!IS_NOLOCK(dev)) {
|
||
|
rc = DosDevIOCtl(
|
||
|
FileHandle, 0x08L, DSK_LOCKDRIVE, 0, 0, 0, 0, 0, 0);
|
||
|
#ifdef DEBUG
|
||
|
if (rc != NO_ERROR) fprintf (stderr, "DosDevIOCtl() returned %d\n", rc);
|
||
|
#endif
|
||
|
}
|
||
|
if (rc == NO_ERROR)
|
||
|
This->fd = _imphandle(FileHandle); else This->fd = -1;
|
||
|
} else
|
||
|
#endif
|
||
|
{
|
||
|
if (IS_SCSI(dev))
|
||
|
This->fd = scsi_open(name, mode, IS_NOLOCK(dev)?0444:0666,
|
||
|
&This->extra_data);
|
||
|
else
|
||
|
This->fd = open(name, mode, IS_NOLOCK(dev)?0444:0666);
|
||
|
}
|
||
|
|
||
|
if(IS_PRIVILEGED(dev) && !(mode2 & NO_PRIV))
|
||
|
drop_privs();
|
||
|
|
||
|
if (This->fd < 0) {
|
||
|
Free(This);
|
||
|
if(errmsg)
|
||
|
#ifdef HAVE_SNPRINTF
|
||
|
snprintf(errmsg, 199, "Can't open %s: %s",
|
||
|
name, strerror(errno));
|
||
|
#else
|
||
|
sprintf(errmsg, "Can't open %s: %s",
|
||
|
name, strerror(errno));
|
||
|
#endif
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if(IS_PRIVILEGED(dev) && !(mode2 & NO_PRIV))
|
||
|
closeExec(This->fd);
|
||
|
|
||
|
#ifdef __EMX__
|
||
|
if (*(name+1) != ':')
|
||
|
#endif
|
||
|
if (fstat(This->fd, &This->stat) < 0){
|
||
|
Free(This);
|
||
|
if(errmsg) {
|
||
|
#ifdef HAVE_SNPRINTF
|
||
|
snprintf(errmsg,199,"Can't stat %s: %s",
|
||
|
name, strerror(errno));
|
||
|
#else
|
||
|
if(strlen(name) > 50) {
|
||
|
sprintf(errmsg,"Can't stat file: %s",
|
||
|
strerror(errno));
|
||
|
} else {
|
||
|
sprintf(errmsg,"Can't stat %s: %s",
|
||
|
name, strerror(errno));
|
||
|
}
|
||
|
#endif
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
#ifndef __EMX__
|
||
|
/* lock the device on writes */
|
||
|
if (locked && lock_dev(This->fd, mode == O_RDWR, dev)) {
|
||
|
if(errmsg)
|
||
|
#ifdef HAVE_SNPRINTF
|
||
|
snprintf(errmsg,199,
|
||
|
"plain floppy: device \"%s\" busy (%s):",
|
||
|
dev ? dev->name : "unknown", strerror(errno));
|
||
|
#else
|
||
|
sprintf(errmsg,
|
||
|
"plain floppy: device \"%s\" busy (%s):",
|
||
|
(dev && strlen(dev->name) < 50) ?
|
||
|
dev->name : "unknown", strerror(errno));
|
||
|
#endif
|
||
|
|
||
|
close(This->fd);
|
||
|
Free(This);
|
||
|
return NULL;
|
||
|
}
|
||
|
#endif
|
||
|
/* set default parameters, if needed */
|
||
|
if (dev){
|
||
|
if ((IS_MFORMAT_ONLY(dev) || !dev->tracks) &&
|
||
|
init_geom(This->fd, dev, orig_dev, &This->stat)){
|
||
|
close(This->fd);
|
||
|
Free(This);
|
||
|
if(errmsg)
|
||
|
sprintf(errmsg,"init: set default params");
|
||
|
return NULL;
|
||
|
}
|
||
|
This->offset = (mt_off_t) dev->offset;
|
||
|
} else
|
||
|
This->offset = 0;
|
||
|
|
||
|
This->refs = 1;
|
||
|
This->Next = 0;
|
||
|
This->Buffer = 0;
|
||
|
|
||
|
if(maxSize) {
|
||
|
if (IS_SCSI(dev)) {
|
||
|
*maxSize = MAX_OFF_T_B(31+log_2(This->scsi_sector_size));
|
||
|
} else {
|
||
|
*maxSize = max_off_t_seek;
|
||
|
}
|
||
|
if(This->offset > *maxSize) {
|
||
|
close(This->fd);
|
||
|
Free(This);
|
||
|
if(errmsg)
|
||
|
sprintf(errmsg,"init: Big disks not supported");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
*maxSize -= This->offset;
|
||
|
}
|
||
|
/* partitioned drive */
|
||
|
|
||
|
/* jpd@usl.edu: assume a partitioned drive on these 2 systems is a ZIP*/
|
||
|
/* or similar drive that must be accessed by low-level scsi commands */
|
||
|
/* AK: introduce new "scsi=1" statement to specifically set
|
||
|
* this option. Indeed, there could conceivably be partitioned
|
||
|
* devices where low level scsi commands will not be needed */
|
||
|
if(IS_SCSI(dev)) {
|
||
|
This->Class = &ScsiClass;
|
||
|
if(This->privileged)
|
||
|
reclaim_privs();
|
||
|
scsi_init(This);
|
||
|
if(This->privileged)
|
||
|
drop_privs();
|
||
|
}
|
||
|
while(!(mode2 & NO_OFFSET) &&
|
||
|
dev && dev->partition && dev->partition <= 4) {
|
||
|
int has_activated, last_end, j;
|
||
|
unsigned char buf[2048];
|
||
|
struct partition *partTable=(struct partition *)(buf+ 0x1ae);
|
||
|
size_t partOff;
|
||
|
|
||
|
/* read the first sector, or part of it */
|
||
|
if (force_read((Stream_t *)This, (char*) buf, 0, 512) != 512)
|
||
|
break;
|
||
|
if( _WORD(buf+510) != 0xaa55)
|
||
|
break;
|
||
|
|
||
|
partOff = BEGIN(partTable[dev->partition]);
|
||
|
if (maxSize) {
|
||
|
if (partOff > *maxSize >> 9) {
|
||
|
close(This->fd);
|
||
|
Free(This);
|
||
|
if(errmsg)
|
||
|
sprintf(errmsg,"init: Big disks not supported");
|
||
|
return NULL;
|
||
|
}
|
||
|
*maxSize -= (mt_off_t) partOff << 9;
|
||
|
}
|
||
|
|
||
|
This->offset += (mt_off_t) partOff << 9;
|
||
|
if(!partTable[dev->partition].sys_ind) {
|
||
|
if(errmsg)
|
||
|
sprintf(errmsg,
|
||
|
"init: non-existant partition");
|
||
|
close(This->fd);
|
||
|
Free(This);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if(!dev->tracks) {
|
||
|
dev->heads = head(partTable[dev->partition].end)+1;
|
||
|
dev->sectors = sector(partTable[dev->partition].end);
|
||
|
dev->tracks = cyl(partTable[dev->partition].end) -
|
||
|
cyl(partTable[dev->partition].start)+1;
|
||
|
}
|
||
|
dev->hidden=dev->sectors*head(partTable[dev->partition].start);
|
||
|
if(!mtools_skip_check &&
|
||
|
consistencyCheck((struct partition *)(buf+0x1ae), 0, 0,
|
||
|
&has_activated, &last_end, &j, dev, 0)) {
|
||
|
fprintf(stderr,
|
||
|
"Warning: inconsistent partition table\n");
|
||
|
fprintf(stderr,
|
||
|
"Possibly unpartitioned device\n");
|
||
|
fprintf(stderr,
|
||
|
"\n*** Maybe try without partition=%d in "
|
||
|
"device definition ***\n\n",
|
||
|
dev->partition);
|
||
|
fprintf(stderr,
|
||
|
"If this is a PCMCIA card, or a disk "
|
||
|
"partitioned on another computer, this "
|
||
|
"message may be in error: add "
|
||
|
"mtools_skip_check=1 to your .mtoolsrc "
|
||
|
"file to suppress this warning\n");
|
||
|
|
||
|
}
|
||
|
break;
|
||
|
/* NOTREACHED */
|
||
|
}
|
||
|
|
||
|
This->lastwhere = -This->offset;
|
||
|
/* provoke a seek on those devices that don't start on a partition
|
||
|
* boundary */
|
||
|
|
||
|
return (Stream_t *) This;
|
||
|
}
|
||
|
|
||
|
int get_fd(Stream_t *Stream)
|
||
|
{
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
return This->fd;
|
||
|
}
|
||
|
|
||
|
void *get_extra_data(Stream_t *Stream)
|
||
|
{
|
||
|
DeclareThis(SimpleFile_t);
|
||
|
|
||
|
return This->extra_data;
|
||
|
}
|