diff options
author | Ian C <ianc@noddybox.co.uk> | 2020-03-20 21:18:16 +0000 |
---|---|---|
committer | Ian C <ianc@noddybox.co.uk> | 2020-03-20 21:18:16 +0000 |
commit | d9beac69eb3b1c3678a3ac9498adff99cde4a8c6 (patch) | |
tree | be53e8925d193de9bbfbc0c549fff1592eabe316 | |
parent | 5106ea51907aa9f6a15070041fefd820671cc4ed (diff) |
Basic keys added and now being read.
-rw-r--r-- | src/czx81.c | 517 |
1 files changed, 387 insertions, 130 deletions
diff --git a/src/czx81.c b/src/czx81.c index 1e50c5c..6d2f60f 100644 --- a/src/czx81.c +++ b/src/czx81.c @@ -66,6 +66,8 @@ /* -------------------------------------------------- GLOBALS */ +static int quit = FALSE; + static Z80Val FRAME_TSTATES = FAST_TSTATES; static Z80Byte mem[0x10000]; @@ -103,68 +105,130 @@ static wchar_t scrcode[0x40] = L'W', L'X', L'Y', L'Z', /* 3c - 3f */ }; +static int key_press_frames = 1; +static int current_key_press_frame = 0; +static int current_key = -1; -/* -------------------------------------------------- ZX81 CODE -*/ -static void RomPatch(const Z80Byte patch[], Z80Word addr) +static int trace = FALSE; + +static struct { - int f; + int key; /* The curses keycode */ + int no; /* The number of keys to press */ + int row[5]; /* The rows for each keypress */ + int bit[5]; /* The bit number for each keypress */ +} keymap[] = +{ + {'1', 1, {3}, {0x01}}, + {'2', 1, {3}, {0x02}}, + {'3', 1, {3}, {0x04}}, + {'4', 1, {3}, {0x08}}, + {'5', 1, {3}, {0x10}}, + + {'6', 1, {4}, {0x10}}, + {'7', 1, {4}, {0x08}}, + {'8', 1, {4}, {0x04}}, + {'9', 1, {4}, {0x02}}, + {'0', 1, {4}, {0x01}}, + + {'q', 1, {2}, {0x01}}, + {'w', 1, {2}, {0x02}}, + {'e', 1, {2}, {0x04}}, + {'r', 1, {2}, {0x08}}, + {'t', 1, {2}, {0x10}}, + + {'y', 1, {5}, {0x10}}, + {'u', 1, {5}, {0x08}}, + {'i', 1, {5}, {0x04}}, + {'o', 1, {5}, {0x02}}, + {'p', 1, {5}, {0x01}}, + + {'a', 1, {1}, {0x01}}, + {'s', 1, {1}, {0x02}}, + {'d', 1, {1}, {0x04}}, + {'f', 1, {1}, {0x08}}, + {'g', 1, {1}, {0x10}}, + + {'h', 1, {6}, {0x10}}, + {'j', 1, {6}, {0x08}}, + {'k', 1, {6}, {0x04}}, + {'l', 1, {6}, {0x02}}, + {'\n', 1, {6}, {0x01}}, + + /* Shift key skipped */ + /* {'', 1, {0}, {0x01}}, */ + {'z', 1, {0}, {0x02}}, + {'x', 1, {0}, {0x04}}, + {'c', 1, {0}, {0x08}}, + {'v', 1, {0}, {0x10}}, + + {'b', 1, {6}, {0x10}}, + {'n', 1, {6}, {0x08}}, + {'m', 1, {6}, {0x04}}, + {'.', 1, {6}, {0x02}}, + {' ', 1, {6}, {0x01}}, + + {ERR, 0, {0}, {0}} +}; - for(f = 0; patch[f] != 0xff; f++) - { - mem[addr++] = patch[f]; - } -} +static char message[1024]; -static int EDCallback(Z80 *z80, Z80Val data) +/* -------------------------------------------------- UTILS +*/ +static long TimeDiff(const struct timespec *start, + const struct timespec *end) { - Z80Word pause; + long diff; - switch((Z80Byte)data) - { - case ED_SAVE: - if (z80->DE.w < 0x8000) - { - /* TODO: Save */ - } + diff = (end->tv_nsec - start->tv_nsec) + + (end->tv_sec - start->tv_sec) * 1000000000L; - break; + return diff; +} - case ED_LOAD: - if (z80->DE.w < 0x8000) - { - /* TODO: Load */ - } - mem[CDFLAG] = 0xc0; +static void ClearScreen(void) +{ + clear(); +} - break; - case ED_WAITKEY: - waitkey = TRUE; - started = TRUE; - break; +static void RenderScreen(void) +{ + static int message_delay = 0; - case ED_ENDWAITKEY: - waitkey = FALSE; - break; + mvprintw(0, 40, "Message:"); - case ED_PAUSE: - waitkey = TRUE; + if (message[0] && message_delay == 0) + { + message_delay = 250; + } - pause = z80->BC.w; + if (message_delay > 0) + { + mvprintw(1, 40, message); + + if (--message_delay == 0) + { + message[0] = 0; + } + } - while(pause-- && !(mem[CDFLAG]&1)) - { - /* TODO: Pause code */ - } + refresh(); +} - waitkey = FALSE; - break; - } - return TRUE; +/* -------------------------------------------------- ZX81 CODE +*/ +static void RomPatch(const Z80Byte patch[], Z80Word addr) +{ + int f; + + for(f = 0; patch[f] != 0xff; f++) + { + mem[addr++] = patch[f]; + } } @@ -239,16 +303,108 @@ static void ZX81HouseKeeping(void) mem[CDFLAG] &= ~1; } - mem[LASTK1] = lastk1^0xff; - mem[LASTK2] = lastk2^0xff; + mem[LASTK1] = lastk1 ^ 0xff; + mem[LASTK2] = lastk2 ^ 0xff; prev_lk1 = lastk1; prev_lk2 = lastk2; } +static void DrawScreen(void) +{ + int fast; + int x,y; + + fast = (FRAME_TSTATES == FAST_TSTATES); + + if (fast) + { + for(y = 0; y < TXT_H; y++) + { + for(x = 0; x < TXT_W; x++) + { + mvprintw(y, x, "%lc", L'\u2592'); + } + } + } + else + { + Z80Byte *scr; + + scr = mem + PEEKW(DFILE); + y = 0; + + while(y < TXT_H) + { + scr++; + x = 0; + + while(*scr != 118 && x < TXT_W) + { + Z80Byte ch = *scr++; + + if (ch & 0x80) + { + attron(A_REVERSE); + } + + mvprintw(y, x, "%lc", scrcode[ch & 0x3f]); + + if (ch & 0x80) + { + attroff(A_REVERSE); + } + + x++; + } + + y++; + } + } +} + +static void Trace(void) +{ + Z80Word addr = z80->PC; + int f; + + for(f = 3; f < 24; f++) + { + mvprintw(f, 40, "%4.4x:", addr); + mvprintw(f, 45, "%-35.35s", Z80Disassemble(z80, &addr)); + } + + f = 26; + + if (f < LINES) mvprintw(f++, 0, "AF = %4.4x", z80->AF.w); + if (f < LINES) mvprintw(f++, 0, "BC = %4.4x", z80->BC.w); + if (f < LINES) mvprintw(f++, 0, "DE = %4.4x", z80->DE.w); + if (f < LINES) mvprintw(f++, 0, "HL = %4.4x", z80->HL.w); + if (f < LINES) mvprintw(f++, 0, "IX = %4.4x", z80->IX.w); + if (f < LINES) mvprintw(f++, 0, "IY = %4.4x", z80->IY.w); + if (f < LINES) mvprintw(f++, 0, "SP = %4.4x", z80->SP); + if (f < LINES) mvprintw(f++, 0, "MATRIX = %2.2x/%2.2x/%2.2x/%2.2x/%2.2x" + "/%2.2x/%2.2x/%2.2x", matrix[0], + matrix[1], + matrix[2], + matrix[3], + matrix[4], + matrix[5], + matrix[6], + matrix[7]); + + refresh(); +} + + static int CheckTimers(Z80 *z80, Z80Val val) { + if (trace) + { + Trace(); + } + if (val >= FRAME_TSTATES) { Z80ResetCycles(z80, val - FRAME_TSTATES); @@ -262,6 +418,8 @@ static int CheckTimers(Z80 *z80, Z80Val val) FRAME_TSTATES = FAST_TSTATES; } + DrawScreen(); + if (z80->SP < 0x8000) { ZX81HouseKeeping(); @@ -276,6 +434,104 @@ static int CheckTimers(Z80 *z80, Z80Val val) } +static void PressKey(int key) +{ + int f; + + if (current_key_press_frame > 0) + { + if (--current_key_press_frame <= 0) + { + current_key_press_frame = 0; + + for(f = 0; f < keymap[current_key].no; f++) + { + matrix[keymap[current_key].row[f]] |= + keymap[current_key].bit[f]; + } + + current_key = -1; + } + } + else + { + if (key != ERR) + { + int n; + + for(n = 0; keymap[n].key != ERR && current_key == -1; n++) + { + if (keymap[n].key == key) + { + current_key_press_frame = key_press_frames; + current_key = n; + + for(f = 0; f < keymap[current_key].no; f++) + { + matrix[keymap[current_key].row[f]] &= + ~keymap[current_key].bit[f]; + } + } + } + } + } +} + + +static int EDCallback(Z80 *z80, Z80Val data) +{ + Z80Word pause; + + switch((Z80Byte)data) + { + case ED_SAVE: + if (z80->DE.w < 0x8000) + { + /* TODO: Save */ + } + + break; + + case ED_LOAD: + if (z80->DE.w < 0x8000) + { + /* TODO: Load */ + } + + mem[CDFLAG] = 0xc0; + + break; + + case ED_WAITKEY: + waitkey = TRUE; + started = TRUE; + break; + + case ED_ENDWAITKEY: + waitkey = FALSE; + break; + + case ED_PAUSE: + waitkey = TRUE; + + pause = z80->BC.w; + + while(pause-- && !(mem[CDFLAG]&1)) + { + ClearScreen(); + PressKey(getch()); + CheckTimers(z80, FRAME_TSTATES); + RenderScreen(); + } + + waitkey = FALSE; + break; + } + + return TRUE; +} + + static Z80Byte ZX81ReadMem(Z80 *z80, Z80Word addr) { return mem[addr]; @@ -295,6 +551,8 @@ static Z80Byte ZX81ReadPort(Z80 *z80, Z80Word port) { Z80Byte b = 0; + sprintf(message,"IN() called with %4.4x MARGIN=%d", port, mem[MARGIN]); + switch(port & 0xff) { case 0xfe: @@ -342,11 +600,42 @@ static Z80Byte ZX81ReadPort(Z80 *z80, Z80Word port) } -void ZX81WritePort(Z80 *z80, Z80Word port, Z80Byte val) +static void ZX81WritePort(Z80 *z80, Z80Word port, Z80Byte val) { } +static void ZX81Reset(void) +{ + int f; + + Z80Reset(z80); + + /* 16K of memory + */ + RAMBOT = 0x4000; + RAMTOP = RAMBOT + 0x4000; + + for(f = RAMBOT; f <= RAMTOP; f++) + { + mem[f] = 0; + } + + /* Keyboard matrix + */ + for(f = 0; f < 8; f++) + { + matrix[f] = 0x1f; + } + + /* Fill the upper 32K with RET opcodes for ULA reads + */ + for(f = 0x8000; f < 0x10000; f++) + { + mem[f] = 0xc9; + } +} + static void ZX81Init(void) { static const Z80Byte save[]= @@ -387,8 +676,6 @@ static void ZX81Init(void) 0xff /* End of patch */ }; - int f; - /* Load and patch ROM */ memcpy(mem, ZX81ROM, ROMLEN); @@ -414,106 +701,65 @@ static void ZX81Init(void) */ memcpy(mem + ROMLEN, mem, ROMLEN); - /* 16K of memory - */ - RAMBOT = 0x4000; - RAMTOP = RAMBOT + 0x4000; - - for(f = RAMBOT; f <= RAMTOP; f++) - { - mem[f] = 0; - } - - /* Keyboard matrix - */ - for(f = 0; f < 8; f++) - { - matrix[f] = 0x1f; - } - - /* Fill the upper 32K with RET opcodes for ULA reads - */ - for(f = 0x8000; f < 0x10000; f++) - { - mem[f] = 0xc9; - } + ZX81Reset(); } -static void DrawScreen(void) +/* -------------------------------------------------- MENU +*/ +static void Menu() { - int fast; - int x,y; + int done = FALSE; - fast = (FRAME_TSTATES == FAST_TSTATES); + erase(); - if (fast) + while(!done) { - for(y = 0; y < TXT_H; y++) - { - for(x = 0; x < TXT_W; x++) - { - mvprintw(y, x, "%lc", L'\u2592'); - } - } - } - else - { - Z80Byte *scr; - - scr = mem + PEEKW(DFILE); - y = 0; + int ch; - while(y < TXT_H) - { - scr++; - x = 0; + mvprintw(0, 0, "F1 .. Tracing %s", trace ? "ON ":"OFF"); + mvprintw(2, 0, "F2 .. Frames per key press: %2d", key_press_frames); + mvprintw(4, 0, "F3 .. Quit"); + mvprintw(6, 0, "Press ESCAPE to resume"); - while(*scr != 118 && x < TXT_W) - { - Z80Byte ch = *scr++; + refresh(); - if (ch & 0x80) - { - attron(A_REVERSE); - } + ch = getch(); - mvprintw(y, x, "%lc", scrcode[ch & 0x3f]); + switch(ch) + { + case KEY_F(1): + trace = !trace; + break; - if (ch & 0x80) + case KEY_F(2): + if (++key_press_frames == 10) { - attroff(A_REVERSE); + key_press_frames = 1; } + break; - x++; - } + case KEY_F(3): + quit = TRUE; + done = TRUE; + break; - y++; + case 27: + done = TRUE; + break; + + default: + break; } } } -/* -------------------------------------------------- UTILS -*/ -static long TimeDiff(const struct timespec *start, - const struct timespec *end) -{ - long diff; - - diff = (end->tv_nsec - start->tv_nsec) + - (end->tv_sec - start->tv_sec) * 1000000000L; - - return diff; -} - - /* -------------------------------------------------- MAIN */ int main(void) { struct timespec start; - int quit = FALSE; setlocale(LC_ALL, ""); @@ -539,6 +785,8 @@ int main(void) clock_gettime(CLOCK_MONOTONIC, &start); + ClearScreen(); + while(!quit) { struct timespec end; @@ -547,18 +795,27 @@ int main(void) Z80Exec(z80); - clear(); - - DrawScreen(); - ch = getch(); - if (ch == 3) + switch(ch) { - quit = TRUE; + case 3: + quit = TRUE; + break; + + case 27: + nodelay(stdscr, FALSE); + Menu(); + ClearScreen(); + nodelay(stdscr, TRUE); + break; + + default: + PressKey(ch); + break; } - refresh(); + RenderScreen(); clock_gettime(CLOCK_MONOTONIC, &end); start = end; |