2020-02-28 09:04:28 -07:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2022-01-16 21:30:43 +01:00
|
|
|
Copyright (C) 2018-22 divingkatae and maximum
|
2020-02-28 09:04:28 -07:00
|
|
|
(theweirdo) spatium
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-02-28 09:04:28 -07:00
|
|
|
(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/>.
|
|
|
|
*/
|
2019-08-21 08:33:01 +02:00
|
|
|
|
2022-01-16 21:30:43 +01:00
|
|
|
/** MPC106 (Grackle) emulation
|
2019-08-21 08:33:01 +02:00
|
|
|
|
2019-08-23 21:30:30 +02:00
|
|
|
Grackle IC is a combined memory and PCI controller manufactured by Motorola.
|
2019-08-21 08:33:01 +02:00
|
|
|
It's the central device in the Gossamer architecture.
|
|
|
|
Manual: https://www.nxp.com/docs/en/reference-manual/MPC106UM.pdf
|
|
|
|
|
2019-08-28 02:39:29 +02:00
|
|
|
This code emulates as much functionality as needed to run PowerMac Beige G3.
|
2019-08-21 08:33:01 +02:00
|
|
|
This implies that
|
|
|
|
- we only support address map B
|
|
|
|
- our virtual device reports revision 4.0 as expected by machine firmware
|
|
|
|
*/
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
#ifndef MPC106_H_
|
|
|
|
#define MPC106_H_
|
|
|
|
|
2021-10-23 20:17:47 +02:00
|
|
|
#include <devices/common/mmiodevice.h>
|
|
|
|
#include <devices/common/pci/pcidevice.h>
|
|
|
|
#include <devices/common/pci/pcihost.h>
|
|
|
|
#include <devices/memctrl/memctrlbase.h>
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
#include <cinttypes>
|
2022-07-17 05:33:06 +02:00
|
|
|
#include <memory>
|
2020-05-12 23:55:45 +05:00
|
|
|
#include <unordered_map>
|
2019-08-21 08:33:01 +02:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
class MPC106 : public MemCtrlBase, public PCIDevice, public PCIHost {
|
2019-08-21 08:33:01 +02:00
|
|
|
public:
|
|
|
|
MPC106();
|
2022-01-16 21:30:43 +01:00
|
|
|
~MPC106() = default;
|
2020-03-14 14:23:46 +01:00
|
|
|
|
2022-07-17 05:33:06 +02:00
|
|
|
static std::unique_ptr<HWComponent> create() {
|
|
|
|
return std::unique_ptr<MPC106>(new MPC106());
|
|
|
|
}
|
|
|
|
|
2022-08-22 03:16:31 -07:00
|
|
|
uint32_t read(uint32_t rgn_start, uint32_t offset, int size);
|
|
|
|
void write(uint32_t rgn_start, uint32_t offset, uint32_t value, int size);
|
2019-08-21 08:33:01 +02:00
|
|
|
|
2022-08-19 20:07:22 +02:00
|
|
|
int device_postinit();
|
|
|
|
|
2019-08-21 08:33:01 +02:00
|
|
|
protected:
|
|
|
|
/* PCI access */
|
|
|
|
uint32_t pci_read(uint32_t size);
|
|
|
|
void pci_write(uint32_t value, uint32_t size);
|
|
|
|
|
2019-08-23 21:30:30 +02:00
|
|
|
/* my own PCI configuration registers access */
|
|
|
|
uint32_t pci_cfg_read(uint32_t reg_offs, uint32_t size);
|
|
|
|
void pci_cfg_write(uint32_t reg_offs, uint32_t value, uint32_t size);
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
bool supports_io_space(void) {
|
|
|
|
return true;
|
|
|
|
};
|
2019-08-23 21:30:30 +02:00
|
|
|
|
2019-10-07 03:21:01 +02:00
|
|
|
void setup_ram(void);
|
|
|
|
|
2019-08-21 08:33:01 +02:00
|
|
|
private:
|
|
|
|
uint8_t my_pci_cfg_hdr[256] = {
|
2020-05-12 23:55:45 +05:00
|
|
|
0x57, 0x10, // vendor ID: Motorola
|
|
|
|
0x02, 0x00, // device ID: MPC106
|
|
|
|
0x06, 0x00, // PCI command
|
|
|
|
0x80, 0x00, // PCI status
|
|
|
|
0x40, // revision ID: 4.0
|
|
|
|
0x00, // standard programming
|
|
|
|
0x00, // subclass code: host bridge
|
|
|
|
0x06, // class code: bridge device
|
|
|
|
0x08, // cache line size
|
|
|
|
0x00, // latency timer
|
|
|
|
0x00, // header type
|
|
|
|
0x00, // BIST Control
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0x00, // Interrupt line
|
|
|
|
0x00, // Interrupt pin
|
|
|
|
0x00, // MIN GNT
|
|
|
|
0x00, // MAX LAT
|
|
|
|
0x00, // Bus number
|
|
|
|
0x00, // Subordinate bus number
|
|
|
|
0x00, // Discount counter
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, // Performance monitor command
|
|
|
|
0x00, 0x00, // Performance monitor mode control
|
2019-10-15 21:19:00 -07:00
|
|
|
0xFF, 0xFF,
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, 0x00, 0x00, 0x00, // Performance monitor counter 0
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Performance monitor counter 1
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Performance monitor counter 2
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Performance monitor counter 3
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0xFF,
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, 0x00, // Power mgt config 1
|
|
|
|
0x00, // Power mgt config 2
|
|
|
|
0xCD, // default value for ODCR
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, // Memory Starting Address
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Extended Memory Starting Address
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Memory Ending Address
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Extended Memory Ending Address
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, // Memory bank enable
|
2019-10-15 21:19:00 -07:00
|
|
|
0xFF, 0xFF,
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, // Memory page mode
|
2019-10-15 21:19:00 -07:00
|
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x10, 0x00, 0x00, 0xFF, // PICR1
|
|
|
|
0x0C, 0x06, 0x0C, 0x00, // PICR2
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0x00, // ECC single-bit error counter
|
|
|
|
0x00, // ECC single-bit error trigger
|
|
|
|
0x04, // Alternate OS visible paramaters 1
|
|
|
|
0x01, // Alternate OS visible paramaters 2
|
2019-10-15 21:19:00 -07:00
|
|
|
|
|
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x01, // Error enabling 1
|
|
|
|
0x00, // Error detection 1
|
2019-10-15 21:19:00 -07:00
|
|
|
0xFF,
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, // 60x bus error status
|
|
|
|
0x00, // Error enabling 2
|
|
|
|
0x00, // Error detection 2
|
2019-10-15 21:19:00 -07:00
|
|
|
0xFF,
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, // PCI bus error status
|
|
|
|
0x00, 0x00, 0x00, 0x00, // 60x/PCI ERROR address
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2019-10-15 21:48:31 -07:00
|
|
|
0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
|
|
0xFF,
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x42, 0x00, 0xFF, 0x0F, // Emulation support config 1
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Modified memory status (no clear)
|
|
|
|
0x20, 0x00, 0x00, 0x00, // Emulation support config 2
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Modified memory status (clear)
|
2019-10-15 21:19:00 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
0x00, 0x00, 0x02, 0xFF, // Memory ctrl config 1
|
|
|
|
0x03, 0x00, 0x00, 0x00, // Memory ctrl config 2
|
|
|
|
0x00, 0x00, 0x00, 0x00, // Memory ctrl config 3
|
|
|
|
0x00, 0x00, 0x10, 0x00 // Memory ctrl config 4
|
2019-08-21 08:33:01 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
uint32_t config_addr;
|
|
|
|
};
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
#endif
|