mirror of
https://github.com/dingusdev/dingusppc.git
synced 2025-01-02 06:30:06 +00:00
456 lines
15 KiB
C++
456 lines
15 KiB
C++
/*
|
|
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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/** @file BigMac Ethernet controller emulation. */
|
|
|
|
#include <devices/deviceregistry.h>
|
|
#include <devices/ethernet/bigmac.h>
|
|
#include <loguru.hpp>
|
|
|
|
BigMac::BigMac(uint8_t id) {
|
|
set_name("BigMac");
|
|
supports_types(HWCompType::MMIO_DEV | HWCompType::ETHER_MAC);
|
|
|
|
this->chip_id = id;
|
|
this->chip_reset();
|
|
}
|
|
|
|
void BigMac::chip_reset() {
|
|
this->event_mask = 0xFFFFU; // disable HW events causing on-chip interrupts
|
|
this->stat = 0;
|
|
|
|
this->phy_reset();
|
|
this->mii_reset();
|
|
this->srom_reset();
|
|
}
|
|
|
|
uint16_t BigMac::read(uint16_t reg_offset) {
|
|
switch (reg_offset) {
|
|
case BigMacReg::XIFC:
|
|
return this->tx_if_ctrl;
|
|
case BigMacReg::CHIP_ID:
|
|
return this->chip_id;
|
|
case BigMacReg::MIF_CSR:
|
|
return (this->mif_csr_old & ~Mif_Data_In) | (this->mii_in_bit << 3);
|
|
case BigMacReg::GLOB_STAT: {
|
|
uint16_t old_stat = this->stat;
|
|
this->stat = 0; // clear-on-read
|
|
return old_stat;
|
|
}
|
|
case BigMacReg::EVENT_MASK:
|
|
return this->event_mask;
|
|
case BigMacReg::SROM_CSR:
|
|
return (this->srom_csr_old & ~Srom_Data_In) | (this->srom_in_bit << 2);
|
|
case BigMacReg::TX_SW_RST:
|
|
return this->tx_reset;
|
|
case BigMacReg::TX_CONFIG:
|
|
return this->tx_config;
|
|
case BigMacReg::PEAK_ATT: {
|
|
uint8_t old_val = this->peak_attempts;
|
|
this->peak_attempts = 0; // clear-on-read
|
|
return old_val;
|
|
}
|
|
case BigMacReg::NC_CNT:
|
|
return this->norm_coll_cnt;
|
|
case BigMacReg::EX_CNT:
|
|
return this->excs_coll_cnt;
|
|
case BigMacReg::LT_CNT:
|
|
return this->late_coll_cnt;
|
|
case BigMacReg::RX_CONFIG:
|
|
return this->rx_config;
|
|
default:
|
|
LOG_F(WARNING, "%s: unimplemented register at 0x%X", this->name.c_str(),
|
|
reg_offset);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void BigMac::write(uint16_t reg_offset, uint16_t value) {
|
|
switch (reg_offset) {
|
|
case BigMacReg::XIFC:
|
|
this->tx_if_ctrl = value;
|
|
break;
|
|
case BigMacReg::TX_FIFO_CSR:
|
|
this->tx_fifo_enable = !!(value & 1);
|
|
this->tx_fifo_size = (((value >> 1) & 0xFF) + 1) << 7;
|
|
break;
|
|
case BigMacReg::TX_FIFO_TH:
|
|
this->tx_fifo_tresh = value;
|
|
break;
|
|
case BigMacReg::RX_FIFO_CSR:
|
|
this->rx_fifo_enable = !!(value & 1);
|
|
this->rx_fifo_size = (((value >> 1) & 0xFF) + 1) << 7;
|
|
break;
|
|
case BigMacReg::MIF_CSR:
|
|
if (value & Mif_Data_Out_En) {
|
|
// send bits one by one on each low-to-high transition of Mif_Clock
|
|
if (((this->mif_csr_old ^ value) & Mif_Clock) && (value & Mif_Clock))
|
|
this->mii_xmit_bit(!!(value & Mif_Data_Out));
|
|
} else {
|
|
if (((this->mif_csr_old ^ value) & Mif_Clock) && (value & Mif_Clock))
|
|
this->mii_rcv_bit();
|
|
}
|
|
this->mif_csr_old = value;
|
|
break;
|
|
case BigMacReg::EVENT_MASK:
|
|
this->event_mask = value;
|
|
break;
|
|
case BigMacReg::SROM_CSR:
|
|
if (value & Srom_Chip_Select) {
|
|
// exchange data on each low-to-high transition of Srom_Clock
|
|
if (((this->srom_csr_old ^ value) & Srom_Clock) && (value & Srom_Clock))
|
|
this->srom_xmit_bit(!!(value & Srom_Data_Out));
|
|
} else {
|
|
this->srom_reset();
|
|
}
|
|
this->srom_csr_old = value;
|
|
break;
|
|
case BigMacReg::TX_SW_RST:
|
|
if (value == 1) {
|
|
LOG_F(INFO, "%s: transceiver soft reset asserted", this->name.c_str());
|
|
this->tx_reset = 0; // acknowledge SW reset
|
|
}
|
|
break;
|
|
case BigMacReg::TX_CONFIG:
|
|
this->tx_config = value;
|
|
break;
|
|
case BigMacReg::NC_CNT:
|
|
this->norm_coll_cnt = value;
|
|
break;
|
|
case BigMacReg::NT_CNT:
|
|
this->net_coll_cnt = value;
|
|
break;
|
|
case BigMacReg::EX_CNT:
|
|
this->excs_coll_cnt = value;
|
|
break;
|
|
case BigMacReg::LT_CNT:
|
|
this->late_coll_cnt = value;
|
|
break;
|
|
case BigMacReg::RNG_SEED:
|
|
this->rng_seed = value;
|
|
break;
|
|
case BigMacReg::RX_SW_RST:
|
|
if (!value) {
|
|
LOG_F(INFO, "%s: receiver soft reset asserted", this->name.c_str());
|
|
}
|
|
break;
|
|
case BigMacReg::RX_CONFIG:
|
|
this->rx_config = value;
|
|
break;
|
|
case BigMacReg::MAC_ADDR_0:
|
|
case BigMacReg::MAC_ADDR_1:
|
|
case BigMacReg::MAC_ADDR_2:
|
|
this->mac_addr_flt[8 - ((reg_offset >> 4) & 0xF)] = value;
|
|
break;
|
|
case BigMacReg::RX_FRM_CNT:
|
|
this->rcv_frame_cnt = value;
|
|
break;
|
|
case BigMacReg::RX_LE_CNT:
|
|
this->len_err_cnt = value;
|
|
break;
|
|
case BigMacReg::RX_AE_CNT:
|
|
this->align_err_cnt = value;
|
|
break;
|
|
case BigMacReg::RX_FE_CNT:
|
|
this->fcs_err_cnt = value;
|
|
break;
|
|
case BigMacReg::RX_CVE_CNT:
|
|
this->cv_err_cnt = value;
|
|
break;
|
|
case BigMacReg::HASH_TAB_0:
|
|
case BigMacReg::HASH_TAB_1:
|
|
case BigMacReg::HASH_TAB_2:
|
|
case BigMacReg::HASH_TAB_3:
|
|
this->hash_table[(reg_offset >> 4) & 3] = value;
|
|
break;
|
|
default:
|
|
LOG_F(WARNING, "%s: unimplemented register at 0x%X is written with 0x%X",
|
|
this->name.c_str(), reg_offset, value);
|
|
}
|
|
}
|
|
|
|
// ================ Media Independent Interface (MII) emulation ================
|
|
bool BigMac::mii_rcv_value(uint16_t& var, uint8_t num_bits, uint8_t next_bit) {
|
|
var = (var << 1) | (next_bit & 1);
|
|
this->mii_bit_counter++;
|
|
if (this->mii_bit_counter >= num_bits) {
|
|
this->mii_bit_counter = 0;
|
|
return true; // all bits have been received -> return true
|
|
}
|
|
return false; // more bits expected
|
|
}
|
|
|
|
void BigMac::mii_rcv_bit() {
|
|
switch(this->mii_state) {
|
|
case MII_FRAME_SM::Preamble:
|
|
this->mii_in_bit = 1; // required for OSX
|
|
this->mii_reset();
|
|
break;
|
|
case MII_FRAME_SM::Turnaround:
|
|
this->mii_in_bit = 0;
|
|
this->mii_bit_counter = 16;
|
|
this->mii_state = MII_FRAME_SM::Read_Data;
|
|
break;
|
|
case MII_FRAME_SM::Read_Data:
|
|
if (this->mii_bit_counter) {
|
|
--this->mii_bit_counter;
|
|
this->mii_in_bit = (this->mii_data >> this->mii_bit_counter) & 1;
|
|
if (!this->mii_bit_counter) {
|
|
this->mii_state = MII_FRAME_SM::Preamble;
|
|
}
|
|
} else { // out of sync (shouldn't happen)
|
|
this->mii_reset();
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Stop:
|
|
this->mii_reset();
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "%s: unhandled state %d in mii_rcv_bit", this->name.c_str(),
|
|
this->mii_state);
|
|
this->mii_reset();
|
|
}
|
|
}
|
|
|
|
void BigMac::mii_xmit_bit(const uint8_t bit_val) {
|
|
switch(this->mii_state) {
|
|
case MII_FRAME_SM::Preamble:
|
|
if (bit_val) {
|
|
this->mii_bit_counter++;
|
|
if (this->mii_bit_counter >= 32) {
|
|
this->mii_state = MII_FRAME_SM::Start;
|
|
this->mii_in_bit = 1; // checked in OSX
|
|
this->mii_bit_counter = 0;
|
|
}
|
|
} else { // zero bit -> out of sync
|
|
this->mii_reset();
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Start:
|
|
if (this->mii_rcv_value(this->mii_start, 2, bit_val)) {
|
|
LOG_F(9, "MII_Start=0x%X", this->mii_start);
|
|
this->mii_state = MII_FRAME_SM::Opcode;
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Opcode:
|
|
if (this->mii_rcv_value(this->mii_opcode, 2, bit_val)) {
|
|
LOG_F(9, "MII_Opcode=0x%X", this->mii_opcode);
|
|
this->mii_state = MII_FRAME_SM::Phy_Address;
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Phy_Address:
|
|
if (this->mii_rcv_value(this->mii_phy_address, 5, bit_val)) {
|
|
LOG_F(9, "MII_PHY_Address=0x%X", this->mii_phy_address);
|
|
this->mii_state = MII_FRAME_SM::Reg_Address;
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Reg_Address:
|
|
if (this->mii_rcv_value(this->mii_reg_address, 5, bit_val)) {
|
|
LOG_F(9, "MII_REG_Address=0x%X", this->mii_reg_address);
|
|
|
|
if (this->mii_start != 1)
|
|
LOG_F(ERROR, "%s: unsupported frame type %d", this->name.c_str(),
|
|
this->mii_start);
|
|
if (this->mii_phy_address)
|
|
LOG_F(ERROR, "%s: unsupported PHY address %d", this->name.c_str(),
|
|
this->mii_phy_address);
|
|
switch (this->mii_opcode) {
|
|
case 1: // write
|
|
this->mii_state = MII_FRAME_SM::Turnaround;
|
|
break;
|
|
case 2: // read
|
|
this->mii_data = this->phy_reg_read(this->mii_reg_address);
|
|
this->mii_state = MII_FRAME_SM::Turnaround;
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "%s: invalid MII opcode %d", this->name.c_str(),
|
|
this->mii_opcode);
|
|
}
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Turnaround:
|
|
if (this->mii_rcv_value(this->mii_turnaround, 2, bit_val)) {
|
|
if (this->mii_turnaround != 2)
|
|
LOG_F(ERROR, "%s: unexpected turnaround 0x%X", this->name.c_str(),
|
|
this->mii_turnaround);
|
|
this->mii_state = MII_FRAME_SM::Write_Data;
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Write_Data:
|
|
if (this->mii_rcv_value(this->mii_data, 16, bit_val)) {
|
|
LOG_F(9, "%s: MII data received = 0x%X", this->name.c_str(),
|
|
this->mii_data);
|
|
this->phy_reg_write(this->mii_reg_address, this->mii_data);
|
|
this->mii_state = MII_FRAME_SM::Stop;
|
|
}
|
|
break;
|
|
case MII_FRAME_SM::Stop:
|
|
if (this->mii_rcv_value(this->mii_stop, 2, bit_val)) {
|
|
LOG_F(9, "MII_Stop=0x%X", this->mii_stop);
|
|
this->mii_reset();
|
|
}
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "%s: unhandled state %d in mii_xmit_bit", this->name.c_str(),
|
|
this->mii_state);
|
|
this->mii_reset();
|
|
}
|
|
}
|
|
|
|
void BigMac::mii_reset() {
|
|
mii_start = 0;
|
|
mii_opcode = 0;
|
|
mii_phy_address = 0;
|
|
mii_reg_address = 0;
|
|
mii_turnaround = 0;
|
|
mii_data = 0;
|
|
mii_stop = 0;
|
|
this->mii_bit_counter = 0;
|
|
this->mii_state = MII_FRAME_SM::Preamble;
|
|
}
|
|
|
|
// ===================== Ethernet PHY interface emulation =====================
|
|
void BigMac::phy_reset() {
|
|
// TODO: add PHY type property to be able to select another PHY (DP83843)
|
|
if (this->chip_id == EthernetCellId::Paddington) {
|
|
this->phy_oui = 0x1E0400; // LXT970 aka ST10040 PHY
|
|
this->phy_model = 0;
|
|
this->phy_rev = 0;
|
|
} else { // assume Heathrow with LXT907 PHY
|
|
this->phy_oui = 0; // LXT907 doesn't support MII, MDIO is pulled low
|
|
this->phy_model = 0;
|
|
this->phy_rev = 0;
|
|
}
|
|
this->phy_anar = 0xA1; // tell the world we support 10BASE-T and 100BASE-TX
|
|
}
|
|
|
|
uint16_t BigMac::phy_reg_read(uint8_t reg_num) {
|
|
switch(reg_num) {
|
|
case PHY_BMCR:
|
|
return this->phy_bmcr;
|
|
case PHY_BMSR:
|
|
return 0x7809; // value from LXT970 datasheet
|
|
case PHY_ID1:
|
|
return (this->phy_oui >> 6) & 0xFFFFU;
|
|
case PHY_ID2:
|
|
return ((this->phy_oui << 10) | (phy_model << 4) | phy_rev) & 0xFFFFU;
|
|
case PHY_ANAR:
|
|
return this->phy_anar;
|
|
default:
|
|
LOG_F(ERROR, "Reading unimplemented PHY register %d", reg_num);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void BigMac::phy_reg_write(uint8_t reg_num, uint16_t value) {
|
|
switch(reg_num) {
|
|
case PHY_BMCR:
|
|
if (value & 0x8000) {
|
|
LOG_F(INFO, "PHY reset requested");
|
|
value &= ~0x8000; // Reset bit is self-clearing
|
|
}
|
|
this->phy_bmcr = value;
|
|
break;
|
|
case PHY_ANAR:
|
|
this->phy_anar = value;
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "Writing unimplemented PHY register %d", reg_num);
|
|
}
|
|
}
|
|
|
|
// ======================== MAC Serial EEPROM emulation ========================
|
|
void BigMac::srom_reset() {
|
|
this->srom_csr_old = 0;
|
|
this->srom_bit_counter = 0;
|
|
this->srom_opcode = 0;
|
|
this->srom_address = 0;
|
|
this->srom_state = Srom_Start;
|
|
}
|
|
|
|
bool BigMac::srom_rcv_value(uint16_t& var, uint8_t num_bits, uint8_t next_bit) {
|
|
var = (var << 1) | (next_bit & 1);
|
|
this->srom_bit_counter++;
|
|
if (this->srom_bit_counter >= num_bits) {
|
|
this->srom_bit_counter = 0;
|
|
return true; // all bits have been received -> return true
|
|
}
|
|
return false; // more bits expected
|
|
}
|
|
|
|
void BigMac::srom_xmit_bit(const uint8_t bit_val) {
|
|
switch(this->srom_state) {
|
|
case Srom_Start:
|
|
if (bit_val)
|
|
this->srom_state = Srom_Opcode;
|
|
else
|
|
this->srom_reset();
|
|
break;
|
|
case Srom_Opcode:
|
|
if (this->srom_rcv_value(this->srom_opcode, 2, bit_val)) {
|
|
switch(this->srom_opcode) {
|
|
case 2: // read
|
|
this->srom_state = Srom_Address;
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "%s: unsupported SROM opcode %d", this->name.c_str(),
|
|
this->srom_opcode);
|
|
this->srom_reset();
|
|
}
|
|
}
|
|
break;
|
|
case Srom_Address:
|
|
if (this->srom_rcv_value(this->srom_address, 6, bit_val)) {
|
|
LOG_F(9, "SROM address received = 0x%X", this->srom_address);
|
|
this->srom_bit_counter = 16;
|
|
this->srom_state = Srom_Read_Data;
|
|
}
|
|
break;
|
|
case Srom_Read_Data:
|
|
if (this->srom_bit_counter) {
|
|
this->srom_bit_counter--;
|
|
this->srom_in_bit = (this->srom_data[this->srom_address] >> this->srom_bit_counter) & 1;
|
|
if (!this->srom_bit_counter) {
|
|
this->srom_address++;
|
|
this->srom_bit_counter = 16;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
LOG_F(ERROR, "%s: unhandled state %d in srom_xmit_bit", this->name.c_str(),
|
|
this->srom_state);
|
|
this->srom_reset();
|
|
}
|
|
}
|
|
|
|
static const DeviceDescription BigMac_Heathrow_Descriptor = {
|
|
BigMac::create_for_heathrow, {}, {}
|
|
};
|
|
|
|
static const DeviceDescription BigMac_Paddington_Descriptor = {
|
|
BigMac::create_for_paddington, {}, {}
|
|
};
|
|
|
|
REGISTER_DEVICE(BigMacHeathrow, BigMac_Heathrow_Descriptor);
|
|
REGISTER_DEVICE(BigMacPaddington, BigMac_Paddington_Descriptor);
|