visualboyadvance-m/src/win32/DirectDraw.cpp

715 lines
19 KiB
C++

// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator.
// Copyright (C) 1999-2003 Forgotten
// Copyright (C) 2004 Forgotten and the VBA development team
// 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 2, 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, write to the Free Software Foundation,
// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#include "stdafx.h"
#define DIRECTDRAW_VERSION 0x0700
#include <ddraw.h>
#include "../System.h"
#include "../gb/gbGlobals.h"
#include "../GBA.h"
#include "../Globals.h"
#include "../Text.h"
#include "../Util.h"
#include "VBA.h"
#include "MainWnd.h"
#include "Reg.h"
#include "resource.h"
#include "Display.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
extern int Init_2xSaI(u32);
extern void winlog(const char *,...);
extern int systemSpeed;
extern int winVideoModeSelect(CWnd *, GUID **);
class DirectDrawDisplay : public IDisplay {
private:
HINSTANCE ddrawDLL;
LPDIRECTDRAW7 pDirectDraw;
LPDIRECTDRAWSURFACE7 ddsPrimary;
LPDIRECTDRAWSURFACE7 ddsOffscreen;
LPDIRECTDRAWSURFACE7 ddsFlip;
LPDIRECTDRAWCLIPPER ddsClipper;
int width;
int height;
bool failed;
bool initializeOffscreen(int w, int h);
public:
DirectDrawDisplay();
virtual ~DirectDrawDisplay();
virtual bool initialize();
virtual void cleanup();
virtual void render();
virtual void checkFullScreen();
virtual void clear();
virtual bool changeRenderSize(int w, int h);
virtual DISPLAY_TYPE getType() { return DIRECT_DRAW; };
virtual void setOption(const char *, int) {}
virtual bool isSkinSupported() { return true; }
virtual int selectFullScreenMode(GUID **);
};
static HRESULT WINAPI checkModesAvailable(LPDDSURFACEDESC2 surf, LPVOID lpContext)
{
if(surf->dwWidth == 320 &&
surf->dwHeight == 240 &&
surf->ddpfPixelFormat.dwRGBBitCount == 16) {
theApp.mode320Available = TRUE;
}
if(surf->dwWidth == 640 &&
surf->dwHeight == 480 &&
surf->ddpfPixelFormat.dwRGBBitCount == 16) {
theApp.mode640Available = TRUE;
}
if(surf->dwWidth == 800 &&
surf->dwHeight == 600 &&
surf->ddpfPixelFormat.dwRGBBitCount == 16) {
theApp.mode800Available = TRUE;
}
if(surf->dwWidth == 1024 &&
surf->dwHeight == 768 &&
surf->ddpfPixelFormat.dwRGBBitCount == 16) {
theApp.mode1024Available = TRUE;
}
if(surf->dwWidth == 1280 &&
surf->dwHeight == 1024 &&
surf->ddpfPixelFormat.dwRGBBitCount == 16) {
theApp.mode1280Available = TRUE;
}
return DDENUMRET_OK;
}
static int ffs(UINT mask)
{
int m = 0;
if (mask) {
while (!(mask & (1 << m)))
m++;
return (m);
}
return (0);
}
DirectDrawDisplay::DirectDrawDisplay()
{
pDirectDraw = NULL;
ddsPrimary = NULL;
ddsOffscreen = NULL;
ddsFlip = NULL;
ddsClipper = NULL;
ddrawDLL = NULL;
width = 0;
height = 0;
failed = false;
}
DirectDrawDisplay::~DirectDrawDisplay()
{
cleanup();
}
void DirectDrawDisplay::cleanup()
{
if(pDirectDraw != NULL) {
if(ddsClipper != NULL) {
ddsClipper->Release();
ddsClipper = NULL;
}
if(ddsFlip != NULL) {
ddsFlip->Release();
ddsFlip = NULL;
}
if(ddsOffscreen != NULL) {
ddsOffscreen->Release();
ddsOffscreen = NULL;
}
if(ddsPrimary != NULL) {
ddsPrimary->Release();
ddsPrimary = NULL;
}
pDirectDraw->Release();
pDirectDraw = NULL;
}
if(ddrawDLL != NULL) {
#ifdef _AFXDLL
AfxFreeLibrary( ddrawDLL );
#else
FreeLibrary( ddrawDLL );
#endif
ddrawDLL = NULL;
}
width = 0;
height = 0;
}
bool DirectDrawDisplay::initialize()
{
GUID *guid = NULL;
if(theApp.ddrawEmulationOnly)
guid = (GUID *)DDCREATE_EMULATIONONLY;
if(theApp.pVideoDriverGUID)
guid = theApp.pVideoDriverGUID;
#ifdef _AFXDLL
ddrawDLL = AfxLoadLibrary("ddraw.dll");
#else
ddrawDLL = LoadLibrary( _T("ddraw.dll") );
#endif
HRESULT (WINAPI *DDrawCreateEx)(GUID *,LPVOID *,REFIID,IUnknown *);
if(ddrawDLL != NULL) {
DDrawCreateEx = (HRESULT (WINAPI *)(GUID *,LPVOID *,REFIID,IUnknown *))
GetProcAddress(ddrawDLL, "DirectDrawCreateEx");
if(DDrawCreateEx == NULL) {
theApp.directXMessage("DirectDrawCreateEx");
return FALSE;
}
} else {
theApp.directXMessage("DDRAW.DLL");
return FALSE;
}
theApp.ddrawUsingEmulationOnly = theApp.ddrawEmulationOnly;
HRESULT hret = DDrawCreateEx(guid,
(void **)&pDirectDraw,
IID_IDirectDraw7,
NULL);
if(hret != DD_OK) {
winlog("Error creating DirectDraw object %08x\n", hret);
if(theApp.ddrawEmulationOnly) {
// disable emulation only setting in case of failure
regSetDwordValue("ddrawEmulationOnly", 0);
}
// errorMessage(myLoadString(IDS_ERROR_DISP_DRAWCREATE), hret);
return FALSE;
}
if(theApp.ddrawDebug) {
DDCAPS driver;
DDCAPS hel;
ZeroMemory(&driver, sizeof(driver));
ZeroMemory(&hel, sizeof(hel));
driver.dwSize = sizeof(driver);
hel.dwSize = sizeof(hel);
pDirectDraw->GetCaps(&driver, &hel);
int i;
DWORD *p = (DWORD *)&driver;
for(i = 0; i < (int)driver.dwSize; i+=4)
winlog("Driver CAPS %2d: %08x\n", i>>2, *p++);
p = (DWORD *)&hel;
for(i = 0; i < (int)hel.dwSize; i+=4)
winlog("HEL CAPS %2d: %08x\n", i>>2, *p++);
}
theApp.mode320Available = false;
theApp.mode640Available = false;
theApp.mode800Available = false;
theApp.mode1024Available = false;
theApp.mode1280Available = false;
// check for available fullscreen modes
pDirectDraw->EnumDisplayModes(DDEDM_STANDARDVGAMODES, NULL, NULL,
checkModesAvailable);
DWORD flags = DDSCL_NORMAL;
if(theApp.videoOption >= VIDEO_320x240)
flags = DDSCL_ALLOWMODEX |
DDSCL_ALLOWREBOOT |
DDSCL_EXCLUSIVE |
DDSCL_FULLSCREEN;
hret = pDirectDraw->SetCooperativeLevel(theApp.m_pMainWnd->GetSafeHwnd(),
flags);
if(hret != DD_OK) {
winlog("Error SetCooperativeLevel %08x\n", hret);
// errorMessage(myLoadString(IDS_ERROR_DISP_DRAWLEVEL), hret);
return FALSE;
}
if(theApp.videoOption > VIDEO_4X) {
hret = pDirectDraw->SetDisplayMode(theApp.fsWidth,
theApp.fsHeight,
theApp.fsColorDepth,
theApp.fsFrequency,
0);
if(hret != DD_OK) {
winlog("Error SetDisplayMode %08x\n", hret);
// errorMessage(myLoadString(IDS_ERROR_DISP_DRAWSET), hret);
return FALSE;
}
}
DDSURFACEDESC2 ddsd;
ZeroMemory(&ddsd,sizeof(ddsd));
ddsd.dwSize = sizeof(ddsd);
ddsd.dwFlags = DDSD_CAPS;
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
if(theApp.videoOption > VIDEO_4X) {
if(theApp.tripleBuffering) {
// setup triple buffering
ddsd.dwFlags |= DDSD_BACKBUFFERCOUNT;
ddsd.ddsCaps.dwCaps |= DDSCAPS_COMPLEX | DDSCAPS_FLIP;
ddsd.dwBackBufferCount = 2;
}
}
hret = pDirectDraw->CreateSurface(&ddsd, &ddsPrimary, NULL);
if(hret != DD_OK) {
winlog("Error primary CreateSurface %08x\n", hret);
// errorMessage(myLoadString(IDS_ERROR_DISP_DRAWSURFACE), hret);
return FALSE;
}
if(theApp.ddrawDebug) {
DDSCAPS2 caps;
ZeroMemory(&caps, sizeof(caps));
ddsPrimary->GetCaps(&caps);
winlog("Primary CAPS 1: %08x\n", caps.dwCaps);
winlog("Primary CAPS 2: %08x\n", caps.dwCaps2);
winlog("Primary CAPS 3: %08x\n", caps.dwCaps3);
winlog("Primary CAPS 4: %08x\n", caps.dwCaps4);
}
if(theApp.videoOption > VIDEO_4X && theApp.tripleBuffering) {
DDSCAPS2 caps;
ZeroMemory(&caps, sizeof(caps));
// this gets the third surface. The front one is the primary,
// the second is the backbuffer and the third is the flip
// surface
caps.dwCaps = DDSCAPS_BACKBUFFER;
hret = ddsPrimary->GetAttachedSurface(&caps, &ddsFlip);
if(hret != DD_OK) {
winlog("Failed to get attached surface %08x", hret);
return FALSE;
}
ddsFlip->AddRef();
clear();
}
// create clipper in all modes to avoid paint problems
// if(videoOption <= VIDEO_4X) {
hret = pDirectDraw->CreateClipper(0, &ddsClipper, NULL);
if(hret == DD_OK) {
ddsClipper->SetHWnd(0, theApp.m_pMainWnd->GetSafeHwnd());
if(theApp.videoOption > VIDEO_4X) {
if(theApp.tripleBuffering)
ddsFlip->SetClipper(ddsClipper);
else
ddsPrimary->SetClipper(ddsClipper);
} else
ddsPrimary->SetClipper(ddsClipper);
}
// }
DDPIXELFORMAT px;
px.dwSize = sizeof(px);
hret = ddsPrimary->GetPixelFormat(&px);
switch(px.dwRGBBitCount) {
case 15:
case 16:
systemColorDepth = 16;
break;
case 24:
systemColorDepth = 24;
theApp.filterFunction = NULL;
break;
case 32:
systemColorDepth = 32;
break;
default:
systemMessage(IDS_ERROR_DISP_COLOR, "Unsupported display setting for color depth: %d bits. \nWindows desktop must be in either 16-bit, 24-bit or 32-bit mode for this program to work in window mode.",px.dwRGBBitCount);
return FALSE;
}
theApp.updateFilter();
theApp.updateIFB();
if(failed)
return false;
return true;
}
bool DirectDrawDisplay::changeRenderSize(int w, int h)
{
if(w != width || h != height) {
if(ddsOffscreen) {
ddsOffscreen->Release();
ddsOffscreen = NULL;
}
if(!initializeOffscreen(w, h)) {
failed = true;
return false;
}
}
return true;
}
bool DirectDrawDisplay::initializeOffscreen(int w, int h)
{
DDSURFACEDESC2 ddsd;
ZeroMemory(&ddsd, sizeof(ddsd));
ddsd.dwSize = sizeof(ddsd);
ddsd.dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH;
ddsd.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN;
if(theApp.ddrawUseVideoMemory)
ddsd.ddsCaps.dwCaps |= (DDSCAPS_LOCALVIDMEM|DDSCAPS_VIDEOMEMORY);
ddsd.dwWidth = w;
ddsd.dwHeight = h;
HRESULT hret = pDirectDraw->CreateSurface(&ddsd, &ddsOffscreen, NULL);
if(hret != DD_OK) {
winlog("Error offscreen CreateSurface %08x\n", hret);
if(theApp.ddrawUseVideoMemory) {
regSetDwordValue("ddrawUseVideoMemory", 0);
}
// errorMessage(myLoadString(IDS_ERROR_DISP_DRAWSURFACE2), hret);
return false;
}
if(theApp.ddrawDebug) {
DDSCAPS2 caps;
ZeroMemory(&caps, sizeof(caps));
ddsOffscreen->GetCaps(&caps);
winlog("Offscreen CAPS 1: %08x\n", caps.dwCaps);
winlog("Offscreen CAPS 2: %08x\n", caps.dwCaps2);
winlog("Offscreen CAPS 3: %08x\n", caps.dwCaps3);
winlog("Offscreen CAPS 4: %08x\n", caps.dwCaps4);
}
DDPIXELFORMAT px;
px.dwSize = sizeof(px);
hret = ddsOffscreen->GetPixelFormat(&px);
if(theApp.ddrawDebug) {
DWORD *pdword = (DWORD *)&px;
for(int ii = 0; ii < 8; ii++) {
winlog("Pixel format %d %08x\n", ii, pdword[ii]);
}
}
switch(px.dwRGBBitCount) {
case 15:
case 16:
systemColorDepth = 16;
break;
case 24:
systemColorDepth = 24;
theApp.filterFunction = NULL;
break;
case 32:
systemColorDepth = 32;
break;
default:
systemMessage(IDS_ERROR_DISP_COLOR, "Unsupported display setting for color depth: %d bits. \nWindows desktop must be in either 16-bit, 24-bit or 32-bit mode for this program to work in window mode.",px.dwRGBBitCount);
return FALSE;
}
if(theApp.ddrawDebug) {
winlog("R Mask: %08x\n", px.dwRBitMask);
winlog("G Mask: %08x\n", px.dwGBitMask);
winlog("B Mask: %08x\n", px.dwBBitMask);
}
systemRedShift = ffs(px.dwRBitMask);
systemGreenShift = ffs(px.dwGBitMask);
systemBlueShift = ffs(px.dwBBitMask);
#ifdef MMX
if(!theApp.disableMMX)
cpu_mmx = theApp.detectMMX();
else
cpu_mmx = 0;
#endif
if((px.dwFlags&DDPF_RGB) != 0 &&
px.dwRBitMask == 0xF800 &&
px.dwGBitMask == 0x07E0 &&
px.dwBBitMask == 0x001F) {
systemGreenShift++;
Init_2xSaI(565);
} else if((px.dwFlags&DDPF_RGB) != 0 &&
px.dwRBitMask == 0x7C00 &&
px.dwGBitMask == 0x03E0 &&
px.dwBBitMask == 0x001F) {
Init_2xSaI(555);
} else if((px.dwFlags&DDPF_RGB) != 0 &&
px.dwRBitMask == 0x001F &&
px.dwGBitMask == 0x07E0 &&
px.dwBBitMask == 0xF800) {
systemGreenShift++;
Init_2xSaI(565);
} else if((px.dwFlags&DDPF_RGB) != 0 &&
px.dwRBitMask == 0x001F &&
px.dwGBitMask == 0x03E0 &&
px.dwBBitMask == 0x7C00) {
Init_2xSaI(555);
} else {
// 32-bit or 24-bit
if(systemColorDepth == 32 || systemColorDepth == 24) {
systemRedShift += 3;
systemGreenShift += 3;
systemBlueShift += 3;
if(systemColorDepth == 32)
Init_2xSaI(32);
}
}
if(theApp.ddrawDebug) {
winlog("R shift: %d\n", systemRedShift);
winlog("G shift: %d\n", systemGreenShift);
winlog("B shift: %d\n", systemBlueShift);
}
utilUpdateSystemColorMaps();
width = w;
height = h;
return true;
}
void DirectDrawDisplay::clear()
{
if(theApp.videoOption <= VIDEO_4X || !theApp.tripleBuffering || ddsFlip == NULL)
return;
DDBLTFX fx;
ZeroMemory(&fx, sizeof(fx));
fx.dwSize = sizeof(fx);
fx.dwFillColor = 0;
ddsFlip->Blt(NULL, NULL, NULL, DDBLT_COLORFILL | DDBLT_WAIT, &fx);
ddsPrimary->Flip(NULL, 0);
ddsFlip->Blt(NULL, NULL, NULL, DDBLT_COLORFILL | DDBLT_WAIT, &fx);
ddsPrimary->Flip(NULL, 0);
ddsFlip->Blt(NULL, NULL, NULL, DDBLT_COLORFILL | DDBLT_WAIT, &fx);
ddsPrimary->Flip(NULL, 0);
}
void DirectDrawDisplay::checkFullScreen()
{
if(theApp.tripleBuffering)
pDirectDraw->FlipToGDISurface();
}
void DirectDrawDisplay::render()
{
HRESULT hret;
unsigned int nBytesPerPixel = systemColorDepth>>3;
if(pDirectDraw == NULL ||
ddsOffscreen == NULL ||
ddsPrimary == NULL)
return;
DDSURFACEDESC2 ddsDesc;
ZeroMemory(&ddsDesc, sizeof(ddsDesc));
ddsDesc.dwSize = sizeof(ddsDesc);
hret = ddsOffscreen->Lock(NULL,
&ddsDesc,
DDLOCK_WRITEONLY|
#ifdef _DEBUG
DDLOCK_NOSYSLOCK|
#endif
DDLOCK_SURFACEMEMORYPTR,
NULL);
if(hret == DDERR_SURFACELOST) {
hret = ddsPrimary->Restore();
if(hret == DD_OK) {
hret = ddsOffscreen->Restore();
if(hret == DD_OK) {
hret = ddsOffscreen->Lock(NULL,
&ddsDesc,
DDLOCK_WRITEONLY|
#ifdef _DEBUG
DDLOCK_NOSYSLOCK|
#endif
DDLOCK_SURFACEMEMORYPTR,
NULL);
}
}
}
if(hret == DD_OK) {
unsigned short pitch = theApp.sizeX * ( systemColorDepth >> 3 ) + 4;
if( theApp.filterFunction ) {
// pixel filter enabled
theApp.filterFunction(
pix + pitch,
pitch,
(u8*)theApp.delta,
(u8*)ddsDesc.lpSurface,
ddsDesc.lPitch,
theApp.sizeX,
theApp.sizeY
);
} else {
// pixel filter disabled
switch( systemColorDepth )
{
case 32:
cpyImg32(
(unsigned char *)ddsDesc.lpSurface,
ddsDesc.lPitch,
pix + pitch,
pitch,
theApp.sizeX,
theApp.sizeY
);
break;
case 16:
cpyImg16(
(unsigned char *)ddsDesc.lpSurface,
ddsDesc.lPitch,
pix + pitch,
pitch,
theApp.sizeX,
theApp.sizeY
);
break;
}
}
if(theApp.showSpeed && (theApp.videoOption > VIDEO_4X || theApp.skin != NULL)) {
char buffer[30];
if(theApp.showSpeed == 1)
sprintf(buffer, "%3d%%", systemSpeed);
else
sprintf(buffer, "%3d%%(%d, %d fps)", systemSpeed,
systemFrameSkip,
theApp.showRenderedFrames);
if(theApp.showSpeedTransparent)
drawTextTransp((u8*)ddsDesc.lpSurface,
ddsDesc.lPitch,
theApp.rect.left+10,
theApp.rect.bottom-10,
buffer);
else
drawText((u8*)ddsDesc.lpSurface,
ddsDesc.lPitch,
theApp.rect.left+10,
theApp.rect.bottom-10,
buffer);
}
} else if(theApp.ddrawDebug)
winlog("Error during lock: %08x\n", hret);
hret = ddsOffscreen->Unlock(NULL);
if(hret == DD_OK) {
if(theApp.vsync && !(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) && !speedup) { // isn't the Flip() call synced unless a certain flag is passed to it?
hret = pDirectDraw->WaitForVerticalBlank(DDWAITVB_BLOCKBEGIN, 0);
}
ddsOffscreen->PageLock(0);
if(theApp.tripleBuffering && theApp.videoOption > VIDEO_4X) {
hret = ddsFlip->Blt(&theApp.dest, ddsOffscreen, NULL, DDBLT_WAIT, NULL);
if(hret == DD_OK) {
if(theApp.menuToggle || !theApp.active) {
pDirectDraw->FlipToGDISurface();
ddsPrimary->SetClipper(ddsClipper);
hret = ddsPrimary->Blt(&theApp.dest, ddsFlip, NULL, DDBLT_ASYNC, NULL);
theApp.m_pMainWnd->DrawMenuBar();
} else
hret = ddsPrimary->Flip(NULL, 0);
}
} else {
hret = ddsPrimary->Blt(&theApp.dest, ddsOffscreen, NULL,DDBLT_ASYNC,NULL);
if(hret == DDERR_SURFACELOST) {
hret = ddsPrimary->Restore();
if(hret == DD_OK) {
hret = ddsPrimary->Blt(&theApp.dest, ddsOffscreen, NULL, DDBLT_ASYNC, NULL);
}
}
}
ddsOffscreen->PageUnlock(0);
} else if(theApp.ddrawDebug)
winlog("Error during unlock: %08x\n", hret);
if(theApp.screenMessage) {
if(((GetTickCount() - theApp.screenMessageTime) < 3000) &&
!theApp.disableStatusMessage) {
ddsPrimary->SetClipper(ddsClipper);
HDC hdc;
ddsPrimary->GetDC(&hdc);
SetTextColor(hdc, RGB(255,0,0));
SetBkMode(hdc,TRANSPARENT);
TextOut(hdc, theApp.dest.left+10, theApp.dest.bottom - 20, theApp.screenMessageBuffer,
(int)_tcslen(theApp.screenMessageBuffer));
ddsPrimary->ReleaseDC(hdc);
} else {
theApp.screenMessage = false;
}
}
if(hret != DD_OK) {
if(theApp.ddrawDebug)
winlog("Error on update screen: %08x\n", hret);
}
}
int DirectDrawDisplay::selectFullScreenMode(GUID **pGUID)
{
return winVideoModeSelect(theApp.m_pMainWnd, pGUID);
}
IDisplay *newDirectDrawDisplay()
{
return new DirectDrawDisplay();
}