538 lines
13 KiB
C
538 lines
13 KiB
C
#ifndef ABSTRACT_H
|
|
#define ABSTRACT_H
|
|
|
|
#include <glob.h>
|
|
#include <ctype.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdbool.h>
|
|
#include <sys/stat.h>
|
|
#include <poll.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
#include <libgen.h>
|
|
#include <string.h>
|
|
#ifdef PROFILE
|
|
#include <time.h>
|
|
#define millis() clock()/1000
|
|
#endif
|
|
|
|
// Lua scripting support
|
|
#ifdef HASLUA
|
|
#include "lua/lua.h"
|
|
#include "lua/lualib.h"
|
|
#include "lua/lauxlib.h"
|
|
|
|
#include "lua.h"
|
|
lua_State* L;
|
|
#endif
|
|
|
|
#define HostOS 0x02
|
|
|
|
/* Externals for abstracted functions need to go here */
|
|
FILE* _sys_fopen_r(uint8* filename);
|
|
int _sys_fseek(FILE* file, long delta, int origin);
|
|
long _sys_ftell(FILE* file);
|
|
long _sys_fread(void* buffer, long size, long count, FILE* file);
|
|
int _sys_fflush(FILE* file);
|
|
int _sys_fclose(FILE* file);
|
|
|
|
/* Memory abstraction functions */
|
|
/*===============================================================================*/
|
|
void _RamLoad(uint8* filename, uint16 address) {
|
|
long l;
|
|
FILE* file = _sys_fopen_r(filename);
|
|
_sys_fseek(file, 0, SEEK_END);
|
|
l = _sys_ftell(file);
|
|
|
|
_sys_fseek(file, 0, SEEK_SET);
|
|
_sys_fread(_RamSysAddr(address), 1, l, file); // (todo) This can overwrite past RAM space
|
|
|
|
_sys_fclose(file);
|
|
}
|
|
|
|
/* Filesystem (disk) abstraction fuctions */
|
|
/*===============================================================================*/
|
|
#define FOLDERCHAR '/'
|
|
|
|
typedef struct {
|
|
uint8 dr;
|
|
uint8 fn[8];
|
|
uint8 tp[3];
|
|
uint8 ex, s1, s2, rc;
|
|
uint8 al[16];
|
|
uint8 cr, r0, r1, r2;
|
|
} CPM_FCB;
|
|
|
|
typedef struct {
|
|
uint8 dr;
|
|
uint8 fn[8];
|
|
uint8 tp[3];
|
|
uint8 ex, s1, s2, rc;
|
|
uint8 al[16];
|
|
} CPM_DIRENTRY;
|
|
|
|
uint8 _sys_exists(uint8* filename) {
|
|
return(!access((const char*)filename, F_OK));
|
|
}
|
|
|
|
FILE* _sys_fopen_r(uint8* filename) {
|
|
return(fopen((const char*)filename, "rb"));
|
|
}
|
|
|
|
FILE* _sys_fopen_w(uint8* filename) {
|
|
return(fopen((const char*)filename, "wb"));
|
|
}
|
|
|
|
FILE* _sys_fopen_rw(uint8* filename) {
|
|
return(fopen((const char*)filename, "r+b"));
|
|
}
|
|
|
|
FILE* _sys_fopen_a(uint8* filename) {
|
|
return(fopen((const char*)filename, "a"));
|
|
}
|
|
|
|
int _sys_fseek(FILE* file, long delta, int origin) {
|
|
return(fseek(file, delta, origin));
|
|
}
|
|
|
|
long _sys_ftell(FILE* file) {
|
|
return(ftell(file));
|
|
}
|
|
|
|
long _sys_fread(void* buffer, long size, long count, FILE* file) {
|
|
return(fread(buffer, size, count, file));
|
|
}
|
|
|
|
long _sys_fwrite(const void* buffer, long size, long count, FILE* file) {
|
|
return(fwrite(buffer, size, count, file));
|
|
}
|
|
|
|
int _sys_fputc(int ch, FILE* file) {
|
|
return(fputc(ch, file));
|
|
}
|
|
|
|
int _sys_feof(FILE* file) {
|
|
return(feof(file));
|
|
}
|
|
|
|
int _sys_fflush(FILE* file) {
|
|
return(fflush(file));
|
|
}
|
|
|
|
int _sys_fclose(FILE* file) {
|
|
return(fclose(file));
|
|
}
|
|
|
|
int _sys_remove(uint8* filename) {
|
|
return(remove((const char*)filename));
|
|
}
|
|
|
|
int _sys_rename(uint8* name1, uint8* name2) {
|
|
return(rename((const char*)name1, (const char*)name2));
|
|
}
|
|
|
|
int _sys_select(uint8* disk) {
|
|
struct stat st;
|
|
return((stat((char*)disk, &st) == 0) && ((st.st_mode & S_IFDIR) != 0));
|
|
}
|
|
|
|
long _sys_filesize(uint8* filename) {
|
|
long l = -1;
|
|
FILE* file = _sys_fopen_r(filename);
|
|
if (file != NULL) {
|
|
_sys_fseek(file, 0, SEEK_END);
|
|
l = _sys_ftell(file);
|
|
_sys_fclose(file);
|
|
}
|
|
return(l);
|
|
}
|
|
|
|
int _sys_openfile(uint8* filename) {
|
|
FILE* file = _sys_fopen_r(filename);
|
|
if (file != NULL)
|
|
_sys_fclose(file);
|
|
return(file != NULL);
|
|
}
|
|
|
|
int _sys_makefile(uint8* filename) {
|
|
FILE* file = _sys_fopen_a(filename);
|
|
if (file != NULL)
|
|
_sys_fclose(file);
|
|
return(file != NULL);
|
|
}
|
|
|
|
int _sys_deletefile(uint8* filename) {
|
|
return(!_sys_remove(filename));
|
|
}
|
|
|
|
int _sys_renamefile(uint8* filename, uint8* newname) {
|
|
return(!_sys_rename(&filename[0], &newname[0]));
|
|
}
|
|
|
|
#ifdef DEBUGLOG
|
|
void _sys_logbuffer(uint8* buffer) {
|
|
FILE* file;
|
|
#ifdef CONSOLELOG
|
|
puts((char*)buffer);
|
|
#else
|
|
uint8 s = 0;
|
|
while (*(buffer + s)) // Computes buffer size
|
|
++s;
|
|
file = _sys_fopen_a((uint8*)LogName);
|
|
_sys_fwrite(buffer, 1, s, file);
|
|
_sys_fclose(file);
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
uint8 _sys_readseq(uint8* filename, long fpos) {
|
|
uint8 result = 0xff;
|
|
uint8 bytesread;
|
|
uint8 dmabuf[128];
|
|
uint8 i;
|
|
|
|
FILE* file = _sys_fopen_r(&filename[0]);
|
|
if (file != NULL) {
|
|
if (!_sys_fseek(file, fpos, 0)) {
|
|
for (i = 0; i < 128; ++i)
|
|
dmabuf[i] = 0x1a;
|
|
bytesread = (uint8)_sys_fread(&dmabuf[0], 1, 128, file);
|
|
if (bytesread) {
|
|
for (i = 0; i < 128; ++i)
|
|
_RamWrite(dmaAddr + i, dmabuf[i]);
|
|
}
|
|
result = bytesread ? 0x00 : 0x01;
|
|
} else {
|
|
result = 0x01;
|
|
}
|
|
_sys_fclose(file);
|
|
} else {
|
|
result = 0x10;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
uint8 _sys_writeseq(uint8* filename, long fpos) {
|
|
uint8 result = 0xff;
|
|
|
|
FILE* file = _sys_fopen_rw(&filename[0]);
|
|
if (file != NULL) {
|
|
if (!_sys_fseek(file, fpos, 0)) {
|
|
if (_sys_fwrite(_RamSysAddr(dmaAddr), 1, 128, file))
|
|
result = 0x00;
|
|
} else {
|
|
result = 0x01;
|
|
}
|
|
_sys_fclose(file);
|
|
} else {
|
|
result = 0x10;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
uint8 _sys_readrand(uint8* filename, long fpos) {
|
|
uint8 result = 0xff;
|
|
uint8 bytesread;
|
|
uint8 dmabuf[128];
|
|
uint8 i;
|
|
long extSize;
|
|
|
|
FILE* file = _sys_fopen_r(&filename[0]);
|
|
if (file != NULL) {
|
|
if (!_sys_fseek(file, fpos, 0)) {
|
|
for (i = 0; i < 128; ++i)
|
|
dmabuf[i] = 0x1a;
|
|
bytesread = (uint8)_sys_fread(&dmabuf[0], 1, 128, file);
|
|
if (bytesread) {
|
|
for (i = 0; i < 128; ++i)
|
|
_RamWrite(dmaAddr + i, dmabuf[i]);
|
|
}
|
|
result = bytesread ? 0x00 : 0x01;
|
|
} else {
|
|
if (fpos >= 65536L * 128) {
|
|
result = 0x06; // seek past 8MB (largest file size in CP/M)
|
|
} else {
|
|
_sys_fseek(file, 0, SEEK_END);
|
|
extSize = _sys_ftell(file);
|
|
// round file size up to next full logical extent
|
|
extSize = 16384 * ((extSize / 16384) + ((extSize % 16384) ? 1 : 0));
|
|
if (fpos < extSize)
|
|
result = 0x01; // reading unwritten data
|
|
else
|
|
result = 0x04; // seek to unwritten extent
|
|
}
|
|
}
|
|
_sys_fclose(file);
|
|
} else {
|
|
result = 0x10;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
uint8 _sys_writerand(uint8* filename, long fpos) {
|
|
uint8 result = 0xff;
|
|
|
|
FILE* file = _sys_fopen_rw(&filename[0]);
|
|
if (file != NULL) {
|
|
if (!_sys_fseek(file, fpos, 0)) {
|
|
if (_sys_fwrite(_RamSysAddr(dmaAddr), 1, 128, file))
|
|
result = 0x00;
|
|
} else {
|
|
result = 0x06;
|
|
}
|
|
_sys_fclose(file);
|
|
} else {
|
|
result = 0x10;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
uint8 _Truncate(char* fn, uint8 rc) {
|
|
uint8 result = 0x00;
|
|
if (truncate(fn, rc * 128))
|
|
result = 0xff;
|
|
return(result);
|
|
}
|
|
|
|
void _MakeUserDir() {
|
|
uint8 dFolder = cDrive + 'A';
|
|
uint8 uFolder = toupper(tohex(userCode));
|
|
|
|
uint8 path[4] = { dFolder, FOLDERCHAR, uFolder, 0 };
|
|
|
|
mkdir((char*)path, S_IRUSR | S_IWUSR | S_IXUSR);
|
|
}
|
|
|
|
uint8 _sys_makedisk(uint8 drive) {
|
|
uint8 result = 0;
|
|
if (drive < 1 || drive>16) {
|
|
result = 0xff;
|
|
} else {
|
|
uint8 dFolder = drive + '@';
|
|
uint8 disk[2] = { dFolder, 0 };
|
|
if (!mkdir((char*)disk, S_IRUSR | S_IWUSR | S_IXUSR)) {
|
|
result = 0xfe;
|
|
} else {
|
|
uint8 path[4] = { dFolder, FOLDERCHAR, '0', 0 };
|
|
mkdir((char*)path, S_IRUSR | S_IWUSR | S_IXUSR);
|
|
}
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
|
|
#ifdef HASLUA
|
|
uint8 _RunLuaScript(char* filename) {
|
|
|
|
L = luaL_newstate();
|
|
luaL_openlibs(L);
|
|
|
|
// Register Lua functions
|
|
lua_register(L, "BdosCall", luaBdosCall);
|
|
lua_register(L, "RamRead", luaRamRead);
|
|
lua_register(L, "RamWrite", luaRamWrite);
|
|
lua_register(L, "RamRead16", luaRamRead16);
|
|
lua_register(L, "RamWrite16", luaRamWrite16);
|
|
lua_register(L, "ReadReg", luaReadReg);
|
|
lua_register(L, "WriteReg", luaWriteReg);
|
|
|
|
int result = luaL_loadfile(L, filename);
|
|
if (result) {
|
|
_puts(lua_tostring(L, -1));
|
|
} else {
|
|
result = lua_pcall(L, 0, LUA_MULTRET, 0);
|
|
if (result)
|
|
_puts(lua_tostring(L, -1));
|
|
}
|
|
|
|
lua_close(L);
|
|
|
|
return(result);
|
|
}
|
|
#endif
|
|
|
|
#ifndef POLLRDBAND
|
|
#define POLLRDBAND 0
|
|
#endif
|
|
#ifndef POLLRDNORM
|
|
#define POLLRDNORM 0
|
|
#endif
|
|
|
|
glob_t pglob;
|
|
int dirPos;
|
|
|
|
static char findNextDirName[17];
|
|
static uint16 fileRecords = 0;
|
|
static uint16 fileExtents = 0;
|
|
static uint16 fileExtentsUsed = 0;
|
|
static uint16 firstFreeAllocBlock;
|
|
|
|
uint8 _findnext(uint8 isdir) {
|
|
uint8 result = 0xff;
|
|
char dir[6] = { '?', FOLDERCHAR, 0, FOLDERCHAR, '*', 0 };
|
|
int i;
|
|
struct stat st;
|
|
uint32 bytes;
|
|
|
|
if (allExtents && fileRecords) {
|
|
// _SearchFirst was called with '?' in the FCB's EX field, so
|
|
// we need to return all file extents.
|
|
// The last found file was large enough that in CP/M it would
|
|
// have another directory entry, so mock up the next entry
|
|
// for the file.
|
|
_mockupDirEntry();
|
|
result = 0;
|
|
} else {
|
|
// Either we're only interested in the first directory entry
|
|
// for each file, or the previously found file has had the
|
|
// required number of dierctory entries returned already.
|
|
dir[0] = filename[0];
|
|
if (allUsers)
|
|
dir[2] = '?';
|
|
else
|
|
dir[2] = filename[2];
|
|
if (!glob(dir, 0, NULL, &pglob)) {
|
|
for (i = dirPos; i < pglob.gl_pathc; ++i) {
|
|
++dirPos;
|
|
strncpy(findNextDirName, pglob.gl_pathv[i], sizeof(findNextDirName) - 1);
|
|
findNextDirName[sizeof(findNextDirName) - 1] = 0;
|
|
_HostnameToFCBname((uint8*)findNextDirName, fcbname);
|
|
if (match(fcbname, pattern) &&
|
|
(stat(findNextDirName, &st) == 0) &&
|
|
((st.st_mode & S_IFREG) != 0) &&
|
|
isxdigit((uint8)findNextDirName[2]) &&
|
|
(isupper((uint8)findNextDirName[2]) || isdigit((uint8)findNextDirName[2]))) {
|
|
if (allUsers)
|
|
currFindUser = isdigit((uint8)findNextDirName[2]) ? findNextDirName[2] - '0' : findNextDirName[2] - 'A' + 10;
|
|
if (isdir) {
|
|
// account for host files that aren't multiples of the block size
|
|
// by rounding their bytes up to the next multiple of blocks
|
|
bytes = st.st_size;
|
|
if (bytes & (BlkSZ - 1))
|
|
bytes = (bytes & ~(BlkSZ - 1)) + BlkSZ;
|
|
// calculate the number of 128 byte records and 16K
|
|
// extents for this file. _mockupDirEntry will use
|
|
// these values to populate the returned directory
|
|
// entry, and decrement the # of records and extents
|
|
// left to process in the file.
|
|
fileRecords = bytes / BlkSZ;
|
|
fileExtents = fileRecords / BlkEX + ((fileRecords & (BlkEX - 1)) ? 1 : 0);
|
|
fileExtentsUsed = 0;
|
|
firstFreeAllocBlock = firstBlockAfterDir;
|
|
_mockupDirEntry();
|
|
} else {
|
|
fileRecords = 0;
|
|
fileExtents = 0;
|
|
fileExtentsUsed = 0;
|
|
firstFreeAllocBlock = firstBlockAfterDir;
|
|
}
|
|
_RamWrite(tmpFCB, filename[0] - '@');
|
|
_HostnameToFCB(tmpFCB, (uint8*)findNextDirName);
|
|
result = 0x00;
|
|
break;
|
|
}
|
|
}
|
|
globfree(&pglob);
|
|
}
|
|
}
|
|
return(result);
|
|
}
|
|
|
|
uint8 _findfirst(uint8 isdir) {
|
|
dirPos = 0; // Set directory search to start from the first position
|
|
_HostnameToFCBname(filename, pattern);
|
|
fileRecords = 0;
|
|
fileExtents = 0;
|
|
fileExtentsUsed = 0;
|
|
return(_findnext(isdir));
|
|
}
|
|
|
|
uint8 _findnextallusers(uint8 isdir) {
|
|
return _findnext(isdir);
|
|
}
|
|
|
|
uint8 _findfirstallusers(uint8 isdir) {
|
|
dirPos = 0;
|
|
strcpy((char*)pattern, "???????????");
|
|
fileRecords = 0;
|
|
fileExtents = 0;
|
|
fileExtentsUsed = 0;
|
|
return(_findnextallusers(isdir));
|
|
}
|
|
|
|
/* Hardware abstraction functions */
|
|
/*===============================================================================*/
|
|
void _HardwareOut(const uint32 Port, const uint32 Value) {
|
|
|
|
}
|
|
|
|
uint32 _HardwareIn(const uint32 Port) {
|
|
return 0;
|
|
}
|
|
|
|
/* Host initialization functions */
|
|
/*===============================================================================*/
|
|
|
|
void _host_init(int argc, char* argv[]) {
|
|
int x = chdir(dirname(argv[0]));
|
|
}
|
|
|
|
/* Console abstraction functions */
|
|
/*===============================================================================*/
|
|
|
|
static struct termios _old_term, _new_term;
|
|
|
|
void _console_init(void) {
|
|
tcgetattr(0, &_old_term);
|
|
|
|
_new_term = _old_term;
|
|
|
|
_new_term.c_lflag &= ~ICANON; /* Input available immediately (no EOL needed) */
|
|
_new_term.c_lflag &= ~ECHO; /* Do not echo input characters */
|
|
_new_term.c_lflag &= ~ISIG; /* ^C and ^Z do not generate signals */
|
|
_new_term.c_iflag &= INLCR; /* Translate NL to CR on input */
|
|
|
|
tcsetattr(0, TCSANOW, &_new_term); /* Apply changes immediately */
|
|
|
|
setvbuf(stdout, (char*)NULL, _IONBF, 0); /* Disable stdout buffering */
|
|
}
|
|
|
|
void _console_reset(void) {
|
|
tcsetattr(0, TCSANOW, &_old_term);
|
|
}
|
|
|
|
int _kbhit(void) {
|
|
struct pollfd pfds[1];
|
|
|
|
pfds[0].fd = STDIN_FILENO;
|
|
pfds[0].events = POLLIN | POLLPRI | POLLRDBAND | POLLRDNORM;
|
|
|
|
return (poll(pfds, 1, 0) == 1) && (pfds[0].revents & (POLLIN | POLLPRI | POLLRDBAND | POLLRDNORM));
|
|
}
|
|
|
|
uint8 _getch(void) {
|
|
return getchar();
|
|
}
|
|
|
|
void _putch(uint8 ch) {
|
|
putchar(ch);
|
|
}
|
|
|
|
uint8 _getche(void) {
|
|
uint8 ch = _getch();
|
|
|
|
_putch(ch);
|
|
|
|
return ch;
|
|
}
|
|
|
|
void _clrscr(void) {
|
|
uint8 ch = system("clear");
|
|
}
|
|
|
|
#endif
|