mirror of
https://github.com/open-simh/simtools.git
synced 2026-01-20 09:55:21 +00:00
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.
220 lines
6.8 KiB
C
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;
|
|
}
|