Mirror of CollapseOS
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

196 lines
4.2KB

  1. /* Common code between forth and stage binaries.
  2. They all run on the same kind of virtual machine: A z80 CPU, 64K of RAM/ROM.
  3. */
  4. #include <string.h>
  5. #include "emul.h"
  6. // Port for block reads. Each read or write has to be done in 5 IO writes:
  7. // 1 - r/w. 1 for read, 2 for write.
  8. // 2 - blkid MSB
  9. // 3 - blkid LSB
  10. // 4 - dest addr MSB
  11. // 5 - dest addr LSB
  12. #define BLK_PORT 0x03
  13. #ifndef BLKFS_PATH
  14. #error BLKFS_PATH needed
  15. #endif
  16. #ifndef FBIN_PATH
  17. #error FBIN_PATH needed
  18. #endif
  19. static Machine m;
  20. static ushort traceval = 0;
  21. static uint64_t blkop = 0; // 5 bytes
  22. static FILE *blkfp;
  23. static uint8_t io_read(int unused, uint16_t addr)
  24. {
  25. addr &= 0xff;
  26. IORD fn = m.iord[addr];
  27. if (fn != NULL) {
  28. return fn();
  29. } else {
  30. fprintf(stderr, "Out of bounds I/O read: %d\n", addr);
  31. return 0;
  32. }
  33. }
  34. static void io_write(int unused, uint16_t addr, uint8_t val)
  35. {
  36. addr &= 0xff;
  37. IOWR fn = m.iowr[addr];
  38. if (fn != NULL) {
  39. fn(val);
  40. } else {
  41. fprintf(stderr, "Out of bounds I/O write: %d / %d (0x%x)\n", addr, val, val);
  42. }
  43. }
  44. static void iowr_blk(uint8_t val)
  45. {
  46. blkop <<= 8;
  47. blkop |= val;
  48. uint8_t rw = blkop >> 32;
  49. if (rw) {
  50. uint16_t blkid = (blkop >> 16);
  51. uint16_t dest = blkop & 0xffff;
  52. blkop = 0;
  53. fseek(blkfp, blkid*1024, SEEK_SET);
  54. if (rw==2) { // write
  55. fwrite(&m.mem[dest], 1024, 1, blkfp);
  56. } else { // read
  57. fread(&m.mem[dest], 1024, 1, blkfp);
  58. }
  59. }
  60. }
  61. static uint8_t mem_read(int unused, uint16_t addr)
  62. {
  63. return m.mem[addr];
  64. }
  65. static void mem_write(int unused, uint16_t addr, uint8_t val)
  66. {
  67. if (addr < m.ramstart) {
  68. fprintf(stderr, "Writing to ROM (%d)!\n", addr);
  69. emul_memdump();
  70. fprintf(stderr, "Press any key to continue...\n");
  71. while (getchar() > 0x100);
  72. }
  73. m.mem[addr] = val;
  74. }
  75. Machine* emul_init()
  76. {
  77. fprintf(stderr, "Using blkfs %s\n", BLKFS_PATH);
  78. blkfp = fopen(BLKFS_PATH, "r+");
  79. if (!blkfp) {
  80. fprintf(stderr, "Can't open\n");
  81. return NULL;
  82. }
  83. fseek(blkfp, 0, SEEK_END);
  84. if (ftell(blkfp) < 100 * 1024) {
  85. fclose(blkfp);
  86. fprintf(stderr, "emul/blkfs too small, something's wrong, aborting.\n");
  87. return NULL;
  88. }
  89. fseek(blkfp, 0, SEEK_SET);
  90. // initialize memory
  91. memset(m.mem, 0, 0x10000);
  92. FILE *bfp = fopen(FBIN_PATH, "r");
  93. if (!bfp) {
  94. fprintf(stderr, "Can't open forth.bin\n");
  95. return NULL;
  96. }
  97. int i = 0;
  98. int c = getc(bfp);
  99. while (c != EOF) {
  100. m.mem[i++] = c;
  101. c = getc(bfp);
  102. }
  103. fclose(bfp);
  104. m.ramstart = 0;
  105. m.minsp = 0xffff;
  106. m.maxix = 0;
  107. for (int i=0; i<0x100; i++) {
  108. m.iord[i] = NULL;
  109. m.iowr[i] = NULL;
  110. }
  111. Z80RESET(&m.cpu);
  112. m.cpu.memRead = mem_read;
  113. m.cpu.memWrite = mem_write;
  114. m.cpu.ioRead = io_read;
  115. m.cpu.ioWrite = io_write;
  116. m.iowr[BLK_PORT] = iowr_blk;
  117. return &m;
  118. }
  119. void emul_deinit()
  120. {
  121. fclose(blkfp);
  122. }
  123. bool emul_step()
  124. {
  125. if (!m.cpu.halted) {
  126. Z80Execute(&m.cpu);
  127. ushort newsp = m.cpu.R1.wr.SP;
  128. if (newsp != 0 && newsp < m.minsp) {
  129. m.minsp = newsp;
  130. }
  131. if (m.cpu.R1.wr.IX > m.maxix) {
  132. m.maxix = m.cpu.R1.wr.IX;
  133. }
  134. return true;
  135. } else {
  136. return false;
  137. }
  138. }
  139. bool emul_steps(unsigned int steps)
  140. {
  141. while (steps) {
  142. if (!emul_step()) {
  143. return false;
  144. }
  145. steps--;
  146. }
  147. return true;
  148. }
  149. void emul_loop()
  150. {
  151. while (emul_step());
  152. }
  153. void emul_trace(ushort addr)
  154. {
  155. ushort newval = m.mem[addr+1] << 8 | m.mem[addr];
  156. if (newval != traceval) {
  157. traceval = newval;
  158. fprintf(stderr, "trace: %04x PC: %04x\n", traceval, m.cpu.PC);
  159. }
  160. }
  161. void emul_memdump()
  162. {
  163. fprintf(stderr, "Dumping memory to memdump. PC %04x\n", m.cpu.PC);
  164. FILE *fp = fopen("memdump", "w");
  165. fwrite(m.mem, 0x10000, 1, fp);
  166. fclose(fp);
  167. }
  168. void emul_debugstr(char *s)
  169. {
  170. sprintf(s, "SP %04x (%04x) IX %04x (%04x)",
  171. m.cpu.R1.wr.SP, m.minsp, m.cpu.R1.wr.IX, m.maxix);
  172. }
  173. void emul_printdebug()
  174. {
  175. fprintf(stderr, "Min SP: %04x\n", m.minsp);
  176. fprintf(stderr, "Max IX: %04x\n", m.maxix);
  177. }