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.
 
 
 
 
 
 

722 lines
14 KiB

#include "pch.h"
#ifndef _WIN32
#include <sys/statvfs.h>
#endif
#include <SDL.h>
namespace dvl {
DWORD last_error;
DWORD GetLastError()
{
return last_error;
}
void SetLastError(DWORD dwErrCode)
{
last_error = dwErrCode;
}
char *_strlwr(char *str)
{
for (char *p = str; *p; ++p) {
*p = tolower(*p);
}
return str;
}
int wsprintfA(LPSTR dest, LPCSTR format, ...)
{
va_list args;
va_start(args, format);
return vsprintf(dest, format, args);
}
int wvsprintfA(LPSTR dest, LPCSTR format, va_list arglist)
{
return vsnprintf(dest, 256, format, arglist);
}
int _strcmpi(const char *_Str1, const char *_Str2)
{
return strcasecmp(_Str1, _Str2);
}
char *_itoa(int _Value, char *_Dest, int _Radix)
{
switch (_Radix) {
case 8:
sprintf(_Dest, "%o", _Value);
break;
case 10:
sprintf(_Dest, "%d", _Value);
break;
case 16:
sprintf(_Dest, "%x", _Value);
break;
default:
UNIMPLEMENTED();
break;
}
return _Dest;
}
DWORD GetTickCount()
{
return SDL_GetTicks();
}
void Sleep(DWORD dwMilliseconds)
{
SDL_Delay(dwMilliseconds);
}
HANDLE FindFirstFileA(LPCSTR lpFileName, LPWIN32_FIND_DATAA lpFindFileData)
{
DUMMY();
return (HANDLE)-1;
}
WINBOOL FindClose(HANDLE hFindFile)
{
UNIMPLEMENTED();
}
/**
* @brief Normally this would get the Windows install, but Diablo uses it to find the old save game folder
*/
UINT GetWindowsDirectoryA(LPSTR lpBuffer, UINT uSize)
{
char *name = SDL_GetPrefPath("diasurgical", "devilution");
strncpy(lpBuffer, name, uSize);
SDL_free(name);
DWORD len = strlen(lpBuffer);
lpBuffer[len - 1] = '\0';
return len - 1;
}
WINBOOL GetDiskFreeSpaceA(LPCSTR lpRootPathName, LPDWORD lpSectorsPerCluster, LPDWORD lpBytesPerSector,
LPDWORD lpNumberOfFreeClusters, LPDWORD lpTotalNumberOfClusters)
{
#ifndef _WIN32
struct statvfs fiData;
int success = statvfs("/", &fiData);
*lpBytesPerSector = fiData.f_frsize;
*lpSectorsPerCluster = 1;
*lpNumberOfFreeClusters = fiData.f_bavail;
*lpTotalNumberOfClusters = fiData.f_blocks;
return success >= 0;
#else
// Todo give windows the real GetDiskFreeSpace
DUMMY();
*lpBytesPerSector = 1;
*lpSectorsPerCluster = 1;
*lpNumberOfFreeClusters = 10 << 20;
*lpTotalNumberOfClusters = 10 << 20;
return true;
#endif
}
/**
* @brief Used for getting save path, by removing up to and including the last "\"
*/
DWORD GetModuleFileNameA(HMODULE hModule, LPSTR lpFilename, DWORD nSize)
{
char *name = SDL_GetPrefPath("diasurgical", "devilution");
strncpy(lpFilename, name, nSize);
SDL_free(name);
DWORD len = strlen(lpFilename);
lpFilename[len - 1] = '\\';
return len;
}
WINBOOL GetComputerNameA(LPSTR lpBuffer, LPDWORD nSize)
{
DUMMY();
strncpy(lpBuffer, "localhost", *nSize);
*nSize = strlen(lpBuffer);
return true;
}
DWORD GetFileVersionInfoSizeA(LPCSTR lptstrFilename, LPDWORD lpdwHandle)
{
DUMMY();
*lpdwHandle = 0;
return 1532;
}
BOOL GetFileVersionInfoA(LPCSTR lptstrFilename, DWORD dwHandle, DWORD dwLen, LPVOID lpData)
{
DUMMY();
*(int *)lpData = 16711836; // TODO use actual version from .rc
return true;
}
BOOL VerQueryValueA(LPCVOID pBlock, LPCSTR lpSubBlock, LPVOID *lplpBuffer, PUINT puLen)
{
DUMMY();
static VS_FIXEDFILEINFO lpBuffer;
// Set internal version, TODO use actual version from .rc
lpBuffer.dwProductVersionMS = 1;
lpBuffer.dwProductVersionMS <<= 16;
lpBuffer.dwProductVersionMS |= 0 & 0xFFFF;
lpBuffer.dwProductVersionLS = 9;
lpBuffer.dwProductVersionLS <<= 16;
lpBuffer.dwProductVersionLS |= 2 & 0xFFFF;
*lplpBuffer = (LPVOID *)&lpBuffer;
return true;
}
DWORD GetCurrentDirectory(DWORD nBufferLength, LPTSTR lpBuffer)
{
char *base_path = SDL_GetBasePath();
if (!base_path) {
base_path = SDL_strdup("./");
}
strncpy(lpBuffer, base_path, nBufferLength);
SDL_free(base_path);
DWORD len = strlen(lpBuffer);
lpBuffer[len - 1] = '\\';
return len;
}
DWORD GetLogicalDriveStringsA(DWORD nBufferLength, LPSTR lpBuffer)
{
DUMMY();
sprintf(lpBuffer, "/");
return 3;
}
UINT GetDriveTypeA(LPCSTR lpRootPathName)
{
return DVL_DRIVE_CDROM;
}
WINBOOL DeleteFileA(LPCSTR lpFileName)
{
UNIMPLEMENTED();
}
WINBOOL CopyFileA(LPCSTR lpExistingFileName, LPCSTR lpNewFileName, WINBOOL bFailIfExists)
{
UNIMPLEMENTED();
}
HFILE OpenFile(LPCSTR lpFileName, LPOFSTRUCT lpReOpenBuff, UINT uStyle)
{
DUMMY();
return DVL_HFILE_ERROR;
}
HWND SetCapture(HWND hWnd)
{
DUMMY_ONCE();
return hWnd;
}
WINBOOL ReleaseCapture()
{
DUMMY_ONCE();
return true;
}
WINBOOL DestroyWindow(HWND hWnd)
{
DUMMY();
return true;
}
HWND GetLastActivePopup(HWND hWnd)
{
UNIMPLEMENTED();
}
HWND GetTopWindow(HWND hWnd)
{
UNIMPLEMENTED();
}
WINBOOL SetForegroundWindow(HWND hWnd)
{
UNIMPLEMENTED();
}
HWND SetFocus(HWND hWnd)
{
UNIMPLEMENTED();
}
HWND FindWindowA(LPCSTR lpClassName, LPCSTR lpWindowName)
{
DUMMY_PRINT("class: %s window: %s", nullstr(lpClassName), nullstr(lpWindowName));
return NULL;
}
void FakeWMDestroy()
{
MainWndProc(NULL, DVL_WM_DESTROY, 0, NULL);
}
HWND CreateWindowExA(
DWORD dwExStyle,
LPCSTR lpClassName,
LPCSTR lpWindowName,
DWORD dwStyle,
int X,
int Y,
int nWidth,
int nHeight,
HWND hWndParent,
HMENU hMenu,
HINSTANCE hInstance,
LPVOID lpParam)
{
if (SDL_Init(SDL_INIT_EVERYTHING) != 0) {
SDL_Log("SDL_Init: %s\n", SDL_GetError());
return NULL;
}
atexit(SDL_Quit);
int flags = SDL_WINDOW_FULLSCREEN_DESKTOP | SDL_WINDOW_INPUT_GRABBED;
if (!fullscreen) {
flags = SDL_WINDOW_RESIZABLE;
}
window = SDL_CreateWindow(lpWindowName, SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, nWidth, nHeight, flags);
atexit(FakeWMDestroy);
return window;
}
BOOL InvalidateRect(HWND hWnd, const RECT *lpRect, BOOL bErase)
{
DUMMY();
return true;
}
/**
* @brief Appears to be used to clear the FB on init
*/
BOOL UpdateWindow(HWND hWnd)
{
DUMMY();
return true;
}
BOOL ShowWindow(HWND hWnd, int nCmdShow)
{
if (nCmdShow == DVL_SW_HIDE) {
SDL_HideWindow(window);
} else if (nCmdShow == DVL_SW_SHOWNORMAL) {
SDL_ShowWindow(window);
}
return true;
}
ATOM RegisterClassExA(const WNDCLASSEXA *lpwcx)
{
DUMMY();
return 1;
}
/**
* @brief Because we don't change resolution it dosen't make sens to use SDL_GetCurrentDisplayMode
*/
int GetSystemMetrics(int nIndex)
{
switch (nIndex) {
case DVL_SM_CXSCREEN:
return SCREEN_WIDTH;
case DVL_SM_CYSCREEN:
return SCREEN_HEIGHT;
}
return 0;
}
/**
* @brief Used for getting a black brush
*/
HGDIOBJ GetStockObject(int i)
{
return NULL;
}
/**
* @brief Used to load window icon
*/
HICON LoadIconA(HINSTANCE hInstance, LPCSTR lpIconName)
{
DUMMY(); //SDL_SetWindowIcon
return NULL;
}
/**
* @brief Used to load small window icon
*/
HANDLE LoadImageA(HINSTANCE hInst, LPCSTR name, UINT type, int cx, int cy, UINT fuLoad)
{
DUMMY();
return NULL;
}
HCURSOR LoadCursorA(HINSTANCE hInstance, LPCSTR lpCursorName)
{
DUMMY(); //SDL_CreateCursor
return NULL;
}
BOOL IsBadReadPtr(const void *lp, UINT_PTR ucb)
{
UNIMPLEMENTED();
return true;
}
BOOL IsBadWritePtr(LPVOID lp, UINT_PTR ucb)
{
UNIMPLEMENTED();
return true;
}
SIZE_T VirtualQuery(LPCVOID lpAddress, PMEMORY_BASIC_INFORMATION lpBuffer, SIZE_T dwLength)
{
UNIMPLEMENTED();
return false;
}
LPVOID VirtualAlloc(LPVOID lpAddress, SIZE_T dwSize, DWORD flAllocationType, DWORD flProtect)
{
UNIMPLEMENTED();
return NULL;
}
BOOL VirtualFree(LPVOID lpAddress, SIZE_T dwSize, DWORD dwFreeType)
{
UNIMPLEMENTED();
return false;
}
void GetLocalTime(LPSYSTEMTIME lpSystemTime)
{
UNIMPLEMENTED();
}
long _findfirst(const char *, struct DVL_finddata_t *)
{
UNIMPLEMENTED();
return -1;
}
int _findnext(long, struct DVL_finddata_t *)
{
UNIMPLEMENTED();
return -1;
}
/**
* @brief Used to shutdown a MS Office 95 tool bar
*/
HWND GetForegroundWindow()
{
return NULL;
}
LPTOP_LEVEL_EXCEPTION_FILTER SetUnhandledExceptionFilter(LPTOP_LEVEL_EXCEPTION_FILTER lpTopLevelExceptionFilter)
{
DUMMY();
return lpTopLevelExceptionFilter;
}
HMODULE GetModuleHandleA(LPCSTR lpModuleName)
{
UNIMPLEMENTED();
return NULL;
}
BOOL GetUserNameA(LPSTR lpBuffer, LPDWORD pcbBuffer)
{
UNIMPLEMENTED();
return false;
}
/**
* @brief Used to shutdown a MS Office 95 tool bar
*/
int GetClassName(HWND hWnd, LPTSTR lpClassName, int nMaxCount)
{
return 0;
}
/**
* @brief Used to find MS Office 95
*/
HRESULT SHGetSpecialFolderLocation(HWND hwnd, int csidl, PIDLIST_ABSOLUTE *ppidl)
{
return 0;
}
/**
* @brief Used to find MS Office 95
*/
HINSTANCE ShellExecuteA(HWND hwnd, LPCSTR lpOperation, LPCSTR lpFile, LPCSTR lpParameters, LPCSTR lpDirectory, INT nShowCmd)
{
return NULL;
}
/**
* @brief Used to find MS Office 95
*/
HWND GetDesktopWindow()
{
return NULL;
}
/**
* @brief Used to find MS Office 95
*/
BOOL SHGetPathFromIDListA(PCIDLIST_ABSOLUTE pidl, LPSTR pszPath)
{
return false;
}
/**
* @brief Used to find MS Office 95
*/
BOOL FindNextFileA(HANDLE hFindFile, LPWIN32_FIND_DATAA lpFindFileData)
{
return false;
}
void GetSystemInfo(LPSYSTEM_INFO lpSystemInfo)
{
DUMMY();
memset(lpSystemInfo, 0, sizeof(*lpSystemInfo));
lpSystemInfo->dwPageSize = 4096;
}
HDC GetDC(HWND hWnd)
{
DUMMY();
return NULL;
}
int ReleaseDC(HWND hWnd, HDC hDC)
{
DUMMY();
return 0;
}
int GetDeviceCaps(HDC hdc, int index)
{
SDL_DisplayMode current;
if (SDL_GetCurrentDisplayMode(0, &current) != 0) {
SDL_Log("SDL_GetCurrentDisplayMode: %s", SDL_GetError());
return 0;
}
if (index == DVL_HORZRES) {
return current.w;
}
if (index == DVL_VERTRES) {
return current.h;
}
return 0;
}
BOOL GetWindowRect(HWND hDlg, tagRECT *Rect)
{
int x, y, w, h;
SDL_GetWindowPosition(window, &x, &y);
SDL_GetWindowSize(window, &w, &h);
Rect->right = x;
Rect->top = y;
Rect->left = w + x;
Rect->bottom = h + y;
return true;
}
UINT GetSystemPaletteEntries(HDC hdc, UINT iStart, UINT cEntries, LPPALETTEENTRY pPalEntries)
{
DUMMY();
return 0;
}
BOOL GetVersionExA(LPOSVERSIONINFOA lpVersionInformation)
{
lpVersionInformation->dwMajorVersion = 5;
lpVersionInformation->dwMinorVersion = 0;
lpVersionInformation->dwPlatformId = DVL_VER_PLATFORM_WIN32_NT;
return true;
}
void lstrcpynA(LPSTR lpString1, LPCSTR lpString2, int iMaxLength)
{
strncpy(lpString1, lpString2, iMaxLength);
}
WINBOOL CreateProcessA(LPCSTR lpApplicationName, LPSTR lpCommandLine, LPSECURITY_ATTRIBUTES lpProcessAttributes,
LPSECURITY_ATTRIBUTES lpThreadAttributes, WINBOOL bInheritHandles, DWORD dwCreationFlags,
LPVOID lpEnvironment, LPCSTR lpCurrentDirectory, LPSTARTUPINFOA lpStartupInfo,
LPPROCESS_INFORMATION lpProcessInformation)
{
UNIMPLEMENTED();
return false;
}
void ExitProcess(UINT uExitCode)
{
UNIMPLEMENTED();
}
DWORD GetCurrentProcessId()
{
UNIMPLEMENTED();
return 0;
}
HANDLE CreateFileMappingA(HANDLE hFile, LPSECURITY_ATTRIBUTES lpFileMappingAttributes, DWORD flProtect,
DWORD dwMaximumSizeHigh, DWORD dwMaximumSizeLow, LPCSTR lpName)
{
DUMMY();
assert(hFile == (HANDLE)-1);
return NULL;
}
LPVOID MapViewOfFile(HANDLE hFileMappingObject, DWORD dwDesiredAccess, DWORD dwFileOffsetHigh,
DWORD dwFileOffsetLow, SIZE_T dwNumberOfBytesToMap)
{
UNIMPLEMENTED();
}
WINBOOL UnmapViewOfFile(LPCVOID lpBaseAddress)
{
UNIMPLEMENTED();
}
DWORD WaitForInputIdle(HANDLE hProcess, DWORD dwMilliseconds)
{
UNIMPLEMENTED();
}
HWND GetWindow(HWND hWnd, UINT uCmd)
{
UNIMPLEMENTED();
}
DWORD GetWindowThreadProcessId(HWND hWnd, LPDWORD lpdwProcessId)
{
UNIMPLEMENTED();
}
DWORD GetPrivateProfileStringA(LPCSTR lpAppName, LPCSTR lpKeyName, LPCSTR lpDefault, LPSTR lpReturnedString,
DWORD nSize, LPCSTR lpFileName)
{
if (!SRegLoadString(lpAppName, lpKeyName, 0, lpReturnedString, nSize)) {
strncpy(lpReturnedString, lpDefault, nSize);
SRegSaveString(lpAppName, lpKeyName, 0, lpReturnedString);
}
}
int MessageBoxA(HWND hWnd, const char *Text, const char *Title, UINT Flags)
{
Uint32 SDLFlags = 0;
if (Flags & DVL_MB_ICONHAND) {
SDLFlags |= SDL_MESSAGEBOX_ERROR;
} else if (Flags & DVL_MB_ICONEXCLAMATION) {
SDLFlags |= SDL_MESSAGEBOX_WARNING;
}
return SDL_ShowSimpleMessageBox(SDLFlags, Title, Text, window) < 0 ? -1 : 0;
}
LSTATUS RegOpenKeyExA(HKEY hKey, LPCSTR lpSubKey, DWORD ulOptions, REGSAM samDesired, PHKEY phkResult)
{
return 1;
};
/**
* @brief This is only ever used to enable or disable the screen saver in a hackish way
* For all other settings operation SReg* from Storm is used instead.
*/
LSTATUS RegQueryValueExA(HKEY hKey, LPCSTR lpValueName, LPDWORD lpReserved, LPDWORD lpType, BYTE *lpData, LPDWORD lpcbData)
{
if (SDL_IsScreenSaverEnabled()) {
lpData[0] = '0';
lpData[1] = '\0';
}
return 1;
};
/**
* @brief This is only ever used to enable or disable the screen saver in a hackish way
* For all other settings operation SReg* from Storm is used instead.
*/
LSTATUS RegSetValueExA(HKEY hKey, LPCSTR lpValueName, DWORD Reserved, DWORD dwType, const BYTE *lpData, DWORD cbData)
{
if (lpData[0] == '0') {
SDL_DisableScreenSaver();
} else {
SDL_EnableScreenSaver();
}
return 1;
};
LSTATUS RegCloseKeyA(HKEY hKey)
{
return 1;
};
void PostQuitMessage(int nExitCode)
{
DUMMY();
PostMessageA(NULL, DVL_WM_QUERYENDSESSION, 0, 0);
}
LRESULT DefWindowProcA(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
{
DUMMY_ONCE();
if (Msg == DVL_WM_QUERYENDSESSION)
exit(0);
return NULL;
}
LONG GetWindowLongA(HWND hWnd, int nIndex)
{
DUMMY();
return 0;
}
LONG SetWindowLongA(HWND hWnd, int nIndex, LONG dwNewLong)
{
DUMMY();
return 0;
}
}