2022-08-06 19:37:10 +02:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2024-04-15 14:12:49 +02:00
|
|
|
Copyright (C) 2018-24 divingkatae and maximum
|
2022-08-06 19:37:10 +02:00
|
|
|
(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 Athens clock generator emulation. */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Athens (Apple part# 343S1191) is a programmable clock generator ASIC
|
|
|
|
used in the PCI Power Macintosh computers.
|
|
|
|
It has two outputs: system clock and video aka dot clock. Both can be
|
|
|
|
programmed using HW jumpers or I2C interface.
|
|
|
|
This high-level emulator focuses on the Athen's I2C interface providing
|
|
|
|
methods for querying the resulting virtual frequences.
|
|
|
|
This is helpful especially for calculating video parameters.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <devices/common/i2c/athens.h>
|
|
|
|
#include <loguru.hpp>
|
|
|
|
|
2022-08-07 15:32:05 +02:00
|
|
|
#include <algorithm>
|
2022-08-06 19:37:10 +02:00
|
|
|
#include <cinttypes>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
AthensClocks::AthensClocks(uint8_t dev_addr)
|
|
|
|
{
|
2024-04-15 14:17:37 +02:00
|
|
|
set_name("Athens");
|
2022-08-06 19:37:10 +02:00
|
|
|
supports_types(HWCompType::I2C_DEV);
|
|
|
|
|
|
|
|
this->my_addr = dev_addr;
|
|
|
|
|
2024-04-15 14:12:49 +02:00
|
|
|
// This initialization is not prescribed
|
|
|
|
// but let's set them to acceptable values anyway
|
|
|
|
this->regs[AthensRegs::D2] = 2;
|
|
|
|
this->regs[AthensRegs::N2] = 2;
|
|
|
|
|
|
|
|
// set P2_MUX2 on power up as follows:
|
|
|
|
// - dot clock VCO is disabled
|
|
|
|
// - dot clock = reference clock / 2
|
2022-08-06 19:37:10 +02:00
|
|
|
this->regs[AthensRegs::P2_MUX2] = 0x62;
|
2024-04-15 14:37:22 +02:00
|
|
|
}
|
2024-04-15 14:12:49 +02:00
|
|
|
|
2024-04-15 14:37:22 +02:00
|
|
|
AthensClocks::AthensClocks(uint8_t dev_addr, const float crystal_freq)
|
|
|
|
: AthensClocks(dev_addr)
|
|
|
|
{
|
|
|
|
this->xtal_freq = crystal_freq;
|
2022-08-06 19:37:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void AthensClocks::start_transaction()
|
|
|
|
{
|
|
|
|
this->pos = 0; // reset read/write position
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AthensClocks::send_subaddress(uint8_t sub_addr)
|
|
|
|
{
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(INFO, "%s: subaddress set to 0x%X", this->name.c_str(), sub_addr);
|
2022-08-06 19:37:10 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AthensClocks::send_byte(uint8_t data)
|
|
|
|
{
|
|
|
|
switch (this->pos) {
|
|
|
|
case 0:
|
|
|
|
this->reg_num = data;
|
|
|
|
this->pos++;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
if (this->reg_num >= ATHENS_NUM_REGS) {
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: invalid register number %d", this->name.c_str(),
|
|
|
|
this->reg_num);
|
2022-08-06 19:37:10 +02:00
|
|
|
return false; // return NACK
|
|
|
|
}
|
|
|
|
this->regs[this->reg_num] = data;
|
|
|
|
break;
|
|
|
|
default:
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: too much data received!", this->name.c_str());
|
2022-08-06 19:37:10 +02:00
|
|
|
return false; // return NACK
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool AthensClocks::receive_byte(uint8_t* p_data)
|
|
|
|
{
|
2022-08-11 01:43:08 +02:00
|
|
|
*p_data = 0x41; // return my ID (value has been guessed)
|
|
|
|
return true;
|
2022-08-06 19:37:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int AthensClocks::get_sys_freq()
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int AthensClocks::get_dot_freq()
|
|
|
|
{
|
|
|
|
static std::vector<int> D2_commons = {7, 11, 13, 14, 15, 17, 23, 31};
|
|
|
|
static std::vector<int> N2_commons = {
|
|
|
|
22, 27, 28, 31, 35, 37, 38, 42, 49, 55, 56, 78, 125
|
|
|
|
};
|
|
|
|
|
2024-04-24 00:31:25 -07:00
|
|
|
if (this->regs[AthensRegs::P2_MUX2] & 0x80) {
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(INFO, "%s: dot clock disabled", this->name.c_str());
|
2022-08-11 01:43:08 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-08-06 19:37:10 +02:00
|
|
|
float out_freq = 0.0f;
|
|
|
|
|
|
|
|
int d2 = this->regs[AthensRegs::D2];
|
|
|
|
int n2 = this->regs[AthensRegs::N2];
|
|
|
|
|
|
|
|
int post_div = 1 << (3 - (this->regs[AthensRegs::P2_MUX2] & 3));
|
|
|
|
|
|
|
|
if (std::find(D2_commons.begin(), D2_commons.end(), d2) == D2_commons.end()) {
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: untested D2 value %d", this->name.c_str(), d2);
|
2022-08-06 19:37:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (std::find(N2_commons.begin(), N2_commons.end(), n2) == N2_commons.end()) {
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: untested N2 value %d", this->name.c_str(), d2);
|
2022-08-06 19:37:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int mux = (this->regs[AthensRegs::P2_MUX2] >> 4) & 3;
|
|
|
|
|
|
|
|
switch (mux) {
|
|
|
|
case 0: // clock source -> dot cock VCO
|
2024-04-15 14:37:22 +02:00
|
|
|
out_freq = this->xtal_freq * ((float)n2 / (float)(d2 * post_div));
|
2022-08-06 19:37:10 +02:00
|
|
|
break;
|
|
|
|
case 1: // clock source -> system clock VCO
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: system clock VCO not supported yet!", this->name.c_str());
|
2022-08-06 19:37:10 +02:00
|
|
|
break;
|
|
|
|
case 2: // clock source -> crystal frequency
|
2024-04-15 14:37:22 +02:00
|
|
|
out_freq = this->xtal_freq / post_div;
|
2022-08-06 19:37:10 +02:00
|
|
|
break;
|
|
|
|
case 3:
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(WARNING, "%s: attempt to use reserved Mux value!", this->name.c_str());
|
2022-08-06 19:37:10 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-04-15 14:17:37 +02:00
|
|
|
LOG_F(INFO, "%s: dot clock frequency set to %f Hz", this->name.c_str(), out_freq);
|
2024-04-15 14:12:49 +02:00
|
|
|
|
2022-08-06 19:37:10 +02:00
|
|
|
return static_cast<int>(out_freq + 0.5f);
|
|
|
|
}
|