/* vim:expandtab:ts=2 sw=2: */ /* Grafx2 - The Ultimate 256-color bitmap paint program Copyright 2018-2019 Thomas Bernard Copyright 2011 Pawel Góralski Copyright 2009 Petter Lindquist Copyright 2008 Yves Rizoud Copyright 2008 Franck Charlet Copyright 2007-2011 Adrien Destugues Copyright 1996-2001 Sunset Design (Guillaume Dorme & Karl Maritaud) Grafx2 is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. Grafx2 is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Grafx2; if not, see */ ///@file c64load.c /// Load C64 .PRG files #include #include #include #include "struct.h" #include "io.h" #include "c64load.h" #include "gfx2mem.h" #include "gfx2log.h" #define CPU_6502_STATIC #define CPU_6502_DEPENDENCIES_H "6502types.h" #define CPU_6502_USE_LOCAL_HEADER #include "6502.h" #define MAX_IO_ACCESS_MSG_COUNT 10 /** * Check if it is a machine langage program with a BASIC * startup line (eg. 10 SYS2061) * @return 0 or the machine language code start address */ word C64_isBinaryProgram(FILE * f) { word prg_load_addr; word addr, next, linenum; word len; char * line; unsigned long start = 0; int i; if (f == NULL) return 0; if (fseek(f, 0, SEEK_SET) < 0) return 0; if (!Read_word_le(f, &prg_load_addr)) return 0; if (prg_load_addr != 0x0801) return 0; addr = prg_load_addr; while (start == 0) { if (!Read_word_le(f, &next)) return 0; if (next == 0) break; if (!Read_word_le(f, &linenum)) return 0; GFX2_Log(GFX2_DEBUG, "$%04x %hu\n", next, linenum); len = next - addr - 4; line = malloc(len); if (!Read_bytes(f, line, len)) { free(line); return 0; } GFX2_LogHexDump(GFX2_DEBUG, "", (byte *)line, 0, len); if ((byte)line[0] == 0x9e) // SYS BASIC token { GFX2_Log(GFX2_DEBUG, "SYS%s\n", line + 1); i = 1; while (line[i] == ' ') i++; start = strtoul(line + i, NULL, 10); } free(line); } return start; } static byte C64_mem_read(void *context, word address) { struct c64state * c64 = (struct c64state *)context; if ((c64->ram[1] & 2) && address >= 0xe000) { GFX2_Log(GFX2_WARNING, "** ROM ** read($%04x)\n", address); // $FFE4 GETIN // make it return 0 (an Z=1) like if no key is pressed if (address == 0xffe4) { c64->keyjoyread++; return 0xA9; // LDA #$xx } if (address == 0xffe5) return 0x00; // 2nd byte of LDA #$00 return 0x60; // RTS } if ((c64->ram[1] & 4) && (((address >= 0xd000) && (address < 0xd800)) || ((address >= 0xdc00) && (address < 0xe000)))) { if ((address & 0xff00) == 0xd000) { if (c64->ioaccess[address - 0xd000] < MAX_IO_ACCESS_MSG_COUNT) { GFX2_Log(GFX2_DEBUG, "** IO ** read($%04x) $%02x\n", address, c64->ram[address]); if (++c64->ioaccess[address - 0xd000] == MAX_IO_ACCESS_MSG_COUNT) GFX2_Log(GFX2_DEBUG, " stopping debug log on $%04x\n", address); } } else { GFX2_Log(GFX2_DEBUG, "** IO ** read($%04x) $%02x\n", address, c64->ram[address]); } if ((address & 0xfffe) == 0xdc00) c64->keyjoyread++; } return c64->ram[address]; } static void C64_mem_write(void *context, word address, byte value) { struct c64state * c64 = (struct c64state *)context; if ((address >= 0xd000 && address < 0xd800) || (address >= 0xdc00 && address < 0xe000)) { if ((address & 0xff00) == 0xd000) { if (c64->ioaccess[address - 0xd000] < MAX_IO_ACCESS_MSG_COUNT) { GFX2_Log(GFX2_DEBUG, "** IO ** write($%04x, $%02x)\n", address, value); if (++c64->ioaccess[address - 0xd000] == MAX_IO_ACCESS_MSG_COUNT) GFX2_Log(GFX2_DEBUG, " stopping debug log on $%04x\n", address); } } else { GFX2_Log(GFX2_DEBUG, "** IO ** write($%04x, $%02x)\n", address, value); } switch (address) { case 0xd011: c64->irqrasterline = (c64->irqrasterline & 0x00ff) | (value >> 7); break; case 0xd012: c64->irqrasterline = (c64->irqrasterline & 0xff00) | value; break; case 0xd018: if (c64->ram[0xd011] & 0x10) // Screen is on c64->fliscreens[(c64->ram[0xd011] - 50) & 7] = value >> 4; break; case 0xd019: // acknowledge rasterirq m6502_irq(c64->cpu, FALSE); break; case 0xd021: // background color { word line = ((word)(c64->ram[0xd011] & 0x80) << 1) | c64->ram[0xd012]; if (line >= 50 && line < 250) c64->backgrounds[line - 50] = value & 0x0f; } break; } } else if (address >= 0xfffa) GFX2_Log(GFX2_DEBUG, "write($%04x, $%02x)\n", address, value); else if (address == 0x314 || address == 0x315) GFX2_Log(GFX2_DEBUG, "write($%04x, $%02x)\n", address, value); c64->ram[address] = value; } int C64_LoadPrg(struct c64state * c64, FILE * file, word start) { M6502 cpu; zusize cycles = 0; zusize next_rasterline = 63; int i, count = 0; byte screen_min = 255; int prg_size; if (c64->ram == NULL) { c64->ram = GFX2_malloc(65536); if (c64->ram == NULL) return 0; } memset(c64->ram, 0, 65536); prg_size = fread(c64->ram + 0x801, 1, 38911, file); GFX2_Log(GFX2_DEBUG, "C64_LoadPrg(%d, $%04x)\n", prg_size, start); if (prg_size < 0) return 0; c64->ram[0x00] = 0x2F; c64->ram[0x01] = 0x37; c64->ram[0x2B] = 0x01; c64->ram[0x2C] = 0x08; c64->ram[0x2D] = (0x7ff + prg_size) & 0xff; c64->ram[0x2E] = (0x7ff + prg_size) >> 8; c64->ram[0x2F] = c64->ram[0x2D]; c64->ram[0x30] = c64->ram[0x2E]; c64->ram[0x31] = c64->ram[0x2D]; c64->ram[0x32] = c64->ram[0x2E]; c64->ram[0xd011] = 0x1B; c64->ram[0xd016] = 0xC8; c64->ram[0xd018] = 0x15; c64->ram[0xd020] = 0xFE; c64->ram[0xd021] = 0xF6; c64->ram[0xd022] = 0xF1; c64->ram[0xd023] = 0xF2; c64->ram[0xd024] = 0xF3; c64->ram[0xd025] = 0xF4; c64->ram[0xd026] = 0xF0; c64->ram[0xd027] = 0xF1; c64->ram[0xd028] = 0xF2; c64->ram[0xd029] = 0xF3; c64->ram[0xd02a] = 0xF4; c64->ram[0xd02b] = 0xF5; c64->ram[0xd02c] = 0xF6; c64->ram[0xd02d] = 0xF7; c64->ram[0xd02e] = 0xFC; c64->ram[0xdd00] = 0x97; c64->cpu = &cpu; memset(&cpu, 0, sizeof(cpu)); cpu.context = (void*)c64; cpu.read = C64_mem_read; cpu.write = C64_mem_write; m6502_power(&cpu, TRUE); cpu.state.pc = start; while (cycles < 10000000 && c64->keyjoyread < 10) { word lastpc = cpu.state.pc; //cycles += m6502_run(&cpu, next_rasterline > cycles ? next_rasterline - cycles : 1); cycles += m6502_run(&cpu, 1); if (cycles >= next_rasterline) { word line = ((word)(c64->ram[0xd011] & 0x80) << 1) | c64->ram[0xd012]; if (++line >= 312) line = 0; c64->ram[0xd012] = line & 0xff; c64->ram[0xd011] = (c64->ram[0xd011] & 0x7f) | ((line >> 1) & 0x80); if (c64->ram[0xd01a] & 1) { if (line == c64->irqrasterline) { GFX2_Log(GFX2_DEBUG, "pc=$%04x Raster IRQ line %hu\n", cpu.state.pc, line); c64->ram[0xd019] |= 0x81; m6502_irq(&cpu, TRUE); } } next_rasterline += 63; } if (lastpc == cpu.state.pc/* && !(c64->ram[0xd01a] & 1)*/) { GFX2_Log(GFX2_DEBUG, "infinite loop detected\n"); if ((c64->ram[0xd011] & 0x10) == 0 // Screen off && (c64->ram[0x315] != 0)) { // fake system interrupt cpu.state.pc = ((word)c64->ram[0x315] << 8) + c64->ram[0x314]; } else break; } } GFX2_Log(GFX2_DEBUG, "%u cycles pc=$%04x\n", cycles, cpu.state.pc); c64->vicmode = (c64->ram[0xd011] & 0x60) | (c64->ram[0xd016] & 0x10); GFX2_Log(GFX2_DEBUG, "$%02x %s%s%s\n", c64->vicmode, (c64->ram[0xd011] & 0x20) ? "BITMAP" : "TEXT", (c64->ram[0xd011] & 0x40) ? " EXTBKG" : "", (c64->ram[0xd016] & 0x10) ? " MULTICOLOR" : ""); GFX2_Log(GFX2_DEBUG, "fliscreens [%x %x %x %x %x %x %x %x]\n", c64->fliscreens[0], c64->fliscreens[1], c64->fliscreens[2], c64->fliscreens[3], c64->fliscreens[4], c64->fliscreens[5], c64->fliscreens[6], c64->fliscreens[7]); for (i = 0; i < 8; i++) { if (c64->fliscreens[i] != 0) count++; if (c64->fliscreens[i] < screen_min) screen_min = c64->fliscreens[i]; } if (count > 1) { GFX2_Log(GFX2_INFO, "FLI MODE DETECTED\n"); c64->vicmode |= C64_VICMODE_FLI; } c64->bitmap = c64->screen = ((c64->ram[0xdd00] & 3) ^ 3) << 14; c64->bitmap += (c64->ram[0xd018] & 0x0f) << 10; c64->bitmap &= ((c64->ram[0xd011] & 0x20) ? 0xe000 : 0xf800); if (c64->vicmode & C64_VICMODE_FLI) c64->screen += screen_min << 10; else c64->screen += (c64->ram[0xd018] & 0xf0) << 6; GFX2_Log(GFX2_DEBUG, "$D018=$%02x bitmap at $%04x, screen at $%04x\n", c64->ram[0xd018], c64->bitmap, c64->screen); return 1; }