Compare commits
No commits in common. "ed3ed7bd03afde78b6e5e6b865a555d4ae55f8fa" and "b667561da9687da684fc26629ef37167f1933d10" have entirely different histories.
ed3ed7bd03
...
b667561da9
@ -5,7 +5,3 @@ add_executable(pat80-emulator main.c)
|
|||||||
|
|
||||||
find_package(Z80 REQUIRED)
|
find_package(Z80 REQUIRED)
|
||||||
target_link_libraries(pat80-emulator Z80)
|
target_link_libraries(pat80-emulator Z80)
|
||||||
|
|
||||||
find_package(Curses REQUIRED)
|
|
||||||
include_directories(${CURSES_INCLUDE_DIR})
|
|
||||||
target_link_libraries(pat80-emulator ${CURSES_LIBRARIES})
|
|
||||||
|
@ -1,15 +1,12 @@
|
|||||||
/**
|
|
||||||
* PAT80 Emulator
|
|
||||||
*
|
|
||||||
* Emulates a PAT80.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <Z/constants/pointer.h> /* Z_NULL */
|
#include <Z/constants/pointer.h> /* Z_NULL */
|
||||||
#include <Z80.h>
|
#include <Z80.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <ncurses.h>
|
|
||||||
|
|
||||||
|
#define CYCLES_PER_FRAME 69888
|
||||||
|
#define CYCLES_AT_INT 24
|
||||||
|
#define CYCLES_PER_INT 32
|
||||||
#define ROM_SIZE 0x8000 /* 32 KiB */
|
#define ROM_SIZE 0x8000 /* 32 KiB */
|
||||||
#define MEMORY_SIZE 0xFFFF /* 64 KiB */
|
#define MEMORY_SIZE 0xFFFF /* 64 KiB */
|
||||||
|
|
||||||
@ -23,6 +20,8 @@ typedef struct {
|
|||||||
zusize cycles;
|
zusize cycles;
|
||||||
zuint8 memory[65536];
|
zuint8 memory[65536];
|
||||||
Z80 cpu;
|
Z80 cpu;
|
||||||
|
Device* devices;
|
||||||
|
zusize device_count;
|
||||||
} Machine;
|
} Machine;
|
||||||
|
|
||||||
|
|
||||||
@ -46,44 +45,45 @@ static zuint8 machine_cpu_in(Machine *self, zuint16 port) {
|
|||||||
int decoded = port & bitmask;
|
int decoded = port & bitmask;
|
||||||
if (decoded <= 0x1F) {
|
if (decoded <= 0x1F) {
|
||||||
// Port 0 (0x00 to 0x1F): terminal
|
// Port 0 (0x00 to 0x1F): terminal
|
||||||
// Read char from stin
|
//printf("TERMINAL_READ");
|
||||||
return getch();
|
//return 'H';
|
||||||
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x3F) {
|
if (decoded <= 0x3F) {
|
||||||
// Port 1 (0x20 to 0x3F): sound card (sn76489)
|
// Port 1 (0x20 to 0x3F): sound card (sn76489)
|
||||||
printw("\nsound_cmd[IN]: Not supported!\n");
|
printf("\nsound_cmd[IN]: Not supported!\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x5F) {
|
if (decoded <= 0x5F) {
|
||||||
// Port 2 (0x40 to 0x5F)
|
// Port 2 (0x40 to 0x5F)
|
||||||
printw("IO_ERROR_IN: No device at port 2\n");
|
printf("IO_ERROR_IN: No device at port 2\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x7F) {
|
if (decoded <= 0x7F) {
|
||||||
// Port 3 (0x60 to 0x7F)
|
// Port 3 (0x60 to 0x7F)
|
||||||
printw("IO_ERROR_IN: No device at port 3\n");
|
printf("IO_ERROR_IN: No device at port 3\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x9F) {
|
if (decoded <= 0x9F) {
|
||||||
// Port 4 (0x80 to 0x9F)
|
// Port 4 (0x80 to 0x9F)
|
||||||
printw("IO_ERROR_IN: No device at port 4\n");
|
printf("IO_ERROR_IN: No device at port 4\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x5F) {
|
if (decoded <= 0x5F) {
|
||||||
// Port 5 (0xA0 to 0xBF)
|
// Port 5 (0xA0 to 0xBF)
|
||||||
printw("IO_ERROR_IN: No device at port 5\n");
|
printf("IO_ERROR_IN: No device at port 5\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x5F) {
|
if (decoded <= 0x5F) {
|
||||||
// Port 6 (0xC0 to 0xDF)
|
// Port 6 (0xC0 to 0xDF)
|
||||||
printw("IO_ERROR_IN: No device at port 6\n");
|
printf("IO_ERROR_IN: No device at port 6\n");
|
||||||
return 0x00;
|
return 0x00;
|
||||||
}
|
}
|
||||||
if (decoded <= 0x5F) {
|
if (decoded <= 0x5F) {
|
||||||
// Port 7 (0xE0 to 0xFF)
|
// Port 7 (0xE0 to 0xFF)
|
||||||
printw("IO_ERROR_IN: No device at port 7\n");
|
printf("IO_ERROR_IN: No device at port 7\n");
|
||||||
} else {
|
} else {
|
||||||
printw("IO_ERROR_IN: Invalid port address: %#04x\n", port);
|
printf("IO_ERROR_IN: Invalid port address: %#04x\n", port);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,32 +97,30 @@ static void machine_cpu_out(Machine *self, zuint16 port, zuint8 value) {
|
|||||||
int decoded = port & bitmask;
|
int decoded = port & bitmask;
|
||||||
if (decoded <= 0x1F) {
|
if (decoded <= 0x1F) {
|
||||||
// Port 0 (0x00 to 0x1F): terminal
|
// Port 0 (0x00 to 0x1F): terminal
|
||||||
attron(A_BOLD);
|
printf("%c", value);
|
||||||
printw("%c", value);
|
|
||||||
attroff(A_BOLD);
|
|
||||||
} else if (decoded <= 0x3F) {
|
} else if (decoded <= 0x3F) {
|
||||||
// Port 1 (0x20 to 0x3F): sound card (sn76489)
|
// Port 1 (0x20 to 0x3F): sound card (sn76489)
|
||||||
printw("\nsound_cmd[%#04x]\n", value);
|
printf("\nsound_cmd[%#04x]\n", value);
|
||||||
} else if (decoded <= 0x5F) {
|
} else if (decoded <= 0x5F) {
|
||||||
// Port 2 (0x40 to 0x5F)
|
// Port 2 (0x40 to 0x5F)
|
||||||
printw("IO_ERROR_OUT: No device at port 2\n");
|
printf("IO_ERROR_OUT: No device at port 2\n");
|
||||||
} else if (decoded <= 0x7F) {
|
} else if (decoded <= 0x7F) {
|
||||||
// Port 3 (0x60 to 0x7F)
|
// Port 3 (0x60 to 0x7F)
|
||||||
printw("IO_ERROR_OUT: No device at port 3\n");
|
printf("IO_ERROR_OUT: No device at port 3\n");
|
||||||
} else if (decoded <= 0x9F) {
|
} else if (decoded <= 0x9F) {
|
||||||
// Port 4 (0x80 to 0x9F)
|
// Port 4 (0x80 to 0x9F)
|
||||||
printw("IO_ERROR_OUT: No device at port 4\n");
|
printf("IO_ERROR_OUT: No device at port 4\n");
|
||||||
} else if (decoded <= 0x5F) {
|
} else if (decoded <= 0x5F) {
|
||||||
// Port 5 (0xA0 to 0xBF)
|
// Port 5 (0xA0 to 0xBF)
|
||||||
printw("IO_ERROR_OUT: No device at port 5\n");
|
printf("IO_ERROR_OUT: No device at port 5\n");
|
||||||
} else if (decoded <= 0x5F) {
|
} else if (decoded <= 0x5F) {
|
||||||
// Port 6 (0xC0 to 0xDF)
|
// Port 6 (0xC0 to 0xDF)
|
||||||
printw("IO_ERROR_OUT: No device at port 6\n");
|
printf("IO_ERROR_OUT: No device at port 6\n");
|
||||||
} else if (decoded <= 0x5F) {
|
} else if (decoded <= 0x5F) {
|
||||||
// Port 7 (0xE0 to 0xFF)
|
// Port 7 (0xE0 to 0xFF)
|
||||||
printw("IO_ERROR_OUT: No device at port 7\n");
|
printf("IO_ERROR_OUT: No device at port 7\n");
|
||||||
} else {
|
} else {
|
||||||
printw("IO_ERROR_OUT: Invalid port address: %#04x\n", port);
|
printf("IO_ERROR_OUT: Invalid port address: %#04x\n", port);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -147,6 +145,8 @@ void machine_initialize(Machine *self) {
|
|||||||
self->cpu.hook = Z_NULL;
|
self->cpu.hook = Z_NULL;
|
||||||
self->cpu.illegal = Z_NULL;
|
self->cpu.illegal = Z_NULL;
|
||||||
self->cpu.options = Z80_MODEL_ZILOG_NMOS;
|
self->cpu.options = Z80_MODEL_ZILOG_NMOS;
|
||||||
|
|
||||||
|
/* Create and initialize devices... */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -169,6 +169,47 @@ void machine_run(Machine *self) {
|
|||||||
z80_run(&self->cpu, Z80_MAXIMUM_CYCLES);
|
z80_run(&self->cpu, Z80_MAXIMUM_CYCLES);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// void machine_run_frame(Machine *self)
|
||||||
|
// {
|
||||||
|
// /* CPU cycles before the INT signal */
|
||||||
|
// self->cycles += z80_execute(&self->cpu, CYCLES_AT_INT - self->cycles);
|
||||||
|
|
||||||
|
// /* CPU cycles during the INT signal */
|
||||||
|
// z80_int(&self->cpu, Z_TRUE);
|
||||||
|
// self->cycles += z80_run(&self->cpu, (CYCLES_AT_INT + CYCLES_PER_INT) - self->cycles);
|
||||||
|
// z80_int(&self->cpu, Z_FALSE);
|
||||||
|
|
||||||
|
// /* CPU cycles after the INT signal */
|
||||||
|
// self->cycles += z80_execute(&self->cpu, CYCLES_PER_FRAME - self->cycles);
|
||||||
|
|
||||||
|
// self->cycles -= CYCLES_PER_FRAME;
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// TEST IMPLEM DEVICE
|
||||||
|
|
||||||
|
zuint8 device_terminal_read(void *context) {
|
||||||
|
return 'H';
|
||||||
|
//return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void device_terminal_write(void *context, zuint8 value) {
|
||||||
|
printf("%c", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
zuint8 device_sound_read(void *context) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void device_sound_write(void *context, zuint8 value) {
|
||||||
|
printf("sound[%c]\n", value);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TEST IMPLEM DEVICE
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
// Parse arguments
|
// Parse arguments
|
||||||
if (argc < 2) {
|
if (argc < 2) {
|
||||||
@ -180,7 +221,7 @@ int main(int argc, char *argv[]) {
|
|||||||
// Setup virtual Pat80 computer
|
// Setup virtual Pat80 computer
|
||||||
Z80 pat80Cpu = {};
|
Z80 pat80Cpu = {};
|
||||||
Machine pat80 = {
|
Machine pat80 = {
|
||||||
/*zusize*/ .cycles = 0,
|
/*zusize*/ .cycles = 100000,
|
||||||
/*Z80*/ .cpu = pat80Cpu
|
/*Z80*/ .cpu = pat80Cpu
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -197,18 +238,23 @@ int main(int argc, char *argv[]) {
|
|||||||
fread(&pat80.memory,ROM_SIZE,1,romFile); // load rom from file into memory, up >
|
fread(&pat80.memory,ROM_SIZE,1,romFile); // load rom from file into memory, up >
|
||||||
fclose(romFile);
|
fclose(romFile);
|
||||||
|
|
||||||
// Init ncurses
|
// Declare system devices
|
||||||
initscr(); // Start curses mode
|
Device patoTerminal = {
|
||||||
raw(); // Line buffering disabled (get character without waiting for ENTER key)
|
.read = *device_terminal_read,
|
||||||
keypad(stdscr, TRUE); // We get F1, F2 etc..
|
.write = *device_terminal_write,
|
||||||
noecho(); // Don't echo() while we do getch
|
};
|
||||||
scrollok(stdscr, TRUE); // Allow scrolling when reached end of window
|
Device patoSound = {
|
||||||
|
.read = *device_sound_read,
|
||||||
|
.write = *device_sound_write,
|
||||||
|
};
|
||||||
|
|
||||||
|
Device patoDevices[2] = {
|
||||||
|
patoTerminal, // Port 0 (0x00 to 0x1F)
|
||||||
|
patoSound // Port 1 (0x20 to 0x3F)
|
||||||
|
};
|
||||||
|
pat80.devices = patoDevices;
|
||||||
|
pat80.device_count = 2;
|
||||||
|
|
||||||
// Start emulated computer
|
|
||||||
machine_reset(&pat80);
|
machine_reset(&pat80);
|
||||||
machine_run(&pat80);
|
machine_run(&pat80);
|
||||||
|
|
||||||
// Stop ncurses
|
|
||||||
endwin(); /* End curses mode */
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user