2019-02-28 21:21:18 -08:00
|
|
|
/* Copyright 2010-2019 Will Scullin <scullin@scullinsteel.com>
|
2013-10-10 11:03:07 -07:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
|
|
* the above copyright notice appear in all copies and that both that
|
|
|
|
* copyright notice and this permission notice appear in supporting
|
|
|
|
* documentation. No representations are made about the suitability of this
|
|
|
|
* software for any purpose. It is provided "as is" without express or
|
|
|
|
* implied warranty.
|
|
|
|
*/
|
|
|
|
|
2019-02-28 21:21:18 -08:00
|
|
|
import RAM from '../ram';
|
|
|
|
import { debug } from '../util';
|
2016-11-21 21:17:34 -08:00
|
|
|
|
2019-12-27 15:04:07 -08:00
|
|
|
export default function LanguageCard(io, rom) {
|
2013-10-10 11:03:07 -07:00
|
|
|
var _rom = rom;
|
|
|
|
var _bank1 = null;
|
|
|
|
var _bank2 = null;
|
|
|
|
var _ram = null;
|
|
|
|
|
|
|
|
var _readbsr = false;
|
|
|
|
var _writebsr = false;
|
|
|
|
var _bsr2 = false;
|
2017-03-05 14:51:59 -08:00
|
|
|
var _prewrite = false;
|
2013-10-10 11:03:07 -07:00
|
|
|
|
|
|
|
var _read1 = null;
|
|
|
|
var _read2 = null;
|
|
|
|
|
|
|
|
var _write1 = null;
|
|
|
|
var _write2 = null;
|
|
|
|
|
|
|
|
function _init() {
|
2019-12-27 15:04:07 -08:00
|
|
|
debug('Language card');
|
2016-11-27 17:28:49 -08:00
|
|
|
|
2013-10-10 11:03:07 -07:00
|
|
|
_bank1 = new RAM(0xd0, 0xdf);
|
|
|
|
_bank2 = new RAM(0xd0, 0xdf);
|
|
|
|
_ram = new RAM(0xe0, 0xff);
|
|
|
|
|
|
|
|
_write1 = _rom;
|
|
|
|
_write2 = _rom;
|
|
|
|
|
|
|
|
_read1 = _rom;
|
|
|
|
_read2 = _rom;
|
|
|
|
}
|
|
|
|
|
|
|
|
function _debug() {
|
2020-01-02 10:57:26 -08:00
|
|
|
// debug.apply(null, arguments);
|
2013-10-10 11:03:07 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_init();
|
|
|
|
|
|
|
|
var LOC = {
|
|
|
|
// Bank 2
|
|
|
|
READBSR2: 0x80,
|
|
|
|
WRITEBSR2: 0x81,
|
|
|
|
OFFBSR2: 0x82,
|
|
|
|
READWRBSR2: 0x83,
|
|
|
|
|
|
|
|
// Bank 1
|
|
|
|
READBSR1: 0x88,
|
|
|
|
WRITEBSR1: 0x89,
|
|
|
|
OFFBSR1: 0x8a,
|
|
|
|
READWRBSR1: 0x8b,
|
|
|
|
};
|
|
|
|
|
2017-12-02 20:45:02 -08:00
|
|
|
function _updateBanks() {
|
|
|
|
if (_readbsr) {
|
|
|
|
_read1 = _bsr2 ? _bank2 : _bank1;
|
|
|
|
_read2 = _ram;
|
|
|
|
} else {
|
|
|
|
_read1 = _rom;
|
|
|
|
_read2 = _rom;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_writebsr) {
|
|
|
|
_write1 = _bsr2 ? _bank2 : _bank1;
|
|
|
|
_write2 = _ram;
|
|
|
|
} else {
|
|
|
|
_write1 = rom;
|
|
|
|
_write2 = rom;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-05 14:51:59 -08:00
|
|
|
function _access(off, val) {
|
2017-08-20 10:46:10 -07:00
|
|
|
var readMode = val === undefined;
|
2013-10-10 11:03:07 -07:00
|
|
|
var result = 0;
|
2020-01-02 10:57:26 -08:00
|
|
|
switch (off & 0x8B) {
|
2020-11-25 17:28:37 -08:00
|
|
|
case LOC.READBSR2: // 0xC080
|
|
|
|
_readbsr = true;
|
|
|
|
_writebsr = false;
|
|
|
|
_bsr2 = true;
|
|
|
|
_prewrite = false;
|
|
|
|
_debug('Bank 2 Read');
|
|
|
|
break;
|
|
|
|
case LOC.WRITEBSR2: // 0xC081
|
|
|
|
_readbsr = false;
|
|
|
|
if (readMode) {
|
|
|
|
_writebsr = _prewrite;
|
|
|
|
}
|
|
|
|
_bsr2 = true;
|
|
|
|
_prewrite = readMode;
|
|
|
|
_debug('Bank 2 Write');
|
|
|
|
break;
|
|
|
|
case LOC.OFFBSR2: // 0xC082
|
|
|
|
_readbsr = false;
|
|
|
|
_writebsr = false;
|
|
|
|
_bsr2 = true;
|
|
|
|
_prewrite = false;
|
|
|
|
_debug('Bank 2 Off');
|
|
|
|
break;
|
|
|
|
case LOC.READWRBSR2: // 0xC083
|
|
|
|
_readbsr = true;
|
|
|
|
if (readMode) {
|
|
|
|
_writebsr = _prewrite;
|
|
|
|
}
|
|
|
|
_bsr2 = true;
|
|
|
|
_prewrite = readMode;
|
|
|
|
_debug('Bank 2 Read/Write');
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LOC.READBSR1: // 0xC088
|
|
|
|
_readbsr = true;
|
|
|
|
_writebsr = false;
|
|
|
|
_bsr2 = false;
|
|
|
|
_prewrite = false;
|
|
|
|
_debug('Bank 1 Read');
|
|
|
|
break;
|
|
|
|
case LOC.WRITEBSR1: // 0xC089
|
|
|
|
_readbsr = false;
|
|
|
|
if (readMode) {
|
|
|
|
_writebsr = _prewrite;
|
|
|
|
}
|
|
|
|
_bsr2 = false;
|
|
|
|
_prewrite = readMode;
|
|
|
|
_debug('Bank 1 Write');
|
|
|
|
break;
|
|
|
|
case LOC.OFFBSR1: // 0xC08A
|
|
|
|
_readbsr = false;
|
|
|
|
_writebsr = false;
|
|
|
|
_bsr2 = false;
|
|
|
|
_prewrite = false;
|
|
|
|
_debug('Bank 1 Off');
|
|
|
|
break;
|
|
|
|
case LOC.READWRBSR1: // 0xC08B
|
|
|
|
_readbsr = true;
|
|
|
|
if (readMode) {
|
|
|
|
_writebsr = _prewrite;
|
|
|
|
}
|
|
|
|
_bsr2 = false;
|
|
|
|
_prewrite = readMode;
|
|
|
|
_debug('Bank 1 Read/Write');
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2013-10-10 11:03:07 -07:00
|
|
|
}
|
2017-03-05 14:51:59 -08:00
|
|
|
|
2017-12-02 20:45:02 -08:00
|
|
|
_updateBanks();
|
2013-10-10 11:03:07 -07:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2016-11-27 17:28:49 -08:00
|
|
|
|
2013-10-10 11:03:07 -07:00
|
|
|
return {
|
|
|
|
start: function() {
|
|
|
|
return 0xd0;
|
|
|
|
},
|
|
|
|
end: function() {
|
|
|
|
return 0xff;
|
|
|
|
},
|
|
|
|
ioSwitch: function(off, val) {
|
|
|
|
return _access(off, val);
|
|
|
|
},
|
2019-12-29 19:13:59 -08:00
|
|
|
read: function(page, off) {
|
2013-10-10 11:03:07 -07:00
|
|
|
var result = 0;
|
|
|
|
if (page < 0xe0) {
|
2019-12-29 19:13:59 -08:00
|
|
|
result = _read1.read(page, off);
|
2013-10-10 11:03:07 -07:00
|
|
|
} else {
|
2019-12-29 19:13:59 -08:00
|
|
|
result = _read2.read(page, off);
|
2013-10-10 11:03:07 -07:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
write: function(page, off, val) {
|
|
|
|
if (page < 0xe0) {
|
|
|
|
_write1.write(page, off, val);
|
|
|
|
} else {
|
|
|
|
_write2.write(page, off, val);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
getState: function() {
|
|
|
|
return {
|
|
|
|
readbsr: _readbsr,
|
|
|
|
writebsr: _writebsr,
|
|
|
|
bsr2: _bsr2,
|
2017-03-05 14:51:59 -08:00
|
|
|
prewrite: _prewrite,
|
2017-12-02 20:45:02 -08:00
|
|
|
ram: _ram.getState(),
|
2013-10-10 11:03:07 -07:00
|
|
|
bank1: _bank1.getState(),
|
|
|
|
bank2: _bank2.getState()
|
|
|
|
};
|
|
|
|
},
|
|
|
|
setState: function(state) {
|
|
|
|
_readbsr = state.readbsr;
|
|
|
|
_writebsr = state.writebsr;
|
|
|
|
_bsr2 = state.bsr2;
|
2017-12-02 20:45:02 -08:00
|
|
|
_prewrite = state.prewrite;
|
|
|
|
_ram.setState(state.ram);
|
2013-10-10 11:03:07 -07:00
|
|
|
_bank1.setState(state.bank1);
|
|
|
|
_bank2.setState(state.bank2);
|
2017-12-02 20:45:02 -08:00
|
|
|
_updateBanks();
|
2013-10-10 11:03:07 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|