/* DingusPPC - The Experimental PowerPC Macintosh emulator Copyright (C) 2018-23 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 Construct the Yosemite machine (Power Macintosh G3 Blue & White). */ #include #include #include #include #include #include #include static const std::vector grackle_irq_map = { {nullptr , DEV_FUN(0x00,0) }, // Grackle {nullptr , DEV_FUN(0x0D,0) }, // Dec21154Yosemite {"pci_J12", DEV_FUN(0x10,0), IntSrc::PCI_J12}, // GPU PCI slot, 66 MHz }; // 33 MHz PCI devices behind the DEC21154 PCI-to-PCI bridge static const std::vector pci_bridge_irq_map = { {"pci_FireWire", DEV_FUN(0x00,0), IntSrc::FIREWIRE}, {"pci_UltraATA", DEV_FUN(0x01,0), IntSrc::ATA }, {"pci_J11" , DEV_FUN(0x02,0), IntSrc::PCI_J11 }, {"pci_J10" , DEV_FUN(0x03,0), IntSrc::PCI_J10 }, {"pci_J9" , DEV_FUN(0x04,0), IntSrc::PCI_J9 }, {nullptr , DEV_FUN(0x05,0) }, // Heathrow {"pci_USB" , DEV_FUN(0x06,0), IntSrc::USB }, }; static void setup_ram_slot(std::string name, int i2c_addr, int capacity_megs) { if (!capacity_megs) return; gMachineObj->add_device(name, std::unique_ptr(new SpdSdram168(i2c_addr))); SpdSdram168* ram_dimm = dynamic_cast(gMachineObj->get_comp_by_name(name)); ram_dimm->set_capacity(capacity_megs); // register RAM DIMM with the I2C bus I2CBus* i2c_bus = dynamic_cast(gMachineObj->get_comp_by_type(HWCompType::I2C_HOST)); i2c_bus->register_device(i2c_addr, ram_dimm); } int initialize_yosemite(std::string& id) { LOG_F(INFO, "Building machine Yosemite..."); // get pointer to the memory controller/primary PCI bridge object MPC106* grackle_obj = dynamic_cast(gMachineObj->get_comp_by_name("Grackle")); grackle_obj->set_irq_map(grackle_irq_map); // get pointer to the bridge of the secondary PCI bus DecPciBridge *sec_bridge = dynamic_cast(gMachineObj->get_comp_by_name("Dec21154Yosemite")); sec_bridge->set_irq_map(pci_bridge_irq_map); // attach PCI devices to the PCI bridges grackle_obj->pci_register_device(DEV_FUN(16,0), dynamic_cast(gMachineObj->get_comp_by_name("AtiRage128"))); // 00:0D.0 PCI Bridge grackle_obj->pci_register_device(DEV_FUN(0x0D,0), dynamic_cast(sec_bridge)); // register CMD646U2 PCI Ultra ATA Controller sec_bridge->pci_register_device(DEV_FUN(1,0), dynamic_cast(gMachineObj->get_comp_by_name("CmdAta"))); sec_bridge->pci_register_device(DEV_FUN(5,0), dynamic_cast(gMachineObj->get_comp_by_name("Heathrow"))); // allocate ROM region if (!grackle_obj->add_rom_region(0xFFF00000, 0x100000)) { LOG_F(ERROR, "Could not allocate ROM region!"); return -1; } // configure RAM slots setup_ram_slot("RAM_DIMM_1", 0x50, GET_INT_PROP("rambank1_size")); setup_ram_slot("RAM_DIMM_2", 0x51, GET_INT_PROP("rambank2_size")); setup_ram_slot("RAM_DIMM_3", 0x52, GET_INT_PROP("rambank3_size")); setup_ram_slot("RAM_DIMM_4", 0x53, GET_INT_PROP("rambank4_size")); // configure CPU clocks uint64_t bus_freq = 66820000ULL; uint64_t timebase_freq = bus_freq / 4; // initialize virtual CPU and request MPC750 CPU aka G3 ppc_cpu_init(grackle_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 yosemite_settings = { {"rambank1_size", new IntProperty(256, vector({8, 16, 32, 64, 128, 256}))}, {"rambank2_size", new IntProperty( 0, vector({0, 8, 16, 32, 64, 128, 256}))}, {"rambank3_size", new IntProperty( 0, vector({0, 8, 16, 32, 64, 128, 256}))}, {"rambank4_size", new IntProperty( 0, vector({0, 8, 16, 32, 64, 128, 256}))}, {"emmo", new BinProperty(0)}, {"hdd_config", new StrProperty("CmdAta0:0")}, {"cdr_config", new StrProperty("Ide0:0")}, }; static vector yosemite_devices = { "Grackle", "Dec21154Yosemite", "CmdAta", "BurgundySnd", "Heathrow", "AtaHardDisk", "AtapiCdrom", }; static const MachineDescription yosemite_descriptor = { .name = "pmg3nw", .description = "Power Macintosh G3 Blue and White", .devices = yosemite_devices, .settings = yosemite_settings, .init_func = &initialize_yosemite }; REGISTER_MACHINE(pmg3nw, yosemite_descriptor);