mirror of
https://github.com/dingusdev/dingusppc.git
synced 2025-08-07 17:25:08 +00:00
Fix TBR range
Fixed an issue where TBR doesn't have full 64-bit range. The original calculation was 64 bit and ended with a ÷ 10^9. This means the max for the upper 32 bits is 2^32/10^9 = 4. The solution is to use a multiplication method that supports a 96 bit product. core/mathutils.h contains functions for that. TBR driving frequency is assumed to be less than 1 GHz. Some minor modification is required for future > 1 GHz support.
This commit is contained in:
49
core/mathutils.h
Normal file
49
core/mathutils.h
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
/*
|
||||||
|
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 <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef MATH_UTILS_H
|
||||||
|
#define MATH_UTILS_H
|
||||||
|
|
||||||
|
inline void _u32xu64(uint32_t a, uint64_t b, uint32_t &hi, uint64_t &lo)
|
||||||
|
{
|
||||||
|
uint64_t p0 = (b & 0xffffffff) * a;
|
||||||
|
uint64_t p1 = (b >> 32) * a;
|
||||||
|
lo = p0 + (p1 << 32);
|
||||||
|
hi = (p1 >> 32) + (lo < p0);
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void _u32xu64(uint32_t a, uint64_t b, uint64_t &hi, uint32_t &lo)
|
||||||
|
{
|
||||||
|
uint64_t p0 = (b & 0xffffffff) * a;
|
||||||
|
uint64_t p1 = (b >> 32) * a;
|
||||||
|
lo = p0;
|
||||||
|
hi = (p0 >> 32) + p1;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline void _u64xu64(uint64_t a, uint64_t b, uint64_t &hi, uint64_t &lo)
|
||||||
|
{
|
||||||
|
uint32_t p0h; uint64_t p0l; _u32xu64(b, a, p0h, p0l);
|
||||||
|
uint64_t p1h; uint32_t p1l; _u32xu64(b >> 32, a, p1h, p1l);
|
||||||
|
lo = p0l + ((uint64_t)p1l << 32);
|
||||||
|
hi = p0h + p1h + (lo < p0l);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // MATH_UTILS_H
|
@@ -161,7 +161,7 @@ extern uint64_t timebase_counter;
|
|||||||
extern uint64_t tbr_wr_timestamp;
|
extern uint64_t tbr_wr_timestamp;
|
||||||
extern uint64_t rtc_timestamp;
|
extern uint64_t rtc_timestamp;
|
||||||
extern uint64_t tbr_wr_value;
|
extern uint64_t tbr_wr_value;
|
||||||
extern uint64_t tbr_freq_hz;
|
extern uint32_t tbr_freq_ghz;
|
||||||
extern uint32_t rtc_lo, rtc_hi;
|
extern uint32_t rtc_lo, rtc_hi;
|
||||||
|
|
||||||
// Additional steps to prevent overflow?
|
// Additional steps to prevent overflow?
|
||||||
|
@@ -69,7 +69,7 @@ int icnt_factor;
|
|||||||
uint64_t tbr_wr_timestamp; // stores vCPU virtual time of the last TBR write
|
uint64_t tbr_wr_timestamp; // stores vCPU virtual time of the last TBR write
|
||||||
uint64_t rtc_timestamp; // stores vCPU virtual time of the last RTC write
|
uint64_t rtc_timestamp; // stores vCPU virtual time of the last RTC write
|
||||||
uint64_t tbr_wr_value; // last value written to the TBR
|
uint64_t tbr_wr_value; // last value written to the TBR
|
||||||
uint64_t tbr_freq_hz; // TBR/RTC driving frequency in Hz
|
uint32_t tbr_freq_ghz; // TBR/RTC driving frequency in GHz expressed as a 32 bit fraction less than 1.0.
|
||||||
uint64_t timebase_counter; // internal timebase counter
|
uint64_t timebase_counter; // internal timebase counter
|
||||||
uint32_t decr; // current value of PPC DEC register
|
uint32_t decr; // current value of PPC DEC register
|
||||||
uint32_t rtc_lo; // MPC601 RTC lower, counts nanoseconds
|
uint32_t rtc_lo; // MPC601 RTC lower, counts nanoseconds
|
||||||
@@ -766,7 +766,7 @@ void ppc_cpu_init(MemCtrlBase* mem_ctrl, uint32_t cpu_version, uint64_t tb_freq)
|
|||||||
tbr_wr_timestamp = 0;
|
tbr_wr_timestamp = 0;
|
||||||
rtc_timestamp = 0;
|
rtc_timestamp = 0;
|
||||||
tbr_wr_value = 0;
|
tbr_wr_value = 0;
|
||||||
tbr_freq_hz = tb_freq;
|
tbr_freq_ghz = (tb_freq << 32) / NS_PER_SEC;
|
||||||
|
|
||||||
exec_flags = 0;
|
exec_flags = 0;
|
||||||
|
|
||||||
|
@@ -22,6 +22,7 @@ along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|||||||
// General opcodes for the processor - ppcopcodes.cpp
|
// General opcodes for the processor - ppcopcodes.cpp
|
||||||
|
|
||||||
#include <core/timermanager.h>
|
#include <core/timermanager.h>
|
||||||
|
#include <core/mathutils.h>
|
||||||
#include "ppcemu.h"
|
#include "ppcemu.h"
|
||||||
#include "ppcmmu.h"
|
#include "ppcmmu.h"
|
||||||
#include <array>
|
#include <array>
|
||||||
@@ -895,7 +896,7 @@ void dppc_interpreter::ppc_mfspr() {
|
|||||||
static inline uint64_t calc_tbr_value()
|
static inline uint64_t calc_tbr_value()
|
||||||
{
|
{
|
||||||
uint64_t diff = get_virt_time_ns() - tbr_wr_timestamp;
|
uint64_t diff = get_virt_time_ns() - tbr_wr_timestamp;
|
||||||
uint64_t tbr_inc = diff * tbr_freq_hz / NS_PER_SEC;
|
uint64_t tbr_inc; uint32_t tbr_inc_lo; _u32xu64(tbr_freq_ghz, diff, tbr_inc, tbr_inc_lo);
|
||||||
return (tbr_wr_value + tbr_inc);
|
return (tbr_wr_value + tbr_inc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user