Timothe Litt 61c8001044 Bug fixes and compiler warnings. Unix readline support, pass 1
Fix various compiler warnings.

Fix bug causing double free when a file isn't found.

Fix bug using uninitialized variable parsing null filename.

Fix bug causing crash when format 3 retrieval pointer encountered.

Add support for readline (command line editing and history on Unix)

Untangle NT I/O so it builds without the direct access SCSI API & works.

Report errors as text messages everywhere.

Add MSVC project files.

Implement most of dir/full

Partially implement XABITM

Add help to command tables.

Allow choice of VMS qualifiers or Unix options.
  mount /write // /dev/cdrom
or
  mount -write /dev/cdrom

Parse quoted strings as command parameters.
   Mount /write "/dev/cdrom"
   search [*...]*.txt "My words for you"

Resolve command, parameter & qualifier ambiguity from tables.

Consolidate the various makefiles into a common file with very small
platform-specific wrappers.  This simplifies maintenance.

Add diskio module to allow easy access to .iso images and simulator files.
Removes requirement for loop device or equivalent.  Builds as a separate
executable.

Writes to the ODS2 volumes are broken.
2016-02-26 17:46:46 -05:00

220 lines
6.8 KiB
C

/* PHYOS2.C v1.2 Physical I/O module for OS2 */
/* This version implemented to read from Floppy or CD by
guessing that A: or B: will be a floppy and anything
else is probably a CD? */
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#define INCL_DOS
#define INCL_DOSDEVIOCTL
#define INCL_NOPMAPI
#include <os2.h>
#include "phyio.h"
#include "ssdef.h"
unsigned init_count = 0;
unsigned read_count = 0;
unsigned write_count = 0;
void phyio_show(void)
{
printf("PHYIO_SHOW Initializations: %d Reads: %d Writes: %d\n",
init_count,read_count,write_count);
}
#pragma pack(1)
/*------------------------------------------------*
* Cat 0x80, Func 0x72: Read Long *
*------------------------------------------------*/
struct ReadLong {
ULONG ID_code;
UCHAR address_mode;
USHORT transfer_count;
ULONG start_sector;
UCHAR reserved;
UCHAR interleave_size;
UCHAR interleave_skip_factor;
};
#pragma pack()
#define SECTORSIZE 2048
struct ReadLong rl = {
0x31304443L, /* "CD01" */
0, /* Address mode */
1, /* Transfer count */
0, /* Start Sector */
0, /* Reserved */
0, /* Interleave size */
0/* Skip factor */
};
#define HANDLE_MAX 32
int hand_count = 0;
struct HANDLE {
HFILE hand_hfile;
int hand_drive;
} handle[HANDLE_MAX];
void phyio_help(FILE *fp ) {
fprintf( fp, "Specify the device to be mounted as a drive letter\n" );
fprintf( fp, "E.g. mount D:\n" );
fprintf( fp, "The drive letter must be between A: and Z:\n" );
fprintf( fp, "The drive is accessed as a physical device\n" );
fprintf( fp, "Use ODS2-Image to work with disk images such as .ISO or simulator files.\n" );
return;
}
unsigned phyio_init(int devlen,char *devnam,unsigned *hand,struct phyio_info *info)
{
if (hand_count < HANDLE_MAX - 1) {
ULONG usAction;
ULONG open_mode;
ULONG bCmd;
ULONG stat;
char device[40];
memcpy(device,devnam,devlen);
device[devlen] = '\0';
handle[hand_count].hand_drive = toupper(*device);
open_mode = OPEN_FLAGS_DASD | OPEN_SHARE_DENYREADWRITE
| OPEN_FLAGS_WRITE_THROUGH | OPEN_FLAGS_FAIL_ON_ERROR
| OPEN_FLAGS_NOINHERIT | OPEN_ACCESS_READWRITE;
stat = DosOpen(device, /* filename to open */
&handle[hand_count].hand_hfile,
/* address of file handle */
&usAction,
/* address to store action taken */
0L,
/* size of new file */
FILE_NORMAL,
/* file's attribute */
FILE_OPEN,
/* action to take if file exists */
open_mode,
/* file's open mode */
0L);
/* Reserved */
if (stat) {
switch (stat) {
case 15:
printf("Drive %s is invalid.\n",device);
break;
case 21:
printf("Drive %s is not ready.\n",device);
break;
case 32:
printf("Drive %s is currently locked by another process.\n",device);
break;
case 65:
printf("Drive %s is a network drive.\n",device);
break;
default:
printf("error %d opening drive %s\n",stat,device);
}
return SS$_PARITY;
}
bCmd = 0;
stat = DosDevIOCtl(handle[hand_count].hand_hfile,IOCTL_DISK,DSK_LOCKDRIVE,&bCmd,1,NULL,NULL,0,NULL);
if (stat) {
printf("Error %d locking drive\n",stat);
return SS$_DEVNOTALLOC;
}
if (handle[hand_count].hand_drive > 'C') {
info->status = PHYIO_READONLY;
} else {
info->status = 0;
}
*hand = hand_count++;
init_count++;
return 1;
} else {
return SS$_IVCHAN;
}
}
unsigned phy_getsect(HFILE hfile,unsigned sector,char *buffer)
{
ULONG ulPinout,ulDinout;
ULONG stat;
char rawsect[SECTORSIZE + 304];
ulPinout = sizeof(rl);
ulDinout = sizeof(rawsect);
rl.start_sector = sector;
stat = DosDevIOCtl(hfile,
IOCTL_CDROMDISK,
CDROMDISK_READLONG,
&rl,sizeof(rl),&ulPinout,
rawsect,sizeof(rawsect),&ulDinout);
if (stat) {
printf("sys%04u: CDROMDISK_READLONG error\n",stat);
return SS$_PARITY;
}
memcpy(buffer,rawsect + 16,SECTORSIZE);
return 1;
}
unsigned phyio_read(unsigned handno,unsigned block,unsigned length,char *buffer)
{
register unsigned sts = 1;
#ifdef DEBUG
printf("PHYIO_READ block %d length %d\n",block,length);
#endif
if (handno >= hand_count) {
sts = SS$_IVCHAN;
} else {
if (handle[handno].hand_drive > 'C') {
register unsigned sect = block * 512 / SECTORSIZE;
register unsigned offset = (block - sect * SECTORSIZE / 512) * 512;
while (length > 0) {
register unsigned transfer;
if (offset == 0 && length >= SECTORSIZE) {
transfer = SECTORSIZE;
if (((sts = phy_getsect(handle[handno].hand_hfile,sect,buffer)) & 1) == 0) break;
} else {
char sector[SECTORSIZE];
if (((sts = phy_getsect(handle[handno].hand_hfile,sect,sector)) & 1) == 0) break;
transfer = SECTORSIZE - offset;
if (transfer > length) transfer = length;
memcpy(buffer,sector + offset,transfer);
}
buffer += transfer;
length -= transfer;
sect++;
offset = 0;
}
} else {
USHORT rc;
ULONG oldpos,lenread;
sts = SS$_PARITY;
rc = DosSetFilePtr(handle[handno].hand_hfile,block * 512,0,&oldpos);
if (rc == 0) {
rc = DosRead(handle[handno].hand_hfile,buffer,length,&lenread);
if (rc == 0 && lenread == length) sts = 1;
}
}
}
if (sts & 1) {
read_count++;
} else {
printf("PHYOS2 Error %d Block %d Length %d\n",sts,block,length);
}
return sts;
}
unsigned phyio_write(unsigned handle,unsigned block,unsigned length,char *buffer)
{
write_count++;
return SS$_WRITLCK;
}