You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

375 lines
11 KiB

#include "dx_stub.h"
#include "miniwin_sdl.h"
#include "../types.h"
#include "stubs.h"
#ifndef NO_GLOBALS
Screen *gpBuffer;
IDirectDraw *lpDDInterface;
IDirectDrawSurface *lpDDSPrimary;
IDirectDrawSurface *lpDDSBackBuf;
IDirectDrawPalette *lpDDPalette;
char gbBackBuf; // unread
char gbEmulate; // unread
#endif
SDL_Window *window;
SDL_Renderer *renderer;
SDL_Texture *texture;
/** 32-bit in-memory backbuffer surface */
SDL_Surface *surface;
/** 8-bit surface wrapper around #gpBuffer */
SDL_Surface *pal_surface;
/** Currently active palette */
SDL_Palette *palette;
/**
* Is #sdl_pal_surface dirty?
*
* This is required so the front buffer would not be updated twice per game loop in unlock_buf_priv()
* which causes the cursor to flicker.
*/
bool surface_dirty;
//
// DirectDraw COM interface stub implementations
//
#define METHOD virtual __stdcall
class StubSurface : public IDirectDrawSurface
{
METHOD HRESULT QueryInterface(REFIID refiid, LPVOID *lpvoid) { UNIMPLEMENTED(); };
METHOD ULONG AddRef() { UNIMPLEMENTED(); };
METHOD ULONG Release() { UNIMPLEMENTED(); };
METHOD HRESULT AddAttachedSurface(LPDIRECTDRAWSURFACE lpDDSAttachedSurface) { UNIMPLEMENTED(); }
METHOD HRESULT AddOverlayDirtyRect(LPRECT lpRect) { UNIMPLEMENTED(); }
METHOD HRESULT Blt(LPRECT lpDestRect, LPDIRECTDRAWSURFACE lpDDSrcSurface, LPRECT lpSrcRect, DWORD dwFlags,
LPDDBLTFX lpDDBltFx)
{
UNIMPLEMENTED();
}
METHOD HRESULT BltBatch(LPDDBLTBATCH lpDDBltBatch, DWORD dwCount, DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT BltFast(DWORD dwX, DWORD dwY, LPDIRECTDRAWSURFACE lpDDSrcSurface, LPRECT lpSrcRect, DWORD dwTrans)
{
DUMMY_ONCE();
assert(lpDDSrcSurface == lpDDSBackBuf);
int w = lpSrcRect->right - lpSrcRect->left + 1;
int h = lpSrcRect->bottom - lpSrcRect->top + 1;
SDL_Rect src_rect = {lpSrcRect->left, lpSrcRect->top, w, h};
SDL_Rect dst_rect = {(int)dwX, (int)dwY, w, h};
// Convert from 8-bit to 32-bit
SDL_CHECK(SDL_BlitSurface(pal_surface, &src_rect, surface, &dst_rect));
surface_dirty = true;
return S_OK;
}
METHOD HRESULT DeleteAttachedSurface(DWORD dwFlags, LPDIRECTDRAWSURFACE lpDDSAttachedSurface) { UNIMPLEMENTED(); }
METHOD HRESULT EnumAttachedSurfaces(LPVOID lpContext, LPDDENUMSURFACESCALLBACK lpEnumSurfacesCallback)
{
UNIMPLEMENTED();
}
METHOD HRESULT EnumOverlayZOrders(DWORD dwFlags, LPVOID lpContext, LPDDENUMSURFACESCALLBACK lpfnCallback)
{
UNIMPLEMENTED();
}
METHOD HRESULT Flip(LPDIRECTDRAWSURFACE lpDDSurfaceTargetOverride, DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT GetAttachedSurface(LPDDSCAPS lpDDSCaps, LPDIRECTDRAWSURFACE *lplpDDAttachedSurface)
{
UNIMPLEMENTED();
}
METHOD HRESULT GetBltStatus(DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT GetCaps(LPDDSCAPS lpDDSCaps) { UNIMPLEMENTED(); }
METHOD HRESULT GetClipper(LPDIRECTDRAWCLIPPER *lplpDDClipper) { UNIMPLEMENTED(); }
METHOD HRESULT GetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) { UNIMPLEMENTED(); }
METHOD HRESULT GetDC(HDC *lphDC)
{
DUMMY_ONCE();
return S_OK;
}
METHOD HRESULT GetFlipStatus(DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT GetOverlayPosition(LPLONG lplX, LPLONG lplY) { UNIMPLEMENTED(); }
METHOD HRESULT GetPalette(LPDIRECTDRAWPALETTE *lplpDDPalette) { UNIMPLEMENTED(); }
METHOD HRESULT GetPixelFormat(LPDDPIXELFORMAT lpDDPixelFormat) { UNIMPLEMENTED(); }
METHOD HRESULT GetSurfaceDesc(LPDDSURFACEDESC lpDDSurfaceDesc) { UNIMPLEMENTED(); }
METHOD HRESULT Initialize(LPDIRECTDRAW lpDD, LPDDSURFACEDESC lpDDSurfaceDesc) { UNIMPLEMENTED(); }
METHOD HRESULT IsLost()
{
DUMMY_ONCE();
return S_OK;
}
METHOD HRESULT Lock(LPRECT lpDestRect, LPDDSURFACEDESC lpDDSurfaceDesc, DWORD dwFlags, HANDLE hEvent)
{
UNIMPLEMENTED();
}
METHOD HRESULT ReleaseDC(HDC hDC)
{
DUMMY_ONCE();
return S_OK;
}
METHOD HRESULT Restore() { UNIMPLEMENTED(); }
METHOD HRESULT SetClipper(LPDIRECTDRAWCLIPPER lpDDClipper) { UNIMPLEMENTED(); }
METHOD HRESULT SetColorKey(DWORD dwFlags, LPDDCOLORKEY lpDDColorKey) { UNIMPLEMENTED(); }
METHOD HRESULT SetOverlayPosition(LONG lX, LONG lY) { UNIMPLEMENTED(); }
METHOD HRESULT SetPalette(LPDIRECTDRAWPALETTE lpDDPalette) { UNIMPLEMENTED(); }
METHOD HRESULT Unlock(LPVOID lpSurfaceData) { UNIMPLEMENTED(); }
METHOD HRESULT UpdateOverlay(LPRECT lpSrcRect, LPDIRECTDRAWSURFACE lpDDDestSurface, LPRECT lpDestRect,
DWORD dwFlags, LPDDOVERLAYFX lpDDOverlayFx)
{
UNIMPLEMENTED();
}
METHOD HRESULT UpdateOverlayDisplay(DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT UpdateOverlayZOrder(DWORD dwFlags, LPDIRECTDRAWSURFACE lpDDSReference) { UNIMPLEMENTED(); }
};
class StubPalette : public IDirectDrawPalette
{
METHOD HRESULT QueryInterface(REFIID refiid, LPVOID *lpvoid) { UNIMPLEMENTED(); };
METHOD ULONG AddRef() { UNIMPLEMENTED(); };
METHOD ULONG Release() { UNIMPLEMENTED(); };
METHOD HRESULT GetCaps(LPDWORD lpdwCaps) { UNIMPLEMENTED(); };
METHOD HRESULT GetEntries(DWORD dwFlags, DWORD dwBase, DWORD dwNumEntries, LPPALETTEENTRY lpEntries)
{
UNIMPLEMENTED();
};
METHOD HRESULT Initialize(LPDIRECTDRAW lpDD, DWORD dwFlags, LPPALETTEENTRY lpDDColorTable) { UNIMPLEMENTED(); };
METHOD HRESULT SetEntries(DWORD dwFlags, DWORD dwStartingEntry, DWORD dwCount, LPPALETTEENTRY lpEntries)
{
UNIMPLEMENTED();
};
};
class StubDraw : public IDirectDraw
{
METHOD HRESULT QueryInterface(REFIID refiid, LPVOID *lpvoid) { UNIMPLEMENTED(); };
METHOD ULONG AddRef() { UNIMPLEMENTED(); };
METHOD ULONG Release() { UNIMPLEMENTED(); };
METHOD HRESULT Compact() { UNIMPLEMENTED(); }
METHOD HRESULT CreateClipper(DWORD dwFlags, LPDIRECTDRAWCLIPPER *lplpDDClipper, IUnknown *pUnkOuter)
{
UNIMPLEMENTED();
}
METHOD HRESULT CreatePalette(DWORD dwFlags, LPPALETTEENTRY lpColorTable, LPDIRECTDRAWPALETTE *lplpDDPalette,
IUnknown *pUnkOuter)
{
UNIMPLEMENTED();
}
METHOD HRESULT CreateSurface(LPDDSURFACEDESC lpDDSurfaceDesc, LPDIRECTDRAWSURFACE *lplpDDSurface,
IUnknown *pUnkOuter)
{
UNIMPLEMENTED();
}
METHOD HRESULT DuplicateSurface(LPDIRECTDRAWSURFACE lpDDSurface, LPDIRECTDRAWSURFACE *lplpDupDDSurface)
{
UNIMPLEMENTED();
}
METHOD HRESULT EnumDisplayModes(DWORD dwFlags, LPDDSURFACEDESC lpDDSurfaceDesc, LPVOID lpContext,
LPDDENUMMODESCALLBACK lpEnumModesCallback)
{
UNIMPLEMENTED();
}
METHOD HRESULT EnumSurfaces(DWORD dwFlags, LPDDSURFACEDESC lpDDSD, LPVOID lpContext,
LPDDENUMSURFACESCALLBACK lpEnumSurfacesCallback)
{
UNIMPLEMENTED();
}
METHOD HRESULT FlipToGDISurface() { UNIMPLEMENTED(); }
METHOD HRESULT GetCaps(LPDDCAPS lpDDDriverCaps, LPDDCAPS lpDDHELCaps) { UNIMPLEMENTED(); }
METHOD HRESULT GetDisplayMode(LPDDSURFACEDESC lpDDSurfaceDesc) { UNIMPLEMENTED(); }
METHOD HRESULT GetFourCCCodes(LPDWORD lpNumCodes, LPDWORD lpCodes) { UNIMPLEMENTED(); }
METHOD HRESULT GetGDISurface(LPDIRECTDRAWSURFACE *lplpGDIDDSurface) { UNIMPLEMENTED(); }
METHOD HRESULT GetMonitorFrequency(LPDWORD lpdwFrequency) { UNIMPLEMENTED(); }
METHOD HRESULT GetScanLine(LPDWORD lpdwScanLine) { UNIMPLEMENTED(); }
METHOD HRESULT GetVerticalBlankStatus(BOOL *lpbIsInVB) { UNIMPLEMENTED(); }
METHOD HRESULT Initialize(GUID *lpGUID) { UNIMPLEMENTED(); }
METHOD HRESULT RestoreDisplayMode() { UNIMPLEMENTED(); }
METHOD HRESULT SetCooperativeLevel(HWND hWnd, DWORD dwFlags) { UNIMPLEMENTED(); }
METHOD HRESULT SetDisplayMode(DWORD dwWidth, DWORD dwHeight, DWORD dwBPP) { UNIMPLEMENTED(); }
METHOD HRESULT WaitForVerticalBlank(DWORD dwFlags, HANDLE hEvent)
{
DUMMY();
return S_OK;
}
};
static StubDraw stub_draw;
static StubSurface stub_surface;
static StubPalette stub_palette;
//
// Main functions
//
void sdl_init_video()
{
SDL_CHECK(SDL_Init(SDL_INIT_VIDEO));
window =
SDL_CreateWindow("devil-test", SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, 640, 480, SDL_WINDOW_SHOWN);
assert(window);
// Hack since ShowCursor is called before dx_init()
SDL_ShowCursor(SDL_DISABLE);
renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_PRESENTVSYNC);
assert(renderer);
// FUTURE: Use SDL_CreateRGBSurfaceWithFormat with SDL_PIXELFORMAT_RGBA8888
surface = SDL_CreateRGBSurface(0, 640, 480, 32, 0xFF000000, 0x00FF0000, 0x0000FF00, 0x000000FF);
assert(surface);
texture =
SDL_CreateTexture(renderer, SDL_PIXELFORMAT_RGBA8888, SDL_TEXTUREACCESS_STREAMING, 640, 480);
assert(texture);
palette = SDL_AllocPalette(256);
assert(palette);
}
void __fastcall dx_init(HWND hWnd)
{
DUMMY();
sdl_init_video();
gbActive = TRUE;
lpDDInterface = &stub_draw;
lpDDSPrimary = &stub_surface;
lpDDSBackBuf = &stub_surface;
lpDDPalette = &stub_palette;
}
/** Copy the palette surface to the main backbuffer */
void sdl_update_entire_surface()
{
assert(surface && pal_surface);
SDL_Rect src_rect = {64, 160, 640, 480};
SDL_CHECK(SDL_BlitSurface(pal_surface, &src_rect, surface, NULL));
}
void sdl_present_surface()
{
assert(!SDL_MUSTLOCK(surface));
SDL_CHECK(SDL_UpdateTexture(texture, NULL, surface->pixels, surface->pitch));
SDL_CHECK(SDL_RenderCopy(renderer, texture, NULL, NULL));
SDL_RenderPresent(renderer);
surface_dirty = false;
}
void __cdecl lock_buf_priv()
{
const int pitch = 640 + 64 + 64;
if (!gpBuffer) {
gpBuffer = (Screen *)malloc(sizeof(Screen));
gpBufEnd += (unsigned int)gpBuffer;
pal_surface = SDL_CreateRGBSurfaceFrom(gpBuffer, pitch, 160 + 480 + 16, 8, pitch, 0, 0, 0, 0);
assert(pal_surface);
SDL_CHECK(SDL_SetSurfacePalette(pal_surface, palette));
}
}
void __cdecl unlock_buf_priv()
{
gpBufEnd -= (unsigned int)gpBufEnd;
if (!surface_dirty) {
return;
}
sdl_present_surface();
}
void __cdecl dx_reinit()
{
UNIMPLEMENTED();
}
//
// Storm functions
//
BOOL STORMAPI SDrawUpdatePalette(unsigned int firstentry, unsigned int numentries, PALETTEENTRY *pPalEntries, int a4)
{
DUMMY_PRINT("first: %d num: %d", firstentry, numentries);
assert(firstentry == 0);
assert(numentries == 256);
SDL_Color colors[256];
for (int i = firstentry; i < numentries; i++) {
SDL_Color *c = &colors[i];
PALETTEENTRY *p = &pPalEntries[i];
c->r = p->peRed;
c->g = p->peGreen;
c->b = p->peBlue;
c->a = SDL_ALPHA_OPAQUE;
}
assert(palette);
SDL_CHECK(SDL_SetPaletteColors(palette, colors, firstentry, numentries));
if (pal_surface) {
sdl_update_entire_surface();
sdl_present_surface();
}
return TRUE;
}
//
// Windows API functions
//
WINBOOL WINAPI SetCursorPos(int X, int Y)
{
DUMMY();
assert(window);
SDL_WarpMouseInWindow(window, X, Y);
return TRUE;
}
int WINAPI ShowCursor(WINBOOL bShow)
{
DUMMY_PRINT("%d", bShow);
if (window) {
SDL_ShowCursor(bShow ? SDL_ENABLE : SDL_DISABLE);
}
return bShow;
}
WINBOOL WINAPI TextOutA(HDC hdc, int x, int y, LPCSTR lpString, int c)
{
DUMMY_ONCE();
assert(window);
SDL_SetWindowTitle(window, lpString);
return TRUE;
}