forked from DavidBuchanan314/6502-emu
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path6850.c
61 lines (51 loc) · 1.39 KB
/
6850.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
#include <stdio.h>
#include <unistd.h>
#include <sys/poll.h>
#include <stdlib.h>
#include "6502.h"
#include "6850.h"
static int n;
static union UartStatusReg uart_SR;
static uint8_t incoming_char;
void init_uart() {
memory[DATA_ADDR] = 0;
uart_SR.byte = 0;
uart_SR.bits.TDRE = 1; // we are always ready to output data
uart_SR.bits.RDRF = 0;
incoming_char = 0;
}
int stdin_ready() {
struct pollfd fds;
fds.fd = 0; // stdin
fds.events = POLLIN;
return poll(&fds, 1, 0) == 1; // timeout = 0
}
void step_uart() {
if (write_addr == &memory[DATA_ADDR]) {
putchar(memory[DATA_ADDR]);
if (memory[DATA_ADDR] == '\b') printf(" \b");
fflush(stdout);
write_addr = NULL;
} else if (read_addr == &memory[DATA_ADDR]) {
uart_SR.bits.RDRF = 0;
read_addr = NULL;
}
/* update input register if empty */
if ((n++ % 10000) == 0) { // polling stdin every cycle is performance intensive. This is a bit of a dirty hack.
if (!uart_SR.bits.RDRF && stdin_ready()) { // the real hardware has no buffer. Remote the RDRF check for more accurate emulation.
if (read(0, &incoming_char, 1) != 1) {
printf("Warning: read() returned 0\n");
}
if (incoming_char == 0x18) { // CTRL+X
printf("\r\n");
exit(0);
}
if (incoming_char == 0x7F) { // Backspace
incoming_char = '\b';
}
uart_SR.bits.RDRF = 1;
}
}
memory[DATA_ADDR] = incoming_char;
memory[CTRL_ADDR] = uart_SR.byte;
}