2009-07-28 21:32:10 +00:00
// Copyright (C) 2003 Dolphin Project.
2008-12-08 05:30:24 +00:00
// 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, version 2.0.
// 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 2.0 for more details.
// A copy of the GPL 2.0 should have been included with the program.
// If not, see http://www.gnu.org/licenses/
// Official SVN repository and contact information can be found at
// http://code.google.com/p/dolphin-emu/
2009-10-07 19:54:56 +00:00
# include "VideoConfig.h"
2009-02-22 21:16:12 +00:00
# include "Setup.h"
2008-12-08 05:30:24 +00:00
# include "MemoryUtil.h"
# include "Thread.h"
2009-07-13 05:38:34 +00:00
# include "Atomic.h"
2008-12-08 05:30:24 +00:00
# include "OpcodeDecoding.h"
2009-10-10 21:19:39 +00:00
# include "CommandProcessor.h"
2009-12-16 17:05:30 +00:00
# include "ChunkFile.h"
2008-12-08 05:30:24 +00:00
# include "Fifo.h"
2009-08-08 01:39:56 +00:00
volatile bool g_bSkipCurrentFrame = false ;
2009-07-02 10:16:06 +00:00
volatile bool g_EFBAccessRequested = false ;
2009-08-15 07:33:35 +00:00
extern u8 * g_pVideoData ;
2009-07-02 10:16:06 +00:00
2009-08-15 07:33:35 +00:00
namespace
{
2009-02-22 22:49:42 +00:00
static volatile bool fifoStateRun = false ;
2010-01-07 20:01:41 +00:00
static volatile bool EmuRunning = false ;
2008-12-08 05:30:24 +00:00
static u8 * videoBuffer ;
2010-01-14 10:52:14 +00:00
static Common : : EventEx fifo_run_event ;
2009-02-20 22:04:52 +00:00
// STATE_TO_SAVE
2008-12-08 05:30:24 +00:00
static int size = 0 ;
2009-02-20 22:04:52 +00:00
} // namespace
2008-12-08 05:30:24 +00:00
void Fifo_DoState ( PointerWrap & p )
{
2009-12-16 17:05:30 +00:00
CommandProcessor : : FifoCriticalEnter ( ) ;
2009-07-03 10:00:09 +00:00
2008-12-08 05:30:24 +00:00
p . DoArray ( videoBuffer , FIFO_SIZE ) ;
p . Do ( size ) ;
2009-02-20 22:04:52 +00:00
int pos = ( int ) ( g_pVideoData - videoBuffer ) ; // get offset
2008-12-08 05:30:24 +00:00
p . Do ( pos ) ; // read or write offset (depends on the mode afaik)
g_pVideoData = & videoBuffer [ pos ] ; // overwrite g_pVideoData -> expected no change when load ss and change when save ss
2009-07-03 10:00:09 +00:00
2009-12-16 17:05:30 +00:00
CommandProcessor : : FifoCriticalLeave ( ) ;
2008-12-08 05:30:24 +00:00
}
void Fifo_Init ( )
{
videoBuffer = ( u8 * ) AllocateMemoryPages ( FIFO_SIZE ) ;
2010-01-03 16:04:40 +00:00
fifo_run_event . Init ( ) ;
2009-02-21 00:54:52 +00:00
fifoStateRun = false ;
2008-12-08 05:30:24 +00:00
}
void Fifo_Shutdown ( )
{
2010-01-03 16:04:40 +00:00
if ( fifoStateRun ) PanicAlert ( " Fifo shutting down while active " ) ;
fifo_run_event . Shutdown ( ) ;
2008-12-08 05:30:24 +00:00
FreeMemoryPages ( videoBuffer , FIFO_SIZE ) ;
}
u8 * FAKE_GetFifoStartPtr ( )
{
return videoBuffer ;
}
u8 * FAKE_GetFifoEndPtr ( )
{
return & videoBuffer [ size ] ;
}
2009-08-15 07:33:35 +00:00
void Fifo_SetRendering ( bool enabled )
{
g_bSkipCurrentFrame = ! enabled ;
2009-08-08 01:39:56 +00:00
}
2009-02-22 22:49:42 +00:00
// Executed from another thread, no the graphics thread!
// Basically, all it does is set a flag so that the loop will eventually exit, then
// waits for the event to be set, which happens when the loop does exit.
// If we look stuck in here, then the video thread is stuck in something and won't exit
// the loop. Switch to the video thread and investigate.
2009-02-20 22:04:52 +00:00
void Fifo_ExitLoop ( )
{
2010-01-03 16:04:40 +00:00
Fifo_ExitLoopNonBlocking ( ) ;
2009-02-20 22:04:52 +00:00
}
2009-03-07 23:34:16 +00:00
// May be executed from any thread, even the graphics thread.
// Created to allow for self shutdown.
2009-08-09 11:03:58 +00:00
void Fifo_ExitLoopNonBlocking ( )
{
2010-01-16 22:37:38 +00:00
// This should break the wait loop in CPU thread
CommandProcessor : : fifo . bFF_GPReadEnable = false ;
CommandProcessor : : SetFifoIdleFromVideoPlugin ( ) ;
// Terminate GPU thread loop
2009-03-07 23:34:16 +00:00
fifoStateRun = false ;
2010-01-03 16:04:40 +00:00
fifo_run_event . Set ( ) ;
}
2010-01-07 20:01:41 +00:00
void Fifo_RunLoop ( bool run )
2010-01-03 16:04:40 +00:00
{
2010-01-07 20:01:41 +00:00
EmuRunning = run ;
if ( run )
fifo_run_event . Set ( ) ;
2009-03-07 23:34:16 +00:00
}
2009-06-15 04:30:02 +00:00
// Description: Fifo_EnterLoop() sends data through this function.
2009-07-17 22:57:02 +00:00
void Fifo_SendFifoData ( u8 * _uData , u32 len )
2009-06-15 04:30:02 +00:00
{
if ( size + len > = FIFO_SIZE )
{
int pos = ( int ) ( g_pVideoData - videoBuffer ) ;
2010-06-14 16:32:40 +00:00
size - = pos ;
if ( size + len > FIFO_SIZE )
2009-06-15 04:30:02 +00:00
{
PanicAlert ( " FIFO out of bounds (sz = %i, at %08x) " , size , pos ) ;
}
2010-06-14 16:32:40 +00:00
memmove ( & videoBuffer [ 0 ] , & videoBuffer [ pos ] , size ) ;
2009-12-16 17:05:30 +00:00
g_pVideoData = videoBuffer ;
2009-06-15 04:30:02 +00:00
}
// Copy new video instructions to videoBuffer for future use in rendering the new picture
memcpy ( videoBuffer + size , _uData , len ) ;
size + = len ;
}
2010-12-11 12:42:55 +00:00
2010-01-23 21:06:12 +00:00
// Description: Main FIFO update loop
// Purpose: Keep the Core HW updated about the CPU-GPU distance
void Fifo_EnterLoop ( const SVideoInitialize & video_initialize )
2008-12-08 05:30:24 +00:00
{
2010-12-11 12:42:55 +00:00
2010-01-23 21:06:12 +00:00
fifoStateRun = true ;
2009-12-31 20:49:04 +00:00
SCPFifoStruct & _fifo = CommandProcessor : : fifo ;
2009-07-23 22:32:21 +00:00
s32 distToSend ;
2008-12-08 05:30:24 +00:00
2010-01-23 21:06:12 +00:00
while ( fifoStateRun )
2009-12-31 20:49:04 +00:00
{
2010-01-23 21:06:12 +00:00
video_initialize . pPeekMessages ( ) ;
2009-07-17 22:57:02 +00:00
2010-08-04 21:02:32 +00:00
VideoFifo_CheckAsyncRequest ( ) ;
2009-08-15 07:33:35 +00:00
2010-06-24 13:28:54 +00:00
// check if we are able to run this buffer
2010-12-11 12:42:55 +00:00
CommandProcessor : : SetStatus ( ) ;
while ( ! CommandProcessor : : interruptWaiting & & _fifo . bFF_GPReadEnable & &
2010-12-27 02:55:35 +00:00
( _fifo . CPReadWriteDistance & & ! AtBreakpoint ( ) | | CommandProcessor : : OnOverflow ) )
2010-01-23 12:50:56 +00:00
{
2010-06-24 13:28:54 +00:00
// while the FIFO is processing data we activate this for sync with emulator thread.
2010-12-11 12:42:55 +00:00
2010-06-24 13:28:54 +00:00
2010-12-11 12:42:55 +00:00
if ( ! fifoStateRun ) break ;
2010-01-23 12:50:56 +00:00
2010-12-11 12:42:55 +00:00
2010-01-23 21:06:12 +00:00
CommandProcessor : : FifoCriticalEnter ( ) ;
// Create pointer to video data and send it to the VideoPlugin
u32 readPtr = _fifo . CPReadPointer ;
u8 * uData = video_initialize . pGetMemoryPointer ( readPtr ) ;
2009-07-20 01:10:00 +00:00
2010-12-11 12:42:55 +00:00
distToSend = 32 ;
if ( readPtr = = _fifo . CPEnd )
readPtr = _fifo . CPBase ;
else
readPtr + = 32 ;
2010-11-28 20:12:41 +00:00
2010-12-05 15:59:11 +00:00
_assert_msg_ ( COMMANDPROCESSOR , ( s32 ) _fifo . CPReadWriteDistance - distToSend > = 0 ,
2010-11-28 20:12:41 +00:00
" Negative fifo.CPReadWriteDistance = %i in FIFO Loop ! \n That can produce inestabilty in the game. Please report it. " , _fifo . CPReadWriteDistance - distToSend ) ;
2009-12-17 04:01:34 +00:00
2010-12-11 12:42:55 +00:00
// Execute new instructions found in uData
Fifo_SendFifoData ( uData , distToSend ) ;
Common : : AtomicStore ( _fifo . CPReadPointer , readPtr ) ;
Common : : AtomicAdd ( _fifo . CPReadWriteDistance , - distToSend ) ;
CommandProcessor : : isFifoBusy = true ;
CommandProcessor : : SetStatus ( ) ;
_fifo . CPCmdIdle = false ;
2009-12-17 04:01:34 +00:00
2010-12-11 12:42:55 +00:00
OpcodeDecoder_Run ( g_bSkipCurrentFrame ) ;
_fifo . CPCmdIdle = true ;
CommandProcessor : : isFifoBusy = false ;
2010-01-23 21:06:12 +00:00
CommandProcessor : : FifoCriticalLeave ( ) ;
2010-01-23 12:50:56 +00:00
2010-01-23 21:06:12 +00:00
// Those two are pretty important and must be called in the FIFO Loop.
// If we don't, s_swapRequested (OGL only) or s_efbAccessRequested won't be set to false
// leading the CPU thread to wait in Video_BeginField or Video_AccessEFB thus slowing things down.
2010-06-24 13:28:54 +00:00
2010-08-04 21:02:32 +00:00
VideoFifo_CheckAsyncRequest ( ) ;
2010-12-11 12:42:55 +00:00
2010-01-23 21:06:12 +00:00
}
2010-06-24 13:28:54 +00:00
2010-01-23 21:06:12 +00:00
CommandProcessor : : SetFifoIdleFromVideoPlugin ( ) ;
2010-01-07 20:01:41 +00:00
if ( EmuRunning )
2010-01-03 18:58:50 +00:00
Common : : YieldCPU ( ) ;
else
fifo_run_event . MsgWait ( ) ;
2009-12-28 19:13:06 +00:00
}
2008-12-08 05:30:24 +00:00
}
2009-06-15 06:39:26 +00:00
2010-12-11 12:42:55 +00:00
bool AtBreakpoint ( )
{
SCPFifoStruct & _fifo = CommandProcessor : : fifo ;
return _fifo . bFF_BPEnable & & ( _fifo . CPReadPointer = = _fifo . CPBreakpoint ) ;
}