2010-10-12 14:29:11 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#ifndef _WIN32
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#include "perfect6502.h"
|
|
|
|
|
|
|
|
/************************************************************
|
|
|
|
*
|
|
|
|
* Interface to OS Library Code / Monitor
|
|
|
|
*
|
|
|
|
************************************************************/
|
|
|
|
|
|
|
|
/* imported by runtime.c */
|
|
|
|
unsigned char A, X, Y, S, P;
|
|
|
|
unsigned short PC;
|
|
|
|
int N, Z, C;
|
|
|
|
|
|
|
|
void
|
|
|
|
init_monitor()
|
|
|
|
{
|
|
|
|
FILE *f;
|
|
|
|
f = fopen("apple1basic.bin", "r");
|
|
|
|
fread(memory + 0xE000, 1, 4096, f);
|
|
|
|
fclose(f);
|
|
|
|
|
|
|
|
memory[0xfffc] = 0x00;
|
|
|
|
memory[0xfffd] = 0xE0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
charout(char ch) {
|
2010-10-20 07:49:51 +00:00
|
|
|
unsigned char S = readSP();
|
|
|
|
unsigned short a = 1 + memory[0x0100+S+1] | memory[0x0100+((S+2) & 0xFF)] << 8;
|
2010-10-12 14:29:11 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Apple I BASIC prints every character received
|
|
|
|
* from the terminal. UNIX terminals do this
|
|
|
|
* anyway, so we have to avoid printing every
|
|
|
|
* line again
|
|
|
|
*/
|
|
|
|
if (a==0xe2a6) /* character echo */
|
|
|
|
return;
|
|
|
|
if (a==0xe2b6) /* CR echo */
|
|
|
|
return;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Apple I BASIC prints a line break and 6 spaces
|
|
|
|
* after every 37 characters. UNIX terminals do
|
|
|
|
* line breaks themselves, so ignore these
|
|
|
|
* characters
|
|
|
|
*/
|
|
|
|
if (a==0xe025 && (ch==10 || ch==' '))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* INPUT */
|
|
|
|
if (a==0xe182) {
|
|
|
|
#if _WIN32
|
|
|
|
if (!isatty(0))
|
|
|
|
return;
|
|
|
|
#else
|
|
|
|
struct stat st;
|
|
|
|
fstat(0, &st);
|
|
|
|
if (S_ISFIFO (st.st_mode))
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
putc(ch, stdout);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
handle_monitor()
|
|
|
|
{
|
|
|
|
if (readRW()) {
|
|
|
|
unsigned short a = readAddressBus();
|
|
|
|
if ((a & 0xFF1F) == 0xD010) {
|
|
|
|
unsigned char c = getchar();
|
|
|
|
if (c == 10)
|
|
|
|
c = 13;
|
|
|
|
c |= 0x80;
|
|
|
|
writeDataBus(c);
|
|
|
|
}
|
|
|
|
if ((a & 0xFF1F) == 0xD011) {
|
|
|
|
if (readPC() == 0xE006)
|
|
|
|
/* if the code is reading a character, we have one ready */
|
|
|
|
writeDataBus(0x80);
|
|
|
|
else
|
|
|
|
/* if the code checks for a STOP condition, nothing is pressed */
|
|
|
|
writeDataBus(0);
|
|
|
|
}
|
|
|
|
if ((a & 0xFF1F) == 0xD012) {
|
|
|
|
/* 0x80 would mean we're not yet ready to receive a character */
|
|
|
|
writeDataBus(0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
unsigned short a = readAddressBus();
|
|
|
|
unsigned char d = readDataBus();
|
|
|
|
if ((a & 0xFF1F) == 0xD012) {
|
|
|
|
unsigned char temp8 = d & 0x7F;
|
|
|
|
if (temp8 == 13)
|
|
|
|
temp8 = 10;
|
|
|
|
charout(temp8);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main()
|
|
|
|
{
|
|
|
|
int clk = 0;
|
|
|
|
|
|
|
|
initAndResetChip();
|
|
|
|
|
|
|
|
/* set up memory for user program */
|
|
|
|
init_monitor();
|
|
|
|
|
|
|
|
/* emulate the 6502! */
|
|
|
|
for (;;) {
|
|
|
|
step();
|
|
|
|
clk = !clk;
|
|
|
|
if (!clk)
|
|
|
|
handle_monitor();
|
|
|
|
|
2010-10-20 07:49:51 +00:00
|
|
|
chipStatus();
|
2010-10-12 14:29:11 +00:00
|
|
|
//if (!(cycle % 1000)) printf("%d\n", cycle);
|
|
|
|
};
|
|
|
|
}
|