fixed some counters problems, should be more smooth and stable.

one of the problems i'm having is that rcntUpdate() should be called more-often, but i don't know where i should put it...

git-svn-id: http://pcsx2-playground.googlecode.com/svn/trunk@238 a6443dda-0b58-4228-96e9-037be469359c
This commit is contained in:
cottonvibes 2008-10-26 04:12:27 +00:00 committed by Gregory Hainaut
parent d72cacbdd4
commit 4c7550971e
2 changed files with 30 additions and 29 deletions

View File

@ -139,12 +139,12 @@ u64 GetCPUTicks()
void UpdateVSyncRate() { void UpdateVSyncRate() {
counters[4].count = 0; counters[4].count = 0; // Set number of scanlines/hblanks to zero
counters[4].mode = MODE_HRENDER; // Counter 4 takes care of hBlanks counters[4].mode = MODE_HRENDER; // Counter 4 takes care of scanlines, so set the mode to HRENDER (drawing part of scanline)
counters[4].sCycle = cpuRegs.cycle; // Update Counter 4's Start Cycle counters[4].sCycle = cpuRegs.cycle; // Update Counter 4's Start Cycle to match CPU's cycle
counters[5].mode = MODE_VRENDER; // Counter 5 takes care of vSync/vBlanks counters[5].mode = MODE_VRENDER; // Counter 5 takes care of vSync/vBlanks
counters[5].rate = MODE_VBLANK1; counters[5].rate = MODE_VBLANK1; // Rate is just used to alternate between vBlank1 and vBlank2 modes (because of interlacing, one half-frame can have a different number of vBlank scanlines)
if (Config.CustomFps > 0) { if (Config.CustomFps > 0) {
iTicks = GetTickFrequency() / Config.CustomFps; iTicks = GetTickFrequency() / Config.CustomFps;
@ -381,16 +381,28 @@ __forceinline void VBlankEnd()
SysUpdate(); // check for and handle keyevents SysUpdate(); // check for and handle keyevents
} }
#define hScanlineOptimization(diff, compareValue, incrementCounters) { \
if (diff >= compareValue) { \
u32 increment = (diff / compareValue); \
SysPrintf("Counters Optimization\n"); \
counters[4].count += increment; \
if (incrementCounters) { \
/*Update counters using the hblank as the clock*/ \
if ((counters[0].mode & 0x83) == 0x83) counters[0].count += (increment * HBLANK_COUNTER_SPEED); \
if ((counters[1].mode & 0x83) == 0x83) counters[1].count += (increment * HBLANK_COUNTER_SPEED); \
if ((counters[2].mode & 0x83) == 0x83) counters[2].count += (increment * HBLANK_COUNTER_SPEED); \
if ((counters[3].mode & 0x83) == 0x83) counters[3].count += (increment * HBLANK_COUNTER_SPEED); \
} \
} \
}
int hScanline() int hScanline()
{ {
u32 difference = (cpuRegs.cycle - counters[4].sCycle); u32 difference = (cpuRegs.cycle - counters[4].sCycle);
if (counters[4].mode & MODE_HBLANK) { //HBLANK Start if (counters[4].mode & MODE_HBLANK) { //HBLANK Start
if (difference >= HBLANK_TIME_ ) { if (difference >= HBLANK_TIME_ ) {
if (difference >= (SCANLINE_ + HBLANK_TIME_)) { //hScanlineOptimization(difference, (SCANLINE_ + HBLANK_TIME_), 1);
//SysPrintf("Counters Optimization 1\n");
counters[4].count += (cpuRegs.cycle - counters[4].sCycle) / (SCANLINE_ + HBLANK_TIME_);
}
rcntStartGate(0); rcntStartGate(0);
psxCheckStartGate(0); psxCheckStartGate(0);
counters[4].sCycle = cpuRegs.cycle; counters[4].sCycle = cpuRegs.cycle;
@ -399,10 +411,7 @@ int hScanline()
} }
else { //HBLANK END / HRENDER Begin else { //HBLANK END / HRENDER Begin
if (difference >= (HRENDER_TIME_)) { if (difference >= (HRENDER_TIME_)) {
if (difference >= (SCANLINE_ + HRENDER_TIME_)) { //hScanlineOptimization(difference, (SCANLINE_ + HRENDER_TIME_), 0);
//SysPrintf("Counters Optimization 2\n");
counters[4].count += (cpuRegs.cycle - counters[4].sCycle) / (SCANLINE_ + HRENDER_TIME_);
}
if (CSRw & 0x4) GSCSRr |= 4; // signal if (CSRw & 0x4) GSCSRr |= 4; // signal
if (!(GSIMR&0x400)) gsIrq(); if (!(GSIMR&0x400)) gsIrq();
if (gates) rcntEndGate(0); if (gates) rcntEndGate(0);
@ -411,10 +420,10 @@ int hScanline()
counters[4].sCycle = cpuRegs.cycle; counters[4].sCycle = cpuRegs.cycle;
counters[4].mode ^= MODE_HBLANK; counters[4].mode ^= MODE_HBLANK;
return 0; return 0; // Count Incremented
} }
} }
return 1; return 1; // Count not Incremented
} }
void vSync() void vSync()
@ -559,7 +568,7 @@ void rcntWmode(int index, u32 value)
case 0: counters[index].rate = 2; break; case 0: counters[index].rate = 2; break;
case 1: counters[index].rate = 32; break; case 1: counters[index].rate = 32; break;
case 2: counters[index].rate = 512; break; case 2: counters[index].rate = 512; break;
case 3: counters[index].rate = PS2HBLANK; break; case 3: counters[index].rate = SCANLINE_; break;
} }
if((counters[index].mode & 0xF) == 0x7) { if((counters[index].mode & 0xF) == 0x7) {
@ -587,19 +596,14 @@ void rcntStartGate(unsigned int mode) {
for (i=0; i <=3; i++) { //Gates for counters for (i=0; i <=3; i++) { //Gates for counters
if ((mode == 0) && ((counters[i].mode & 0x83) == 0x83)) counters[i].count++; //Update counters using the hblank as the clock if ((mode == 0) && ((counters[i].mode & 0x83) == 0x83)) counters[i].count += HBLANK_COUNTER_SPEED; //Update counters using the hblank as the clock
if (!(gates & (1<<i))) continue; if (!(gates & (1<<i))) continue;
if ((counters[i].mode & 0x8) != mode) continue; if ((counters[i].mode & 0x8) != mode) continue;
//SysPrintf("Gate %d mode %d Start\n", i, (counters[i].mode & 0x30) >> 4);
switch (counters[i].mode & 0x30) { switch (counters[i].mode & 0x30) {
case 0x00: //Count When Signal is low (off) case 0x00: //Count When Signal is low (off)
counters[i].count = rcntRcount(i);
rcntUpd(i); rcntUpd(i);
counters[i].mode &= ~0x80; counters[i].mode |= 0x80;
break;
case 0x20: //Reset and start counting on Vsync end
//Do Nothing
break; break;
case 0x10: //Reset and start counting on Vsync start case 0x10: //Reset and start counting on Vsync start
case 0x30: //Reset and start counting on Vsync start and end case 0x30: //Reset and start counting on Vsync start and end
@ -607,7 +611,6 @@ void rcntStartGate(unsigned int mode) {
rcntReset(i); rcntReset(i);
counters[i].target &= 0xffff; counters[i].target &= 0xffff;
break; break;
} }
} }
} }
@ -618,14 +621,12 @@ void rcntEndGate(unsigned int mode) {
for(i=0; i <=3; i++) { //Gates for counters for(i=0; i <=3; i++) { //Gates for counters
if (!(gates & (1<<i))) continue; if (!(gates & (1<<i))) continue;
if ((counters[i].mode & 0x8) != mode) continue; if ((counters[i].mode & 0x8) != mode) continue;
//SysPrintf("Gate %d mode %d End\n", i, (counters[i].mode & 0x30) >> 4);
switch (counters[i].mode & 0x30) { switch (counters[i].mode & 0x30) {
case 0x00: //Count When Signal is low (off) case 0x00: //Count When Signal is low (off)
counters[i].count = rcntRcount(i);
rcntUpd(i); rcntUpd(i);
counters[i].mode |= 0x80; counters[i].mode &= ~0x80;
break;
case 0x10: //Reset and start counting on Vsync start
//Do Nothing
break; break;
case 0x20: //Reset and start counting on Vsync end case 0x20: //Reset and start counting on Vsync end
case 0x30: //Reset and start counting on Vsync start and end case 0x30: //Reset and start counting on Vsync start and end

View File

@ -61,7 +61,7 @@ typedef struct {
#define SCANLINES_VBLANK2_PAL 20 // scanlines used for vblank2 (odd interlace) #define SCANLINES_VBLANK2_PAL 20 // scanlines used for vblank2 (odd interlace)
//------------------------------------------------------------------ //------------------------------------------------------------------
// PAL Timing Information!!! // Timing (PAL/NTSC) Information!!!
//------------------------------------------------------------------ //------------------------------------------------------------------
#define SCANLINE_ (u32)((Config.PsxType&1) ? SCANLINE_PAL : SCANLINE_NTSC) * HBLANK_TIMER_SLOWDOWN #define SCANLINE_ (u32)((Config.PsxType&1) ? SCANLINE_PAL : SCANLINE_NTSC) * HBLANK_TIMER_SLOWDOWN
#define HRENDER_TIME_ (u32)((Config.PsxType&1) ? HRENDER_TIME_PAL : HRENDER_TIME_NTSC) * HBLANK_TIMER_SLOWDOWN #define HRENDER_TIME_ (u32)((Config.PsxType&1) ? HRENDER_TIME_PAL : HRENDER_TIME_NTSC) * HBLANK_TIMER_SLOWDOWN