mirror of
https://github.com/dingusdev/dingusppc.git
synced 2025-01-15 01:31:34 +00:00
212 lines
7.0 KiB
C++
212 lines
7.0 KiB
C++
/*
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
|
Copyright (C) 2018-24 divingkatae and maximum
|
|
(theweirdo) spatium
|
|
|
|
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/** Highspeed Memory Controller emulation.
|
|
|
|
Author: Max Poliakovski
|
|
*/
|
|
|
|
#include <devices/deviceregistry.h>
|
|
#include <devices/common/hwcomponent.h>
|
|
#include <devices/memctrl/hmc.h>
|
|
#include <loguru.hpp>
|
|
|
|
HMC::HMC() : MemCtrlBase()
|
|
{
|
|
this->name = "Highspeed Memory Controller";
|
|
|
|
supports_types(HWCompType::MEM_CTRL | HWCompType::MMIO_DEV);
|
|
|
|
// add memory mapped I/O region for the HMC control register
|
|
add_mmio_region(0x50F40000, 0x10000, this);
|
|
|
|
this->ctrl_reg = 0ULL;
|
|
this->bit_pos = 0;
|
|
}
|
|
|
|
uint32_t HMC::read(uint32_t rgn_start, uint32_t offset, int size)
|
|
{
|
|
if (!offset)
|
|
return !!(this->ctrl_reg & (1ULL << this->bit_pos++));
|
|
else
|
|
return 0; // FIXME: what should be returned for invalid offsets?
|
|
}
|
|
|
|
void HMC::write(uint32_t rgn_start, uint32_t offset, uint32_t value, int size)
|
|
{
|
|
uint64_t bit;
|
|
|
|
switch(offset) {
|
|
case 0:
|
|
bit = 1ULL << this->bit_pos++;
|
|
this->ctrl_reg = (value & 1) ? this->ctrl_reg | bit :
|
|
this->ctrl_reg & ~bit;
|
|
if (this->bit_pos >= HMC_CTRL_BITS) {
|
|
this->bit_pos = 0;
|
|
if (((this->ctrl_reg >> HMC_RAM_CFG_POS) & 3) != this->bank_config) {
|
|
this->bank_config = (this->ctrl_reg >> HMC_RAM_CFG_POS) & 3;
|
|
this->remap_ram_regions();
|
|
}
|
|
}
|
|
break;
|
|
case 8: // writing to HMCBase + 8 resets internal bit position
|
|
this->bit_pos = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void HMC::remap_ram_regions() {
|
|
uint32_t bank_b_addr;
|
|
|
|
switch (this->bank_config & 3) {
|
|
case BANK_CFG_128MB:
|
|
bank_b_addr = BANK_B_START;
|
|
break;
|
|
case BANK_CFG_2MB:
|
|
bank_b_addr = this->mb_bank_size + BANK_SIZE_2MB;
|
|
break;
|
|
case BANK_CFG_8MB:
|
|
bank_b_addr = this->mb_bank_size + BANK_SIZE_8MB;
|
|
break;
|
|
case BANK_CFG_32MB:
|
|
bank_b_addr = this->mb_bank_size + BANK_SIZE_32MB;
|
|
break;
|
|
}
|
|
|
|
if (this->bank_b_size && this->bank_b_start != bank_b_addr) {
|
|
AddressMapEntry *ref_entry = find_range(this->bank_b_start);
|
|
if (ref_entry) {
|
|
ref_entry->end = bank_b_addr + (ref_entry->end - ref_entry->start);
|
|
ref_entry->start = bank_b_addr;
|
|
|
|
this->bank_b_start = bank_b_addr;
|
|
LOG_F(INFO, "%s: successfully relocated bank B mem region to 0x%X",
|
|
this->name.c_str(), bank_b_addr);
|
|
} else
|
|
LOG_F(ERROR, "%s: failed to relocate bank B mem region to 0x%X",
|
|
this->name.c_str(), bank_b_addr);
|
|
}
|
|
}
|
|
|
|
int HMC::install_ram(uint32_t mb_bank_size, uint32_t bank_a_size, uint32_t bank_b_size) {
|
|
if (mb_bank_size != BANK_SIZE_4MB && mb_bank_size != BANK_SIZE_8MB) {
|
|
LOG_F(ERROR, "%s: invalid motherboard bank size %d", this->name.c_str(),
|
|
mb_bank_size);
|
|
return -1;
|
|
}
|
|
|
|
if (!bank_a_size && bank_b_size) {
|
|
LOG_F(ERROR, "%s: bank A can't be empty while bank B is not empty",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
|
|
if (!this->add_ram_region(BANK_MB_START, mb_bank_size)) {
|
|
LOG_F(ERROR, "%s: could not allocate motherboard RAM region!", this->name.c_str());
|
|
return -1;
|
|
}
|
|
|
|
this->mb_bank_start = BANK_MB_START;
|
|
this->mb_bank_size = mb_bank_size;
|
|
this->bank_a_start = -1;
|
|
this->bank_a_size = bank_a_size;
|
|
this->bank_b_start = -1;
|
|
this->bank_b_size = bank_b_size;
|
|
|
|
if (bank_a_size) {
|
|
// create alias for RAM bank A (required for memory sizing)
|
|
if (!this->add_ram_region(BANK_A_ALIAS, bank_a_size)) {
|
|
LOG_F(ERROR, "%s: could not allocate region for bank A!",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
|
|
this->bank_a_start = BANK_A_ALIAS;
|
|
|
|
uint32_t offset = 0;
|
|
uint32_t size = bank_a_size;
|
|
|
|
// make the main region for the bank A starting right after
|
|
// the motherboard bank
|
|
if (bank_a_size > BANK_SIZE_120MB) {
|
|
// For a full 128MB bank, the lower part of this region is hidden
|
|
// by the motherboard bank. Set up the partial mirror!
|
|
offset = mb_bank_size;
|
|
size = BANK_SIZE_120MB;
|
|
}
|
|
|
|
if (!this->add_mem_mirror_partial(mb_bank_size, BANK_A_ALIAS,
|
|
offset, size)) {
|
|
LOG_F(ERROR, "%s: could not create mirror for RAM bank A!",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
|
|
// Create additional aliases for bank A if the installed memory is
|
|
// smaller than 8 MB. That's because HWInit always searches those areas
|
|
// for the warm start signature.
|
|
if (bank_a_size < BANK_SIZE_8MB) {
|
|
for (uint32_t alias_start = BANK_B_START + bank_a_size - BANK_SIZE_8MB;
|
|
alias_start < BANK_B_START; alias_start += bank_a_size) {
|
|
if (!this->add_mem_mirror(alias_start, this->bank_a_start)) {
|
|
LOG_F(ERROR, "%s: could not create alias for RAM bank A!",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bank_b_size) {
|
|
if (!this->add_ram_region(BANK_B_START, bank_b_size)) {
|
|
LOG_F(ERROR, "%s: could not allocate region for bank B!",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
|
|
this->bank_b_start = BANK_B_START;
|
|
|
|
// Create additional aliases for bank B if the installed memory is
|
|
// smaller than 8 MB. That's because HWInit always searches those areas
|
|
// for the warm start signature.
|
|
if (bank_b_size < BANK_SIZE_8MB) {
|
|
for (uint32_t alias_start = BANK_A_ALIAS + bank_b_size - BANK_SIZE_8MB;
|
|
alias_start < BANK_A_ALIAS; alias_start += bank_b_size) {
|
|
if (!this->add_mem_mirror(alias_start, this->bank_b_start)) {
|
|
LOG_F(ERROR, "%s: could not create alias for RAM bank B!",
|
|
this->name.c_str());
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
this->remap_ram_regions();
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const DeviceDescription Hmc_Descriptor = {
|
|
HMC::create, {}, {}
|
|
};
|
|
|
|
REGISTER_DEVICE(HMC, Hmc_Descriptor);
|