mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-01 02:31:00 +00:00
414 lines
12 KiB
Plaintext
414 lines
12 KiB
Plaintext
//
|
|
// IIgsMemoryMapTests.mm
|
|
// Clock SignalTests
|
|
//
|
|
// Created by Thomas Harte on 25/10/2020.
|
|
// Copyright © 2020 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#import <XCTest/XCTest.h>
|
|
|
|
#include "../../../Machines/Apple/AppleIIgs/MemoryMap.hpp"
|
|
|
|
namespace {
|
|
// Test the ROM03-era mapping hardware.
|
|
struct MemoryMap: public Apple::IIgs::MemoryMap {
|
|
MemoryMap() : Apple::IIgs::MemoryMap(true) {}
|
|
};
|
|
}
|
|
|
|
@interface IIgsMemoryMapTests : XCTestCase
|
|
@end
|
|
|
|
@implementation IIgsMemoryMapTests {
|
|
MemoryMap _memoryMap;
|
|
std::vector<uint8_t> _ram;
|
|
std::vector<uint8_t> _rom;
|
|
}
|
|
|
|
- (void)setUp {
|
|
_ram.resize((128 + 8 * 1024) * 1024);
|
|
_rom.resize(256 * 1024);
|
|
_memoryMap.set_storage(_ram, _rom);
|
|
|
|
// If this isn't the first test run, RAM and ROM may have old values.
|
|
// Initialise to a known state.
|
|
memset(_ram.data(), 0, _ram.size());
|
|
memset(_rom.data(), 0, _rom.size());
|
|
}
|
|
|
|
- (void)write:(uint8_t)value address:(uint32_t)address {
|
|
const auto ®ion = MemoryMapRegion(_memoryMap, address);
|
|
XCTAssertFalse(region.flags & MemoryMap::Region::IsIO);
|
|
MemoryMapWrite(_memoryMap, region, address, &value);
|
|
}
|
|
|
|
- (uint8_t)readAddress:(uint32_t)address {
|
|
const auto ®ion = MemoryMapRegion(_memoryMap, address);
|
|
uint8_t value;
|
|
MemoryMapRead(region, address, &value);
|
|
return value;
|
|
}
|
|
|
|
- (void)testAllRAM {
|
|
// Disable IO/LC 'shadowing', to give linear memory up to bank $80.
|
|
_memoryMap.set_shadow_register(0x5f);
|
|
|
|
// Fill memory via the map.
|
|
for(int address = 0x00'0000; address < 0x80'0000; ++address) {
|
|
const uint8_t value = uint8_t(address ^ (address >> 8) ^ (address >> 16));
|
|
[self write:value address:address];
|
|
}
|
|
|
|
// Test by direct access.
|
|
for(int address = 0x00'0000; address < 0x80'0000; ++address) {
|
|
const uint8_t value = uint8_t(address ^ (address >> 8) ^ (address >> 16));
|
|
XCTAssertEqual([self readAddress:address], value);
|
|
}
|
|
}
|
|
|
|
- (void)testROMIsReadonly {
|
|
_rom[0] = 0xc0;
|
|
|
|
// Test that ROM can be read in the correct location.
|
|
XCTAssertEqual([self readAddress:0xfc'0000], 0xc0);
|
|
|
|
// Try writing to it, and check that nothing happened.
|
|
[self write:0xfc address:0xfc'0000];
|
|
XCTAssertEqual(_rom[0], 0xc0);
|
|
}
|
|
|
|
/// Tests that the same portion of ROM is visible in banks $00, $01, $e0 and $e1.
|
|
- (void)testROMVisibility {
|
|
_rom.back() = 0xa8;
|
|
auto test_bank = [self](uint32_t bank) {
|
|
const uint32_t address = bank | 0xffff;
|
|
XCTAssertEqual([self readAddress:address], 0xa8);
|
|
};
|
|
|
|
test_bank(0x00'0000);
|
|
test_bank(0x01'0000);
|
|
test_bank(0xe0'0000);
|
|
test_bank(0xe1'0000);
|
|
}
|
|
|
|
// TODO: edit and reenable shadowing tests below, once I clear up whether shadowing is based on
|
|
// logical or physical address.
|
|
|
|
/// Tests that writes to $00:$0400 and to $01:$0400 are subsequently visible at $e0:$0400 and $e1:$0400.
|
|
/*- (void)testShadowing {
|
|
[self write:0xab address:0x00'0400];
|
|
[self write:0xcd address:0x01'0400];
|
|
XCTAssertEqual([self readAddress:0xe0'0400], 0xab);
|
|
XCTAssertEqual([self readAddress:0xe1'0400], 0xcd);
|
|
}
|
|
|
|
/// Tests that a write to bank $00 which via the auxiliary switches is redirected to bank $01 is then
|
|
/// mirrored to $e1.
|
|
- (void)testAuxiliaryShadowing {
|
|
// Select the alternate text page 1.
|
|
_memoryMap.access(0xc001, false); // Set 80STORE.
|
|
_memoryMap.access(0xc055, false); // Set PAGE2.
|
|
// These two things together should enable auxiliary memory for text page 1.
|
|
// No, really.
|
|
|
|
// Enable shadowing of text page 1.
|
|
_memoryMap.set_shadow_register(0x00);
|
|
|
|
// Establish a different value in bank $e1, then write
|
|
// to bank $00 and check banks $01 and $e1.
|
|
[self write: 0xcb address:0xe1'0400];
|
|
[self write: 0xde address:0x00'0400];
|
|
|
|
XCTAssertEqual([self readAddress:0xe1'0400], 0xde);
|
|
XCTAssertEqual([self readAddress:0x01'0400], 0xde);
|
|
|
|
// Reset the $e1 page version and check all three detinations.
|
|
[self write: 0xcb address:0xe1'0400];
|
|
|
|
XCTAssertEqual([self readAddress:0xe1'0400], 0xcb);
|
|
XCTAssertEqual([self readAddress:0x00'0400], 0xde);
|
|
XCTAssertEqual([self readAddress:0x01'0400], 0xde);
|
|
}*/
|
|
|
|
- (void)testE0E1RAMConsistent {
|
|
// Do some random language card paging, to hit set_language_card.
|
|
_memoryMap.set_state_register(0x00);
|
|
_memoryMap.set_state_register(0xff);
|
|
|
|
[self write: 0x12 address:0xe0'0000];
|
|
[self write: 0x34 address:0xe1'0000];
|
|
|
|
XCTAssertEqual(_ram[_ram.size() - 128*1024], 0x12);
|
|
XCTAssertEqual(_ram[_ram.size() - 64*1024], 0x34);
|
|
}
|
|
|
|
- (void)testAuxiliarySwitches {
|
|
// Inhibit IO/LC 'shadowing'.
|
|
_memoryMap.set_shadow_register(0x40);
|
|
|
|
// Check that all writes and reads currently occur to main RAM.
|
|
XCTAssertEqual(_memoryMap.get_state_register() & 0xf0, 0x00);
|
|
for(int c = 0; c < 65536; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c], value);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Set writing to auxiliary memory.
|
|
// Reading should still be from main.
|
|
_memoryMap.access(0xc005, false);
|
|
XCTAssertEqual(_memoryMap.get_state_register() & 0xf0, 0x10);
|
|
for(int c = 0x0200; c < 0xc000; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c + 64*1024], value);
|
|
XCTAssertEqual([self readAddress:c], 0);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Switch reading and writing.
|
|
_memoryMap.access(0xc004, false);
|
|
_memoryMap.access(0xc003, false);
|
|
XCTAssertEqual(_memoryMap.get_state_register() & 0xf0, 0x20);
|
|
for(int c = 0x0200; c < 0xc000; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c], value);
|
|
XCTAssertEqual([self readAddress:c], 0);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Test main zero page.
|
|
for(int c = 0x0000; c < 0x0200; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c], value);
|
|
XCTAssertEqual([self readAddress:c], value);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Enable the alternate zero page.
|
|
_memoryMap.access(0xc009, false);
|
|
XCTAssertEqual(_memoryMap.get_state_register() & 0xf0, 0xa0);
|
|
for(int c = 0x0000; c < 0x0200; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c + 64*1024], value);
|
|
XCTAssertEqual([self readAddress:c], value);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Enable 80STORE and PAGE2 and test for access to the second video page.
|
|
_memoryMap.access(0xc001, false);
|
|
_memoryMap.access(0xc055, true);
|
|
XCTAssertEqual(_memoryMap.get_state_register() & 0xf0, 0xe0);
|
|
for(int c = 0x0400; c < 0x0800; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c + 64*1024], value);
|
|
XCTAssertEqual([self readAddress:c], value);
|
|
}
|
|
|
|
// Reset.
|
|
memset(_ram.data(), 0, 128*1024);
|
|
|
|
// Enable HIRES and test for access to the second video page.
|
|
_memoryMap.access(0xc057, true);
|
|
for(int c = 0x2000; c < 0x4000; c++) {
|
|
const uint8_t value = c ^ (c >> 8);
|
|
[self write:value address:c];
|
|
XCTAssertEqual(_ram[c + 64*1024], value);
|
|
XCTAssertEqual([self readAddress:c], value);
|
|
}
|
|
}
|
|
|
|
- (void)testJSONExamples {
|
|
NSArray<NSDictionary *> *const tests =
|
|
[NSJSONSerialization JSONObjectWithData:
|
|
[NSData dataWithContentsOfURL:
|
|
[[NSBundle bundleForClass:[self class]]
|
|
URLForResource:@"mm"
|
|
withExtension:@"json"
|
|
subdirectory:@"IIgs Memory Map"]]
|
|
options:0
|
|
error:nil];
|
|
|
|
[tests enumerateObjectsUsingBlock:^(NSDictionary * _Nonnull test, NSUInteger index, BOOL * _Nonnull stop) {
|
|
NSLog(@"Test index %lu", static_cast<unsigned long>(index));
|
|
|
|
// Apply state.
|
|
const bool highRes = [test[@"hires"] boolValue];
|
|
const bool lcw = [test[@"lcw"] boolValue];
|
|
const bool store80 = [test[@"80store"] boolValue];
|
|
const uint8_t shadow = [test[@"shadow"] integerValue];
|
|
const uint8_t state = [test[@"state"] integerValue];
|
|
|
|
_memoryMap.access(0xc056 + highRes, false);
|
|
_memoryMap.access(0xc080 + lcw, true);
|
|
_memoryMap.access(0xc080 + lcw, true);
|
|
_memoryMap.access(0xc000 + store80, false);
|
|
_memoryMap.set_shadow_register(shadow);
|
|
_memoryMap.set_state_register(state);
|
|
|
|
// Test results.
|
|
auto testMemory =
|
|
^(NSString *type, void (^ applyTest)(int logical, int physical, const MemoryMap::Region ®ion)) {
|
|
for(NSArray<NSNumber *> *region in test[type]) {
|
|
const auto logicalStart = [region[0] intValue];
|
|
const auto logicalEnd = [region[1] intValue];
|
|
const auto physicalStart = [region[2] intValue];
|
|
const auto physicalEnd = [region[3] intValue];
|
|
|
|
if(physicalEnd == physicalStart && physicalStart == 0) {
|
|
continue;
|
|
}
|
|
|
|
int physical = physicalStart;
|
|
for(int logical = logicalStart; logical < logicalEnd; logical++) {
|
|
const auto ®ion = self->_memoryMap.regions[self->_memoryMap.region_map[logical]];
|
|
|
|
// Don't worry about IO pages here; they'll be compared shortly.
|
|
if(!(region.flags & MemoryMap::Region::IsIO)) {
|
|
const auto ®ion = self->_memoryMap.regions[self->_memoryMap.region_map[logical]];
|
|
applyTest(logical, physical, region);
|
|
|
|
if(*stop) {
|
|
NSLog(@"Logical page %04x should be mapped to %@ physical %04x",
|
|
logical,
|
|
type,
|
|
physical);
|
|
|
|
NSLog(@"Stopping after first failure");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if(physical != physicalEnd) ++physical;
|
|
}
|
|
}
|
|
};
|
|
|
|
auto physicalOffset =
|
|
^(const uint8_t *pointer) {
|
|
// Check for a mapping to RAM.
|
|
if(pointer >= self->_ram.data() && pointer < &(*self->_ram.end())) {
|
|
int foundPhysical = int(pointer - self->_ram.data()) >> 8;
|
|
|
|
// This emulator maps a contiguous 8mb + 128kb of RAM such that the
|
|
// first 8mb resides up to physical location 0x8000, and the final
|
|
// 128kb sits from locatio 0xe000. So adjust for that here.
|
|
if(foundPhysical >= 0x8000) {
|
|
foundPhysical += 0xe000 - 0x8000;
|
|
}
|
|
|
|
return foundPhysical;
|
|
}
|
|
|
|
// Check for a mapping to ROM.
|
|
if(pointer >= self->_rom.data() && pointer < &(*self->_rom.end())) {
|
|
// This emulator uses a separate store for ROM, which sholud appear in
|
|
// the memory map from locatio 0xfc00.
|
|
return 0xfc00 + (int(pointer - self->_rom.data()) >> 8);
|
|
}
|
|
|
|
return -1;
|
|
};
|
|
|
|
// Test read pointers.
|
|
testMemory(@"read", ^(int logical, int physical, const MemoryMap::Region ®ion) {
|
|
XCTAssert(region.read != nullptr);
|
|
if(region.read == nullptr) {
|
|
*stop = YES;
|
|
return;
|
|
}
|
|
|
|
// Compare to correct value.
|
|
const int foundPhysical = physicalOffset(®ion.read[logical << 8]);
|
|
if(physical != foundPhysical) {
|
|
*stop = YES;
|
|
return;
|
|
}
|
|
});
|
|
|
|
// Test write pointers.
|
|
testMemory(@"write", ^(int logical, int physical, const MemoryMap::Region ®ion) {
|
|
// This emulator guards writes to ROM by setting those pointers to nullptr;
|
|
// so allow a nullptr write target if ROM is mapped here.
|
|
if(region.write == nullptr && physical >= 0xfc00) {
|
|
return;
|
|
}
|
|
|
|
XCTAssert(region.write != nullptr);
|
|
if(region.write == nullptr) {
|
|
*stop = YES;
|
|
return;
|
|
}
|
|
|
|
// Compare to correct value.
|
|
const int foundPhysical = physicalOffset(®ion.write[logical << 8]);
|
|
if(physical != foundPhysical) {
|
|
*stop = YES;
|
|
return;
|
|
}
|
|
});
|
|
|
|
// Test shadowed regions.
|
|
bool shouldBeShadowed = false;
|
|
int logical = 0;
|
|
for(NSNumber *next in test[@"shadowed"]) {
|
|
while(logical < [next intValue]) {
|
|
[[maybe_unused]] const auto ®ion =
|
|
self->_memoryMap.regions[self->_memoryMap.region_map[logical]];
|
|
const bool isShadowed =
|
|
IsShadowed(_memoryMap, region, (logical << 8));
|
|
|
|
XCTAssertEqual(
|
|
isShadowed,
|
|
shouldBeShadowed,
|
|
@"Logical page %04x %@ subject to shadowing", logical, shouldBeShadowed ? @"should be" : @"should not be");
|
|
|
|
++logical;
|
|
}
|
|
shouldBeShadowed ^= true;
|
|
}
|
|
|
|
// Test IO regions.
|
|
bool shouldBeIO = false;
|
|
logical = 0;
|
|
for(NSNumber *next in test[@"io"]) {
|
|
while(logical < [next intValue]) {
|
|
const auto ®ion =
|
|
self->_memoryMap.regions[self->_memoryMap.region_map[logical]];
|
|
|
|
// This emulator marks card pages as IO because it uses IO to mean
|
|
// "anything that isn't the built-in RAM". Just don't test card pages.
|
|
const bool isIO =
|
|
region.flags & MemoryMap::Region::IsIO &&
|
|
(((logical & 0xff) < 0xc1) || ((logical & 0xff) > 0xcf));
|
|
|
|
XCTAssertEqual(
|
|
isIO,
|
|
shouldBeIO,
|
|
@"Logical page %04x %@ marked as IO", logical, shouldBeIO ? @"should be" : @"should not be");
|
|
|
|
++logical;
|
|
}
|
|
shouldBeIO ^= true;
|
|
}
|
|
}];
|
|
}
|
|
|
|
@end
|