/* DingusPPC - The Experimental PowerPC Macintosh emulator Copyright (C) 2018-22 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 . */ /** Constructs a Gazelle style machine. This family includes Power Macintosh 5500, 6500 and Twentieth Anniversary Mac. */ #include #include #include #include #include #include #include #include #include #include static std::vector psx_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), }, // OHare {"pci_E1", DEV_FUN(0x11,0), IntSrc::PCI_E}, {"pci_F1", DEV_FUN(0x12,0), IntSrc::PCI_F}, }; // TODO: move it to /cpu/ppc int get_cpu_pll_value(const uint64_t cpu_freq_hz) { switch (cpu_freq_hz / 1000000) { case 225: return 7; // 4.5:1 ratio case 250: return 11; // 5:1 ratio case 275: return 9; // 5.5:1 ratio default: ABORT_F("Unsupported CPU frequency %llu Hz", cpu_freq_hz); } } int initialize_gazelle(std::string& id) { LOG_F(INFO, "Building machine Gazelle..."); PCIHost *pci_host = dynamic_cast(gMachineObj->get_comp_by_name("PsxPci1")); pci_host->set_irq_map(psx_irq_map); // register O'Hare I/O controller with the main PCI bus pci_host->pci_register_device( DEV_FUN(0x10,0), dynamic_cast(gMachineObj->get_comp_by_name("OHare"))); PsxCtrl* psx_obj = dynamic_cast(gMachineObj->get_comp_by_name("Psx")); // allocate ROM region if (!psx_obj->add_rom_region(0xFFC00000, 0x400000)) { LOG_F(ERROR, "Could not allocate ROM region!"); return -1; } // insert RAM DIMMs psx_obj->insert_ram_dimm(0, GET_INT_PROP("rambank1_size") * DRAM_CAP_1MB); psx_obj->insert_ram_dimm(1, GET_INT_PROP("rambank2_size") * DRAM_CAP_1MB); psx_obj->insert_ram_dimm(2, GET_INT_PROP("rambank3_size") * DRAM_CAP_1MB); psx_obj->insert_ram_dimm(3, GET_INT_PROP("rambank4_size") * DRAM_CAP_1MB); psx_obj->insert_ram_dimm(4, GET_INT_PROP("rambank5_size") * DRAM_CAP_1MB); // allocate and map physical RAM psx_obj->map_phys_ram(); // configure CPU clocks uint64_t bus_freq = 50000000ULL; uint64_t timebase_freq = bus_freq / 4; // init virtual CPU and request MPC603ev ppc_cpu_init(psx_obj, PPC_VER::MPC603EV, false, timebase_freq); // CPU frequency is hardcoded to 225 MHz for now ppc_state.spr[SPR::HID1] = get_cpu_pll_value(225000000) << 28; return 0; } static const PropMap pm6500_settings = { {"rambank1_size", new IntProperty(32, std::vector({ 4, 8, 16, 32}))}, {"rambank2_size", new IntProperty( 0, std::vector({0, 4, 8, 16, 32}))}, {"rambank3_size", new IntProperty( 0, std::vector({0, 4, 8, 16, 32}))}, {"rambank4_size", new IntProperty( 0, std::vector({0, 4, 8, 16, 32}))}, {"rambank5_size", new IntProperty( 0, std::vector({0, 4, 8, 16, 32}))}, {"emmo", new BinProperty(0)}, }; static std::vector pm6500_devices = { "Psx", "PsxPci1", "OHare" }; static const MachineDescription pm6500_descriptor = { .name = "pm6500", .description = "Power Macintosh 6500", .devices = pm6500_devices, .settings = pm6500_settings, .init_func = &initialize_gazelle }; REGISTER_MACHINE(pm6500, pm6500_descriptor);