mirror of
https://github.com/FunkyFr3sh/cnc-ddraw.git
synced 2025-03-24 17:49:52 +01:00
Re-implement software renderer in GDI
This commit is contained in:
parent
a0c80f3e83
commit
84acbbc0c3
2
Makefile
2
Makefile
@ -7,7 +7,7 @@ REV=$(shell sh -c 'git rev-parse --short @{0}')
|
|||||||
all:
|
all:
|
||||||
sed 's/__REV__/$(REV)/g' ddraw.rc.in > ddraw.rc
|
sed 's/__REV__/$(REV)/g' ddraw.rc.in > ddraw.rc
|
||||||
$(WINDRES) -J rc ddraw.rc ddraw.rc.o
|
$(WINDRES) -J rc ddraw.rc ddraw.rc.o
|
||||||
$(CC) $(CFLAGS) -shared -o ddraw.dll main.c mouse.c palette.c surface.c clipper.c render.c screenshot.c ddraw.def ddraw.rc.o $(LIBS)
|
$(CC) $(CFLAGS) -shared -o ddraw.dll main.c mouse.c palette.c surface.c clipper.c render.c render_soft.c screenshot.c ddraw.def ddraw.rc.o $(LIBS)
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
rm -f ddraw.dll
|
rm -f ddraw.dll
|
||||||
|
17
main.c
17
main.c
@ -38,6 +38,7 @@ BOOL screenshot(struct IDirectDrawSurfaceImpl *);
|
|||||||
IDirectDrawImpl *ddraw = NULL;
|
IDirectDrawImpl *ddraw = NULL;
|
||||||
|
|
||||||
DWORD WINAPI render_main(void);
|
DWORD WINAPI render_main(void);
|
||||||
|
DWORD WINAPI render_soft_main(void);
|
||||||
|
|
||||||
HRESULT __stdcall ddraw_Compact(IDirectDrawImpl *This)
|
HRESULT __stdcall ddraw_Compact(IDirectDrawImpl *This)
|
||||||
{
|
{
|
||||||
@ -247,7 +248,7 @@ HRESULT __stdcall ddraw_SetDisplayMode(IDirectDrawImpl *This, DWORD width, DWORD
|
|||||||
|
|
||||||
if(This->render.thread == NULL)
|
if(This->render.thread == NULL)
|
||||||
{
|
{
|
||||||
This->render.thread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)render_main, NULL, 0, NULL);
|
This->render.thread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)This->renderer, NULL, 0, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
return DD_OK;
|
return DD_OK;
|
||||||
@ -389,7 +390,7 @@ HRESULT __stdcall ddraw_SetCooperativeLevel(IDirectDrawImpl *This, HWND hWnd, DW
|
|||||||
memset(&pfd, 0, sizeof(PIXELFORMATDESCRIPTOR));
|
memset(&pfd, 0, sizeof(PIXELFORMATDESCRIPTOR));
|
||||||
pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR);
|
pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR);
|
||||||
pfd.nVersion = 1;
|
pfd.nVersion = 1;
|
||||||
pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER;
|
pfd.dwFlags = PFD_DRAW_TO_WINDOW | PFD_DOUBLEBUFFER | (This->renderer == render_main ? PFD_SUPPORT_OPENGL : 0);
|
||||||
pfd.iPixelType = PFD_TYPE_RGBA;
|
pfd.iPixelType = PFD_TYPE_RGBA;
|
||||||
pfd.cColorBits = ddraw->render.bpp ? ddraw->render.bpp : ddraw->mode.dmBitsPerPel;
|
pfd.cColorBits = ddraw->render.bpp ? ddraw->render.bpp : ddraw->mode.dmBitsPerPel;
|
||||||
pfd.iLayerType = PFD_MAIN_PLANE;
|
pfd.iLayerType = PFD_MAIN_PLANE;
|
||||||
@ -692,5 +693,17 @@ HRESULT WINAPI DirectDrawCreate(GUID FAR* lpGUID, LPDIRECTDRAW FAR* lplpDD, IUnk
|
|||||||
This->vhack = 0;
|
This->vhack = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GetPrivateProfileStringA("ddraw", "renderer", "opengl", tmp, sizeof(tmp), ini_path);
|
||||||
|
if(tolower(tmp[0]) == 's' || tolower(tmp[0]) == 'g')
|
||||||
|
{
|
||||||
|
printf("DirectDrawCreate: Using software renderer\n");
|
||||||
|
This->renderer = render_soft_main;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
printf("DirectDrawCreate: Using OpenGL renderer\n");
|
||||||
|
This->renderer = render_main;
|
||||||
|
}
|
||||||
|
|
||||||
return DD_OK;
|
return DD_OK;
|
||||||
}
|
}
|
||||||
|
1
main.h
1
main.h
@ -78,6 +78,7 @@ typedef struct IDirectDrawImpl
|
|||||||
BOOL vsync;
|
BOOL vsync;
|
||||||
float sensitivity;
|
float sensitivity;
|
||||||
BOOL vhack;
|
BOOL vhack;
|
||||||
|
DWORD WINAPI (*renderer)(void);
|
||||||
|
|
||||||
} IDirectDrawImpl;
|
} IDirectDrawImpl;
|
||||||
|
|
||||||
|
98
render_soft.c
Normal file
98
render_soft.c
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2011 Toni Spets <toni.spets@iki.fi>
|
||||||
|
*
|
||||||
|
* Permission to use, copy, modify, and distribute this software for any
|
||||||
|
* purpose with or without fee is hereby granted, provided that the above
|
||||||
|
* copyright notice and this permission notice appear in all copies.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "main.h"
|
||||||
|
#include "surface.h"
|
||||||
|
|
||||||
|
DWORD WINAPI render_soft_main(void)
|
||||||
|
{
|
||||||
|
HDC memDC = CreateCompatibleDC(ddraw->render.hDC);
|
||||||
|
HBITMAP surface = CreateCompatibleBitmap(ddraw->render.hDC, ddraw->width, ddraw->height);
|
||||||
|
|
||||||
|
PBITMAPINFO bmi = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(BITMAPINFO) + (sizeof(RGBQUAD) * 256) + 1024);
|
||||||
|
|
||||||
|
bmi->bmiHeader.biSize = sizeof(BITMAPINFO);
|
||||||
|
bmi->bmiHeader.biWidth = ddraw->width;
|
||||||
|
bmi->bmiHeader.biHeight = -ddraw->height;
|
||||||
|
bmi->bmiHeader.biPlanes = 1;
|
||||||
|
bmi->bmiHeader.biBitCount = 8;
|
||||||
|
bmi->bmiHeader.biCompression = BI_RGB;
|
||||||
|
|
||||||
|
SelectObject(memDC, surface);
|
||||||
|
|
||||||
|
DWORD tick_start = 0;
|
||||||
|
DWORD tick_end = 0;
|
||||||
|
DWORD frame_len = 0;
|
||||||
|
|
||||||
|
if(ddraw->render.maxfps < 0)
|
||||||
|
{
|
||||||
|
ddraw->render.maxfps = ddraw->mode.dmDisplayFrequency;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(ddraw->render.maxfps > 0)
|
||||||
|
{
|
||||||
|
frame_len = 1000.0f / ddraw->render.maxfps;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (ddraw->render.run)
|
||||||
|
{
|
||||||
|
if(ddraw->render.maxfps > 0)
|
||||||
|
{
|
||||||
|
tick_start = GetTickCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
EnterCriticalSection(&ddraw->cs);
|
||||||
|
if (ddraw->primary && ddraw->primary->palette)
|
||||||
|
{
|
||||||
|
if (ddraw->primary->palette->data_rgb == NULL)
|
||||||
|
{
|
||||||
|
ddraw->primary->palette->data_rgb = &bmi->bmiColors[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
SetDIBits(memDC, surface, 0, ddraw->height, ddraw->primary->surface, bmi, DIB_RGB_COLORS);
|
||||||
|
|
||||||
|
if (ddraw->render.width != ddraw->width || ddraw->render.height != ddraw->height)
|
||||||
|
{
|
||||||
|
StretchBlt(ddraw->render.hDC, 0, 0, ddraw->render.width, ddraw->render.height, memDC, 0, 0, ddraw->width, ddraw->height, SRCCOPY);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
BitBlt(ddraw->render.hDC, 0, 0, ddraw->width, ddraw->height, memDC, 0, 0, SRCCOPY);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LeaveCriticalSection(&ddraw->cs);
|
||||||
|
|
||||||
|
if(ddraw->render.maxfps > 0)
|
||||||
|
{
|
||||||
|
tick_end = GetTickCount();
|
||||||
|
|
||||||
|
if(tick_end - tick_start < frame_len)
|
||||||
|
{
|
||||||
|
Sleep( frame_len - (tick_end - tick_start) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SetEvent(ddraw->render.ev);
|
||||||
|
}
|
||||||
|
|
||||||
|
HeapFree(GetProcessHeap(), 0, bmi);
|
||||||
|
DeleteObject(surface);
|
||||||
|
DeleteDC(memDC);
|
||||||
|
return TRUE;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user