/*
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 .
*/
/** @file Constructs a Catalyst (Power Macintosh 7200) machine. */
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
static std::vector bandit1_irq_map = {
{nullptr , DEV_FUN(0x0B,0), IntSrc::BANDIT1},
{"pci_A1", DEV_FUN(0x0D,0), IntSrc::PCI_A},
{"pci_B1", DEV_FUN(0x0E,0), IntSrc::PCI_B},
{"pci_C1", DEV_FUN(0x0F,0), IntSrc::PCI_C},
{nullptr , DEV_FUN(0x10,0), }, // GrandCentral
};
int initialize_catalyst(std::string& id)
{
LOG_F(INFO, "Building machine Catalyst...");
PlatinumCtrl* platinum_obj;
PCIHost *pci_host = dynamic_cast(gMachineObj->get_comp_by_name("Bandit1"));
pci_host->set_irq_map(bandit1_irq_map);
// get (raw) pointer to the I/O controller
GrandCentral* gc_obj = dynamic_cast(gMachineObj->get_comp_by_name("GrandCentral"));
// connect GrandCentral I/O controller to the PCI1 bus
pci_host->pci_register_device(DEV_FUN(0x10,0), gc_obj);
// attach IOBus Device #1 0xF301A000
gMachineObj->add_device("BoardReg1", std::unique_ptr(
new BoardRegister("Board Register 1",
0x3F | // pull up all PRSNT bits
((GET_BIN_PROP("emmo") ^ 1) << 8) | // factory tests (active low)
(0 << 11) | // 2-bit box ID
0xE000U // pull up unused bits
)));
gc_obj->attach_iodevice(0, dynamic_cast(gMachineObj->get_comp_by_name("BoardReg1")));
// get (raw) pointer to the memory controller
platinum_obj = dynamic_cast(gMachineObj->get_comp_by_name("Platinum"));
// allocate ROM region
if (!platinum_obj->add_rom_region(0xFFC00000, 0x400000)) {
LOG_F(ERROR, "Could not allocate ROM region!");
return -1;
}
// plug 8MB RAM DIMM into slot #0
platinum_obj->insert_ram_dimm(0, DRAM_CAP_8MB);
// allocate and map physical RAM
platinum_obj->map_phys_ram();
std::string cpu = GET_STR_PROP("cpu");
if (cpu == "601") {
// init virtual CPU and request MPC601
ppc_cpu_init(platinum_obj, PPC_VER::MPC601, true, 7833600ULL);
}
else if (cpu == "750") {
// configure CPU clocks
uint64_t bus_freq = 50000000ULL;
uint64_t timebase_freq = bus_freq / 4;
// initialize virtual CPU and request MPC750 CPU aka G3
ppc_cpu_init(platinum_obj, PPC_VER::MPC750, false, timebase_freq);
// set CPU PLL ratio to 3.5
ppc_state.spr[SPR::HID1] = 0xE << 28;
}
return 0;
}
static const PropMap pm7200_settings = {
{"rambank1_size",
new IntProperty(16, vector({4, 8, 16, 32, 64, 128}))},
{"rambank2_size",
new IntProperty( 0, vector({0, 4, 8, 16, 32, 64, 128}))},
{"rambank3_size",
new IntProperty( 0, vector({0, 4, 8, 16, 32, 64, 128}))},
{"rambank4_size",
new IntProperty( 0, vector({0, 4, 8, 16, 32, 64, 128}))},
{"emmo",
new BinProperty(0)},
{"cpu",
new StrProperty("601", vector({"601", "750"}))},
};
static vector pm7200_devices = {
"Platinum", "Bandit1", "GrandCentral"
};
static const MachineDescription pm7200_descriptor = {
.name = "pm7200",
.description = "Power Macintosh 7200",
.devices = pm7200_devices,
.settings = pm7200_settings,
.init_func = &initialize_catalyst
};
REGISTER_MACHINE(pm7200, pm7200_descriptor);