mirror of
https://github.com/LIV2/WinUAE.git
synced 2025-12-06 00:12:52 +00:00
3903 lines
108 KiB
C++
3903 lines
108 KiB
C++
/*
|
|
* UAE - The Un*x Amiga Emulator
|
|
*
|
|
* Memory management
|
|
*
|
|
* (c) 1995 Bernd Schmidt
|
|
*/
|
|
|
|
#define DEBUG_STUPID 0
|
|
|
|
// don't touch
|
|
#define ACA500x_DEVELOPMENT 0
|
|
|
|
#include "sysconfig.h"
|
|
#include "sysdeps.h"
|
|
|
|
#include "options.h"
|
|
#include "uae.h"
|
|
#include "memory.h"
|
|
#include "rommgr.h"
|
|
#include "ersatz.h"
|
|
#include "zfile.h"
|
|
#include "custom.h"
|
|
#include "events.h"
|
|
#include "newcpu.h"
|
|
#include "autoconf.h"
|
|
#include "savestate.h"
|
|
#include "ar.h"
|
|
#include "crc32.h"
|
|
#include "gui.h"
|
|
#include "cdtv.h"
|
|
#include "akiko.h"
|
|
#include "arcadia.h"
|
|
#include "enforcer.h"
|
|
#include "threaddep/thread.h"
|
|
#include "gayle.h"
|
|
#include "debug.h"
|
|
#include "debugmem.h"
|
|
#include "gfxboard.h"
|
|
#include "cpuboard.h"
|
|
#include "uae/ppc.h"
|
|
#include "devices.h"
|
|
#include "inputdevice.h"
|
|
#include "casablanca.h"
|
|
|
|
bool canbang;
|
|
static bool rom_write_enabled;
|
|
#ifdef JIT
|
|
/* Set by each memory handler that does not simply access real memory. */
|
|
int special_mem, special_mem_default;
|
|
/* do not use get_n_addr */
|
|
int jit_n_addr_unsafe;
|
|
#endif
|
|
static int mem_hardreset;
|
|
static bool roms_modified;
|
|
|
|
#define FLASHEMU 0
|
|
|
|
static bool isdirectjit (void)
|
|
{
|
|
return currprefs.cachesize && !currprefs.comptrustbyte;
|
|
}
|
|
|
|
static bool canjit (void)
|
|
{
|
|
if (currprefs.cpu_model < 68020 || currprefs.address_space_24)
|
|
return false;
|
|
return true;
|
|
}
|
|
static bool needmman (void)
|
|
{
|
|
if (!jit_direct_compatible_memory)
|
|
return false;
|
|
#ifdef _WIN32
|
|
return true;
|
|
#endif
|
|
if (canjit ())
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
static void nocanbang (void)
|
|
{
|
|
if (canbang) {
|
|
write_log(_T("Switching JIT direct off!\n"));
|
|
}
|
|
canbang = 0;
|
|
}
|
|
|
|
uae_u8 ce_banktype[65536];
|
|
uae_u8 ce_cachable[65536];
|
|
|
|
static size_t bootrom_filepos, chip_filepos, bogo_filepos, a3000lmem_filepos, a3000hmem_filepos, mem25bit_filepos;
|
|
|
|
/* Set if we notice during initialization that settings changed,
|
|
and we must clear all memory to prevent bogus contents from confusing
|
|
the Kickstart. */
|
|
static bool need_hardreset;
|
|
static int bogomem_aliasing;
|
|
|
|
/* The address space setting used during the last reset. */
|
|
static bool last_address_space_24;
|
|
|
|
addrbank *mem_banks[MEMORY_BANKS];
|
|
|
|
/* This has two functions. It either holds a host address that, when added
|
|
to the 68k address, gives the host address corresponding to that 68k
|
|
address (in which case the value in this array is even), OR it holds the
|
|
same value as mem_banks, for those banks that have baseaddr==0. In that
|
|
case, bit 0 is set (the memory access routines will take care of it). */
|
|
|
|
uae_u8 *baseaddr[MEMORY_BANKS];
|
|
|
|
#ifdef NO_INLINE_MEMORY_ACCESS
|
|
__inline__ uae_u32 longget (uaecptr addr)
|
|
{
|
|
return call_mem_get_func (get_mem_bank (addr).lget, addr);
|
|
}
|
|
__inline__ uae_u32 wordget (uaecptr addr)
|
|
{
|
|
return call_mem_get_func (get_mem_bank (addr).wget, addr);
|
|
}
|
|
__inline__ uae_u32 byteget (uaecptr addr)
|
|
{
|
|
return call_mem_get_func (get_mem_bank (addr).bget, addr);
|
|
}
|
|
__inline__ void longput (uaecptr addr, uae_u32 l)
|
|
{
|
|
call_mem_put_func (get_mem_bank (addr).lput, addr, l);
|
|
}
|
|
__inline__ void wordput (uaecptr addr, uae_u32 w)
|
|
{
|
|
call_mem_put_func (get_mem_bank (addr).wput, addr, w);
|
|
}
|
|
__inline__ void byteput (uaecptr addr, uae_u32 b)
|
|
{
|
|
call_mem_put_func (get_mem_bank (addr).bput, addr, b);
|
|
}
|
|
#endif
|
|
|
|
extern bool debugmem_initialized;
|
|
|
|
bool real_address_allowed(void)
|
|
{
|
|
return debugmem_initialized == false;
|
|
}
|
|
|
|
int addr_valid (const TCHAR *txt, uaecptr addr, uae_u32 len)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (ab == 0 || !(ab->flags & (ABFLAG_RAM | ABFLAG_ROM)) || addr < 0x100 || len > 16777215 || !valid_address (addr, len)) {
|
|
write_log (_T("corrupt %s pointer %x (%d) detected!\n"), txt, addr, len);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int illegal_count;
|
|
/* A dummy bank that only contains zeros */
|
|
|
|
static uae_u32 REGPARAM3 dummy_lget (uaecptr) REGPARAM;
|
|
static uae_u32 REGPARAM3 dummy_wget (uaecptr) REGPARAM;
|
|
static uae_u32 REGPARAM3 dummy_bget (uaecptr) REGPARAM;
|
|
static void REGPARAM3 dummy_lput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 dummy_wput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 dummy_bput (uaecptr, uae_u32) REGPARAM;
|
|
static int REGPARAM3 dummy_check (uaecptr addr, uae_u32 size) REGPARAM;
|
|
|
|
/* fake UAE ROM */
|
|
|
|
extern addrbank fakeuaebootrom_bank;
|
|
MEMORY_FUNCTIONS(fakeuaebootrom);
|
|
|
|
#define MAX_ILG 1000
|
|
#define NONEXISTINGDATA 0
|
|
//#define NONEXISTINGDATA 0xffffffff
|
|
|
|
static bool map_uae_boot_rom_direct(void)
|
|
{
|
|
if (!fakeuaebootrom_bank.allocated_size) {
|
|
fakeuaebootrom_bank.start = 0xf00000;
|
|
fakeuaebootrom_bank.reserved_size = 65536;
|
|
fakeuaebootrom_bank.mask = fakeuaebootrom_bank.reserved_size - 1;
|
|
if (!mapped_malloc (&fakeuaebootrom_bank))
|
|
return false;
|
|
// create jump table to real uae boot rom
|
|
for (int i = 0xff00; i < 0xfff8; i += 8) {
|
|
uae_u8 *p = fakeuaebootrom_bank.baseaddr + i;
|
|
p[0] = 0x4e;
|
|
p[1] = 0xf9;
|
|
uaecptr p2 = rtarea_base + i;
|
|
p[2] = p2 >> 24;
|
|
p[3] = p2 >> 16;
|
|
p[4] = p2 >> 8;
|
|
p[5] = p2 >> 0;
|
|
}
|
|
}
|
|
map_banks(&fakeuaebootrom_bank, 0xf0, 1, 1);
|
|
write_log(_T("Mapped fake UAE Boot ROM jump table.\n"));
|
|
return true;
|
|
}
|
|
|
|
// if access looks like old style hook call and f00000 space is unused, redirect to UAE boot ROM.
|
|
static bool maybe_map_boot_rom(uaecptr addr)
|
|
{
|
|
if (currprefs.uaeboard >= 2 && get_mem_bank_real(0xf00000) == &dummy_bank && !currprefs.mmu_model) {
|
|
uae_u32 pc = M68K_GETPC;
|
|
if (addr >= 0xf0ff00 && addr <= 0xf0fff8 && (((valid_address(pc, 2) && (pc < 0xf00000 || pc >= 0x01000000) && !currprefs.cpu_compatible) || (pc == addr && currprefs.cpu_compatible)))) {
|
|
bool check2 = currprefs.cpu_compatible;
|
|
if (!check2) {
|
|
uae_u32 w = get_word(pc);
|
|
// JSR xxxxxxxx or JSR (an)
|
|
if (w == 0x4eb9 || (w & 0xfff8) == 0x4e90)
|
|
check2 = true;
|
|
}
|
|
if (check2) {
|
|
if (map_uae_boot_rom_direct()) {
|
|
if (get_mem_bank_real(0xf00000) == &fakeuaebootrom_bank) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static void dummylog(int rw, uaecptr addr, int size, uae_u32 val, int ins)
|
|
{
|
|
/* ignore Zorro3 expansion space */
|
|
if (addr >= AUTOCONFIG_Z3 && addr <= AUTOCONFIG_Z3 + 0x200)
|
|
return;
|
|
/* autoconfig and extended rom */
|
|
if (addr >= 0xe00000 && addr <= 0xf7ffff)
|
|
return;
|
|
/* motherboard ram */
|
|
if (addr >= 0x08000000 && addr <= 0x08000007)
|
|
return;
|
|
if (addr >= 0x07f00000 && addr <= 0x07f00007)
|
|
return;
|
|
if (addr >= 0x07f7fff0 && addr <= 0x07ffffff)
|
|
return;
|
|
if (debugmem_extinvalidmem(addr, val, rw ? (1 << size) : -(1 << size)))
|
|
return;
|
|
if ((illegal_count >= MAX_ILG && MAX_ILG > 0) && !memwatch_access_validator)
|
|
return;
|
|
if (MAX_ILG >= 0)
|
|
illegal_count++;
|
|
if (ins) {
|
|
write_log (_T("WARNING: Illegal opcode %cget at %08x PC=%x\n"),
|
|
size == sz_word ? 'w' : 'l', addr, M68K_GETPC);
|
|
} else if (rw) {
|
|
write_log (_T("Illegal %cput at %08x=%08x PC=%x\n"),
|
|
size == sz_byte ? 'b' : size == sz_word ? 'w' : 'l', addr, val, M68K_GETPC);
|
|
} else {
|
|
write_log (_T("Illegal %cget at %08x PC=%x\n"),
|
|
size == sz_byte ? 'b' : size == sz_word ? 'w' : 'l', addr, M68K_GETPC);
|
|
}
|
|
}
|
|
|
|
// 250ms delay
|
|
static void gary_wait(uaecptr addr, int size, bool write)
|
|
{
|
|
static int cnt = 50;
|
|
|
|
#if 0
|
|
int lines = 313 * 12;
|
|
while (lines-- > 0)
|
|
x_do_cycles(228 * CYCLE_UNIT);
|
|
#endif
|
|
|
|
if (cnt > 0) {
|
|
write_log (_T("Gary timeout: %08x %d %c PC=%08x\n"), addr, size, write ? 'W' : 'R', M68K_GETPC);
|
|
cnt--;
|
|
}
|
|
}
|
|
|
|
static bool gary_nonrange(uaecptr addr)
|
|
{
|
|
if (currprefs.cs_fatgaryrev < 0)
|
|
return false;
|
|
if (addr < 0xb80000)
|
|
return false;
|
|
if (addr >= 0xd00000 && addr < 0xdc0000)
|
|
return true;
|
|
if (addr >= 0xdd0000 && addr < 0xde0000)
|
|
return true;
|
|
if (addr >= 0xdf8000 && addr < 0xe00000)
|
|
return false;
|
|
if (addr >= 0xe80000 && addr < 0xf80000)
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
void dummy_put (uaecptr addr, int size, uae_u32 val)
|
|
{
|
|
#if FLASHEMU
|
|
if (addr >= 0xf00000 && addr < 0xf80000 && size < 2)
|
|
flash_write(addr, val);
|
|
#endif
|
|
|
|
if (gary_nonrange(addr) || (size > sz_byte && gary_nonrange(addr + (1 << size) - 1))) {
|
|
if (gary_timeout)
|
|
gary_wait(addr, size, true);
|
|
if (gary_toenb && currprefs.mmu_model)
|
|
hardware_exception2(addr, val, false, false, size);
|
|
}
|
|
}
|
|
|
|
static uae_u32 nonexistingdata(void)
|
|
{
|
|
if (currprefs.cs_unmapped_space == 1)
|
|
return 0x00000000;
|
|
if (currprefs.cs_unmapped_space == 2)
|
|
return 0xffffffff;
|
|
return NONEXISTINGDATA;
|
|
}
|
|
|
|
uae_u32 dummy_get_safe(uaecptr addr, int size, bool inst, uae_u32 defvalue)
|
|
{
|
|
uae_u32 v = defvalue;
|
|
uae_u32 mask = size == sz_long ? 0xffffffff : (1 << ((1 << size) * 8)) - 1;
|
|
if (currprefs.cpu_model >= 68040)
|
|
return v & mask;
|
|
if (!currprefs.cpu_compatible)
|
|
return v & mask;
|
|
if (currprefs.address_space_24)
|
|
addr &= 0x00ffffff;
|
|
if (addr >= 0x10000000)
|
|
return v & mask;
|
|
// CD32 and B2000
|
|
if (currprefs.cs_unmapped_space == 1)
|
|
return 0;
|
|
if (currprefs.cs_unmapped_space == 2)
|
|
return 0xffffffff & mask;
|
|
if ((currprefs.cpu_model <= 68010) || (currprefs.cpu_model == 68020 && (currprefs.chipset_mask & CSMASK_AGA) && currprefs.address_space_24)) {
|
|
// if executed from ROM: always return zero
|
|
uaecptr pc = m68k_getpc();
|
|
addrbank *ab = &get_mem_bank(pc);
|
|
if (ab->flags & ABFLAG_ROM) {
|
|
return 0;
|
|
}
|
|
if (size == sz_long) {
|
|
v = regs.irc & 0xffff;
|
|
if (addr & 1)
|
|
v = (v << 8) | (v >> 8);
|
|
v = (v << 16) | v;
|
|
} else if (size == sz_word) {
|
|
v = regs.irc & 0xffff;
|
|
if (addr & 1)
|
|
v = (v << 8) | (v >> 8);
|
|
} else {
|
|
v = regs.irc;
|
|
v = (addr & 1) ? (v & 0xff) : ((v >> 8) & 0xff);
|
|
}
|
|
}
|
|
return v & mask;
|
|
}
|
|
|
|
uae_u32 dummy_get (uaecptr addr, int size, bool inst, uae_u32 defvalue)
|
|
{
|
|
uae_u32 v = defvalue;
|
|
|
|
#if FLASHEMU
|
|
if (addr >= 0xf00000 && addr < 0xf80000 && size < 2) {
|
|
if (addr < 0xf60000)
|
|
return flash_read(addr);
|
|
return 8;
|
|
}
|
|
#endif
|
|
#if ACA500x_DEVELOPMENT
|
|
if (addr == 0xb03000) {
|
|
return 0xffff;
|
|
}
|
|
if (addr == 0xb07000) {
|
|
return 0x0000;
|
|
}
|
|
if (addr == 0xb2f800) {
|
|
return 0xffff;
|
|
}
|
|
if (addr == 0xb3b800) {
|
|
return 0x0000;
|
|
}
|
|
if (addr == 0xb3f800) {
|
|
return currprefs.cpu_model > 68000 ? 0x0000 : 0xffff;
|
|
}
|
|
if (addr == 0xb0b000) {
|
|
extern bool isideint(void);
|
|
return isideint() ? 0xffff : 0x0000;
|
|
}
|
|
#endif
|
|
|
|
if ((size == sz_word || size == sz_long) && inst && maybe_map_boot_rom(addr)) {
|
|
if (size == sz_word)
|
|
return get_word(addr);
|
|
return get_long(addr);
|
|
}
|
|
|
|
if (gary_nonrange(addr) || (size > sz_byte && gary_nonrange(addr + (1 << size) - 1))) {
|
|
if (gary_timeout)
|
|
gary_wait(addr, size, false);
|
|
if (gary_toenb)
|
|
hardware_exception2(addr, 0, true, false, size);
|
|
return v;
|
|
}
|
|
|
|
return dummy_get_safe(addr, size, inst, defvalue);
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 dummy_lget (uaecptr addr)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(0, addr, sz_long, 0, 0);
|
|
return dummy_get(addr, sz_long, false, nonexistingdata());
|
|
}
|
|
uae_u32 REGPARAM2 dummy_lgeti (uaecptr addr)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(0, addr, sz_long, 0, 1);
|
|
return dummy_get(addr, sz_long, true, nonexistingdata());
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 dummy_wget (uaecptr addr)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(0, addr, sz_word, 0, 0);
|
|
return dummy_get(addr, sz_word, false, nonexistingdata());
|
|
}
|
|
uae_u32 REGPARAM2 dummy_wgeti (uaecptr addr)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(0, addr, sz_word, 0, 1);
|
|
return dummy_get(addr, sz_word, true, nonexistingdata());
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 dummy_bget (uaecptr addr)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(0, addr, sz_byte, 0, 0);
|
|
return dummy_get(addr, sz_byte, false, nonexistingdata());
|
|
}
|
|
|
|
static void REGPARAM2 dummy_lput (uaecptr addr, uae_u32 l)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(1, addr, sz_long, l, 0);
|
|
dummy_put(addr, sz_long, l);
|
|
}
|
|
static void REGPARAM2 dummy_wput (uaecptr addr, uae_u32 w)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(1, addr, sz_word, w, 0);
|
|
dummy_put(addr, sz_word, w);
|
|
}
|
|
static void REGPARAM2 dummy_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
dummylog(1, addr, sz_byte, b, 0);
|
|
dummy_put(addr, sz_byte, b);
|
|
}
|
|
|
|
static int REGPARAM2 dummy_check (uaecptr addr, uae_u32 size)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static void REGPARAM2 none_put (uaecptr addr, uae_u32 v)
|
|
{
|
|
}
|
|
static uae_u32 REGPARAM2 ones_get (uaecptr addr)
|
|
{
|
|
return 0xffffffff;
|
|
}
|
|
|
|
addrbank *get_sub_bank(uaecptr *paddr)
|
|
{
|
|
int i;
|
|
uaecptr addr = *paddr;
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
struct addrbank_sub *sb = ab->sub_banks;
|
|
if (!sb)
|
|
return &dummy_bank;
|
|
for (i = 0; sb[i].bank; i++) {
|
|
int offset = addr & 65535;
|
|
if (offset < sb[i + 1].offset) {
|
|
uae_u32 mask = sb[i].mask;
|
|
uae_u32 maskval = sb[i].maskval;
|
|
if ((offset & mask) == maskval) {
|
|
*paddr = addr - sb[i].suboffset;
|
|
return sb[i].bank;
|
|
}
|
|
}
|
|
}
|
|
*paddr = addr - sb[i - 1].suboffset;
|
|
return sb[i - 1].bank;
|
|
}
|
|
uae_u32 REGPARAM3 sub_bank_lget (uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->lget(addr);
|
|
}
|
|
uae_u32 REGPARAM3 sub_bank_wget(uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->wget(addr);
|
|
}
|
|
uae_u32 REGPARAM3 sub_bank_bget(uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->bget(addr);
|
|
}
|
|
void REGPARAM3 sub_bank_lput(uaecptr addr, uae_u32 v) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
ab->lput(addr, v);
|
|
}
|
|
void REGPARAM3 sub_bank_wput(uaecptr addr, uae_u32 v) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
ab->wput(addr, v);
|
|
}
|
|
void REGPARAM3 sub_bank_bput(uaecptr addr, uae_u32 v) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
ab->bput(addr, v);
|
|
}
|
|
uae_u32 REGPARAM3 sub_bank_lgeti(uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->lgeti(addr);
|
|
}
|
|
uae_u32 REGPARAM3 sub_bank_wgeti(uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->wgeti(addr);
|
|
}
|
|
int REGPARAM3 sub_bank_check(uaecptr addr, uae_u32 size) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->check(addr, size);
|
|
}
|
|
uae_u8 *REGPARAM3 sub_bank_xlate(uaecptr addr) REGPARAM
|
|
{
|
|
addrbank *ab = get_sub_bank(&addr);
|
|
return ab->xlateaddr(addr);
|
|
}
|
|
|
|
|
|
/* Chip memory */
|
|
|
|
static uae_u32 chipmem_full_mask;
|
|
static uae_u32 chipmem_full_size;
|
|
|
|
static int REGPARAM3 chipmem_check (uaecptr addr, uae_u32 size) REGPARAM;
|
|
static uae_u8 *REGPARAM3 chipmem_xlate (uaecptr addr) REGPARAM;
|
|
|
|
#ifdef AGA
|
|
|
|
/* AGA ce-chipram access */
|
|
|
|
static void ce2_timeout (void)
|
|
{
|
|
wait_cpu_cycle_read (0, -1);
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_lget_ce2 (uaecptr addr)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
ce2_timeout ();
|
|
return do_get_mem_long (m);
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_wget_ce2 (uaecptr addr)
|
|
{
|
|
uae_u16 *m, v;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
ce2_timeout ();
|
|
v = do_get_mem_word (m);
|
|
return v;
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_bget_ce2 (uaecptr addr)
|
|
{
|
|
addr &= chipmem_bank.mask;
|
|
ce2_timeout ();
|
|
return chipmem_bank.baseaddr[addr];
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_lput_ce2 (uaecptr addr, uae_u32 l)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
ce2_timeout ();
|
|
do_put_mem_long (m, l);
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_wput_ce2 (uaecptr addr, uae_u32 w)
|
|
{
|
|
uae_u16 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
ce2_timeout ();
|
|
do_put_mem_word (m, w);
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_bput_ce2 (uaecptr addr, uae_u32 b)
|
|
{
|
|
addr &= chipmem_bank.mask;
|
|
ce2_timeout ();
|
|
chipmem_bank.baseaddr[addr] = b;
|
|
}
|
|
|
|
#endif
|
|
|
|
static uae_u32 REGPARAM2 chipmem_lget (uaecptr addr)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
return do_get_mem_long (m);
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_wget (uaecptr addr)
|
|
{
|
|
uae_u16 *m, v;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
v = do_get_mem_word (m);
|
|
return v;
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_bget (uaecptr addr)
|
|
{
|
|
uae_u8 v;
|
|
addr &= chipmem_bank.mask;
|
|
v = chipmem_bank.baseaddr[addr];
|
|
return v;
|
|
}
|
|
|
|
void REGPARAM2 chipmem_lput (uaecptr addr, uae_u32 l)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
do_put_mem_long (m, l);
|
|
}
|
|
|
|
void REGPARAM2 chipmem_wput (uaecptr addr, uae_u32 w)
|
|
{
|
|
uae_u16 *m;
|
|
|
|
addr &= chipmem_bank.mask;
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
do_put_mem_word (m, w);
|
|
}
|
|
|
|
void REGPARAM2 chipmem_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
addr &= chipmem_bank.mask;
|
|
chipmem_bank.baseaddr[addr] = b;
|
|
}
|
|
|
|
/* cpu chipmem access inside agnus addressable ram but no ram available */
|
|
static uae_u32 chipmem_dummy (void)
|
|
{
|
|
/* not really right but something random that has more ones than zeros.. */
|
|
return 0xffff & ~((1 << (uaerand () & 31)) | (1 << (uaerand () & 31)));
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_dummy_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
}
|
|
static void REGPARAM2 chipmem_dummy_wput (uaecptr addr, uae_u32 b)
|
|
{
|
|
}
|
|
static void REGPARAM2 chipmem_dummy_lput (uaecptr addr, uae_u32 b)
|
|
{
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_dummy_bget (uaecptr addr)
|
|
{
|
|
return chipmem_dummy ();
|
|
}
|
|
static uae_u32 REGPARAM2 chipmem_dummy_wget (uaecptr addr)
|
|
{
|
|
return chipmem_dummy ();
|
|
}
|
|
static uae_u32 REGPARAM2 chipmem_dummy_lget (uaecptr addr)
|
|
{
|
|
return (chipmem_dummy () << 16) | chipmem_dummy ();
|
|
}
|
|
|
|
static uae_u32 chipmem_noise(uae_u32 addr)
|
|
{
|
|
// not yet implemented
|
|
return 0;
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_agnus_lget (uaecptr addr)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size - 3)
|
|
return chipmem_noise(addr);
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
return do_get_mem_long (m);
|
|
}
|
|
|
|
uae_u32 REGPARAM2 chipmem_agnus_wget (uaecptr addr)
|
|
{
|
|
uae_u16 *m;
|
|
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size - 1)
|
|
return chipmem_noise(addr);
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
return do_get_mem_word (m);
|
|
}
|
|
|
|
static uae_u32 REGPARAM2 chipmem_agnus_bget (uaecptr addr)
|
|
{
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size)
|
|
return chipmem_noise(addr);
|
|
return chipmem_bank.baseaddr[addr];
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_agnus_lput (uaecptr addr, uae_u32 l)
|
|
{
|
|
uae_u32 *m;
|
|
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size - 3)
|
|
return;
|
|
m = (uae_u32 *)(chipmem_bank.baseaddr + addr);
|
|
do_put_mem_long (m, l);
|
|
}
|
|
|
|
void REGPARAM2 chipmem_agnus_wput (uaecptr addr, uae_u32 w)
|
|
{
|
|
uae_u16 *m;
|
|
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size - 1)
|
|
return;
|
|
m = (uae_u16 *)(chipmem_bank.baseaddr + addr);
|
|
do_put_mem_word (m, w);
|
|
}
|
|
|
|
static void REGPARAM2 chipmem_agnus_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
addr &= chipmem_full_mask;
|
|
if (addr >= chipmem_full_size)
|
|
return;
|
|
chipmem_bank.baseaddr[addr] = b;
|
|
}
|
|
|
|
static int REGPARAM2 chipmem_check (uaecptr addr, uae_u32 size)
|
|
{
|
|
// Check if chip ram is in two "banks" (ECS 0.5m+0.5m config)
|
|
if (((addr & ~chipmem_bank.mask) & chipmem_full_mask) || (((addr + size) & ~chipmem_bank.mask) & chipmem_full_mask)) {
|
|
return 0;
|
|
}
|
|
addr &= chipmem_bank.mask;
|
|
return (addr + size) <= chipmem_full_size;
|
|
}
|
|
|
|
static uae_u8 *REGPARAM2 chipmem_xlate (uaecptr addr)
|
|
{
|
|
addr &= chipmem_bank.mask;
|
|
return chipmem_bank.baseaddr + addr;
|
|
}
|
|
|
|
STATIC_INLINE void REGPARAM2 chipmem_lput_bigmem (uaecptr addr, uae_u32 v)
|
|
{
|
|
put_long (addr, v);
|
|
}
|
|
STATIC_INLINE void REGPARAM2 chipmem_wput_bigmem (uaecptr addr, uae_u32 v)
|
|
{
|
|
put_word (addr, v);
|
|
}
|
|
STATIC_INLINE void REGPARAM2 chipmem_bput_bigmem (uaecptr addr, uae_u32 v)
|
|
{
|
|
put_byte (addr, v);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_lget_bigmem (uaecptr addr)
|
|
{
|
|
return get_long (addr);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_wget_bigmem (uaecptr addr)
|
|
{
|
|
return get_word (addr);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_bget_bigmem (uaecptr addr)
|
|
{
|
|
return get_byte (addr);
|
|
}
|
|
STATIC_INLINE int REGPARAM2 chipmem_check_bigmem (uaecptr addr, uae_u32 size)
|
|
{
|
|
return valid_address (addr, size);
|
|
}
|
|
STATIC_INLINE uae_u8* REGPARAM2 chipmem_xlate_bigmem (uaecptr addr)
|
|
{
|
|
return get_real_address (addr);
|
|
}
|
|
|
|
STATIC_INLINE void REGPARAM2 chipmem_lput_debugmem(uaecptr addr, uae_u32 v)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
debugmem_chiphit(addr, v, 4);
|
|
put_long(addr, v);
|
|
}
|
|
STATIC_INLINE void REGPARAM2 chipmem_wput_debugmem(uaecptr addr, uae_u32 v)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
debugmem_chiphit(addr, v, 2);
|
|
put_word(addr, v);
|
|
}
|
|
STATIC_INLINE void REGPARAM2 chipmem_bput_debugmem(uaecptr addr, uae_u32 v)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
debugmem_chiphit(addr, v, 1);
|
|
put_byte(addr, v);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_lget_debugmem(uaecptr addr)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
return debugmem_chiphit(addr, 0, -4);
|
|
return get_long(addr);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_wget_debugmem(uaecptr addr)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
return debugmem_chiphit(addr, 0, -2);
|
|
return get_word(addr);
|
|
}
|
|
STATIC_INLINE uae_u32 REGPARAM2 chipmem_bget_debugmem(uaecptr addr)
|
|
{
|
|
if (addr < debugmem_chiplimit)
|
|
return debugmem_chiphit(addr, 0, -1);
|
|
return get_byte(addr);
|
|
}
|
|
STATIC_INLINE int REGPARAM2 chipmem_check_debugmem(uaecptr addr, uae_u32 size)
|
|
{
|
|
return valid_address(addr, size);
|
|
}
|
|
STATIC_INLINE uae_u8* REGPARAM2 chipmem_xlate_debugmem(uaecptr addr)
|
|
{
|
|
return get_real_address(addr);
|
|
}
|
|
|
|
uae_u32 (REGPARAM2 *chipmem_lget_indirect)(uaecptr);
|
|
uae_u32 (REGPARAM2 *chipmem_wget_indirect)(uaecptr);
|
|
uae_u32 (REGPARAM2 *chipmem_bget_indirect)(uaecptr);
|
|
void (REGPARAM2 *chipmem_lput_indirect)(uaecptr, uae_u32);
|
|
void (REGPARAM2 *chipmem_wput_indirect)(uaecptr, uae_u32);
|
|
void (REGPARAM2 *chipmem_bput_indirect)(uaecptr, uae_u32);
|
|
int (REGPARAM2 *chipmem_check_indirect)(uaecptr, uae_u32);
|
|
uae_u8 *(REGPARAM2 *chipmem_xlate_indirect)(uaecptr);
|
|
|
|
void chipmem_setindirect(void)
|
|
{
|
|
if (debugmem_bank.baseaddr && debugmem_chiplimit) {
|
|
chipmem_lget_indirect = chipmem_lget_debugmem;
|
|
chipmem_wget_indirect = chipmem_wget_debugmem;
|
|
chipmem_bget_indirect = chipmem_bget_debugmem;
|
|
chipmem_lput_indirect = chipmem_lput_debugmem;
|
|
chipmem_wput_indirect = chipmem_wput_debugmem;
|
|
chipmem_bput_indirect = chipmem_bput_debugmem;
|
|
chipmem_check_indirect = chipmem_check_bigmem;
|
|
chipmem_xlate_indirect = chipmem_xlate_bigmem;
|
|
} else if (currprefs.z3chipmem.size) {
|
|
chipmem_lget_indirect = chipmem_lget_bigmem;
|
|
chipmem_wget_indirect = chipmem_wget_bigmem;
|
|
chipmem_bget_indirect = chipmem_bget_bigmem;
|
|
chipmem_lput_indirect = chipmem_lput_bigmem;
|
|
chipmem_wput_indirect = chipmem_wput_bigmem;
|
|
chipmem_bput_indirect = chipmem_bput_bigmem;
|
|
chipmem_check_indirect = chipmem_check_bigmem;
|
|
chipmem_xlate_indirect = chipmem_xlate_bigmem;
|
|
} else {
|
|
chipmem_lget_indirect = chipmem_lget;
|
|
chipmem_wget_indirect = chipmem_agnus_wget;
|
|
chipmem_bget_indirect = chipmem_agnus_bget;
|
|
chipmem_lput_indirect = chipmem_lput;
|
|
chipmem_wput_indirect = chipmem_agnus_wput;
|
|
chipmem_bput_indirect = chipmem_agnus_bput;
|
|
chipmem_check_indirect = chipmem_check;
|
|
chipmem_xlate_indirect = chipmem_xlate;
|
|
}
|
|
}
|
|
|
|
/* Slow memory */
|
|
|
|
MEMORY_FUNCTIONS(bogomem);
|
|
|
|
/* A3000 motherboard fast memory */
|
|
|
|
MEMORY_FUNCTIONS(a3000lmem);
|
|
MEMORY_FUNCTIONS(a3000hmem);
|
|
|
|
/* 25bit memory (0x01000000) */
|
|
|
|
MEMORY_FUNCTIONS(mem25bit);
|
|
|
|
/* debugger memory */
|
|
|
|
MEMORY_FUNCTIONS(debugmem);
|
|
MEMORY_WGETI(debugmem);
|
|
MEMORY_LGETI(debugmem);
|
|
|
|
/* Kick memory */
|
|
|
|
uae_u16 kickstart_version;
|
|
|
|
/*
|
|
* A1000 kickstart RAM handling
|
|
*
|
|
* RESET instruction unhides boot ROM and disables write protection
|
|
* write access to boot ROM hides boot ROM and enables write protection
|
|
*
|
|
*/
|
|
static int a1000_kickstart_mode;
|
|
static uae_u8 *a1000_bootrom;
|
|
static void a1000_handle_kickstart (int mode)
|
|
{
|
|
if (!a1000_bootrom)
|
|
return;
|
|
protect_roms (false);
|
|
if (mode == 0) {
|
|
a1000_kickstart_mode = 0;
|
|
memcpy (kickmem_bank.baseaddr, kickmem_bank.baseaddr + ROM_SIZE_256, ROM_SIZE_256);
|
|
kickstart_version = (kickmem_bank.baseaddr[ROM_SIZE_256 + 12] << 8) | kickmem_bank.baseaddr[ROM_SIZE_256 + 13];
|
|
} else {
|
|
a1000_kickstart_mode = 1;
|
|
memcpy (kickmem_bank.baseaddr, a1000_bootrom, ROM_SIZE_256);
|
|
kickstart_version = 0;
|
|
}
|
|
if (kickstart_version == 0xffff)
|
|
kickstart_version = 0;
|
|
}
|
|
|
|
void a1000_reset (void)
|
|
{
|
|
a1000_handle_kickstart (1);
|
|
}
|
|
|
|
static void REGPARAM3 kickmem_lput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 kickmem_wput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 kickmem_bput (uaecptr, uae_u32) REGPARAM;
|
|
|
|
MEMORY_BGET(kickmem);
|
|
MEMORY_WGET(kickmem);
|
|
MEMORY_LGET(kickmem);
|
|
MEMORY_CHECK(kickmem);
|
|
MEMORY_XLATE(kickmem);
|
|
|
|
static void REGPARAM2 kickmem_lput (uaecptr addr, uae_u32 b)
|
|
{
|
|
uae_u32 *m;
|
|
if (currprefs.rom_readwrite && rom_write_enabled) {
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u32 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_long (m, b);
|
|
#if 0
|
|
if (addr == ROM_SIZE_512-4) {
|
|
rom_write_enabled = false;
|
|
write_log (_T("ROM write disabled\n"));
|
|
}
|
|
#endif
|
|
} else if (a1000_kickstart_mode) {
|
|
if (addr >= 0xfc0000) {
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u32 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_long (m, b);
|
|
return;
|
|
} else
|
|
a1000_handle_kickstart (0);
|
|
} else if (currprefs.illegal_mem) {
|
|
write_log (_T("Illegal kickmem lput at %08x PC=%08x\n"), addr, M68K_GETPC);
|
|
}
|
|
}
|
|
|
|
static void REGPARAM2 kickmem_wput (uaecptr addr, uae_u32 b)
|
|
{
|
|
uae_u16 *m;
|
|
if (currprefs.rom_readwrite && rom_write_enabled) {
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u16 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_word (m, b);
|
|
} else if (a1000_kickstart_mode) {
|
|
if (addr >= 0xfc0000) {
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u16 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_word (m, b);
|
|
return;
|
|
} else
|
|
a1000_handle_kickstart (0);
|
|
} else if (currprefs.illegal_mem) {
|
|
write_log (_T("Illegal kickmem wput at %08x PC=%08x\n"), addr, M68K_GETPC);
|
|
}
|
|
}
|
|
|
|
static void REGPARAM2 kickmem_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.rom_readwrite && rom_write_enabled) {
|
|
addr &= kickmem_bank.mask;
|
|
kickmem_bank.baseaddr[addr] = b;
|
|
} else if (a1000_kickstart_mode) {
|
|
if (addr >= 0xfc0000) {
|
|
addr &= kickmem_bank.mask;
|
|
kickmem_bank.baseaddr[addr] = b;
|
|
return;
|
|
} else
|
|
a1000_handle_kickstart (0);
|
|
} else if (currprefs.illegal_mem) {
|
|
write_log (_T("Illegal kickmem bput at %08x PC=%08x\n"), addr, M68K_GETPC);
|
|
}
|
|
}
|
|
|
|
static void REGPARAM2 kickmem2_lput (uaecptr addr, uae_u32 l)
|
|
{
|
|
uae_u32 *m;
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u32 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_long (m, l);
|
|
}
|
|
|
|
static void REGPARAM2 kickmem2_wput (uaecptr addr, uae_u32 w)
|
|
{
|
|
uae_u16 *m;
|
|
addr &= kickmem_bank.mask;
|
|
m = (uae_u16 *)(kickmem_bank.baseaddr + addr);
|
|
do_put_mem_word (m, w);
|
|
}
|
|
|
|
static void REGPARAM2 kickmem2_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
addr &= kickmem_bank.mask;
|
|
kickmem_bank.baseaddr[addr] = b;
|
|
}
|
|
|
|
/* CD32/CDTV extended kick memory */
|
|
|
|
static int extendedkickmem_type;
|
|
|
|
#define EXTENDED_ROM_CD32 1
|
|
#define EXTENDED_ROM_CDTV 2
|
|
#define EXTENDED_ROM_KS 3
|
|
#define EXTENDED_ROM_ARCADIA 4
|
|
#define EXTENDED_ROM_ALG 5
|
|
|
|
static void REGPARAM3 extendedkickmem_lput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem_wput (uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem_bput (uaecptr, uae_u32) REGPARAM;
|
|
|
|
MEMORY_BGET(extendedkickmem);
|
|
MEMORY_WGET(extendedkickmem);
|
|
MEMORY_LGET(extendedkickmem);
|
|
MEMORY_CHECK(extendedkickmem);
|
|
MEMORY_XLATE(extendedkickmem);
|
|
|
|
static void REGPARAM2 extendedkickmem_lput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem lput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem_wput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem wput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem lput at %08x\n"), addr);
|
|
}
|
|
|
|
|
|
static void REGPARAM3 extendedkickmem2a_lput(uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem2a_wput(uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem2a_bput(uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem2b_lput(uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem2b_wput(uaecptr, uae_u32) REGPARAM;
|
|
static void REGPARAM3 extendedkickmem2b_bput(uaecptr, uae_u32) REGPARAM;
|
|
|
|
MEMORY_BGET(extendedkickmem2a);
|
|
MEMORY_WGET(extendedkickmem2a);
|
|
MEMORY_LGET(extendedkickmem2a);
|
|
MEMORY_CHECK(extendedkickmem2a);
|
|
MEMORY_XLATE(extendedkickmem2a);
|
|
|
|
MEMORY_BGET(extendedkickmem2b);
|
|
MEMORY_WGET(extendedkickmem2b);
|
|
MEMORY_LGET(extendedkickmem2b);
|
|
MEMORY_CHECK(extendedkickmem2b);
|
|
MEMORY_XLATE(extendedkickmem2b);
|
|
|
|
static void REGPARAM2 extendedkickmem2a_lput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem2a lput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem2a_wput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem2a wput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem2a_bput (uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log (_T("Illegal extendedkickmem2a lput at %08x\n"), addr);
|
|
}
|
|
|
|
static void REGPARAM2 extendedkickmem2b_lput(uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log(_T("Illegal extendedkickmem2b lput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem2b_wput(uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log(_T("Illegal extendedkickmem2b wput at %08x\n"), addr);
|
|
}
|
|
static void REGPARAM2 extendedkickmem2b_bput(uaecptr addr, uae_u32 b)
|
|
{
|
|
if (currprefs.illegal_mem)
|
|
write_log(_T("Illegal extendedkickmem2b lput at %08x\n"), addr);
|
|
}
|
|
|
|
|
|
/* Default memory access functions */
|
|
|
|
int REGPARAM2 default_check (uaecptr a, uae_u32 b)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int be_cnt, be_recursive;
|
|
|
|
uae_u8 *REGPARAM2 default_xlate (uaecptr addr)
|
|
{
|
|
if (be_recursive) {
|
|
cpu_halt(CPU_HALT_OPCODE_FETCH_FROM_NON_EXISTING_ADDRESS);
|
|
return kickmem_xlate(2);
|
|
}
|
|
|
|
if (maybe_map_boot_rom(addr))
|
|
return get_real_address(addr);
|
|
|
|
be_recursive++;
|
|
int size = currprefs.cpu_model >= 68020 ? 4 : 2;
|
|
if (quit_program == 0) {
|
|
/* do this only in 68010+ mode, there are some tricky A500 programs.. */
|
|
if ((currprefs.cpu_model > 68000 || !currprefs.cpu_compatible) && !currprefs.mmu_model) {
|
|
#if defined(ENFORCER)
|
|
enforcer_disable ();
|
|
#endif
|
|
if (be_cnt < 3) {
|
|
int i, j;
|
|
uaecptr a2 = addr - 32;
|
|
uaecptr a3 = m68k_getpc () - 32;
|
|
write_log (_T("Your Amiga program just did something terribly stupid %08X PC=%08X\n"), addr, M68K_GETPC);
|
|
if (debugging || DEBUG_STUPID) {
|
|
activate_debugger ();
|
|
m68k_dumpstate(NULL, 0xffffffff);
|
|
}
|
|
for (i = 0; i < 10; i++) {
|
|
write_log (_T("%08X "), i >= 5 ? a3 : a2);
|
|
for (j = 0; j < 16; j += 2) {
|
|
write_log (_T(" %04X"), get_word (i >= 5 ? a3 : a2));
|
|
if (i >= 5) a3 += 2; else a2 += 2;
|
|
}
|
|
write_log (_T("\n"));
|
|
}
|
|
memory_map_dump();
|
|
m68k_dumpstate(NULL, 0xffffffff);
|
|
}
|
|
if (0 || (gary_toenb && (gary_nonrange(addr) || (size > 1 && gary_nonrange(addr + size - 1))))) {
|
|
hardware_exception2(addr, 0, true, true, size);
|
|
} else {
|
|
cpu_halt(CPU_HALT_OPCODE_FETCH_FROM_NON_EXISTING_ADDRESS);
|
|
}
|
|
}
|
|
}
|
|
be_recursive--;
|
|
return kickmem_xlate (2); /* So we don't crash. */
|
|
}
|
|
|
|
/* Address banks */
|
|
|
|
addrbank dummy_bank = {
|
|
dummy_lget, dummy_wget, dummy_bget,
|
|
dummy_lput, dummy_wput, dummy_bput,
|
|
default_xlate, dummy_check, NULL, NULL, NULL,
|
|
dummy_lgeti, dummy_wgeti,
|
|
ABFLAG_NONE, S_READ, S_WRITE
|
|
};
|
|
|
|
addrbank ones_bank = {
|
|
ones_get, ones_get, ones_get,
|
|
none_put, none_put, none_put,
|
|
default_xlate, dummy_check, NULL, NULL, _T("Ones"),
|
|
dummy_lgeti, dummy_wgeti,
|
|
ABFLAG_NONE, S_READ, S_WRITE
|
|
};
|
|
|
|
addrbank chipmem_bank = {
|
|
chipmem_lget, chipmem_wget, chipmem_bget,
|
|
chipmem_lput, chipmem_wput, chipmem_bput,
|
|
chipmem_xlate, chipmem_check, NULL, _T("chip"), _T("Chip memory"),
|
|
chipmem_lget, chipmem_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CHIPRAM | ABFLAG_CACHE_ENABLE_BOTH, 0, 0
|
|
};
|
|
|
|
addrbank chipmem_dummy_bank = {
|
|
chipmem_dummy_lget, chipmem_dummy_wget, chipmem_dummy_bget,
|
|
chipmem_dummy_lput, chipmem_dummy_wput, chipmem_dummy_bput,
|
|
default_xlate, dummy_check, NULL, NULL, _T("Dummy Chip memory"),
|
|
dummy_lgeti, dummy_wgeti,
|
|
ABFLAG_IO | ABFLAG_CHIPRAM, S_READ, S_WRITE
|
|
};
|
|
|
|
|
|
#ifdef AGA
|
|
addrbank chipmem_bank_ce2 = {
|
|
chipmem_lget_ce2, chipmem_wget_ce2, chipmem_bget_ce2,
|
|
chipmem_lput_ce2, chipmem_wput_ce2, chipmem_bput_ce2,
|
|
chipmem_xlate, chipmem_check, NULL, NULL, _T("Chip memory (68020 'ce')"),
|
|
chipmem_lget_ce2, chipmem_wget_ce2,
|
|
ABFLAG_RAM | ABFLAG_CHIPRAM | ABFLAG_CACHE_ENABLE_BOTH, S_READ, S_WRITE
|
|
};
|
|
#endif
|
|
|
|
addrbank bogomem_bank = {
|
|
bogomem_lget, bogomem_wget, bogomem_bget,
|
|
bogomem_lput, bogomem_wput, bogomem_bput,
|
|
bogomem_xlate, bogomem_check, NULL, _T("bogo"), _T("Slow memory"),
|
|
bogomem_lget, bogomem_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_BOTH, 0, 0
|
|
};
|
|
|
|
addrbank mem25bit_bank = {
|
|
mem25bit_lget, mem25bit_wget, mem25bit_bget,
|
|
mem25bit_lput, mem25bit_wput, mem25bit_bput,
|
|
mem25bit_xlate, mem25bit_check, NULL, _T("25bitmem"), _T("25bit memory"),
|
|
mem25bit_lget, mem25bit_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL, 0, 0
|
|
};
|
|
|
|
addrbank debugmem_bank = {
|
|
debugmem_lget, debugmem_wget, debugmem_bget,
|
|
debugmem_lput, debugmem_wput, debugmem_bput,
|
|
debugmem_xlate, debugmem_check, NULL, _T("debugmem"), _T("debugger memory"),
|
|
debugmem_lgeti, debugmem_wgeti,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL, 0, 0
|
|
};
|
|
|
|
addrbank a3000lmem_bank = {
|
|
a3000lmem_lget, a3000lmem_wget, a3000lmem_bget,
|
|
a3000lmem_lput, a3000lmem_wput, a3000lmem_bput,
|
|
a3000lmem_xlate, a3000lmem_check, NULL, _T("ramsey_low"), _T("RAMSEY memory (low)"),
|
|
a3000lmem_lget, a3000lmem_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, 0
|
|
};
|
|
|
|
addrbank a3000hmem_bank = {
|
|
a3000hmem_lget, a3000hmem_wget, a3000hmem_bget,
|
|
a3000hmem_lput, a3000hmem_wput, a3000hmem_bput,
|
|
a3000hmem_xlate, a3000hmem_check, NULL, _T("ramsey_high"), _T("RAMSEY memory (high)"),
|
|
a3000hmem_lget, a3000hmem_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, 0
|
|
};
|
|
|
|
addrbank kickmem_bank = {
|
|
kickmem_lget, kickmem_wget, kickmem_bget,
|
|
kickmem_lput, kickmem_wput, kickmem_bput,
|
|
kickmem_xlate, kickmem_check, NULL, _T("kick"), _T("Kickstart ROM"),
|
|
kickmem_lget, kickmem_wget,
|
|
ABFLAG_ROM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, S_WRITE
|
|
};
|
|
|
|
addrbank kickram_bank = {
|
|
kickmem_lget, kickmem_wget, kickmem_bget,
|
|
kickmem2_lput, kickmem2_wput, kickmem2_bput,
|
|
kickmem_xlate, kickmem_check, NULL, NULL, _T("Kickstart Shadow RAM"),
|
|
kickmem_lget, kickmem_wget,
|
|
ABFLAG_UNK | ABFLAG_SAFE | ABFLAG_CACHE_ENABLE_ALL, 0, S_WRITE
|
|
};
|
|
|
|
addrbank extendedkickmem_bank = {
|
|
extendedkickmem_lget, extendedkickmem_wget, extendedkickmem_bget,
|
|
extendedkickmem_lput, extendedkickmem_wput, extendedkickmem_bput,
|
|
extendedkickmem_xlate, extendedkickmem_check, NULL, NULL, _T("Extended Kickstart ROM"),
|
|
extendedkickmem_lget, extendedkickmem_wget,
|
|
ABFLAG_ROM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, S_WRITE
|
|
};
|
|
addrbank extendedkickmem2a_bank = {
|
|
extendedkickmem2a_lget, extendedkickmem2a_wget, extendedkickmem2a_bget,
|
|
extendedkickmem2a_lput, extendedkickmem2a_wput, extendedkickmem2a_bput,
|
|
extendedkickmem2a_xlate, extendedkickmem2a_check, NULL, _T("rom_a8"), _T("Extended 2nd Kickstart ROM"),
|
|
extendedkickmem2a_lget, extendedkickmem2a_wget,
|
|
ABFLAG_ROM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, S_WRITE
|
|
};
|
|
addrbank extendedkickmem2b_bank = {
|
|
extendedkickmem2b_lget, extendedkickmem2b_wget, extendedkickmem2b_bget,
|
|
extendedkickmem2b_lput, extendedkickmem2b_wput, extendedkickmem2b_bput,
|
|
extendedkickmem2b_xlate, extendedkickmem2b_check, NULL, _T("rom_b0"), _T("Extended 3nd Kickstart ROM"),
|
|
extendedkickmem2b_lget, extendedkickmem2b_wget,
|
|
ABFLAG_ROM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, S_WRITE
|
|
};
|
|
addrbank fakeuaebootrom_bank = {
|
|
fakeuaebootrom_lget, fakeuaebootrom_wget, mem25bit_bget,
|
|
fakeuaebootrom_lput, fakeuaebootrom_wput, mem25bit_bput,
|
|
fakeuaebootrom_xlate, fakeuaebootrom_check, NULL, _T("*"), _T("fakeuaerom"),
|
|
fakeuaebootrom_lget, fakeuaebootrom_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE, 0, 0
|
|
};
|
|
|
|
MEMORY_FUNCTIONS(custmem1);
|
|
MEMORY_FUNCTIONS(custmem2);
|
|
|
|
addrbank custmem1_bank = {
|
|
custmem1_lget, custmem1_wget, custmem1_bget,
|
|
custmem1_lput, custmem1_wput, custmem1_bput,
|
|
custmem1_xlate, custmem1_check, NULL, _T("custmem1"), _T("Non-autoconfig RAM #1"),
|
|
custmem1_lget, custmem1_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, 0
|
|
};
|
|
addrbank custmem2_bank = {
|
|
custmem2_lget, custmem2_wget, custmem2_bget,
|
|
custmem2_lput, custmem2_wput, custmem2_bput,
|
|
custmem2_xlate, custmem2_check, NULL, _T("custmem2"), _T("Non-autoconfig RAM #2"),
|
|
custmem2_lget, custmem2_wget,
|
|
ABFLAG_RAM | ABFLAG_THREADSAFE | ABFLAG_CACHE_ENABLE_ALL | ABFLAG_DIRECTACCESS, 0, 0
|
|
};
|
|
|
|
static bool singlebit(uae_u32 v)
|
|
{
|
|
while (v && !(v & 1))
|
|
v >>= 1;
|
|
return (v & ~1) == 0;
|
|
}
|
|
|
|
static void allocate_memory_custombanks(void)
|
|
{
|
|
if (custmem1_bank.reserved_size != currprefs.custom_memory_sizes[0]) {
|
|
mapped_free(&custmem1_bank);
|
|
custmem1_bank.reserved_size = currprefs.custom_memory_sizes[0];
|
|
// custmem1 and 2 can have non-power of 2 size so only set correct mask if size is power of 2.
|
|
custmem1_bank.mask = singlebit(custmem1_bank.reserved_size) ? custmem1_bank.reserved_size - 1 : -1;
|
|
custmem1_bank.start = currprefs.custom_memory_addrs[0];
|
|
if (custmem1_bank.reserved_size) {
|
|
if (!mapped_malloc(&custmem1_bank))
|
|
custmem1_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
if (custmem2_bank.reserved_size != currprefs.custom_memory_sizes[1]) {
|
|
mapped_free(&custmem2_bank);
|
|
custmem2_bank.reserved_size = currprefs.custom_memory_sizes[1];
|
|
custmem2_bank.mask = singlebit(custmem2_bank.reserved_size) ? custmem2_bank.reserved_size - 1 : -1;
|
|
custmem2_bank.start = currprefs.custom_memory_addrs[1];
|
|
if (custmem2_bank.reserved_size) {
|
|
if (!mapped_malloc(&custmem2_bank))
|
|
custmem2_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
#define fkickmem_size ROM_SIZE_512
|
|
static int a3000_f0;
|
|
void a3000_fakekick (int map)
|
|
{
|
|
static uae_u8 *kickstore;
|
|
|
|
protect_roms (false);
|
|
if (map) {
|
|
uae_u8 *fkickmemory = a3000lmem_bank.baseaddr + a3000lmem_bank.reserved_size - fkickmem_size;
|
|
if (fkickmemory[2] == 0x4e && fkickmemory[3] == 0xf9 && fkickmemory[4] == 0x00) {
|
|
if (!kickstore)
|
|
kickstore = xmalloc (uae_u8, fkickmem_size);
|
|
memcpy (kickstore, kickmem_bank.baseaddr, fkickmem_size);
|
|
if (fkickmemory[5] == 0xfc) {
|
|
memcpy (kickmem_bank.baseaddr, fkickmemory, fkickmem_size / 2);
|
|
memcpy (kickmem_bank.baseaddr + fkickmem_size / 2, fkickmemory, fkickmem_size / 2);
|
|
extendedkickmem_bank.reserved_size = 65536;
|
|
extendedkickmem_bank.label = _T("rom_f0");
|
|
extendedkickmem_bank.mask = extendedkickmem_bank.reserved_size - 1;
|
|
mapped_malloc (&extendedkickmem_bank);
|
|
memcpy (extendedkickmem_bank.baseaddr, fkickmemory + fkickmem_size / 2, 65536);
|
|
map_banks (&extendedkickmem_bank, 0xf0, 1, 1);
|
|
a3000_f0 = 1;
|
|
} else {
|
|
memcpy (kickmem_bank.baseaddr, fkickmemory, fkickmem_size);
|
|
}
|
|
}
|
|
} else {
|
|
if (a3000_f0) {
|
|
map_banks (&dummy_bank, 0xf0, 1, 1);
|
|
mapped_free (&extendedkickmem_bank);
|
|
a3000_f0 = 0;
|
|
}
|
|
if (kickstore)
|
|
memcpy (kickmem_bank.baseaddr, kickstore, fkickmem_size);
|
|
xfree (kickstore);
|
|
kickstore = NULL;
|
|
}
|
|
protect_roms (true);
|
|
}
|
|
|
|
static bool is_alg_rom(const TCHAR *name)
|
|
{
|
|
struct romdata *rd = getromdatabypath(name);
|
|
if (!rd)
|
|
return false;
|
|
return (rd->type & ROMTYPE_ALG) != 0;
|
|
}
|
|
|
|
static void descramble_alg(uae_u8 *data, int size)
|
|
{
|
|
uae_u8 *tbuf = xmalloc(uae_u8, size);
|
|
memcpy(tbuf, data, size);
|
|
for (int mode = 0; mode < 3; mode++) {
|
|
if ((data[8] == 0x4a && data[9] == 0xfc))
|
|
break;
|
|
for (int s = 0; s < size; s++) {
|
|
int d = s;
|
|
if (mode == 0) {
|
|
if (s & 0x2000)
|
|
d ^= 0x1000;
|
|
if (s & 0x8000)
|
|
d ^= 0x4000;
|
|
} else if (mode == 1) {
|
|
if (s & 0x2000)
|
|
d ^= 0x1000;
|
|
} else {
|
|
if ((~s) & 0x2000)
|
|
d ^= 0x1000;
|
|
if (s & 0x8000)
|
|
d ^= 0x4000;
|
|
d ^= 0x20000;
|
|
}
|
|
data[d] = tbuf[s];
|
|
}
|
|
}
|
|
xfree(tbuf);
|
|
}
|
|
|
|
static const uae_char *kickstring = "exec.library";
|
|
|
|
static int read_kickstart (struct zfile *f, uae_u8 *mem, int size, int dochecksum, int noalias)
|
|
{
|
|
uae_char buffer[11];
|
|
int i, j, oldpos;
|
|
int cr = 0, kickdisk = 0;
|
|
|
|
if (size < 0) {
|
|
zfile_fseek (f, 0, SEEK_END);
|
|
size = zfile_ftell32(f) & ~0x3ff;
|
|
zfile_fseek (f, 0, SEEK_SET);
|
|
}
|
|
oldpos = zfile_ftell32(f);
|
|
i = zfile_fread (buffer, 1, sizeof(buffer), f);
|
|
if (i < sizeof(buffer))
|
|
return 0;
|
|
if (!memcmp (buffer, "KICK", 4)) {
|
|
zfile_fseek (f, 512, SEEK_SET);
|
|
kickdisk = 1;
|
|
#if 0
|
|
} else if (size >= ROM_SIZE_512 && !memcmp (buffer, "AMIG", 4)) {
|
|
/* ReKick */
|
|
zfile_fseek (f, oldpos + 0x6c, SEEK_SET);
|
|
cr = 2;
|
|
#endif
|
|
} else if (memcmp ((uae_char*)buffer, "AMIROMTYPE1", 11) != 0) {
|
|
zfile_fseek (f, oldpos, SEEK_SET);
|
|
} else {
|
|
cloanto_rom = 1;
|
|
cr = 1;
|
|
}
|
|
|
|
memset (mem, 0, size);
|
|
if (size >= 131072) {
|
|
for (i = 0; i < 8; i++) {
|
|
mem[size - 16 + i * 2 + 1] = 0x18 + i;
|
|
}
|
|
mem[size - 20] = size >> 24;
|
|
mem[size - 19] = size >> 16;
|
|
mem[size - 18] = size >> 8;
|
|
mem[size - 17] = size >> 0;
|
|
}
|
|
|
|
i = zfile_fread (mem, 1, size, f);
|
|
|
|
if (kickdisk && i > ROM_SIZE_256)
|
|
i = ROM_SIZE_256;
|
|
#if 0
|
|
if (i >= ROM_SIZE_256 && (i != ROM_SIZE_256 && i != ROM_SIZE_512 && i != ROM_SIZE_512 * 2 && i != ROM_SIZE_512 * 4)) {
|
|
notify_user (NUMSG_KSROMREADERROR);
|
|
return 0;
|
|
}
|
|
#endif
|
|
if (i < size - 20)
|
|
kickstart_fix_checksum (mem, size);
|
|
j = 1;
|
|
while (j < i)
|
|
j <<= 1;
|
|
i = j;
|
|
|
|
if (!noalias && i == size / 2)
|
|
memcpy (mem + size / 2, mem, size / 2);
|
|
|
|
if (cr) {
|
|
if (!decode_rom (mem, size, cr, i))
|
|
return 0;
|
|
}
|
|
|
|
if (size <= 256)
|
|
return size;
|
|
|
|
if (currprefs.cs_a1000ram && i < ROM_SIZE_256) {
|
|
int off = 0;
|
|
if (!a1000_bootrom)
|
|
a1000_bootrom = xcalloc (uae_u8, ROM_SIZE_256);
|
|
while (off + i < ROM_SIZE_256) {
|
|
memcpy (a1000_bootrom + off, kickmem_bank.baseaddr, i);
|
|
off += i;
|
|
}
|
|
memset (kickmem_bank.baseaddr, 0, kickmem_bank.allocated_size);
|
|
a1000_handle_kickstart (1);
|
|
dochecksum = 0;
|
|
i = ROM_SIZE_512;
|
|
}
|
|
|
|
for (j = 0; j < 256 && i >= ROM_SIZE_256; j++) {
|
|
if (!memcmp (mem + j, kickstring, strlen (kickstring) + 1))
|
|
break;
|
|
}
|
|
|
|
if (j == 256 || i < ROM_SIZE_256)
|
|
dochecksum = 0;
|
|
if (dochecksum)
|
|
kickstart_checksum (mem, size);
|
|
return i;
|
|
}
|
|
|
|
static bool load_extendedkickstart (const TCHAR *romextfile, int type)
|
|
{
|
|
struct zfile *f;
|
|
int size, off;
|
|
bool ret = false;
|
|
|
|
if (_tcslen (romextfile) == 0)
|
|
return false;
|
|
if (is_arcadia_rom (romextfile) == ARCADIA_BIOS) {
|
|
extendedkickmem_type = EXTENDED_ROM_ARCADIA;
|
|
return false;
|
|
}
|
|
if (is_alg_rom(romextfile)) {
|
|
type = EXTENDED_ROM_ALG;
|
|
|
|
}
|
|
f = read_rom_name (romextfile);
|
|
if (!f) {
|
|
notify_user (NUMSG_NOEXTROM);
|
|
return false;
|
|
}
|
|
zfile_fseek (f, 0, SEEK_END);
|
|
size = zfile_ftell32(f);
|
|
extendedkickmem_bank.reserved_size = ROM_SIZE_512;
|
|
off = 0;
|
|
if (type == 0) {
|
|
if (currprefs.cs_cd32cd) {
|
|
extendedkickmem_type = EXTENDED_ROM_CD32;
|
|
} else if (currprefs.cs_cdtvcd || currprefs.cs_cdtvram) {
|
|
extendedkickmem_type = EXTENDED_ROM_CDTV;
|
|
} else if (size > 300000) {
|
|
uae_u8 data[2] = { 0 };
|
|
zfile_fseek(f, off, SEEK_SET);
|
|
zfile_fread(data, sizeof(data), 1, f);
|
|
if (data[0] == 0x11 && data[1] == 0x11) {
|
|
if (need_uae_boot_rom(&currprefs) != 0xf00000)
|
|
extendedkickmem_type = EXTENDED_ROM_CDTV;
|
|
} else {
|
|
extendedkickmem_type = EXTENDED_ROM_CD32;
|
|
}
|
|
} else if (need_uae_boot_rom (&currprefs) != 0xf00000) {
|
|
extendedkickmem_type = EXTENDED_ROM_CDTV;
|
|
}
|
|
} else {
|
|
extendedkickmem_type = type;
|
|
}
|
|
if (extendedkickmem_type) {
|
|
zfile_fseek (f, off, SEEK_SET);
|
|
switch (extendedkickmem_type) {
|
|
case EXTENDED_ROM_CDTV:
|
|
extendedkickmem_bank.label = _T("rom_f0");
|
|
mapped_malloc (&extendedkickmem_bank);
|
|
extendedkickmem_bank.start = 0xf00000;
|
|
break;
|
|
case EXTENDED_ROM_CD32:
|
|
extendedkickmem_bank.label = _T("rom_e0");
|
|
mapped_malloc (&extendedkickmem_bank);
|
|
extendedkickmem_bank.start = 0xe00000;
|
|
break;
|
|
case EXTENDED_ROM_ALG:
|
|
extendedkickmem_bank.label = _T("rom_f0");
|
|
mapped_malloc(&extendedkickmem_bank);
|
|
extendedkickmem_bank.start = 0xf00000;
|
|
break;
|
|
}
|
|
if (extendedkickmem_bank.baseaddr) {
|
|
read_kickstart (f, extendedkickmem_bank.baseaddr, extendedkickmem_bank.allocated_size, 0, 1);
|
|
if (extendedkickmem_type == EXTENDED_ROM_ALG)
|
|
descramble_alg(extendedkickmem_bank.baseaddr, 262144);
|
|
extendedkickmem_bank.mask = extendedkickmem_bank.allocated_size - 1;
|
|
ret = true;
|
|
}
|
|
}
|
|
zfile_fclose (f);
|
|
return ret;
|
|
}
|
|
|
|
|
|
extern unsigned char arosrom[];
|
|
extern unsigned int arosrom_len;
|
|
extern int seriallog;
|
|
static bool load_kickstart_replacement(void)
|
|
{
|
|
struct zfile *f;
|
|
f = zfile_fopen_data(_T("aros.gz"), arosrom_len, arosrom);
|
|
if (!f)
|
|
return false;
|
|
f = zfile_gunzip(f);
|
|
if (!f)
|
|
return false;
|
|
|
|
extendedkickmem_bank.reserved_size = ROM_SIZE_512;
|
|
extendedkickmem_bank.mask = ROM_SIZE_512 - 1;
|
|
extendedkickmem_bank.label = _T("rom_e0");
|
|
extendedkickmem_type = EXTENDED_ROM_KS;
|
|
mapped_malloc(&extendedkickmem_bank);
|
|
read_kickstart(f, extendedkickmem_bank.baseaddr, ROM_SIZE_512, 0, 1);
|
|
|
|
|
|
kickmem_bank.reserved_size = ROM_SIZE_512;
|
|
kickmem_bank.mask = ROM_SIZE_512 - 1;
|
|
read_kickstart(f, kickmem_bank.baseaddr, ROM_SIZE_512, 1, 0);
|
|
|
|
zfile_fclose(f);
|
|
|
|
seriallog = -1;
|
|
|
|
// if 68000-68020 config without any other fast ram with m68k aros: enable special extra RAM.
|
|
if (currprefs.cpu_model <= 68020 &&
|
|
currprefs.cachesize == 0 &&
|
|
currprefs.fastmem[0].size == 0 &&
|
|
currprefs.z3fastmem[0].size == 0 &&
|
|
currprefs.mbresmem_high.size == 0 &&
|
|
currprefs.mbresmem_low.size == 0 &&
|
|
currprefs.cpuboardmem1.size == 0) {
|
|
|
|
changed_prefs.custom_memory_addrs[0] = currprefs.custom_memory_addrs[0] = 0xa80000;
|
|
changed_prefs.custom_memory_sizes[0] = currprefs.custom_memory_sizes[0] = 512 * 1024;
|
|
changed_prefs.custom_memory_mask[0] = currprefs.custom_memory_mask[0] = 0;
|
|
changed_prefs.custom_memory_addrs[1] = currprefs.custom_memory_addrs[1] = 0xb00000;
|
|
changed_prefs.custom_memory_sizes[1] = currprefs.custom_memory_sizes[1] = 512 * 1024;
|
|
changed_prefs.custom_memory_mask[1] = currprefs.custom_memory_mask[1] = 0;
|
|
|
|
allocate_memory_custombanks();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int patch_shapeshifter (uae_u8 *kickmemory)
|
|
{
|
|
/* Patch Kickstart ROM for ShapeShifter - from Christian Bauer.
|
|
* Changes 'lea $400,a0' and 'lea $1000,a0' to 'lea $3000,a0' for
|
|
* ShapeShifter compatibility.
|
|
*/
|
|
int i, patched = 0;
|
|
uae_u8 kickshift1[] = { 0x41, 0xf8, 0x04, 0x00 };
|
|
uae_u8 kickshift2[] = { 0x41, 0xf8, 0x10, 0x00 };
|
|
uae_u8 kickshift3[] = { 0x43, 0xf8, 0x04, 0x00 };
|
|
|
|
for (i = 0x200; i < 0x300; i++) {
|
|
if (!memcmp (kickmemory + i, kickshift1, sizeof (kickshift1)) ||
|
|
!memcmp (kickmemory + i, kickshift2, sizeof (kickshift2)) ||
|
|
!memcmp (kickmemory + i, kickshift3, sizeof (kickshift3))) {
|
|
kickmemory[i + 2] = 0x30;
|
|
write_log (_T("Kickstart KickShifted @%04X\n"), i);
|
|
patched++;
|
|
}
|
|
}
|
|
return patched;
|
|
}
|
|
|
|
/* disable incompatible drivers */
|
|
static int patch_residents (uae_u8 *kickmemory, int size)
|
|
{
|
|
int i, j, patched = 0;
|
|
const uae_char *residents[] = { "NCR scsi.device", NULL };
|
|
// "scsi.device", "carddisk.device", "card.resource" };
|
|
uaecptr base = size == ROM_SIZE_512 ? 0xf80000 : 0xfc0000;
|
|
|
|
if (is_device_rom(&currprefs, ROMTYPE_SCSI_A4000T, 0) < 0) {
|
|
for (i = 0; i < size - 100; i++) {
|
|
if (kickmemory[i] == 0x4a && kickmemory[i + 1] == 0xfc) {
|
|
uaecptr addr;
|
|
addr = (kickmemory[i + 2] << 24) | (kickmemory[i + 3] << 16) | (kickmemory[i + 4] << 8) | (kickmemory[i + 5] << 0);
|
|
if (addr != i + base)
|
|
continue;
|
|
addr = (kickmemory[i + 14] << 24) | (kickmemory[i + 15] << 16) | (kickmemory[i + 16] << 8) | (kickmemory[i + 17] << 0);
|
|
if (addr >= base && addr < base + size) {
|
|
j = 0;
|
|
while (residents[j]) {
|
|
if (!memcmp (residents[j], kickmemory + addr - base, strlen (residents[j]) + 1)) {
|
|
TCHAR *s = au (residents[j]);
|
|
write_log (_T("KSPatcher: '%s' at %08X disabled\n"), s, i + base);
|
|
xfree (s);
|
|
kickmemory[i] = 0x4b; /* destroy RTC_MATCHWORD */
|
|
patched++;
|
|
break;
|
|
}
|
|
j++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return patched;
|
|
}
|
|
|
|
static void patch_kick (void)
|
|
{
|
|
int patched = 0;
|
|
if (kickmem_bank.allocated_size >= ROM_SIZE_512 && currprefs.kickshifter)
|
|
patched += patch_shapeshifter (kickmem_bank.baseaddr);
|
|
patched += patch_residents (kickmem_bank.baseaddr, kickmem_bank.allocated_size);
|
|
if (extendedkickmem_bank.baseaddr) {
|
|
patched += patch_residents (extendedkickmem_bank.baseaddr, extendedkickmem_bank.allocated_size);
|
|
if (patched)
|
|
kickstart_fix_checksum (extendedkickmem_bank.baseaddr, extendedkickmem_bank.allocated_size);
|
|
}
|
|
if (patched)
|
|
kickstart_fix_checksum (kickmem_bank.baseaddr, kickmem_bank.allocated_size);
|
|
}
|
|
|
|
static struct zfile *get_kickstart_filehandle(struct uae_prefs *p)
|
|
{
|
|
struct zfile *f;
|
|
TCHAR tmprom[MAX_DPATH], tmprom2[MAX_DPATH];
|
|
|
|
f = read_rom_name(p->romfile);
|
|
_tcscpy(tmprom, p->romfile);
|
|
_tcscpy(tmprom2, p->romfile);
|
|
if (f == NULL) {
|
|
_stprintf(tmprom2, _T("%s%s"), start_path_data, p->romfile);
|
|
f = rom_fopen(tmprom2, _T("rb"), ZFD_NORMAL);
|
|
if (f == NULL) {
|
|
_stprintf(tmprom2, _T("%sroms/kick.rom"), start_path_data);
|
|
f = rom_fopen(tmprom2, _T("rb"), ZFD_NORMAL);
|
|
if (f == NULL) {
|
|
_stprintf(tmprom2, _T("%skick.rom"), start_path_data);
|
|
f = rom_fopen(tmprom2, _T("rb"), ZFD_NORMAL);
|
|
if (f == NULL) {
|
|
_stprintf(tmprom2, _T("%s../shared/rom/kick.rom"), start_path_data);
|
|
f = rom_fopen(tmprom2, _T("rb"), ZFD_NORMAL);
|
|
if (f == NULL) {
|
|
_stprintf(tmprom2, _T("%s../System/rom/kick.rom"), start_path_data);
|
|
f = rom_fopen(tmprom2, _T("rb"), ZFD_NORMAL);
|
|
if (f == NULL) {
|
|
f = read_rom_name_guess(tmprom, tmprom2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (f) {
|
|
_tcscpy(p->romfile, tmprom2);
|
|
}
|
|
return f;
|
|
}
|
|
|
|
extern struct zfile *read_executable_rom(struct zfile*, int size, int blocks);
|
|
static const uae_u8 romend[20] = {
|
|
0x00, 0x08, 0x00, 0x00,
|
|
0x00, 0x18, 0x00, 0x19, 0x00, 0x1a, 0x00, 0x1b, 0x00, 0x1c, 0x00, 0x1d, 0x00, 0x1e, 0x00, 0x1f
|
|
};
|
|
|
|
static int load_kickstart (void)
|
|
{
|
|
TCHAR tmprom[MAX_DPATH];
|
|
cloanto_rom = 0;
|
|
if (!_tcscmp(currprefs.romfile, _T(":AROS"))) {
|
|
return load_kickstart_replacement();
|
|
}
|
|
_tcscpy(tmprom, currprefs.romfile);
|
|
struct zfile *f = get_kickstart_filehandle(&currprefs);
|
|
addkeydir (currprefs.romfile);
|
|
if (f == NULL) /* still no luck */
|
|
goto err;
|
|
|
|
if (f != NULL) {
|
|
int filesize, size, maxsize;
|
|
int kspos = ROM_SIZE_512;
|
|
int extpos = 0;
|
|
bool singlebigrom = false;
|
|
|
|
uae_u8 tmp[8] = { 0 };
|
|
zfile_fread(tmp, sizeof tmp, 1, f);
|
|
|
|
maxsize = ROM_SIZE_512;
|
|
|
|
if ((tmp[0] == 0x00 && tmp[1] == 0x00 && tmp[2] == 0x03 && tmp[3] == 0xf3 &&
|
|
tmp[4] == 0x00 && tmp[5] == 0x00 && tmp[6] == 0x00 && tmp[7] == 0x00) ||
|
|
(tmp[0] == 0x7f && tmp[1] == 'E' && tmp[2] == 'L' && tmp[3] == 'F')) {
|
|
struct zfile *zf = read_executable_rom(f, ROM_SIZE_512, 3);
|
|
if (zf) {
|
|
int size = zfile_size32(zf);
|
|
zfile_fclose(f);
|
|
f = zf;
|
|
if (size > ROM_SIZE_512) {
|
|
maxsize = zfile_size32(zf);
|
|
singlebigrom = true;
|
|
extendedkickmem2a_bank.reserved_size = 524288;
|
|
extendedkickmem2a_bank.mask = extendedkickmem2a_bank.allocated_size - 1;
|
|
extendedkickmem2a_bank.start = size > 2 * ROM_SIZE_512 ? 0xa00000 : 0xa80000;
|
|
mapped_malloc(&extendedkickmem2a_bank);
|
|
extendedkickmem2b_bank.reserved_size = 524288;
|
|
extendedkickmem2b_bank.mask = extendedkickmem2a_bank.allocated_size - 1;
|
|
extendedkickmem2a_bank.start = extendedkickmem2a_bank.start + 524288;
|
|
mapped_malloc(&extendedkickmem2a_bank);
|
|
read_kickstart(f, extendedkickmem2a_bank.baseaddr, 524288, 0, 1);
|
|
read_kickstart(f, extendedkickmem2b_bank.baseaddr, 524288, 0, 1);
|
|
memset(kickmem_bank.baseaddr, 0, ROM_SIZE_512);
|
|
memcpy(kickmem_bank.baseaddr, extendedkickmem2a_bank.baseaddr, 0xd0);
|
|
memcpy(kickmem_bank.baseaddr + ROM_SIZE_512 - 20, romend, sizeof(romend));
|
|
kickstart_fix_checksum(kickmem_bank.baseaddr, ROM_SIZE_512);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!singlebigrom) {
|
|
zfile_fseek(f, 0, SEEK_END);
|
|
filesize = zfile_ftell32(f);
|
|
zfile_fseek(f, 0, SEEK_SET);
|
|
if (!singlebigrom) {
|
|
if (filesize == 1760 * 512) {
|
|
filesize = ROM_SIZE_256;
|
|
maxsize = ROM_SIZE_256;
|
|
}
|
|
if (filesize == ROM_SIZE_512 + 8) {
|
|
/* GVP 0xf0 kickstart */
|
|
zfile_fseek(f, 8, SEEK_SET);
|
|
}
|
|
if (filesize >= ROM_SIZE_512 * 2) {
|
|
struct romdata *rd = getromdatabyzfile(f);
|
|
zfile_fseek(f, kspos, SEEK_SET);
|
|
}
|
|
if (filesize >= ROM_SIZE_512 * 4) {
|
|
kspos = ROM_SIZE_512 * 3;
|
|
extpos = 0;
|
|
zfile_fseek(f, kspos, SEEK_SET);
|
|
}
|
|
}
|
|
size = read_kickstart(f, kickmem_bank.baseaddr, maxsize, 1, 0);
|
|
if (size == 0)
|
|
goto err;
|
|
kickmem_bank.mask = size - 1;
|
|
kickmem_bank.reserved_size = size;
|
|
if (filesize >= ROM_SIZE_512 * 2 && !extendedkickmem_type) {
|
|
extendedkickmem_bank.reserved_size = ROM_SIZE_512;
|
|
if (currprefs.cs_cdtvcd || currprefs.cs_cdtvram) {
|
|
extendedkickmem_type = EXTENDED_ROM_CDTV;
|
|
extendedkickmem_bank.reserved_size *= 2;
|
|
extendedkickmem_bank.label = _T("rom_f0");
|
|
extendedkickmem_bank.start = 0xf00000;
|
|
} else {
|
|
extendedkickmem_type = EXTENDED_ROM_KS;
|
|
extendedkickmem_bank.label = _T("rom_e0");
|
|
extendedkickmem_bank.start = 0xe00000;
|
|
}
|
|
mapped_malloc(&extendedkickmem_bank);
|
|
zfile_fseek(f, extpos, SEEK_SET);
|
|
read_kickstart(f, extendedkickmem_bank.baseaddr, extendedkickmem_bank.allocated_size, 0, 1);
|
|
extendedkickmem_bank.mask = extendedkickmem_bank.allocated_size - 1;
|
|
}
|
|
if (filesize > ROM_SIZE_512 * 2) {
|
|
extendedkickmem2a_bank.reserved_size = ROM_SIZE_512;
|
|
mapped_malloc(&extendedkickmem2a_bank);
|
|
extendedkickmem2b_bank.reserved_size = ROM_SIZE_512;
|
|
mapped_malloc(&extendedkickmem2b_bank);
|
|
zfile_fseek(f, extpos + ROM_SIZE_512, SEEK_SET);
|
|
read_kickstart(f, extendedkickmem2a_bank.baseaddr, ROM_SIZE_512, 0, 1);
|
|
zfile_fseek(f, extpos + ROM_SIZE_512 * 2, SEEK_SET);
|
|
read_kickstart(f, extendedkickmem2b_bank.baseaddr, ROM_SIZE_512, 0, 1);
|
|
extendedkickmem2a_bank.mask = extendedkickmem2a_bank.allocated_size - 1;
|
|
extendedkickmem2a_bank.start = 0xa80000;
|
|
extendedkickmem2b_bank.mask = extendedkickmem2b_bank.allocated_size - 1;
|
|
extendedkickmem2b_bank.start = 0xb00000;
|
|
}
|
|
}
|
|
}
|
|
|
|
#if defined(AMIGA)
|
|
chk_sum:
|
|
#endif
|
|
|
|
kickstart_version = (kickmem_bank.baseaddr[12] << 8) | kickmem_bank.baseaddr[13];
|
|
if (kickstart_version == 0xffff) {
|
|
// 1.0-1.1 and older
|
|
kickstart_version = (kickmem_bank.baseaddr[16] << 8) | kickmem_bank.baseaddr[17];
|
|
if (kickstart_version > 33)
|
|
kickstart_version = 0;
|
|
}
|
|
zfile_fclose (f);
|
|
return 1;
|
|
err:
|
|
_tcscpy (currprefs.romfile, tmprom);
|
|
zfile_fclose (f);
|
|
return 0;
|
|
}
|
|
|
|
static void set_direct_memory(addrbank *ab)
|
|
{
|
|
if (!(ab->flags & ABFLAG_DIRECTACCESS))
|
|
return;
|
|
ab->baseaddr_direct_r = ab->baseaddr;
|
|
if (!(ab->flags & ABFLAG_ROM))
|
|
ab->baseaddr_direct_w = ab->baseaddr;
|
|
}
|
|
|
|
#ifndef NATMEM_OFFSET
|
|
|
|
bool mapped_malloc (addrbank *ab)
|
|
{
|
|
ab->startmask = ab->start;
|
|
ab->startaccessmask = ab->start & ab->mask;
|
|
ab->baseaddr = xcalloc (uae_u8, ab->reserved_size + 4);
|
|
ab->allocated_size = ab->baseaddr != NULL ? ab->reserved_size : 0;
|
|
ab->baseaddr_direct_r = NULL;
|
|
ab->baseaddr_direct_w = NULL;
|
|
ab->flags &= ~ABFLAG_MAPPED;
|
|
set_direct_memory(ab);
|
|
return ab->baseaddr != NULL;
|
|
}
|
|
|
|
void mapped_free (addrbank *ab)
|
|
{
|
|
xfree(ab->baseaddr);
|
|
ab->flags &= ~ABFLAG_MAPPED;
|
|
ab->allocated_size = 0;
|
|
ab->baseaddr = NULL;
|
|
}
|
|
|
|
#else
|
|
|
|
#include <uae/mman.h>
|
|
|
|
shmpiece *shm_start;
|
|
|
|
static void dumplist (void)
|
|
{
|
|
shmpiece *x = shm_start;
|
|
write_log (_T("Start Dump:\n"));
|
|
while (x) {
|
|
write_log (_T("this=%p,Native %p,id %d,prev=%p,next=%p,size=0x%08x\n"),
|
|
x, x->native_address, x->id, x->prev, x->next, x->size);
|
|
x = x->next;
|
|
}
|
|
write_log (_T("End Dump:\n"));
|
|
}
|
|
|
|
static shmpiece *find_shmpiece (uae_u8 *base, bool safe)
|
|
{
|
|
shmpiece *x = shm_start;
|
|
|
|
while (x && x->native_address != base)
|
|
x = x->next;
|
|
if (!x) {
|
|
if (safe || bogomem_aliasing)
|
|
return 0;
|
|
write_log (_T("NATMEM: Failure to find mapping at %08lx, %p\n"), base - NATMEM_OFFSET, base);
|
|
nocanbang ();
|
|
return 0;
|
|
}
|
|
return x;
|
|
}
|
|
|
|
static void delete_shmmaps (uae_u32 start, uae_u32 size)
|
|
{
|
|
if (!needmman ())
|
|
return;
|
|
|
|
while (size) {
|
|
uae_u8 *base = mem_banks[bankindex (start)]->baseaddr;
|
|
if (base) {
|
|
shmpiece *x;
|
|
//base = ((uae_u8*)NATMEM_OFFSET)+start;
|
|
|
|
x = find_shmpiece (base, true);
|
|
if (!x)
|
|
return;
|
|
|
|
if (x->size > size) {
|
|
if (isdirectjit ())
|
|
write_log (_T("NATMEM WARNING: size mismatch mapping at %08x (size %08x, delsize %08x)\n"),start,x->size,size);
|
|
size = x->size;
|
|
}
|
|
|
|
uae_shmdt (x->native_address);
|
|
size -= x->size;
|
|
start += x->size;
|
|
if (x->next)
|
|
x->next->prev = x->prev; /* remove this one from the list */
|
|
if (x->prev)
|
|
x->prev->next = x->next;
|
|
else
|
|
shm_start = x->next;
|
|
xfree (x);
|
|
} else {
|
|
size -= 0x10000;
|
|
start += 0x10000;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void add_shmmaps (uae_u32 start, addrbank *what)
|
|
{
|
|
shmpiece *x = shm_start;
|
|
shmpiece *y;
|
|
uae_u8 *base = what->baseaddr;
|
|
|
|
if (!needmman ())
|
|
return;
|
|
|
|
if (!base)
|
|
return;
|
|
|
|
if (what->jit_read_flag && what->jit_write_flag)
|
|
return;
|
|
|
|
x = find_shmpiece (base, false);
|
|
if (!x)
|
|
return;
|
|
|
|
y = xmalloc (shmpiece, 1);
|
|
*y = *x;
|
|
base = ((uae_u8 *) NATMEM_OFFSET) + start;
|
|
y->native_address = (uae_u8*)uae_shmat (what, y->id, base, 0, NULL);
|
|
if (y->native_address == (void *) -1) {
|
|
write_log (_T("NATMEM: Failure to map existing at %08x (%p)\n"), start, base);
|
|
dumplist ();
|
|
nocanbang ();
|
|
return;
|
|
}
|
|
y->next = shm_start;
|
|
y->prev = NULL;
|
|
if (y->next)
|
|
y->next->prev = y;
|
|
shm_start = y;
|
|
}
|
|
|
|
#define MAPPED_MALLOC_DEBUG 1
|
|
|
|
bool mapped_malloc (addrbank *ab)
|
|
{
|
|
int id;
|
|
void *answer;
|
|
shmpiece *x;
|
|
bool rtgmem = (ab->flags & ABFLAG_RTG) != 0;
|
|
static int recurse;
|
|
|
|
if (ab->allocated_size) {
|
|
write_log(_T("mapped_malloc with memory bank '%s' already allocated!?\n"), ab->name);
|
|
}
|
|
ab->allocated_size = 0;
|
|
ab->baseaddr_direct_r = NULL;
|
|
ab->baseaddr_direct_w = NULL;
|
|
ab->flags &= ~ABFLAG_MAPPED;
|
|
|
|
if (ab->label && ab->label[0] == '*') {
|
|
if (ab->start == 0 || ab->start == 0xffffffff) {
|
|
write_log(_T("mapped_malloc(*) without start address!\n"));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
struct uae_mman_data md = { 0 };
|
|
uaecptr start = ab->start;
|
|
if (uae_mman_info(ab, &md)) {
|
|
start = md.start;
|
|
}
|
|
ab->startmask = start;
|
|
ab->startaccessmask = start & ab->mask;
|
|
if (!md.directsupport || (ab->flags & ABFLAG_ALLOCINDIRECT)) {
|
|
if (!(ab->flags & ABFLAG_ALLOCINDIRECT)) {
|
|
if (canbang) {
|
|
error_log(_T("JIT direct switched off: %s\n"), ab->name);
|
|
}
|
|
nocanbang();
|
|
}
|
|
ab->flags &= ~ABFLAG_DIRECTMAP;
|
|
if (ab->flags & ABFLAG_NOALLOC) {
|
|
ab->allocated_size = ab->reserved_size;
|
|
#if MAPPED_MALLOC_DEBUG
|
|
write_log(_T("mapped_malloc noalloc %s\n"), ab->name);
|
|
#endif
|
|
return true;
|
|
}
|
|
ab->baseaddr = xcalloc (uae_u8, ab->reserved_size + 4);
|
|
if (ab->baseaddr) {
|
|
// fill end of ram with ILLEGAL to catch direct PC falling out of RAM.
|
|
put_long_host(ab->baseaddr + ab->reserved_size, 0x4afc4afc);
|
|
ab->allocated_size = ab->reserved_size;
|
|
}
|
|
set_direct_memory(ab);
|
|
#if MAPPED_MALLOC_DEBUG
|
|
write_log(_T("mapped_malloc nodirect %s %p\n"), ab->name, ab->baseaddr);
|
|
#endif
|
|
return ab->baseaddr != NULL;
|
|
}
|
|
|
|
id = uae_shmget (UAE_IPC_PRIVATE, ab, 0x1ff);
|
|
if (id == -1) {
|
|
nocanbang ();
|
|
if (recurse)
|
|
return NULL;
|
|
recurse++;
|
|
mapped_malloc (ab);
|
|
recurse--;
|
|
return ab->baseaddr != NULL;
|
|
}
|
|
if (!(ab->flags & ABFLAG_NOALLOC)) {
|
|
answer = uae_shmat (ab, id, NULL, 0, &md);
|
|
uae_shmctl (id, UAE_IPC_RMID, NULL);
|
|
} else {
|
|
write_log(_T("MMAN: mapped_malloc using existing baseaddr %p\n"), ab->baseaddr);
|
|
answer = ab->baseaddr;
|
|
}
|
|
if (answer != (void *) -1) {
|
|
x = xmalloc (shmpiece, 1);
|
|
x->native_address = (uae_u8*)answer;
|
|
x->id = id;
|
|
x->size = ab->reserved_size;
|
|
x->name = ab->label;
|
|
x->next = shm_start;
|
|
x->prev = NULL;
|
|
if (x->next)
|
|
x->next->prev = x;
|
|
shm_start = x;
|
|
ab->baseaddr = x->native_address;
|
|
if (ab->baseaddr) {
|
|
if (md.hasbarrier) {
|
|
// fill end of ram with ILLEGAL to catch direct PC falling out of RAM.
|
|
put_long_host(ab->baseaddr + ab->reserved_size, 0x4afc4afc);
|
|
}
|
|
ab->allocated_size = ab->reserved_size;
|
|
}
|
|
ab->flags |= ABFLAG_DIRECTMAP;
|
|
set_direct_memory(ab);
|
|
#if MAPPED_MALLOC_DEBUG
|
|
write_log(_T("mapped_malloc direct %s %p\n"), ab->name, ab->baseaddr);
|
|
#endif
|
|
return ab->baseaddr != NULL;
|
|
}
|
|
if (recurse)
|
|
return NULL;
|
|
nocanbang ();
|
|
recurse++;
|
|
mapped_malloc (ab);
|
|
recurse--;
|
|
set_direct_memory(ab);
|
|
#if MAPPED_MALLOC_DEBUG
|
|
write_log(_T("mapped_malloc indirect %s %p\n"), ab->name, ab->baseaddr);
|
|
#endif
|
|
return ab->baseaddr != NULL;
|
|
}
|
|
|
|
#endif
|
|
|
|
static void init_mem_banks (void)
|
|
{
|
|
// unsigned so i << 16 won't overflow to negative when i >= 32768
|
|
for (unsigned int i = 0; i < MEMORY_BANKS; i++)
|
|
put_mem_bank (i << 16, &dummy_bank, 0);
|
|
#ifdef NATMEM_OFFSET
|
|
delete_shmmaps (0, 0xFFFF0000);
|
|
#endif
|
|
}
|
|
|
|
static void map_banks_set(addrbank *bank, int start, int size, int realsize)
|
|
{
|
|
bank->startmask = start << 16;
|
|
map_banks(bank, start, size, realsize);
|
|
}
|
|
|
|
static void allocate_memory (void)
|
|
{
|
|
bogomem_aliasing = 0;
|
|
|
|
bool bogoreset = (bogomem_bank.flags & ABFLAG_NOALLOC) != 0 &&
|
|
(chipmem_bank.reserved_size != currprefs.chipmem.size || bogomem_bank.reserved_size != currprefs.bogomem.size);
|
|
|
|
mapped_free(&fakeuaebootrom_bank);
|
|
|
|
if (bogoreset) {
|
|
mapped_free(&chipmem_bank);
|
|
mapped_free(&bogomem_bank);
|
|
}
|
|
|
|
/* emulate 0.5M+0.5M with 1M Agnus chip ram aliasing */
|
|
if (currprefs.chipmem.size == 0x80000 && currprefs.bogomem.size >= 0x80000 &&
|
|
(currprefs.chipset_mask & CSMASK_ECS_AGNUS) && !(currprefs.chipset_mask & CSMASK_AGA) && currprefs.cpu_model < 68020) {
|
|
if ((chipmem_bank.reserved_size != currprefs.chipmem.size || bogomem_bank.reserved_size != currprefs.bogomem.size || chipmem_full_size != 0x80000 * 2)) {
|
|
int memsize1, memsize2;
|
|
mapped_free (&chipmem_bank);
|
|
mapped_free (&bogomem_bank);
|
|
bogomem_bank.reserved_size = 0;
|
|
memsize1 = chipmem_bank.reserved_size = currprefs.chipmem.size;
|
|
memsize2 = bogomem_bank.reserved_size = currprefs.bogomem.size;
|
|
chipmem_bank.mask = chipmem_bank.reserved_size - 1;
|
|
chipmem_bank.start = chipmem_start_addr;
|
|
chipmem_full_mask = bogomem_bank.reserved_size * 2 - 1;
|
|
chipmem_full_size = 0x80000 * 2;
|
|
chipmem_bank.reserved_size = memsize1 + memsize2;
|
|
mapped_malloc (&chipmem_bank);
|
|
chipmem_bank.reserved_size = currprefs.chipmem.size;
|
|
chipmem_bank.allocated_size = currprefs.chipmem.size;
|
|
bogomem_bank.baseaddr = chipmem_bank.baseaddr + memsize1;
|
|
bogomem_bank.mask = bogomem_bank.reserved_size - 1;
|
|
bogomem_bank.start = bogomem_start_addr;
|
|
bogomem_bank.flags |= ABFLAG_NOALLOC;
|
|
bogomem_bank.allocated_size = bogomem_bank.reserved_size;
|
|
if (chipmem_bank.baseaddr == 0) {
|
|
write_log (_T("Fatal error: out of memory for chipmem.\n"));
|
|
chipmem_bank.reserved_size = 0;
|
|
} else {
|
|
need_hardreset = true;
|
|
}
|
|
}
|
|
bogomem_aliasing = 1;
|
|
} else if (currprefs.chipmem.size == 0x80000 && currprefs.bogomem.size >= 0x80000 &&
|
|
!(currprefs.chipset_mask & CSMASK_ECS_AGNUS) && currprefs.cs_1mchipjumper && currprefs.cpu_model < 68020) {
|
|
if ((chipmem_bank.reserved_size != currprefs.chipmem.size || bogomem_bank.reserved_size != currprefs.bogomem.size || chipmem_full_size != chipmem_bank.reserved_size)) {
|
|
int memsize1, memsize2;
|
|
mapped_free (&chipmem_bank);
|
|
mapped_free (&bogomem_bank);
|
|
bogomem_bank.reserved_size = 0;
|
|
memsize1 = chipmem_bank.reserved_size = currprefs.chipmem.size;
|
|
memsize2 = bogomem_bank.reserved_size = currprefs.bogomem.size;
|
|
chipmem_bank.mask = chipmem_bank.reserved_size - 1;
|
|
chipmem_bank.start = chipmem_start_addr;
|
|
chipmem_full_mask = chipmem_bank.reserved_size - 1;
|
|
chipmem_full_size = chipmem_bank.reserved_size;
|
|
chipmem_bank.reserved_size = memsize1 + memsize2;
|
|
mapped_malloc (&chipmem_bank);
|
|
chipmem_bank.reserved_size = currprefs.chipmem.size;
|
|
chipmem_bank.allocated_size = currprefs.chipmem.size;
|
|
bogomem_bank.baseaddr = chipmem_bank.baseaddr + memsize1;
|
|
bogomem_bank.mask = bogomem_bank.reserved_size - 1;
|
|
bogomem_bank.start = chipmem_bank.start + currprefs.chipmem.size;
|
|
bogomem_bank.flags |= ABFLAG_NOALLOC;
|
|
if (chipmem_bank.baseaddr == 0) {
|
|
write_log (_T("Fatal error: out of memory for chipmem.\n"));
|
|
chipmem_bank.reserved_size = 0;
|
|
} else {
|
|
need_hardreset = true;
|
|
}
|
|
}
|
|
bogomem_aliasing = 2;
|
|
}
|
|
|
|
if (chipmem_bank.reserved_size != currprefs.chipmem.size || bogoreset) {
|
|
int memsize;
|
|
mapped_free (&chipmem_bank);
|
|
chipmem_bank.flags &= ~ABFLAG_NOALLOC;
|
|
|
|
memsize = chipmem_bank.reserved_size = chipmem_full_size = currprefs.chipmem.size;
|
|
chipmem_full_mask = chipmem_bank.mask = chipmem_bank.reserved_size - 1;
|
|
chipmem_bank.start = chipmem_start_addr;
|
|
if (!currprefs.cachesize && memsize < 0x100000)
|
|
memsize = 0x100000;
|
|
if (memsize > 0x100000 && memsize < 0x200000)
|
|
memsize = 0x200000;
|
|
chipmem_bank.reserved_size = memsize;
|
|
mapped_malloc (&chipmem_bank);
|
|
chipmem_bank.reserved_size = currprefs.chipmem.size;
|
|
chipmem_bank.allocated_size = currprefs.chipmem.size;
|
|
if (chipmem_bank.baseaddr == 0) {
|
|
write_log (_T("Fatal error: out of memory for chipmem.\n"));
|
|
chipmem_bank.reserved_size = 0;
|
|
} else {
|
|
need_hardreset = true;
|
|
if (memsize > chipmem_bank.allocated_size)
|
|
memset (chipmem_bank.baseaddr + chipmem_bank.allocated_size, 0xff, memsize - chipmem_bank.allocated_size);
|
|
}
|
|
currprefs.chipset_mask = changed_prefs.chipset_mask;
|
|
chipmem_full_mask = chipmem_bank.allocated_size - 1;
|
|
if (!currprefs.cachesize) {
|
|
if (currprefs.chipset_mask & CSMASK_ECS_AGNUS) {
|
|
if (chipmem_bank.allocated_size < 0x100000)
|
|
chipmem_full_mask = 0x100000 - 1;
|
|
if (chipmem_bank.allocated_size > 0x100000 && chipmem_bank.allocated_size < 0x200000)
|
|
chipmem_full_mask = chipmem_bank.mask = 0x200000 - 1;
|
|
} else if (currprefs.cs_1mchipjumper) {
|
|
chipmem_full_mask = 0x80000 - 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bogomem_bank.reserved_size != currprefs.bogomem.size || bogoreset) {
|
|
if (!(bogomem_bank.reserved_size == 0x200000 && currprefs.bogomem.size == 0x180000)) {
|
|
mapped_free (&bogomem_bank);
|
|
bogomem_bank.flags &= ~ABFLAG_NOALLOC;
|
|
bogomem_bank.reserved_size = 0;
|
|
|
|
bogomem_bank.reserved_size = currprefs.bogomem.size;
|
|
if (bogomem_bank.reserved_size >= 0x180000)
|
|
bogomem_bank.reserved_size = 0x200000;
|
|
bogomem_bank.mask = bogomem_bank.reserved_size - 1;
|
|
bogomem_bank.start = bogomem_start_addr;
|
|
|
|
if (bogomem_bank.reserved_size) {
|
|
if (!mapped_malloc (&bogomem_bank)) {
|
|
write_log (_T("Out of memory for bogomem.\n"));
|
|
bogomem_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
need_hardreset = true;
|
|
}
|
|
}
|
|
if (debugmem_bank.reserved_size != currprefs.debugmem_size) {
|
|
mapped_free(&debugmem_bank);
|
|
|
|
debugmem_bank.reserved_size = currprefs.debugmem_size;
|
|
debugmem_bank.mask = debugmem_bank.reserved_size - 1;
|
|
debugmem_bank.start = currprefs.debugmem_start;
|
|
if (debugmem_bank.reserved_size) {
|
|
if (!mapped_malloc(&debugmem_bank)) {
|
|
write_log(_T("Out of memory for debugger memory.\n"));
|
|
debugmem_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
}
|
|
if (mem25bit_bank.reserved_size != currprefs.mem25bit.size) {
|
|
mapped_free(&mem25bit_bank);
|
|
|
|
mem25bit_bank.reserved_size = currprefs.mem25bit.size;
|
|
mem25bit_bank.mask = mem25bit_bank.reserved_size - 1;
|
|
mem25bit_bank.start = 0x01000000;
|
|
if (mem25bit_bank.reserved_size) {
|
|
if (!mapped_malloc(&mem25bit_bank)) {
|
|
write_log(_T("Out of memory for 25 bit memory.\n"));
|
|
mem25bit_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
need_hardreset = true;
|
|
}
|
|
if (a3000lmem_bank.reserved_size != currprefs.mbresmem_low.size) {
|
|
mapped_free (&a3000lmem_bank);
|
|
|
|
a3000lmem_bank.reserved_size = currprefs.mbresmem_low.size;
|
|
a3000lmem_bank.mask = a3000lmem_bank.reserved_size - 1;
|
|
a3000lmem_bank.start = 0x08000000 - a3000lmem_bank.reserved_size;
|
|
if (a3000lmem_bank.reserved_size) {
|
|
if (!mapped_malloc (&a3000lmem_bank)) {
|
|
write_log (_T("Out of memory for a3000lowmem.\n"));
|
|
a3000lmem_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
need_hardreset = true;
|
|
}
|
|
if (a3000hmem_bank.reserved_size != currprefs.mbresmem_high.size) {
|
|
mapped_free (&a3000hmem_bank);
|
|
|
|
a3000hmem_bank.reserved_size = currprefs.mbresmem_high.size;
|
|
a3000hmem_bank.mask = a3000hmem_bank.reserved_size - 1;
|
|
a3000hmem_bank.start = 0x08000000;
|
|
if (a3000hmem_bank.reserved_size) {
|
|
if (!mapped_malloc (&a3000hmem_bank)) {
|
|
write_log (_T("Out of memory for a3000highmem.\n"));
|
|
a3000hmem_bank.reserved_size = 0;
|
|
}
|
|
}
|
|
need_hardreset = true;
|
|
}
|
|
|
|
allocate_memory_custombanks();
|
|
|
|
if (savestate_state == STATE_RESTORE) {
|
|
if (bootrom_filepos) {
|
|
if (currprefs.uaeboard < 0)
|
|
currprefs.uaeboard = 0;
|
|
protect_roms (false);
|
|
restore_ram (bootrom_filepos, rtarea_bank.baseaddr);
|
|
protect_roms (true);
|
|
if (currprefs.uaeboard >= 2) {
|
|
map_banks_set(&rtarea_bank, rtarea_base >> 16, 1, 0);
|
|
}
|
|
}
|
|
restore_ram (chip_filepos, chipmem_bank.baseaddr);
|
|
if (bogomem_bank.allocated_size > 0)
|
|
restore_ram (bogo_filepos, bogomem_bank.baseaddr);
|
|
if (mem25bit_bank.allocated_size > 0)
|
|
restore_ram(mem25bit_filepos, mem25bit_bank.baseaddr);
|
|
if (a3000lmem_bank.allocated_size > 0)
|
|
restore_ram (a3000lmem_filepos, a3000lmem_bank.baseaddr);
|
|
if (a3000hmem_bank.allocated_size > 0)
|
|
restore_ram (a3000hmem_filepos, a3000hmem_bank.baseaddr);
|
|
}
|
|
#ifdef AGA
|
|
chipmem_bank_ce2.baseaddr = chipmem_bank.baseaddr;
|
|
#endif
|
|
bootrom_filepos = 0;
|
|
chip_filepos = 0;
|
|
bogo_filepos = 0;
|
|
a3000lmem_filepos = 0;
|
|
a3000hmem_filepos = 0;
|
|
cpuboard_init();
|
|
}
|
|
|
|
static void setmemorywidth(struct ramboard *mb, addrbank *ab)
|
|
{
|
|
if (!ab || !ab->allocated_size)
|
|
return;
|
|
if (!mb->force16bit)
|
|
return;
|
|
for (int i = (ab->start >> 16); i < ((ab->start + ab->allocated_size) >> 16); i++) {
|
|
if (ce_banktype[i] == CE_MEMBANK_FAST32)
|
|
ce_banktype[i] = CE_MEMBANK_FAST16;
|
|
if (ce_banktype[i] == CE_MEMBANK_CHIP32)
|
|
ce_banktype[i] = CE_MEMBANK_CHIP16;
|
|
}
|
|
}
|
|
|
|
static void fill_ce_banks (void)
|
|
{
|
|
int i;
|
|
|
|
if (currprefs.cpu_model <= 68010) {
|
|
memset (ce_banktype, CE_MEMBANK_FAST16, sizeof ce_banktype);
|
|
} else {
|
|
memset (ce_banktype, CE_MEMBANK_FAST32, sizeof ce_banktype);
|
|
}
|
|
|
|
addrbank *ab = &get_mem_bank(0);
|
|
if (ab && (ab->flags & ABFLAG_CHIPRAM)) {
|
|
for (i = 0; i < (0x200000 >> 16); i++) {
|
|
ce_banktype[i] = (currprefs.cs_mbdmac || (currprefs.chipset_mask & CSMASK_AGA)) ? CE_MEMBANK_CHIP32 : CE_MEMBANK_CHIP16;
|
|
}
|
|
}
|
|
if (!currprefs.cs_slowmemisfast) {
|
|
for (i = (0xc00000 >> 16); i < (0xe00000 >> 16); i++)
|
|
ce_banktype[i] = ce_banktype[0];
|
|
for (i = (bogomem_bank.start >> 16); i < ((bogomem_bank.start + bogomem_bank.allocated_size) >> 16); i++)
|
|
ce_banktype[i] = ce_banktype[0];
|
|
}
|
|
for (i = (0xd00000 >> 16); i < (0xe00000 >> 16); i++) {
|
|
ce_banktype[i] = CE_MEMBANK_CHIP16;
|
|
}
|
|
for (i = (0xa00000 >> 16); i < (0xc00000 >> 16); i++) {
|
|
addrbank *b;
|
|
ce_banktype[i] = CE_MEMBANK_CIA;
|
|
b = &get_mem_bank (i << 16);
|
|
if (b && !(b->flags & ABFLAG_CIA)) {
|
|
ce_banktype[i] = CE_MEMBANK_FAST32;
|
|
}
|
|
}
|
|
// CD32 ROM is 16-bit
|
|
if (currprefs.cs_cd32cd) {
|
|
for (i = (0xe00000 >> 16); i < (0xe80000 >> 16); i++)
|
|
ce_banktype[i] = CE_MEMBANK_FAST16;
|
|
for (i = (0xf80000 >> 16); i <= (0xff0000 >> 16); i++)
|
|
ce_banktype[i] = CE_MEMBANK_FAST16;
|
|
}
|
|
|
|
// A4000T NCR is 32-bit
|
|
if (is_device_rom(&currprefs, ROMTYPE_SCSI_A4000T, 0) >= 0) {
|
|
ce_banktype[0xdd0000 >> 16] = CE_MEMBANK_FAST32;
|
|
}
|
|
|
|
if (currprefs.cs_toshibagary) {
|
|
for (i = (0xe80000 >> 16); i < (0xf80000 >> 16); i++)
|
|
ce_banktype[i] = CE_MEMBANK_CHIP16;
|
|
}
|
|
|
|
if (currprefs.cs_romisslow) {
|
|
for (i = (0xe00000 >> 16); i < (0xe80000 >> 16); i++)
|
|
ce_banktype[i] = CE_MEMBANK_CHIP16;
|
|
for (i = (0xf80000 >> 16); i < (0x1000000 >> 16); i++)
|
|
ce_banktype[i] = CE_MEMBANK_CHIP16;
|
|
}
|
|
|
|
setmemorywidth(&currprefs.chipmem, &chipmem_bank);
|
|
setmemorywidth(&currprefs.bogomem, &bogomem_bank);
|
|
setmemorywidth(&currprefs.z3chipmem, &z3chipmem_bank);
|
|
setmemorywidth(&currprefs.mbresmem_low, &a3000lmem_bank);
|
|
for (int i = 0; i < MAX_RAM_BOARDS; i++) {
|
|
setmemorywidth(&currprefs.z3fastmem[i], &z3fastmem_bank[i]);
|
|
setmemorywidth(&currprefs.fastmem[i], &fastmem_bank[i]);
|
|
}
|
|
|
|
if (currprefs.address_space_24) {
|
|
for (i = 1; i < 256; i++)
|
|
memcpy(&ce_banktype[i * 256], &ce_banktype[0], 256);
|
|
}
|
|
|
|
}
|
|
|
|
static int overlay_state;
|
|
|
|
void map_overlay (int chip)
|
|
{
|
|
int size;
|
|
addrbank *cb;
|
|
|
|
if (chip < 0)
|
|
chip = overlay_state;
|
|
|
|
if (currprefs.cs_compatible == CP_CASABLANCA) {
|
|
casablanca_map_overlay();
|
|
return;
|
|
}
|
|
|
|
size = chipmem_bank.allocated_size >= 0x180000 ? (chipmem_bank.allocated_size >> 16) : 32;
|
|
if (bogomem_aliasing)
|
|
size = 8;
|
|
cb = &chipmem_bank;
|
|
#ifdef AGA
|
|
#if 0
|
|
if (currprefs.cpu_cycle_exact && currprefs.cpu_model >= 68020)
|
|
cb = &chipmem_bank_ce2;
|
|
#endif
|
|
#endif
|
|
if (chip) {
|
|
map_banks (&dummy_bank, 0, size, 0);
|
|
if (!isdirectjit ()) {
|
|
if ((currprefs.chipset_mask & CSMASK_ECS_AGNUS) && bogomem_bank.allocated_size == 0) {
|
|
map_banks(cb, 0, size, chipmem_bank.allocated_size);
|
|
int start = chipmem_bank.allocated_size >> 16;
|
|
if (chipmem_bank.allocated_size < 0x100000) {
|
|
if (currprefs.cs_1mchipjumper) {
|
|
int dummy = (0x100000 - chipmem_bank.allocated_size) >> 16;
|
|
map_banks (&chipmem_dummy_bank, start, dummy, 0);
|
|
map_banks (&chipmem_dummy_bank, start + 16, dummy, 0);
|
|
}
|
|
} else if (chipmem_bank.allocated_size < 0x200000 && chipmem_bank.allocated_size > 0x100000) {
|
|
int dummy = (0x200000 - chipmem_bank.allocated_size) >> 16;
|
|
map_banks (&chipmem_dummy_bank, start, dummy, 0);
|
|
}
|
|
} else {
|
|
int mapsize = 32;
|
|
if ((chipmem_bank.allocated_size >> 16) > mapsize)
|
|
mapsize = chipmem_bank.allocated_size >> 16;
|
|
map_banks(cb, 0, mapsize, chipmem_bank.allocated_size);
|
|
}
|
|
} else {
|
|
map_banks (cb, 0, chipmem_bank.allocated_size >> 16, 0);
|
|
}
|
|
} else {
|
|
addrbank *rb = NULL;
|
|
if (size < 32 && bogomem_aliasing == 0)
|
|
size = 32;
|
|
cb = get_mem_bank_real(0xf00000);
|
|
if (!rb && cb && (cb->flags & ABFLAG_ROM) && get_word (0xf00000) == 0x1114)
|
|
rb = cb;
|
|
cb = get_mem_bank_real(0xe00000);
|
|
if (!rb && cb && (cb->flags & ABFLAG_ROM) && get_word (0xe00000) == 0x1114)
|
|
rb = cb;
|
|
if (!rb)
|
|
rb = &kickmem_bank;
|
|
map_banks (rb, 0, size, 0x80000);
|
|
}
|
|
initramboard(&chipmem_bank, &currprefs.chipmem);
|
|
overlay_state = chip;
|
|
fill_ce_banks ();
|
|
cpuboard_overlay_override();
|
|
if (!isrestore () && valid_address (regs.pc, 4))
|
|
m68k_setpc_normal (m68k_getpc ());
|
|
}
|
|
|
|
void memory_clear (void)
|
|
{
|
|
mem_hardreset = 0;
|
|
if (savestate_state == STATE_RESTORE)
|
|
return;
|
|
if (chipmem_bank.baseaddr)
|
|
memset(chipmem_bank.baseaddr, 0, chipmem_bank.allocated_size);
|
|
if (bogomem_bank.baseaddr)
|
|
memset(bogomem_bank.baseaddr, 0, bogomem_bank.allocated_size);
|
|
if (mem25bit_bank.baseaddr)
|
|
memset(mem25bit_bank.baseaddr, 0, mem25bit_bank.allocated_size);
|
|
if (a3000lmem_bank.baseaddr)
|
|
memset(a3000lmem_bank.baseaddr, 0, a3000lmem_bank.allocated_size);
|
|
if (a3000hmem_bank.baseaddr)
|
|
memset(a3000hmem_bank.baseaddr, 0, a3000hmem_bank.allocated_size);
|
|
expansion_clear ();
|
|
cpuboard_clear();
|
|
}
|
|
|
|
static void restore_roms(void)
|
|
{
|
|
roms_modified = false;
|
|
protect_roms (false);
|
|
write_log (_T("ROM loader.. (%s)\n"), currprefs.romfile);
|
|
kickstart_rom = 1;
|
|
a1000_handle_kickstart (0);
|
|
xfree (a1000_bootrom);
|
|
a1000_bootrom = 0;
|
|
a1000_kickstart_mode = 0;
|
|
|
|
memcpy(currprefs.romfile, changed_prefs.romfile, sizeof currprefs.romfile);
|
|
memcpy(currprefs.romextfile, changed_prefs.romextfile, sizeof currprefs.romextfile);
|
|
need_hardreset = true;
|
|
mapped_free(&extendedkickmem_bank);
|
|
mapped_free(&extendedkickmem2a_bank);
|
|
mapped_free(&extendedkickmem2b_bank);
|
|
extendedkickmem_bank.reserved_size = 0;
|
|
extendedkickmem2a_bank.reserved_size = 0;
|
|
extendedkickmem2b_bank.reserved_size = 0;
|
|
extendedkickmem_type = 0;
|
|
load_extendedkickstart (currprefs.romextfile, 0);
|
|
load_extendedkickstart (currprefs.romextfile2, EXTENDED_ROM_CDTV);
|
|
kickmem_bank.mask = ROM_SIZE_512 - 1;
|
|
if (!load_kickstart ()) {
|
|
if (_tcslen (currprefs.romfile) > 0) {
|
|
error_log (_T("Failed to open '%s'\n"), currprefs.romfile);
|
|
notify_user (NUMSG_NOROM);
|
|
}
|
|
load_kickstart_replacement ();
|
|
} else {
|
|
struct romdata *rd = getromdatabydata (kickmem_bank.baseaddr, kickmem_bank.reserved_size);
|
|
if (rd) {
|
|
write_log (_T("Known ROM '%s' loaded\n"), rd->name);
|
|
#if 1
|
|
if ((rd->cpu & 8) && changed_prefs.cpu_model < 68030) {
|
|
notify_user (NUMSG_KS68030PLUS);
|
|
uae_restart (-1, NULL);
|
|
} else if ((rd->cpu & 3) == 3 && changed_prefs.cpu_model != 68030) {
|
|
notify_user (NUMSG_KS68030);
|
|
uae_restart (-1, NULL);
|
|
} else if ((rd->cpu & 3) == 1 && changed_prefs.cpu_model < 68020) {
|
|
notify_user (NUMSG_KS68EC020);
|
|
uae_restart (-1, NULL);
|
|
} else if ((rd->cpu & 3) == 2 && (changed_prefs.cpu_model < 68020 || changed_prefs.address_space_24)) {
|
|
notify_user (NUMSG_KS68020);
|
|
uae_restart (-1, NULL);
|
|
}
|
|
#endif
|
|
if (rd->cloanto)
|
|
cloanto_rom = 1;
|
|
kickstart_rom = 0;
|
|
if ((rd->type & (ROMTYPE_SPECIALKICK | ROMTYPE_KICK)) == ROMTYPE_KICK)
|
|
kickstart_rom = 1;
|
|
if ((rd->cpu & 4) && currprefs.cs_compatible) {
|
|
/* A4000 ROM = need ramsey, gary and ide */
|
|
if (currprefs.cs_ramseyrev < 0)
|
|
changed_prefs.cs_ramseyrev = currprefs.cs_ramseyrev = 0x0f;
|
|
changed_prefs.cs_fatgaryrev = currprefs.cs_fatgaryrev = 0;
|
|
if (currprefs.cs_ide != IDE_A4000)
|
|
changed_prefs.cs_ide = currprefs.cs_ide = -1;
|
|
}
|
|
} else {
|
|
write_log (_T("Unknown ROM '%s' loaded\n"), currprefs.romfile);
|
|
}
|
|
}
|
|
patch_kick ();
|
|
write_log (_T("ROM loader end\n"));
|
|
protect_roms (true);
|
|
}
|
|
|
|
bool read_kickstart_version(struct uae_prefs *p)
|
|
{
|
|
kickstart_version = 0;
|
|
cloanto_rom = 0;
|
|
struct zfile *z = get_kickstart_filehandle(p);
|
|
if (!z)
|
|
return false;
|
|
uae_u8 mem[32] = { 0 };
|
|
read_kickstart(z, mem, sizeof mem, 0, 0);
|
|
zfile_fclose(z);
|
|
kickstart_version = (mem[12] << 8) | mem[13];
|
|
if (kickstart_version == 0xffff) {
|
|
// 1.0-1.1 and older
|
|
kickstart_version = (mem[16] << 8) | mem[17];
|
|
if (kickstart_version > 33)
|
|
kickstart_version = 0;
|
|
}
|
|
write_log(_T("KS ver = %d (0x%02x)\n"), kickstart_version & 255, kickstart_version);
|
|
return true;
|
|
}
|
|
|
|
void reload_roms(void)
|
|
{
|
|
if (roms_modified)
|
|
restore_roms();
|
|
}
|
|
|
|
void memory_restore(void)
|
|
{
|
|
last_address_space_24 = currprefs.address_space_24;
|
|
cpuboard_map();
|
|
map_banks_set(&kickmem_bank, 0xF8, 8, 0);
|
|
}
|
|
|
|
void memory_reset (void)
|
|
{
|
|
int bnk, bnk_end;
|
|
bool gayleorfatgary;
|
|
|
|
alg_flag = 0;
|
|
need_hardreset = false;
|
|
rom_write_enabled = true;
|
|
jit_n_addr_unsafe = 0;
|
|
/* Use changed_prefs, as m68k_reset is called later. */
|
|
if (last_address_space_24 != changed_prefs.address_space_24)
|
|
need_hardreset = true;
|
|
last_address_space_24 = changed_prefs.address_space_24;
|
|
|
|
if (mem_hardreset > 2)
|
|
memory_init ();
|
|
|
|
memset(ce_cachable, CACHE_ENABLE_INS, sizeof ce_cachable);
|
|
|
|
be_cnt = be_recursive = 0;
|
|
currprefs.chipmem.size = changed_prefs.chipmem.size;
|
|
currprefs.bogomem.size = changed_prefs.bogomem.size;
|
|
currprefs.mbresmem_low.size = changed_prefs.mbresmem_low.size;
|
|
currprefs.mbresmem_high.size = changed_prefs.mbresmem_high.size;
|
|
currprefs.cs_ksmirror_e0 = changed_prefs.cs_ksmirror_e0;
|
|
currprefs.cs_ksmirror_a8 = changed_prefs.cs_ksmirror_a8;
|
|
currprefs.cs_ciaoverlay = changed_prefs.cs_ciaoverlay;
|
|
currprefs.cs_cdtvram = changed_prefs.cs_cdtvram;
|
|
currprefs.cs_a1000ram = changed_prefs.cs_a1000ram;
|
|
currprefs.cs_ide = changed_prefs.cs_ide;
|
|
currprefs.cs_fatgaryrev = changed_prefs.cs_fatgaryrev;
|
|
currprefs.cs_ramseyrev = changed_prefs.cs_ramseyrev;
|
|
currprefs.cs_unmapped_space = changed_prefs.cs_unmapped_space;
|
|
cpuboard_reset(mem_hardreset);
|
|
|
|
gayleorfatgary = ((currprefs.chipset_mask & CSMASK_AGA) || currprefs.cs_pcmcia || currprefs.cs_ide > 0 || currprefs.cs_mbdmac) && !currprefs.cs_cd32cd;
|
|
|
|
init_mem_banks ();
|
|
allocate_memory ();
|
|
chipmem_setindirect ();
|
|
|
|
if (mem_hardreset > 1 || ((roms_modified || a1000_bootrom) && is_hardreset())
|
|
|| _tcscmp (currprefs.romfile, changed_prefs.romfile) != 0
|
|
|| _tcscmp (currprefs.romextfile, changed_prefs.romextfile) != 0)
|
|
{
|
|
restore_roms();
|
|
}
|
|
|
|
if ((cloanto_rom || extendedkickmem_bank.allocated_size) && currprefs.maprom && currprefs.maprom < 0x01000000) {
|
|
currprefs.maprom = changed_prefs.maprom = 0x00a80000;
|
|
if (extendedkickmem2a_bank.allocated_size || extendedkickmem2b_bank.allocated_size) // can't do if 2M ROM
|
|
currprefs.maprom = changed_prefs.maprom = 0;
|
|
}
|
|
|
|
map_banks (&custom_bank, 0xC0, 0xE0 - 0xC0, 0);
|
|
map_banks (&cia_bank, 0xA0, 32, 0);
|
|
if (!currprefs.cs_a1000ram && currprefs.cs_rtc != 3)
|
|
/* D80000 - DDFFFF not mapped (A1000 or A2000 = custom chips) */
|
|
map_banks (&dummy_bank, 0xD8, 6, 0);
|
|
|
|
/* map "nothing" to 0x200000 - 0x9FFFFF (0xBEFFFF if Gayle or Fat Gary) */
|
|
bnk = chipmem_bank.allocated_size >> 16;
|
|
if (bnk < 0x20 + (currprefs.fastmem[0].size >> 16))
|
|
bnk = 0x20 + (currprefs.fastmem[0].size >> 16);
|
|
bnk_end = currprefs.cs_cd32cd ? 0xBE : (gayleorfatgary ? 0xBF : 0xA0);
|
|
map_banks (&dummy_bank, bnk, bnk_end - bnk, 0);
|
|
if (gayleorfatgary) {
|
|
// a3000 or a4000 = custom chips from 0xc0 to 0xd0
|
|
if (currprefs.cs_ide == IDE_A4000 || currprefs.cs_mbdmac)
|
|
map_banks (&dummy_bank, 0xd0, 8, 0);
|
|
else
|
|
map_banks (&dummy_bank, 0xc0, 0xd8 - 0xc0, 0);
|
|
} else if (currprefs.cs_cd32cd) {
|
|
// CD32: 0xc0 to 0xd0
|
|
map_banks(&dummy_bank, 0xd0, 8, 0);
|
|
// strange 64k custom mirror
|
|
map_banks(&custom_bank, 0xb9, 1, 0);
|
|
}
|
|
|
|
if (bogomem_bank.baseaddr) {
|
|
int t = currprefs.bogomem.size >> 16;
|
|
if (t > 0x1C)
|
|
t = 0x1C;
|
|
if (t > 0x18 && ((currprefs.chipset_mask & CSMASK_AGA) || (currprefs.cpu_model >= 68020 && !currprefs.address_space_24)))
|
|
t = 0x18;
|
|
if (bogomem_aliasing == 2)
|
|
map_banks (&bogomem_bank, 0x08, t, 0);
|
|
else
|
|
map_banks (&bogomem_bank, 0xC0, t, 0);
|
|
initramboard(&bogomem_bank, &currprefs.bogomem);
|
|
}
|
|
if (currprefs.cs_ide || currprefs.cs_pcmcia) {
|
|
if (currprefs.cs_ide == IDE_A600A1200 || currprefs.cs_pcmcia) {
|
|
map_banks (&gayle_bank, 0xD8, 6, 0);
|
|
map_banks (&gayle2_bank, 0xDD, 2, 0);
|
|
}
|
|
if (currprefs.cs_ide == IDE_A4000 || is_device_rom(&currprefs, ROMTYPE_SCSI_A4000T, 0))
|
|
map_banks (&gayle_bank, 0xDD, 1, 0);
|
|
if (currprefs.cs_ide < 0 && !currprefs.cs_pcmcia)
|
|
map_banks (&gayle_bank, 0xD8, 6, 0);
|
|
if (currprefs.cs_ide < 0)
|
|
map_banks (&gayle_bank, 0xDD, 1, 0);
|
|
}
|
|
if (currprefs.cs_rtc == 3) // A2000 clock
|
|
map_banks (&clock_bank, 0xD8, 4, 0);
|
|
if (currprefs.cs_rtc == 1 || currprefs.cs_rtc == 2 || currprefs.cs_cdtvram)
|
|
map_banks (&clock_bank, 0xDC, 1, 0);
|
|
else if (currprefs.cs_ksmirror_a8 || currprefs.cs_ide > 0 || currprefs.cs_pcmcia)
|
|
map_banks (&clock_bank, 0xDC, 1, 0); /* none clock */
|
|
if (currprefs.cs_fatgaryrev >= 0 || currprefs.cs_ramseyrev >= 0)
|
|
map_banks (&mbres_bank, 0xDE, 1, 0);
|
|
#ifdef CD32
|
|
if (currprefs.cs_cd32c2p || currprefs.cs_cd32cd || currprefs.cs_cd32nvram) {
|
|
map_banks (&akiko_bank, AKIKO_BASE >> 16, 1, 0);
|
|
map_banks (&gayle2_bank, 0xDD, 2, 0);
|
|
}
|
|
#endif
|
|
if (mem25bit_bank.baseaddr) {
|
|
map_banks(&mem25bit_bank, mem25bit_bank.start >> 16, mem25bit_bank.allocated_size >> 16, 0);
|
|
initramboard(&mem25bit_bank, &currprefs.mem25bit);
|
|
}
|
|
if (a3000lmem_bank.baseaddr) {
|
|
map_banks(&a3000lmem_bank, a3000lmem_bank.start >> 16, a3000lmem_bank.allocated_size >> 16, 0);
|
|
initramboard(&a3000lmem_bank, &currprefs.mbresmem_low);
|
|
}
|
|
if (a3000hmem_bank.baseaddr) {
|
|
map_banks(&a3000hmem_bank, a3000hmem_bank.start >> 16, a3000hmem_bank.allocated_size >> 16, 0);
|
|
initramboard(&a3000hmem_bank, &currprefs.mbresmem_high);
|
|
}
|
|
if (debugmem_bank.baseaddr) {
|
|
map_banks(&debugmem_bank, debugmem_bank.start >> 16, debugmem_bank.allocated_size >> 16, 0);
|
|
}
|
|
cpuboard_map();
|
|
map_banks_set(&kickmem_bank, 0xF8, 8, 0);
|
|
if (currprefs.maprom) {
|
|
if (!cpuboard_maprom())
|
|
map_banks_set(&kickram_bank, currprefs.maprom >> 16, extendedkickmem2a_bank.allocated_size ? 32 : (extendedkickmem_bank.allocated_size ? 16 : 8), 0);
|
|
}
|
|
/* map beta Kickstarts at 0x200000/0xC00000/0xF00000 */
|
|
if (kickmem_bank.baseaddr[0] == 0x11 && kickmem_bank.baseaddr[2] == 0x4e && kickmem_bank.baseaddr[3] == 0xf9 && kickmem_bank.baseaddr[4] == 0x00) {
|
|
uae_u32 addr = kickmem_bank.baseaddr[5];
|
|
if (addr == 0x20 && currprefs.chipmem.size <= 0x200000 && currprefs.fastmem[0].size == 0)
|
|
map_banks_set(&kickmem_bank, addr, 8, 0);
|
|
if (addr == 0xC0 && currprefs.bogomem.size == 0)
|
|
map_banks_set(&kickmem_bank, addr, 8, 0);
|
|
if (addr == 0xF0)
|
|
map_banks_set(&kickmem_bank, addr, 8, 0);
|
|
}
|
|
|
|
if (a1000_bootrom)
|
|
a1000_handle_kickstart (1);
|
|
|
|
#ifdef AUTOCONFIG
|
|
expansion_map();
|
|
#endif
|
|
|
|
if (a3000_f0)
|
|
map_banks_set(&extendedkickmem_bank, 0xf0, 1, 0);
|
|
|
|
/* Map the chipmem into all of the lower 8MB */
|
|
map_overlay (1);
|
|
|
|
switch (extendedkickmem_type) {
|
|
case EXTENDED_ROM_KS:
|
|
map_banks_set(&extendedkickmem_bank, 0xE0, 8, 0);
|
|
break;
|
|
#ifdef CDTV
|
|
case EXTENDED_ROM_CDTV:
|
|
map_banks_set(&extendedkickmem_bank, 0xF0, extendedkickmem_bank.allocated_size == 2 * ROM_SIZE_512 ? 16 : 8, 0);
|
|
break;
|
|
#endif
|
|
#ifdef CD32
|
|
case EXTENDED_ROM_CD32:
|
|
map_banks_set(&extendedkickmem_bank, 0xE0, 8, 0);
|
|
break;
|
|
#endif
|
|
case EXTENDED_ROM_ALG:
|
|
map_banks_set(&extendedkickmem_bank, 0xF0, 4, 0);
|
|
alg_map_banks();
|
|
break;
|
|
}
|
|
|
|
#ifdef AUTOCONFIG
|
|
if ((need_uae_boot_rom (&currprefs) && currprefs.uaeboard == 0) || currprefs.uaeboard == 1)
|
|
map_banks_set(&rtarea_bank, rtarea_base >> 16, 1, 0);
|
|
#endif
|
|
|
|
if ((cloanto_rom || currprefs.cs_ksmirror_e0) && (currprefs.maprom != 0xe00000) && !extendedkickmem_type) {
|
|
map_banks(&kickmem_bank, 0xE0, 8, 0);
|
|
}
|
|
if (currprefs.cs_ksmirror_a8) {
|
|
if (extendedkickmem2a_bank.allocated_size && extendedkickmem2a_bank.baseaddr) {
|
|
map_banks_set(&extendedkickmem2a_bank, extendedkickmem2a_bank.start >> 16, extendedkickmem2a_bank.allocated_size >> 16, 0);
|
|
if (extendedkickmem2b_bank.allocated_size && extendedkickmem2b_bank.baseaddr) {
|
|
map_banks_set(&extendedkickmem2b_bank, extendedkickmem2b_bank.start >> 16, extendedkickmem2b_bank.allocated_size >> 16, 0);
|
|
}
|
|
} else {
|
|
struct romdata *rd = getromdatabypath (currprefs.cartfile);
|
|
if (!rd || rd->id != 63) {
|
|
if (extendedkickmem_type == EXTENDED_ROM_CD32 || extendedkickmem_type == EXTENDED_ROM_KS)
|
|
map_banks(&extendedkickmem_bank, 0xb0, 8, 0);
|
|
else
|
|
map_banks(&kickmem_bank, 0xb0, 8, 0);
|
|
map_banks(&kickmem_bank, 0xa8, 8, 0);
|
|
}
|
|
}
|
|
} else if (extendedkickmem2a_bank.allocated_size && extendedkickmem2a_bank.baseaddr) {
|
|
map_banks_set(&extendedkickmem2a_bank, extendedkickmem2a_bank.start >> 16, extendedkickmem2a_bank.allocated_size >> 16, 0);
|
|
if (extendedkickmem2b_bank.allocated_size && extendedkickmem2b_bank.baseaddr) {
|
|
map_banks_set(&extendedkickmem2b_bank, extendedkickmem2b_bank.start >> 16, extendedkickmem2b_bank.allocated_size >> 16, 0);
|
|
}
|
|
}
|
|
|
|
#ifdef ARCADIA
|
|
if (is_arcadia_rom (currprefs.romextfile) == ARCADIA_BIOS) {
|
|
if (_tcscmp (currprefs.romextfile, changed_prefs.romextfile) != 0)
|
|
memcpy (currprefs.romextfile, changed_prefs.romextfile, sizeof currprefs.romextfile);
|
|
if (_tcscmp (currprefs.cartfile, changed_prefs.cartfile) != 0)
|
|
memcpy (currprefs.cartfile, changed_prefs.cartfile, sizeof currprefs.cartfile);
|
|
arcadia_unmap ();
|
|
is_arcadia_rom (currprefs.romextfile);
|
|
is_arcadia_rom (currprefs.cartfile);
|
|
arcadia_map_banks ();
|
|
}
|
|
#endif
|
|
|
|
#ifdef ACTION_REPLAY
|
|
#ifdef ARCADIA
|
|
if (!arcadia_bios) {
|
|
#endif
|
|
action_replay_memory_reset ();
|
|
#ifdef ARCADIA
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
for (int i = 0; i < 2; i++) {
|
|
if (currprefs.custom_memory_sizes[i]) {
|
|
map_banks (i == 0 ? &custmem1_bank : &custmem2_bank,
|
|
currprefs.custom_memory_addrs[i] >> 16,
|
|
currprefs.custom_memory_sizes[i] >> 16, 0);
|
|
if (currprefs.custom_memory_mask[i]) {
|
|
for (int j = currprefs.custom_memory_addrs[i]; j & currprefs.custom_memory_mask[i]; j += currprefs.custom_memory_sizes[i]) {
|
|
map_banks(i == 0 ? &custmem1_bank : &custmem2_bank, j >> 16, currprefs.custom_memory_sizes[i] >> 16, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mem_hardreset) {
|
|
memory_clear ();
|
|
}
|
|
write_log (_T("memory init end\n"));
|
|
}
|
|
|
|
|
|
void memory_init (void)
|
|
{
|
|
init_mem_banks ();
|
|
virtualdevice_init ();
|
|
|
|
chipmem_bank.reserved_size = 0;
|
|
bogomem_bank.reserved_size = 0;
|
|
kickmem_bank.baseaddr = NULL;
|
|
extendedkickmem_bank.baseaddr = NULL;
|
|
extendedkickmem_bank.reserved_size = 0;
|
|
extendedkickmem2a_bank.baseaddr = NULL;
|
|
extendedkickmem2a_bank.reserved_size = 0;
|
|
extendedkickmem2b_bank.baseaddr = NULL;
|
|
extendedkickmem2b_bank.reserved_size = 0;
|
|
extendedkickmem_type = 0;
|
|
chipmem_bank.baseaddr = 0;
|
|
mem25bit_bank.reserved_size = mem25bit_bank.reserved_size = 0;
|
|
a3000lmem_bank.reserved_size = a3000hmem_bank.reserved_size = 0;
|
|
a3000lmem_bank.baseaddr = a3000hmem_bank.baseaddr = NULL;
|
|
bogomem_bank.baseaddr = NULL;
|
|
custmem1_bank.reserved_size = custmem2_bank.reserved_size = 0;
|
|
custmem1_bank.baseaddr = NULL;
|
|
custmem2_bank.baseaddr = NULL;
|
|
|
|
kickmem_bank.reserved_size = ROM_SIZE_512;
|
|
mapped_malloc (&kickmem_bank);
|
|
if (kickmem_bank.baseaddr) {
|
|
memset(kickmem_bank.baseaddr, 0, ROM_SIZE_512);
|
|
}
|
|
_tcscpy (currprefs.romfile, _T("<none>"));
|
|
currprefs.romextfile[0] = 0;
|
|
cpuboard_reset(1);
|
|
|
|
#ifdef ACTION_REPLAY
|
|
action_replay_unload (0);
|
|
action_replay_load ();
|
|
action_replay_init (1);
|
|
#ifdef ACTION_REPLAY_HRTMON
|
|
hrtmon_load ();
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
void memory_cleanup (void)
|
|
{
|
|
mapped_free(&mem25bit_bank);
|
|
mapped_free(&a3000lmem_bank);
|
|
mapped_free(&a3000hmem_bank);
|
|
mapped_free(&bogomem_bank);
|
|
mapped_free(&kickmem_bank);
|
|
xfree(a1000_bootrom);
|
|
mapped_free(&chipmem_bank);
|
|
mapped_free(&custmem1_bank);
|
|
mapped_free(&custmem2_bank);
|
|
mapped_free(&fakeuaebootrom_bank);
|
|
|
|
bogomem_bank.baseaddr = NULL;
|
|
kickmem_bank.baseaddr = NULL;
|
|
mem25bit_bank.baseaddr = NULL;
|
|
a3000lmem_bank.baseaddr = a3000hmem_bank.baseaddr = NULL;
|
|
a1000_bootrom = NULL;
|
|
a1000_kickstart_mode = 0;
|
|
chipmem_bank.baseaddr = NULL;
|
|
custmem1_bank.baseaddr = NULL;
|
|
custmem2_bank.baseaddr = NULL;
|
|
|
|
cpuboard_cleanup();
|
|
#ifdef ACTION_REPLAY
|
|
action_replay_cleanup();
|
|
#endif
|
|
#ifdef ARCADIA
|
|
arcadia_unmap ();
|
|
#endif
|
|
}
|
|
|
|
void set_roms_modified(void)
|
|
{
|
|
roms_modified = true;
|
|
}
|
|
|
|
void memory_hardreset (int mode)
|
|
{
|
|
if (mode + 1 > mem_hardreset)
|
|
mem_hardreset = mode + 1;
|
|
}
|
|
|
|
// do not map if it conflicts with custom banks
|
|
void map_banks_cond (addrbank *bank, int start, int size, int realsize)
|
|
{
|
|
for (int i = 0; i < MAX_CUSTOM_MEMORY_ADDRS; i++) {
|
|
int cstart = currprefs.custom_memory_addrs[i] >> 16;
|
|
if (!cstart)
|
|
continue;
|
|
int csize = currprefs.custom_memory_sizes[i] >> 16;
|
|
if (!csize)
|
|
continue;
|
|
if (start <= cstart && start + size >= cstart)
|
|
return;
|
|
if (cstart <= start && (cstart + size >= start || start + size > cstart))
|
|
return;
|
|
}
|
|
map_banks (bank, start, size, realsize);
|
|
}
|
|
|
|
#ifdef WITH_THREADED_CPU
|
|
|
|
struct addrbank_thread {
|
|
addrbank *orig;
|
|
addrbank ab;
|
|
};
|
|
|
|
#define MAX_THREAD_BANKS 200
|
|
static addrbank_thread *thread_banks[MAX_THREAD_BANKS];
|
|
addrbank *thread_mem_banks[MEMORY_BANKS];
|
|
static int thread_banks_used;
|
|
|
|
static void REGPARAM2 threadcpu_lput(uaecptr addr, uae_u32 l)
|
|
{
|
|
//write_log(_T("LPUT %08x %08x %08x\n"), addr, l, M68K_GETPC);
|
|
|
|
process_cpu_indirect_memory_write(addr, l, 2);
|
|
}
|
|
static void REGPARAM2 threadcpu_wput(uaecptr addr, uae_u32 w)
|
|
{
|
|
//write_log(_T("WPUT %08x %08x %08x\n"), addr, w, M68K_GETPC);
|
|
|
|
process_cpu_indirect_memory_write(addr, w, 1);
|
|
}
|
|
static void REGPARAM2 threadcpu_bput(uaecptr addr, uae_u32 b)
|
|
{
|
|
//write_log(_T("BPUT %08x %08x %08x\n"), addr, b, M68K_GETPC);
|
|
|
|
process_cpu_indirect_memory_write(addr, b, 0);
|
|
}
|
|
static uae_u32 REGPARAM2 threadcpu_lget(uaecptr addr)
|
|
{
|
|
uae_u32 v = process_cpu_indirect_memory_read(addr, 2);
|
|
|
|
//write_log(_T("LGET %08x %08x %08x\n"), addr, v, M68K_GETPC);
|
|
|
|
return v;
|
|
}
|
|
static uae_u32 REGPARAM2 threadcpu_wget(uaecptr addr)
|
|
{
|
|
uae_u32 v = process_cpu_indirect_memory_read(addr, 1);
|
|
|
|
//write_log(_T("WGET %08x %08x %08x\n"), addr, v, M68K_GETPC);
|
|
|
|
return v;
|
|
}
|
|
uae_u32 REGPARAM2 threadcpu_bget(uaecptr addr)
|
|
{
|
|
uae_u32 v = process_cpu_indirect_memory_read(addr, 0);
|
|
|
|
//write_log(_T("BGET %08x %08x %08x\n"), addr, v, M68K_GETPC);
|
|
|
|
return v;
|
|
}
|
|
|
|
static addrbank *get_bank_cpu_thread(addrbank *bank)
|
|
{
|
|
if ((bank->flags & ABFLAG_THREADSAFE) && !(bank->flags & ABFLAG_IO))
|
|
return bank;
|
|
if (bank == &dummy_bank)
|
|
return bank;
|
|
|
|
for (int i = 0; i < thread_banks_used; i++) {
|
|
if (thread_banks[i]->orig == bank) {
|
|
return &thread_banks[i]->ab;
|
|
}
|
|
}
|
|
struct addrbank_thread *at = thread_banks[thread_banks_used];
|
|
if (!at)
|
|
at = xcalloc(addrbank_thread, 1);
|
|
thread_banks[thread_banks_used++] = at;
|
|
at->orig = bank;
|
|
memcpy(&at->ab, bank, sizeof addrbank);
|
|
addrbank *tb = &at->ab;
|
|
tb->jit_read_flag = S_READ;
|
|
tb->jit_write_flag = S_WRITE;
|
|
tb->lget = threadcpu_lget;
|
|
tb->wget = threadcpu_wget;
|
|
tb->bget = threadcpu_bget;
|
|
tb->lput = threadcpu_lput;
|
|
tb->wput = threadcpu_wput;
|
|
tb->bput = threadcpu_bput;
|
|
// wgeti/lgeti should always point to real RAM
|
|
return tb;
|
|
}
|
|
#endif
|
|
|
|
static void set_memory_cacheable(int bnr, addrbank *bank)
|
|
{
|
|
uae_u8 cc = bank->flags >> ABFLAG_CACHE_SHIFT;
|
|
if (!currprefs.mmu_model) {
|
|
// if no MMU emulation, make sure chip RAM is not data cacheable
|
|
if (bank->flags & ABFLAG_CHIPRAM) {
|
|
cc &= ~(CACHE_ENABLE_DATA | CACHE_ENABLE_DATA_BURST);
|
|
}
|
|
}
|
|
ce_cachable[bnr] = cc;
|
|
}
|
|
|
|
|
|
static void map_banks2 (addrbank *bank, int start, int size, int realsize, int quick)
|
|
{
|
|
int bnr, old;
|
|
unsigned long int hioffs = 0, endhioffs = 0x100;
|
|
uae_u32 realstart = start;
|
|
addrbank *orig_bank = NULL;
|
|
|
|
bank->flags |= ABFLAG_MAPPED;
|
|
|
|
#ifdef WITH_THREADED_CPU
|
|
if (currprefs.cpu_thread) {
|
|
addrbank *b = bank;
|
|
bank = get_bank_cpu_thread(bank);
|
|
if (b != bank)
|
|
orig_bank = b;
|
|
}
|
|
#endif
|
|
|
|
if (quick <= 0)
|
|
old = debug_bankchange (-1);
|
|
flush_icache(3); /* Sure don't want to keep any old mappings around! */
|
|
#ifdef NATMEM_OFFSET
|
|
if (!quick)
|
|
delete_shmmaps (start << 16, size << 16);
|
|
#endif
|
|
|
|
if (!realsize)
|
|
realsize = size << 16;
|
|
|
|
if ((size << 16) < realsize) {
|
|
write_log (_T("Broken mapping, size=%x, realsize=%x\nStart is %x\n"),
|
|
size, realsize, start);
|
|
}
|
|
|
|
#ifndef ADDRESS_SPACE_24BIT
|
|
if (start >= 0x100) {
|
|
int real_left = 0;
|
|
for (bnr = start; bnr < start + size; bnr++) {
|
|
if (!real_left) {
|
|
realstart = bnr;
|
|
real_left = realsize >> 16;
|
|
#ifdef NATMEM_OFFSET
|
|
if (!quick)
|
|
add_shmmaps (realstart << 16, bank);
|
|
#endif
|
|
}
|
|
put_mem_bank (bnr << 16, bank, realstart << 16);
|
|
set_memory_cacheable(bnr, bank);
|
|
#ifdef WITH_THREADED_CPU
|
|
if (currprefs.cpu_thread) {
|
|
if (orig_bank)
|
|
put_mem_bank(bnr << 16, orig_bank, realstart << 16);
|
|
thread_mem_banks[bnr] = orig_bank;
|
|
}
|
|
#endif
|
|
real_left--;
|
|
}
|
|
if (quick <= 0)
|
|
debug_bankchange (old);
|
|
return;
|
|
}
|
|
#endif
|
|
if (last_address_space_24)
|
|
endhioffs = 0x10000;
|
|
#ifdef ADDRESS_SPACE_24BIT
|
|
endhioffs = 0x100;
|
|
#endif
|
|
for (hioffs = 0; hioffs < endhioffs; hioffs += 0x100) {
|
|
int real_left = 0;
|
|
for (bnr = start; bnr < start + size; bnr++) {
|
|
if (!real_left) {
|
|
realstart = bnr + hioffs;
|
|
real_left = realsize >> 16;
|
|
#ifdef NATMEM_OFFSET
|
|
if (!quick)
|
|
add_shmmaps (realstart << 16, bank);
|
|
#endif
|
|
}
|
|
put_mem_bank ((bnr + hioffs) << 16, bank, realstart << 16);
|
|
set_memory_cacheable(bnr + hioffs, bank);
|
|
#ifdef WITH_THREADED_CPU
|
|
if (currprefs.cpu_thread) {
|
|
if (orig_bank)
|
|
put_mem_bank((bnr + hioffs) << 16, bank, realstart << 16);
|
|
thread_mem_banks[bnr + hioffs] = orig_bank;
|
|
}
|
|
#endif
|
|
real_left--;
|
|
}
|
|
}
|
|
if (quick <= 0)
|
|
debug_bankchange (old);
|
|
fill_ce_banks ();
|
|
}
|
|
|
|
#ifdef WITH_PPC
|
|
static void ppc_generate_map_banks(addrbank *bank, int start, int size)
|
|
{
|
|
uae_u32 bankaddr = start << 16;
|
|
uae_u32 banksize = size << 16;
|
|
if (bank->sub_banks) {
|
|
uae_u32 subbankaddr = bankaddr;
|
|
addrbank *ab = NULL;
|
|
for (int i = 0; i <= 65536; i += MEMORY_MIN_SUBBANK) {
|
|
uae_u32 addr = bankaddr + i;
|
|
addrbank *ab2 = get_sub_bank(&addr);
|
|
if (ab2 != ab && ab != NULL) {
|
|
ppc_map_banks(subbankaddr, (bankaddr + i) - subbankaddr, ab->name, (ab->flags & ABFLAG_PPCIOSPACE) ? NULL : ab->baseaddr, ab == &dummy_bank);
|
|
subbankaddr = bankaddr + i;
|
|
}
|
|
ab = ab2;
|
|
}
|
|
} else {
|
|
uae_u8 *baseaddr = bank->baseaddr;
|
|
if (baseaddr) {
|
|
baseaddr += bankaddr - bank->start;
|
|
}
|
|
// ABFLAG_PPCIOSPACE = map as indirect even if baseaddr is non-NULL
|
|
ppc_map_banks(bankaddr, banksize, bank->name, (bank->flags & ABFLAG_PPCIOSPACE) ? NULL : baseaddr, bank == &dummy_bank);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
static addrbank *highram_temp_bank[65536 - 0x100];
|
|
|
|
void restore_banks(void)
|
|
{
|
|
for (int bnr = 0x100; bnr < 65536; bnr++) {
|
|
if (highram_temp_bank[bnr - 0x100]) {
|
|
map_banks(highram_temp_bank[bnr - 0x100], bnr, 1, 0);
|
|
} else {
|
|
map_banks(&dummy_bank, bnr, 1, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
void map_banks (addrbank *bank, int start, int size, int realsize)
|
|
{
|
|
if (start == 0xffffffff)
|
|
return;
|
|
|
|
if ((bank->jit_read_flag | bank->jit_write_flag) & S_N_ADDR) {
|
|
jit_n_addr_unsafe = 1;
|
|
}
|
|
|
|
if (start >= 0x100) {
|
|
int real_left = 0;
|
|
for (int bnr = start; bnr < start + size; bnr++) {
|
|
highram_temp_bank[bnr - 0x100] = bank;
|
|
}
|
|
if (currprefs.address_space_24)
|
|
return;
|
|
}
|
|
map_banks2 (bank, start, size, realsize, 0);
|
|
#ifdef WITH_PPC
|
|
ppc_generate_map_banks(bank, start, size);
|
|
#endif
|
|
}
|
|
|
|
bool validate_banks_z3(addrbank *bank, int start, int size)
|
|
{
|
|
if (start < 0x1000 || size <= 0) {
|
|
error_log(_T("Z3 invalid map_banks(%s) start=%08x size=%08x\n"), bank->name, start << 16, size << 16);
|
|
cpu_halt(CPU_HALT_AUTOCONFIG_CONFLICT);
|
|
return false;
|
|
}
|
|
if (size > 0x4000 || start + size > 0xf000) {
|
|
error_log(_T("Z3 invalid map_banks(%s) start=%08x size=%08x\n"), bank->name, start << 16, size << 16);
|
|
return false;
|
|
}
|
|
for (int i = start; i < start + size; i++) {
|
|
addrbank *ab = &get_mem_bank(start << 16);
|
|
if (ab != &dummy_bank && ab != bank) {
|
|
error_log(_T("Z3 map_banks(%s) attempting to override existing memory bank '%s' at %08x!\n"), bank->name, ab->name, i << 16);
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void map_banks_z3(addrbank *bank, int start, int size)
|
|
{
|
|
if (!validate_banks_z3(bank, start, size))
|
|
return;
|
|
map_banks(bank, start, size, 0);
|
|
}
|
|
|
|
bool validate_banks_z2(addrbank *bank, int start, int size)
|
|
{
|
|
if (start < 0x20 || (start >= 0xa0 && start < 0xe9) || start >= 0xf0) {
|
|
error_log(_T("Z2 map_banks(%s) with invalid start address %08X\n"), bank->name, start << 16);
|
|
cpu_halt(CPU_HALT_AUTOCONFIG_CONFLICT);
|
|
return false;
|
|
}
|
|
if (start >= 0xe9) {
|
|
if (start + size > 0xf0) {
|
|
error_log(_T("Z2 map_banks(%s) with invalid region %08x - %08X\n"), bank->name, start << 16, (start + size) << 16);
|
|
cpu_halt(CPU_HALT_AUTOCONFIG_CONFLICT);
|
|
return false;
|
|
}
|
|
} else {
|
|
if (start + size > 0xa0) {
|
|
error_log(_T("Z2 map_banks(%s) with invalid region %08x - %08X\n"), bank->name, start << 16, (start + size) << 16);
|
|
cpu_halt(CPU_HALT_AUTOCONFIG_CONFLICT);
|
|
return false;
|
|
}
|
|
}
|
|
if (size <= 0 || size > 0x80) {
|
|
error_log(_T("Z2 map_banks(%s) with invalid size %08x\n"), bank->name, size);
|
|
cpu_halt(CPU_HALT_AUTOCONFIG_CONFLICT);
|
|
return false;
|
|
}
|
|
for (int i = start; i < start + size; i++) {
|
|
addrbank *ab = &get_mem_bank(start << 16);
|
|
if (ab != &dummy_bank && bank != &dummy_bank) {
|
|
error_log(_T("Z2 map_banks(%s) attempting to override existing memory bank '%s' at %08x!\n"), bank->name, ab->name, i << 16);
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
void map_banks_z2 (addrbank *bank, int start, int size)
|
|
{
|
|
if (!validate_banks_z2(bank, start, size))
|
|
return;
|
|
map_banks (bank, start, size, 0);
|
|
}
|
|
|
|
uae_u32 map_banks_z2_autosize(addrbank *bank, int start)
|
|
{
|
|
uae_u32 size = expansion_board_size(bank);
|
|
if (!size) {
|
|
error_log(_T("Z2 map_banks(%s) %08x, invalid size!\n"), bank->name, start << 16);
|
|
return 0;
|
|
}
|
|
map_banks_z2(bank, start, size >> 16);
|
|
return size;
|
|
}
|
|
|
|
void map_banks_quick (addrbank *bank, int start, int size, int realsize)
|
|
{
|
|
map_banks2 (bank, start, size, realsize, 1);
|
|
#ifdef WITH_PPC
|
|
ppc_generate_map_banks(bank, start, size);
|
|
#endif
|
|
}
|
|
void map_banks_nojitdirect (addrbank *bank, int start, int size, int realsize)
|
|
{
|
|
map_banks2 (bank, start, size, realsize, -1);
|
|
#ifdef WITH_PPC
|
|
ppc_generate_map_banks(bank, start, size);
|
|
#endif
|
|
}
|
|
|
|
#ifdef SAVESTATE
|
|
|
|
/* memory save/restore code */
|
|
|
|
uae_u8 *save_bootrom(size_t *len)
|
|
{
|
|
if (!uae_boot_rom_type)
|
|
return 0;
|
|
*len = uae_boot_rom_size;
|
|
return rtarea_bank.baseaddr;
|
|
}
|
|
|
|
uae_u8 *save_cram(size_t *len)
|
|
{
|
|
*len = chipmem_bank.allocated_size;
|
|
return chipmem_bank.baseaddr;
|
|
}
|
|
|
|
uae_u8 *save_bram(size_t *len)
|
|
{
|
|
*len = bogomem_bank.allocated_size;
|
|
return bogomem_bank.baseaddr;
|
|
}
|
|
|
|
static uae_u8 *save_mem25bitram(size_t *len)
|
|
{
|
|
*len = mem25bit_bank.allocated_size;
|
|
return mem25bit_bank.baseaddr;
|
|
}
|
|
|
|
uae_u8 *save_a3000lram(size_t *len)
|
|
{
|
|
*len = a3000lmem_bank.allocated_size;
|
|
return a3000lmem_bank.baseaddr;
|
|
}
|
|
|
|
uae_u8 *save_a3000hram(size_t *len)
|
|
{
|
|
*len = a3000hmem_bank.allocated_size;
|
|
return a3000hmem_bank.baseaddr;
|
|
}
|
|
|
|
void restore_bootrom(int len, size_t filepos)
|
|
{
|
|
bootrom_filepos = filepos;
|
|
}
|
|
|
|
void restore_cram(int len, size_t filepos)
|
|
{
|
|
chip_filepos = filepos;
|
|
changed_prefs.chipmem.size = len;
|
|
}
|
|
|
|
void restore_bram(int len, size_t filepos)
|
|
{
|
|
bogo_filepos = filepos;
|
|
changed_prefs.bogomem.size = len;
|
|
}
|
|
|
|
void restore_a3000lram(int len, size_t filepos)
|
|
{
|
|
a3000lmem_filepos = filepos;
|
|
changed_prefs.mbresmem_low.size = len;
|
|
}
|
|
|
|
void restore_a3000hram(int len, size_t filepos)
|
|
{
|
|
a3000hmem_filepos = filepos;
|
|
changed_prefs.mbresmem_high.size = len;
|
|
}
|
|
|
|
uae_u8 *restore_rom(uae_u8 *src)
|
|
{
|
|
uae_u32 crc32, mem_start, mem_size, mem_type, version;
|
|
TCHAR *s, *romn;
|
|
int i, crcdet;
|
|
struct romlist *rl = romlist_getit ();
|
|
|
|
mem_start = restore_u32 ();
|
|
mem_size = restore_u32 ();
|
|
mem_type = restore_u32 ();
|
|
version = restore_u32 ();
|
|
crc32 = restore_u32 ();
|
|
romn = restore_string ();
|
|
crcdet = 0;
|
|
for (i = 0; i < romlist_count (); i++) {
|
|
if (rl[i].rd->crc32 == crc32 && crc32) {
|
|
if (zfile_exists (rl[i].path)) {
|
|
switch (mem_type)
|
|
{
|
|
case 0:
|
|
_tcsncpy (changed_prefs.romfile, rl[i].path, 255);
|
|
break;
|
|
case 1:
|
|
_tcsncpy (changed_prefs.romextfile, rl[i].path, 255);
|
|
break;
|
|
}
|
|
write_log (_T("ROM '%s' = '%s'\n"), romn, rl[i].path);
|
|
crcdet = 1;
|
|
} else {
|
|
write_log (_T("ROM '%s' = '%s' invalid rom scanner path!"), romn, rl[i].path);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
s = restore_string ();
|
|
if (!crcdet) {
|
|
if (zfile_exists (s)) {
|
|
switch (mem_type)
|
|
{
|
|
case 0:
|
|
_tcsncpy (changed_prefs.romfile, s, 255);
|
|
break;
|
|
case 1:
|
|
_tcsncpy (changed_prefs.romextfile, s, 255);
|
|
break;
|
|
}
|
|
write_log (_T("ROM detected (path) as '%s'\n"), s);
|
|
crcdet = 1;
|
|
}
|
|
}
|
|
xfree (s);
|
|
if (!crcdet)
|
|
write_log (_T("WARNING: ROM '%s' %d.%d (CRC32=%08x %08x-%08x) not found!\n"),
|
|
romn, version >> 16, version & 0xffff, crc32, mem_start, mem_start + mem_size - 1);
|
|
xfree (romn);
|
|
return src;
|
|
}
|
|
|
|
uae_u8 *save_rom(int first, size_t *len, uae_u8 *dstptr)
|
|
{
|
|
static int count;
|
|
uae_u8 *dst, *dstbak;
|
|
uae_u8 *mem_real_start;
|
|
uae_u32 version;
|
|
TCHAR *path;
|
|
int mem_start, mem_size, mem_type, saverom;
|
|
int i;
|
|
TCHAR tmpname[1000];
|
|
|
|
version = 0;
|
|
saverom = 0;
|
|
if (first)
|
|
count = 0;
|
|
for (;;) {
|
|
mem_type = count;
|
|
mem_size = 0;
|
|
switch (count) {
|
|
case 0: /* Kickstart ROM */
|
|
mem_start = 0xf80000;
|
|
mem_real_start = kickmem_bank.baseaddr;
|
|
mem_size = kickmem_bank.allocated_size;
|
|
path = currprefs.romfile;
|
|
/* 256KB or 512KB ROM? */
|
|
for (i = 0; i < mem_size / 2 - 4; i++) {
|
|
if (get_long(i + mem_start) != get_long(i + mem_start + mem_size / 2))
|
|
break;
|
|
}
|
|
if (i == mem_size / 2 - 4) {
|
|
mem_size /= 2;
|
|
mem_start += ROM_SIZE_256;
|
|
}
|
|
version = get_long(mem_start + 12); /* version+revision */
|
|
_stprintf (tmpname, _T("Kickstart %d.%d"), get_word(mem_start + 12), get_word(mem_start + 14));
|
|
break;
|
|
case 1: /* Extended ROM */
|
|
if (!extendedkickmem_type)
|
|
break;
|
|
mem_start = extendedkickmem_bank.start;
|
|
mem_real_start = extendedkickmem_bank.baseaddr;
|
|
mem_size = extendedkickmem_bank.allocated_size;
|
|
path = currprefs.romextfile;
|
|
version = get_long(mem_start + 12); /* version+revision */
|
|
if (version == 0xffffffff)
|
|
version = get_long(mem_start + 16);
|
|
_stprintf (tmpname, _T("Extended"));
|
|
break;
|
|
default:
|
|
return 0;
|
|
}
|
|
count++;
|
|
if (mem_size)
|
|
break;
|
|
}
|
|
if (dstptr)
|
|
dstbak = dst = dstptr;
|
|
else
|
|
dstbak = dst = xmalloc (uae_u8, 4 + 4 + 4 + 4 + 4 + 256 + 256 + mem_size);
|
|
save_u32 (mem_start);
|
|
save_u32 (mem_size);
|
|
save_u32 (mem_type);
|
|
save_u32 (version);
|
|
save_u32 (get_crc32 (mem_real_start, mem_size));
|
|
save_string (tmpname);
|
|
save_string (path);
|
|
if (saverom) {
|
|
for (i = 0; i < mem_size; i++)
|
|
*dst++ = get_byte(mem_start + i);
|
|
}
|
|
*len = dst - dstbak;
|
|
return dstbak;
|
|
}
|
|
|
|
#endif /* SAVESTATE */
|
|
|
|
static void REGPARAM2 empty_put(uaecptr addr, uae_u32 v)
|
|
{
|
|
}
|
|
|
|
void loadboardfile(addrbank *ab, struct boardloadfile * lf)
|
|
{
|
|
if (!ab->baseaddr)
|
|
return;
|
|
if (!lf->loadfile[0])
|
|
return;
|
|
struct zfile* zf = zfile_fopen(lf->loadfile, _T("rb"));
|
|
if (zf) {
|
|
int size = lf->filesize;
|
|
if (!size) {
|
|
size = ab->allocated_size;
|
|
}
|
|
else if (lf->loadoffset + size > ab->allocated_size)
|
|
size = ab->allocated_size - lf->loadoffset;
|
|
if (size > 0) {
|
|
int total = zfile_fread(ab->baseaddr + lf->loadoffset, 1, size, zf);
|
|
write_log(_T("Expansion file '%s': load %u bytes, offset %u, start addr %08x\n"),
|
|
lf->loadfile, total, lf->loadoffset, ab->start + lf->loadoffset);
|
|
}
|
|
zfile_fclose(zf);
|
|
}
|
|
else {
|
|
write_log(_T("Couldn't open expansion file '%s'\n"), lf->loadfile);
|
|
}
|
|
}
|
|
|
|
void initramboard(addrbank *ab, struct ramboard *rb)
|
|
{
|
|
ab->flags &= ~ABFLAG_NODMA;
|
|
if (rb->nodma)
|
|
ab->flags |= ABFLAG_NODMA;
|
|
if (!ab->baseaddr)
|
|
return;
|
|
loadboardfile(ab, &rb->lf);
|
|
if (rb->readonly) {
|
|
ab->lput = empty_put;
|
|
ab->wput = empty_put;
|
|
ab->bput = empty_put;
|
|
ab->jit_write_flag = 0;
|
|
}
|
|
}
|
|
|
|
|
|
/* memory helpers */
|
|
|
|
void memcpyha_safe(uaecptr dst, const uae_u8 *src, int size)
|
|
{
|
|
if (!addr_valid (_T("memcpyha"), dst, size))
|
|
return;
|
|
while (size--)
|
|
put_byte (dst++, *src++);
|
|
}
|
|
void memcpyha(uaecptr dst, const uae_u8 *src, int size)
|
|
{
|
|
while (size--)
|
|
put_byte (dst++, *src++);
|
|
}
|
|
void memcpyah_safe(uae_u8 *dst, uaecptr src, int size)
|
|
{
|
|
if (!addr_valid (_T("memcpyah"), src, size))
|
|
return;
|
|
while (size--)
|
|
*dst++ = get_byte (src++);
|
|
}
|
|
void memcpyah (uae_u8 *dst, uaecptr src, int size)
|
|
{
|
|
while (size--)
|
|
*dst++ = get_byte (src++);
|
|
}
|
|
uae_char *strcpyah_safe (uae_char *dst, uaecptr src, int maxsize)
|
|
{
|
|
uae_char *res = dst;
|
|
uae_u8 b;
|
|
dst[0] = 0;
|
|
do {
|
|
if (!addr_valid (_T("_tcscpyah"), src, 1))
|
|
return res;
|
|
b = get_byte (src++);
|
|
*dst++ = b;
|
|
*dst = 0;
|
|
maxsize--;
|
|
if (maxsize <= 1)
|
|
break;
|
|
} while (b);
|
|
return res;
|
|
}
|
|
uaecptr strcpyha_safe (uaecptr dst, const uae_char *src)
|
|
{
|
|
uaecptr res = dst;
|
|
uae_u8 b;
|
|
do {
|
|
if (!addr_valid (_T("_tcscpyha"), dst, 1))
|
|
return res;
|
|
b = *src++;
|
|
put_byte (dst++, b);
|
|
} while (b);
|
|
return res;
|
|
}
|
|
|
|
uae_u32 memory_get_longi(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return call_mem_get_func(ab->lgeti, addr);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_r + addr;
|
|
return do_get_mem_long((uae_u32 *)m);
|
|
}
|
|
}
|
|
uae_u32 memory_get_wordi(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return call_mem_get_func(ab->wgeti, addr);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_r + addr;
|
|
return do_get_mem_word((uae_u16*)m);
|
|
}
|
|
}
|
|
uae_u32 memory_get_long(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return call_mem_get_func(ab->lget, addr);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_r + addr;
|
|
return do_get_mem_long((uae_u32*)m);
|
|
}
|
|
}
|
|
uae_u32 memory_get_word(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return call_mem_get_func(ab->wget, addr);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_r + addr;
|
|
return do_get_mem_word((uae_u16*)m);
|
|
}
|
|
}
|
|
uae_u32 memory_get_byte(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return call_mem_get_func(ab->bget, addr);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_r + addr;
|
|
return *m;
|
|
}
|
|
}
|
|
|
|
void memory_put_long(uaecptr addr, uae_u32 v)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_w) {
|
|
call_mem_put_func(ab->lput, addr, v);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_w + addr;
|
|
do_put_mem_long((uae_u32*)m, v);
|
|
}
|
|
}
|
|
void memory_put_word(uaecptr addr, uae_u32 v)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_w) {
|
|
call_mem_put_func(ab->wput, addr, v);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_w + addr;
|
|
do_put_mem_word((uae_u16*)m, v);
|
|
}
|
|
}
|
|
void memory_put_byte(uaecptr addr, uae_u32 v)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_w) {
|
|
call_mem_put_func(ab->bput, addr, v);
|
|
} else {
|
|
uae_u8 *m;
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
m = ab->baseaddr_direct_w + addr;
|
|
*m = (uae_u8)v;
|
|
}
|
|
}
|
|
|
|
uae_u8 *memory_get_real_address(uaecptr addr)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return get_mem_bank(addr).xlateaddr(addr);
|
|
} else {
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
return ab->baseaddr_direct_r + addr;
|
|
}
|
|
}
|
|
|
|
int memory_valid_address(uaecptr addr, uae_u32 size)
|
|
{
|
|
addrbank *ab = &get_mem_bank(addr);
|
|
if (!ab->baseaddr_direct_r) {
|
|
return get_mem_bank(addr).check(addr, size);
|
|
}
|
|
addr -= ab->startaccessmask;
|
|
addr &= ab->mask;
|
|
return addr + size <= ab->allocated_size;
|
|
}
|
|
|
|
void dma_put_word(uaecptr addr, uae_u16 v)
|
|
{
|
|
addrbank* ab = &get_mem_bank(addr);
|
|
if (ab->flags & ABFLAG_NODMA)
|
|
return;
|
|
put_word(addr, v);
|
|
}
|
|
void dma_put_byte(uaecptr addr, uae_u8 v)
|
|
{
|
|
addrbank* ab = &get_mem_bank(addr);
|
|
if (ab->flags & ABFLAG_NODMA)
|
|
return;
|
|
put_byte(addr, v);
|
|
}
|
|
uae_u16 dma_get_word(uaecptr addr)
|
|
{
|
|
addrbank* ab = &get_mem_bank(addr);
|
|
if (ab->flags & ABFLAG_NODMA)
|
|
return 0xffff;
|
|
return get_word(addr);
|
|
}
|
|
uae_u8 dma_get_byte(uaecptr addr)
|
|
{
|
|
addrbank* ab = &get_mem_bank(addr);
|
|
if (ab->flags & ABFLAG_NODMA)
|
|
return 0xff;
|
|
return get_byte(addr);
|
|
}
|