Files
open-simh.simtools/extracters/dbtap/tapeio.c
John Forecast b08ebe8eb0 New tape manipulation tools
rawtap	Allows extract, create and append operations on .tap files.

cpytap	Copies a .tap file to a new .tap file while allowing file level edits; skip file, replace file,
		append files and insert files. Any files copied from the original source .tap will have
		their internal record structure maintained.

cosy		COSY is the compressed format used by the CDC1700. This program allows for
		extraction of all files from an archive and the creation of a new archive. It assumes
		that you would have used raw tap about to have extracted the COSY file from a
		tape.

dbtap	Utility to read, write and list .tap containers written in the DOS/BATCH-11 format. It
		understands ascii and binary modes and can be used to transfer files in and out of
		most PDP-11 operating systems (not sure about RSTS/E), early VMS and early
		TOPS-10 systems.
2017-03-14 13:44:02 -07:00

615 lines
13 KiB
C
Raw Blame History

This file contains invisible Unicode characters
This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
/* tapeio.c: Tape I/O routines
Copyright (c) 2017, John Forecast
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
JOHN FORECAST BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Except as contained in this notice, the name of John Forecast shall not
be used in advertising or otherwise to promote the sale, use or other dealings
in this Software without prior written authorization from John Forecast.
*/
#include <stddef.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <sys/stat.h>
#include "tapeio.h"
FILE *tfile = NULL;
char buffer[MAXRCLNT];
int rLength, occupied;
static int verifyFormat(void);
/*++
* OpenTapeForRead
*
* Open an existing SIMH .tap format file for read access. If the file is
* successfully opened, scan the file to determine if it is a valid .tap
* format file and whether there are error records present.
*
* Inputs:
*
* name - name of the file
*
* Outputs:
*
* None
*
* Returns:
*
* TIO_SUCCESS - file successfully opened, format is valid
* TIO_ERROR - file successfully opened, error records present
* TIO_CORRUPT - file successfully opened, format is invalid
* TIO_OPENFAIL - file open failed
*
* Note the file remains open if the return status is TIO_SUCCESS or
* TIO_ERROR
*
--*/
int OpenTapeForRead(
char *name
)
{
if ((tfile = fopen(name, "r")) != NULL) {
int status;
status = verifyFormat();
rewind(tfile);
if ((status != TIO_SUCCESS) && (status != TIO_ERROR))
CloseTape();
return status;
}
return TIO_OPENFAIL;
}
/*++
* OpenTapeForWrite
*
* Create a new SIMH .tap format fiole for write access. Two tape marks are
* written to the file and the file handle is rewound to the beginning of the
* file. If the file already exists, and error (TIO_CREATEFAIL) is returned.
*
* Inputs:
*
* name - name of the file
*
* Outputs:
*
* None
*
* Returns:
*
* TIO_SUCCESS - file successfully created
* TIO_IOERROR - I/O error writing the initial file contents
* TIO_CREATEFAIL - file create failed
*
--*/
int OpenTapeForWrite(
char *name
)
{
/*
* Fail if the file exists
*/
if (access(name, F_OK) == 0)
return TIO_CREATEFAIL;
if ((tfile = fopen(name, "w+")) != NULL) {
uint32 tm = 0;
int status = TIO_SUCCESS;
/*
* Write 2 tape marks
*/
if (fwrite(&tm, sizeof(tm), 2, tfile) != 2)
status = TIO_IOERROR;
if (status == TIO_SUCCESS) {
rewind(tfile);
return TIO_SUCCESS;
}
/*
* Failed to write the 2 tape marks. Try to delete the file before
* returning an error.
*/
CloseTape();
unlink(name);
return TIO_IOERROR;
}
return TIO_CREATEFAIL;
}
/*++
* OpenTapeForAppend
*
* Open an existing SIMH .tap format file for write access, leaving the
* file handle positioned just before the final tape mark. If the file is
* successfully opened, scan the file to determine if it is a valid .tap
* format file and whether there are error records present.
*
* Inputs:
*
* name - name of the file
*
* Outputs:
*
* None
*
* Returns:
*
* TIO_SUCCESS - file successfully opened, format is valid
* TIO_ERROR - file successfully opened, error records present
* TIO_CORRUPT - file successfully opened, format is invalid
* TIO_OPENFAIL - file open failed
*
* Note the file remains open if the return status is TIO_SUCCESS or
* TIO_ERROR
*
--*/
int OpenTapeForAppend(
char *name
)
{
if ((tfile =fopen(name, "r+")) != NULL) {
int status;
status = verifyFormat();
if ((status != TIO_SUCCESS) && (status != TIO_ERROR))
CloseTape();
return status;
}
return TIO_OPENFAIL;
}
/*++
* CloseTape
*
* If the tape is open, close it.
*
* Inputs:
*
* None
*
* Outputs:
*
* None
*
* Returns:
*
* None
*
--*/
void CloseTape(void)
{
if (tfile != NULL)
fclose(tfile);
tfile = NULL;
}
/*++
* verifyFormat
*
* Verify the format of the SIMH .tap file. If the format is valid, leave
* the file handle positioned right before the last tape mark in the file.
*
* Inputs:
*
* None
*
* Outputs:
*
* None
*
* Returns:
*
* TIO_SUCCESS - file format is correct
* TIO_ERROR - file format is correct, error records detected
* TIO_CORRUPT - file format is invalid
* TIO_IOERROR - I/O errror while processing file
*
--*/
static int verifyFormat(void)
{
int errorCount = 0, tmSeen = 0;
uint8 meta[4];
uint32 header, bc;
off_t position;
struct stat stat;
/*
* Determine the size of the file.
*/
fstat(fileno(tfile), &stat);
for (;;) {
position = ftello(tfile);
/*
* If we are position at the end of file, there is a tape mark missing.
* Treat it as though there is one present.
*/
if (position == stat.st_size)
return TIO_SUCCESS;
if (fread(meta, sizeof(meta), 1, tfile) != 1)
return TIO_CORRUPT;
bc = (((unsigned int)meta[3]) << 24) |
(((unsigned int)meta[2]) << 16) |
(((unsigned int)meta[1]) << 8) |
(unsigned int)meta[0];
switch (bc) {
case ST_TM:
if (++tmSeen <= 1)
break;
/* Treat second TM in a row as end of medium */
/* FALLTHROUGH */
case ST_EOM:
if (fseek(tfile, -sizeof(meta), SEEK_CUR) != 0)
return TIO_IOERROR;
return errorCount ? TIO_ERROR : TIO_SUCCESS;
case ST_GAP:
break;
default:
/*
* Record descriptor
*/
tmSeen = 0;
header = bc;
if ((bc & ST_ERROR) != 0)
errorCount++;
if ((bc & ST_MBZ) != 0)
return TIO_CORRUPT;
bc = RECLEN(bc & ST_LENGTH);
/*
* Check if we are seeking outside of the file. If so, this is not
* a .tap container file.
*/
if ((position + bc + (2 * sizeof(meta))) > (unsigned long long)stat.st_size)
return TIO_CORRUPT;
if (fseek(tfile, bc, SEEK_CUR) != 0)
return TIO_CORRUPT;
if (fread(meta, sizeof(meta), 1, tfile) != 1)
return TIO_CORRUPT;
bc = (((unsigned int)meta[3]) << 24) |
(((unsigned int)meta[2]) << 16) |
(((unsigned int)meta[1]) << 8) |
(unsigned int)meta[0];
if (header != bc)
return TIO_CORRUPT;
}
}
}
/*++
* ReadTapeRecord
*
* Read the next record from the tape into the specified buffer. If the
* buffer is smaller than the record, the entire record will be consumed.
*
* Inputs:
*
* buf - pointer to the buffer to receive the data
* len - length of the buffer
*
* Outputs:
*
* None
*
* Returns:
*
* ST_EOM - end of medium detected
* ST_TM - tape mark detected
* Other - record length (including error flag)
* if the buffer is smaller than the record, the
* length returned will be that of the buffer
*
--*/
uint32 ReadTapeRecord(
void *buf,
int len
)
{
long pos = ftell(tfile);
uint8 meta[4];
uint32 bc, erflag, length;
/*
* Note: any I/O errors are treated as "end of medium" detection.
*/
if (fread(meta, sizeof(meta), 1, tfile) != 1)
return ST_EOM;
bc = (((unsigned int)meta[3]) << 24) |
(((unsigned int)meta[2]) << 16) |
(((unsigned int)meta[1]) << 8) |
(unsigned int)meta[0];
switch (bc) {
case ST_EOM:
case ST_TM:
return bc;
default:
erflag = bc & ST_ERROR;
bc &= ST_LENGTH;
length = (uint32)len;
if (bc < length)
length = bc;
if (fread(buf, sizeof(uint8), length, tfile) != length)
return ST_EOM;
/*
* Now position the file after this record.
*/
pos += RECLEN(bc) + (2 * sizeof(meta));
if (fseek(tfile, pos, SEEK_SET) != 0)
return ST_EOM;
return erflag | length;
}
return ST_EOM;
}
/*++
* ReadTapeRecordLength
*
* Get the length of the next record on the tape without actually reading
* the data.
*
* Inputs:
*
* None
*
* Outputs:
*
* None
*
* Returns:
*
* ST_EOM - end of medium detected
* ST_TM - tape mark detected
* Other - record length (including error flag)
* if the buffer is smaller than the record, the
* length returned will be that of the buffer
*
--*/
uint32 ReadTapeRecordLength(void)
{
long pos = ftell(tfile);
uint8 meta[4];
uint32 bc, erflag;
/*
* Note: any I/O errors are treated as "end of medium" detection.
*/
if (fread(meta, sizeof(meta), 1, tfile) != 1)
return ST_EOM;
bc = (((unsigned int)meta[3]) << 24) |
(((unsigned int)meta[2]) << 16) |
(((unsigned int)meta[1]) << 8) |
(unsigned int)meta[0];
switch (bc) {
case ST_EOM:
case ST_TM:
return bc;
default:
erflag = bc & ST_ERROR;
bc &= ST_LENGTH;
/*
* Now position the file after this record.
*/
pos += RECLEN(bc) + (2 * sizeof(meta));
if (fseek(tfile, pos, SEEK_SET) != 0)
return ST_EOM;
return erflag | bc;
}
return ST_EOM;
}
/*++
* WriteTapeRecord
*
* Write a record to the tape at it's current position.
*
* Inputs:
*
* buf - pointer to the buffer to be written
* len - length of the buffer
*
* Outputs:
*
* None
*
* Returns:
*
* 0 if record was written successfully, -1 if write failed
*
--*/
int WriteTapeRecord(
void *buf,
int len
)
{
uint8 meta[4];
int datalen;
meta[0] = len & 0xFF;
meta[1] = (len >> 8) & 0xFF;
meta[2] = (len >> 16) & 0xFF;
meta[3] = (len >> 24) & 0xFF;
datalen = (len + 1) & ~1;
if (fwrite(meta, sizeof(meta), 1, tfile) != 1)
return -1;
if (fwrite(buf, datalen, 1, tfile) != 1)
return -1;
if (fwrite(meta, sizeof(meta), 1, tfile) != 1)
return -1;
return 0;
}
/*++
* WriteTapeMark
*
* Write a tape mark to the tape at it's current position and, optionally,
* backup to before the tape mark.
*
* Inputs:
*
* backup - if 1, reposition the tape to before the tape mark
*
* Outputs:
*
* None
*
* Returns:
*
* 0 if tape mark was written successfully, -1 if write failed
*
--*/
int WriteTapeMark(
int backup
)
{
uint32 tm = 0;
if (fwrite(&tm, sizeof(tm), 1, tfile) != 1)
return -1;
if (backup)
if (fseek(tfile, -sizeof(tm), SEEK_CUR) != 0)
return -1;
return 0;
}
/*++
* initTapeBuffering
*
* Initialize variables for writes to tape for ASCII mode transfers
* (translates LF -> CRLF).
*
* Inputs:
*
* reclen - size of the tape record buffer to use
*
* Outputs:
*
* None
*
* Returns:
*
* None
*
--*/
void initTapeBuffering(
int reclen
)
{
rLength = reclen;
occupied = 0;
}
/*++
* flushTapeBuffering
*
* Flush any pending data out to the current tape.
*
* Inputs:
*
* None
*
* Outputs:
*
* None
*
* Returns:
*
* 0 if data was successfully flushed, -1 if write failed
*
--*/
int flushTapeBuffering(void)
{
uint32 count = occupied;
occupied = 0;
if (count != 0)
return WriteTapeRecord(buffer, count);
return 0;
}
/*++
* writeTapeBuffering
*
* Write a character to the current tape, buffering the data into records.
*
* Inputs:
*
* ch - the character to be output
*
* Outputs:
*
* None
*
* Returns:
*
* 0 if character was successfully buffered or written to tape, -1 if
* write failed
*
--*/
int writeTapeBuffering(
char ch
)
{
buffer[occupied++] = ch;
if (occupied == rLength) {
occupied = 0;
return WriteTapeRecord(buffer, rLength);
}
return 0;
}