Subversion Repositories Games.Carmageddon

Rev

Rev 11 | Go to most recent revision | Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
1 pmbaty 1
#include "brender/brender.h"
2
#include "car.h"
3
#include "dinput.h"
4
#include "errors.h"
5
#include "globvars.h"
6
#include "grafdata.h"
7
#include "graphics.h"
8
#include "harness/config.h"
9
#include "harness/hooks.h"
10
#include "harness/os.h"
11
#include "harness/trace.h"
12
#include "input.h"
13
#include "loadsave.h"
14
#include "main.h"
15
#include "pd/sys.h"
16
#include "sound.h"
17
#include "ssdx.h"
18
#include "utility.h"
19
#include <errno.h>
20
#include <stdio.h>
21
#include <string.h>
22
#include <sys/stat.h>
23
#include <time.h>
24
 
25
#include "harness/win95_polyfill.h"
26
 
27
#define GFX_INIT_STRING_32X20X8 "320x200 init string"
28
#define GFX_INIT_STRING_64X48X8 "640x480 init string"
29
 
30
#define KEYDOWN(var, key) (var[key] & 0x80)
31
 
32
int gExtra_mem;
33
int gReplay_override;
34
tGraf_spec gGraf_specs[2] = {
35
    { 8, 1, 0, 320, 200, 0, 0, "32X20X8", GFX_INIT_STRING_32X20X8, 320, 320, 200, NULL },
36
    { 8, 1, 0, 640, 480, 0, 0, "64X48X8", GFX_INIT_STRING_64X48X8, 640, 480, 480, NULL }
37
};
38
int gASCII_table[128];
39
tU32 gKeyboard_bits[8];
40
int gASCII_shift_table[128];
41
char gNetwork_profile_fname[256];
42
tS32 gJoystick_min1y;
43
tS32 gJoystick_min2y;
44
tS32 gJoystick_min2x;
45
tS32 gRaw_joystick2y;
46
tS32 gRaw_joystick2x;
47
tS32 gRaw_joystick1y;
48
tS32 gRaw_joystick1x;
49
tS32 gJoystick_range2y;
50
tS32 gJoystick_range2x;
51
tS32 gJoystick_range1y;
52
tS32 gJoystick_range1x;
53
int gNo_voodoo;
54
int gSwitched_resolution;
55
br_pixelmap* gReal_back_screen;
56
tS32 gJoystick_min1x;
57
br_pixelmap* gTemp_screen;
58
int gGfx_initialized; // maybe renamed here
59
tU32 gUpper_loop_limit;
60
int gReal_back_screen_locked;
61
tU32 gScan_code[123]; // was tU8 [123][2] in symbol dump
62
 
63
// Added by dethrace. Windows-specific. Original variable names unknown.
64
int gWin32_fatal_error_exit_code;
65
int gWin32_show_fatal_error_message;
66
char gWin32_fatal_error_message[512];
67
int gWin32_action_replay_buffer_allocated;
68
void* gWin32_action_replay_buffer;
69
int gWin32_action_replay_buffer_size;
70
void* gWin32_hwnd;
71
int gWin32_lbutton_down;
72
int gWin32_rbutton_down;
73
PALETTEENTRY_ gWin32_palette[256];
74
br_diaghandler gWin32_br_diaghandler;
75
int gWin32_window_has_focus = 1;
76
int gNetwork_profile_file_exists = 0;
77
 
78
char* _unittest_last_fatal_error;
79
 
80
int original_main(int pArgc, char** pArgv);
81
 
82
/*static*/ void KeyboardHandler(void);
83
/*static*/ int KeyDown(tU8 pScan_code);
84
/*static*/ void KeyTranslation(tU8 pKey_index, tU8 pScan_code_1, tU8 pScan_code_2);
85
/*static*/ void KeyBegin(void);
86
/*static*/ void KeyEnd(void);
87
/*static*/ int KeyDown22(int pKey_index);
88
static void Win32ReleaseInputDevice(void);
89
static void Win32PumpMessages(void);
90
static HARNESS_NORETURN void Win32FatalError(char* pStr_1, char* pStr_2);
91
static void Win32CreateWindow(void);
92
static void Win32InitScreen(void);
93
/*static*/ void Copy8BitTo16BitPixelmap(br_pixelmap* pDst, br_pixelmap* pSrc, br_pixelmap* pPalette);
94
/*static*/ void SwapBackScreen(void);
95
/*static*/ void ReallyCopyBackScreen(int pRendering_area_only, int pClear_top_and_bottom);
96
/*static*/ void CopyBackScreen(int pRendering_area_only);
97
static void Win32SetPaletteEntries(uint8_t* entries, int pFirst_colour, int pCount);
98
static void Win32InitInputDevice(void);
99
static void Win32AllocateActionReplayBuffer(void);
100
static void Usage(char* pProgpath);
101
/*static*/ int OurGetChar(void);
102
/*static*/ int InitJoysticks(void);
103
/*static*/ tU32 ReadJoystickAxis(int pBit);
104
static void Win32BRenderWarningFunc(char* msg);
105
static void Win32BRenderFailureFunc(char* msg);
106
 
107
extern void QuitGame(void);
108
 
109
void KeyboardHandler(void) {
110
    //tU8 scan_code; // Pierre-Marie Baty -- unused variable
111
    //tU8 up; // Pierre-Marie Baty -- unused variable
112
    //static tU8 extended; // Pierre-Marie Baty -- unused variable
113
    LOG_TRACE("()");
114
    NOT_IMPLEMENTED();
115
}
116
 
117
int KeyDown(tU8 pScan_code) {
118
    NOT_IMPLEMENTED();
119
}
120
 
121
void KeyTranslation(tU8 pKey_index, tU8 pScan_code_1, tU8 pScan_code_2) {
122
    LOG_TRACE("(%d, %d, %d)", pKey_index, pScan_code_1, pScan_code_2);
123
    NOT_IMPLEMENTED();
124
}
125
 
126
void KeyBegin(void) {
127
    gScan_code[KEY_0] = DIK_0;
128
    gScan_code[KEY_1] = DIK_1;
129
    gScan_code[KEY_2] = DIK_2;
130
    gScan_code[KEY_3] = DIK_3;
131
    gScan_code[KEY_4] = DIK_4;
132
    gScan_code[KEY_5] = DIK_5;
133
    gScan_code[KEY_6] = DIK_6;
134
    gScan_code[KEY_7] = DIK_7;
135
    gScan_code[KEY_8] = DIK_8;
136
    gScan_code[KEY_9] = DIK_9;
137
    gScan_code[KEY_A] = DIK_A;
138
    gScan_code[KEY_B] = DIK_B;
139
    gScan_code[KEY_C] = DIK_C;
140
    gScan_code[KEY_D] = DIK_D;
141
    gScan_code[KEY_E] = DIK_E;
142
    gScan_code[KEY_F] = DIK_F;
143
    gScan_code[KEY_G] = DIK_G;
144
    gScan_code[KEY_H] = DIK_H;
145
    gScan_code[KEY_I] = DIK_I;
146
    gScan_code[KEY_J] = DIK_J;
147
    gScan_code[KEY_K] = DIK_K;
148
    gScan_code[KEY_L] = DIK_L;
149
    gScan_code[KEY_M] = DIK_M;
150
    gScan_code[KEY_N] = DIK_N;
151
    gScan_code[KEY_O] = DIK_O;
152
    gScan_code[KEY_P] = DIK_P;
153
    gScan_code[KEY_Q] = DIK_Q;
154
    gScan_code[KEY_R] = DIK_R;
155
    gScan_code[KEY_S] = DIK_S;
156
    gScan_code[KEY_T] = DIK_T;
157
    gScan_code[KEY_U] = DIK_U;
158
    gScan_code[KEY_V] = DIK_V;
159
    gScan_code[KEY_W] = DIK_W;
160
    gScan_code[KEY_X] = DIK_X;
161
    gScan_code[KEY_Y] = DIK_Y;
162
    gScan_code[KEY_Z] = DIK_Z;
163
    gScan_code[KEY_GRAVE] = DIK_GRAVE;
164
    gScan_code[KEY_MINUS] = DIK_MINUS;
165
    gScan_code[KEY_EQUALS] = DIK_EQUALS;
166
    gScan_code[KEY_BACKSPACE] = DIK_BACK;
167
    gScan_code[KEY_RETURN] = DIK_RETURN;
168
    gScan_code[KEY_KP_ENTER] = DIK_NUMPADENTER;
169
    gScan_code[KEY_CAPSLOCK] = DIK_CAPSLOCK;
170
    gScan_code[KEY_TAB] = DIK_TAB;
171
    gScan_code[KEY_SLASH] = DIK_SLASH;
172
    gScan_code[KEY_UNKNOWN_55] = 0;
173
    gScan_code[KEY_SEMICOLON] = DIK_SEMICOLON;
174
    gScan_code[KEY_APOSTROPHE] = DIK_APOSTROPHE;
175
    gScan_code[KEY_PERIOD] = DIK_PERIOD;
176
    gScan_code[KEY_COMMA] = DIK_COMMA;
177
    gScan_code[KEY_LBRACKET] = DIK_LBRACKET;
178
    gScan_code[KEY_RBRACKET] = DIK_RBRACKET;
179
    gScan_code[KEY_ESCAPE] = DIK_ESCAPE;
180
    gScan_code[KEY_INSERT] = DIK_INSERT;
181
    gScan_code[KEY_DELETE] = DIK_DELETE;
182
    gScan_code[KEY_HOME] = DIK_HOME;
183
    gScan_code[KEY_END] = DIK_END;
184
    gScan_code[KEY_PAGEUP] = DIK_PGUP;
185
    gScan_code[KEY_PAGEDOWN] = DIK_PGDN;
186
    gScan_code[KEY_LEFT] = DIK_LEFT;
187
    gScan_code[KEY_RIGHT] = DIK_RIGHT;
188
    gScan_code[KEY_UP] = DIK_UP;
189
    gScan_code[KEY_DOWN] = DIK_DOWN;
190
    gScan_code[KEY_KP_NUMLOCK] = DIK_NUMLOCK;
191
    gScan_code[KEY_KP_DIVIDE] = DIK_DIVIDE;
192
    gScan_code[KEY_KP_MULTIPLY] = DIK_MULTIPLY;
193
    gScan_code[KEY_KP_MINUS] = DIK_SUBTRACT;
194
    gScan_code[KEY_KP_PLUS] = DIK_ADD;
195
    gScan_code[KEY_KP_PERIOD] = DIK_DECIMAL;
196
    gScan_code[KEY_KP_0] = DIK_NUMPAD0;
197
    gScan_code[KEY_KP_1] = DIK_NUMPAD1;
198
    gScan_code[KEY_KP_2] = DIK_NUMPAD2;
199
    gScan_code[KEY_KP_3] = DIK_NUMPAD3;
200
    gScan_code[KEY_KP_4] = DIK_NUMPAD4;
201
    gScan_code[KEY_KP_5] = DIK_NUMPAD5;
202
    gScan_code[KEY_KP_6] = DIK_NUMPAD6;
203
    gScan_code[KEY_KP_7] = DIK_NUMPAD7;
204
    gScan_code[KEY_KP_8] = DIK_NUMPAD8;
205
    gScan_code[KEY_KP_9] = DIK_NUMPAD9;
206
    gScan_code[KEY_F1] = DIK_F1;
207
    gScan_code[KEY_F2] = DIK_F2;
208
    gScan_code[KEY_F3] = DIK_F3;
209
    gScan_code[KEY_F4] = DIK_F4;
210
    gScan_code[KEY_F5] = DIK_F5;
211
    gScan_code[KEY_F6] = DIK_F6;
212
    gScan_code[KEY_F7] = DIK_F7;
213
    gScan_code[KEY_F8] = DIK_F8;
214
    gScan_code[KEY_F9] = DIK_F9;
215
    gScan_code[KEY_F10] = DIK_F10;
216
    gScan_code[KEY_F11] = DIK_F11;
217
    gScan_code[KEY_F12] = DIK_F12;
218
    gScan_code[KEY_PRTSCN] = DIK_SYSRQ;
219
    gScan_code[KEY_SCRLK] = DIK_SCROLL;
220
    gScan_code[KEY_SPACE] = DIK_SPACE;
221
    gScan_code[KEY_RSHIFT] = DIK_RSHIFT;
222
    gScan_code[KEY_RALT] = DIK_RALT;
223
    gScan_code[KEY_RCTRL] = DIK_RCONTROL;
224
    gScan_code[KEY_LSHIFT] = DIK_LSHIFT;
225
    gScan_code[KEY_LALT] = DIK_LALT;
226
    gScan_code[KEY_LCTRL] = DIK_LCONTROL;
227
    gScan_code[KEY_KP_EQUALS] = 0;
228
    gScan_code[KEY_BACKSLASH] = DIK_BACKSLASH;
229
    gScan_code[KEY_CTRL_ANY_2] = 0xff;
230
    gScan_code[KEY_PAUSE] = 0xff;
231
    gScan_code[KEY_SHIFT_ANY] = 0xff;
232
    gScan_code[KEY_ALT_ANY] = 0xff;
233
    gScan_code[KEY_CTRL_ANY] = 0xff;
234
}
235
 
236
void KeyEnd(void) {
237
    LOG_TRACE("()");
238
    NOT_IMPLEMENTED();
239
}
240
 
241
int KeyDown22(int pKey_index) {
242
    LOG_TRACE("(%d)", pKey_index);
243
    NOT_IMPLEMENTED();
244
}
245
 
246
void PDSetKeyArray(int* pKeys, int pMark) {
247
    int i;
248
    uint8_t keystate[256];
249
    LOG_TRACE10("(%p, %d)", pKeys, pMark);
250
 
251
    gKeys_pressed = 0;
252
    Win32PumpMessages();
253
    DirectInputDevice_GetDeviceState(COUNT_OF(keystate), keystate);
254
 
255
    for (i = 0; i < COUNT_OF(gScan_code); i++) {
256
        if (gScan_code[i] == 0) {
257
            continue;
258
        }
259
        if (gScan_code[i] == 0xff) {
260
            switch (i) {
261
            case KEY_SHIFT_ANY:
262
                if (KEYDOWN(keystate, gScan_code[KEY_LSHIFT]) || KEYDOWN(keystate, gScan_code[KEY_RSHIFT])) {
263
                    pKeys[KEY_SHIFT_ANY] = pMark;
264
                } else {
265
                    if (pKeys[KEY_SHIFT_ANY] == pMark) {
266
                        pKeys[KEY_SHIFT_ANY] = 0;
267
                    }
268
                }
269
                break;
270
            case KEY_ALT_ANY:
271
                if (KEYDOWN(keystate, gScan_code[KEY_LALT]) || KEYDOWN(keystate, gScan_code[KEY_RALT])) {
272
                    pKeys[KEY_ALT_ANY] = pMark;
273
                } else {
274
                    if (pKeys[KEY_ALT_ANY] == pMark) {
275
                        pKeys[KEY_ALT_ANY] = 0;
276
                    }
277
                }
278
                break;
279
            case KEY_CTRL_ANY:
280
            case KEY_CTRL_ANY_2:
281
                if (KEYDOWN(keystate, gScan_code[KEY_LCTRL]) || KEYDOWN(keystate, gScan_code[KEY_RCTRL])) {
282
                    pKeys[KEY_CTRL_ANY] = pMark;
283
                    pKeys[KEY_CTRL_ANY_2] = pMark;
284
                } else {
285
                    if (pKeys[KEY_CTRL_ANY] == pMark || pKeys[KEY_CTRL_ANY_2] == pMark) {
286
                        pKeys[KEY_CTRL_ANY] = 0;
287
                        pKeys[KEY_CTRL_ANY_2] = 0;
288
                    }
289
                }
290
                break;
291
            default:
292
                continue;
293
            }
294
        } else if (KEYDOWN(keystate, gScan_code[i])) {
295
            pKeys[i] = pMark;
296
            gKeys_pressed = i + (gKeys_pressed << 8) + 1;
297
        } else {
298
            if (pKeys[i] == pMark) {
299
                pKeys[i] = 0;
300
            }
301
        }
302
    }
303
}
304
 
305
void Win32ReleaseInputDevice(void) {
306
}
307
 
308
int PDGetASCIIFromKey(int pKey) {
309
    LOG_TRACE("(%d)", pKey);
310
 
311
    /* The Windows Carmageddon executable uses PDKeyDown here. The German DOS executable uses PDKeyDown3. */
312
    if (PDKeyDown3(KEY_SHIFT_ANY)) {
313
        return gASCII_shift_table[pKey];
314
    } else {
315
        return gASCII_table[pKey];
316
    }
317
}
318
 
319
void Win32PumpMessages(void) {
320
    MSG_ msg; // [esp+Ch] [ebp-20h] BYREF
321
 
322
    PDUnlockRealBackScreen();
323
    while ((!gWin32_window_has_focus || PeekMessageA_(&msg, 0, 0, 0, 1u)) && (gWin32_window_has_focus || GetMessageA_(&msg, 0, 0, 0) != -1)) {
324
        if (msg.message == WM_QUIT) {
325
            dr_dprintf("WM_QUIT received.");
326
            if (gWin32_window_has_focus) {
327
                dr_dprintf("Active, so lock the surface");
328
                PDLockRealBackScreen();
329
                dr_dprintf("QuitGame being called...");
330
                QuitGame();
331
            }
332
            PDShutdownSystem();
333
        }
334
        TranslateMessage_(&msg);
335
        DispatchMessageA_(&msg);
336
    }
337
    PDLockRealBackScreen();
338
}
339
 
340
void PDFatalError(char* pThe_str) {
341
    LOG_TRACE("(\"%s\")", pThe_str);
342
 
343
    dr_dprintf("FATAL ERROR: %s", pThe_str);
344
    Win32FatalError(pThe_str, "");
345
}
346
 
347
void Win32FatalError(char* pStr_1, char* pStr_2) {
348
    gWin32_show_fatal_error_message = 1;
349
    sprintf(gWin32_fatal_error_message, "%s\n%s", pStr_1, pStr_2);
350
    gWin32_fatal_error_exit_code = 15;
351
    PDShutdownSystem();
352
}
353
 
354
void PDNonFatalError(char* pThe_str) {
355
    LOG_TRACE("(\"%s\")", pThe_str);
356
    NOT_IMPLEMENTED();
357
}
358
 
359
void PDInitialiseSystem(void) {
360
    tPath_name the_path;
361
    FILE* f;
362
    int len;
363
 
364
    Win32AllocateActionReplayBuffer();
365
    gBack_screen = NULL;
366
    gScreen = NULL;
367
    Win32CreateWindow();
368
    ShowCursor_(0);
369
    KeyBegin();
370
 
371
    PathCat(the_path, gApplication_path, "KEYBOARD.COK");
372
    f = fopen(the_path, "rb");
373
    if (f == NULL) {
374
        if (harness_game_info.defines.requires_ascii_table) {
375
#if !defined(DETHRACE_FIX_BUGS)
376
            PDFatalError("This .exe must have KEYBOARD.COK in the DATA folder.");
377
#endif
378
        }
379
        // dethrace: demos do not ship with KEYBOARD.COK file
380
        memcpy(gASCII_table, harness_game_info.defines.ascii_table, sizeof(gASCII_table));
381
        memcpy(gASCII_shift_table, harness_game_info.defines.ascii_shift_table, sizeof(gASCII_shift_table));
382
    } else {
383
        PathCat(the_path, gApplication_path, "KEYBOARD.COK");
384
        f = fopen(the_path, "rb");
385
        if (f == NULL) {
386
            PDFatalError("This .exe must have KEYBOARD.COK in the DATA folder.");
387
        }
388
        fseek(f, 0, 2);
389
        len = ftell(f) / 2;
390
        rewind(f);
391
        fread(gASCII_table, len, 1u, f);
392
        fread(gASCII_shift_table, len, 1u, f);
393
        fclose(f);
394
    }
395
    Win32InitInputDevice();
396
}
397
 
398
void Win32CreateWindow(void) {
399
    // wnd_class.style = 3;
400
    // wnd_class.lpfnWndProc = window_proc;
401
    // wnd_class.cbClsExtra = 0;
402
    // wnd_class.cbWndExtra = 0;
403
    // wnd_class.hInstance = gWin32_hinst;
404
    // wnd_class.hIcon = LoadIconA(0, (LPCSTR)0x7F00);
405
    // wnd_class.hCursor = LoadCursorA(0, (LPCSTR)0x7F00);
406
    // wnd_class.hbrBackground = (HBRUSH)GetStockObject(4);
407
    // wnd_class.lpszMenuName = 0;
408
    // wnd_class.lpszClassName = "CarmageddonClass";
409
    // LOWORD(v5) = RegisterClassA(&wnd_class);
410
    // v5 = (unsigned __int16)v5;
411
    // v2 = gWin32_hinst;
412
 
413
    // int height = GetSystemMetrics(SM_CYSCREEN);
414
    // int width = GetSystemMetrics(SM_CXSCREEN);
415
 
416
    int width = gGraf_specs[gGraf_spec_index].total_width;
417
    int height = gGraf_specs[gGraf_spec_index].total_height;
418
    // WS_VISIBLE | WS_POPUP
419
    gWin32_hwnd = CreateWindowExA_(0, "CarmageddonClass", "Carmageddon", 0x90000000, 0, 0, width, height, 0, NULL, NULL, NULL);
420
    SSDXGetWindowRect(gWin32_hwnd);
421
    // hdc = GetDC(gWin32_hwnd);
422
    // GetSystemPaletteEntries(hdc, 0, 256u, &gWin32_palette);
423
    // ReleaseDC(gWin32_hwnd, hdc);
424
    // UpdateWindow(gWin32_hwnd);
425
    // SetFocus(gWin32_hwnd);
426
}
427
 
428
void PDShutdownSystem(void) {
429
    static int been_here = 0;
430
    LOG_TRACE("()");
431
 
432
    if (!been_here) {
433
        been_here = 1;
434
        dr_dprintf("PDShutdownSystem()...");
435
        SSDXRelease();
436
        Win32ReleaseInputDevice();
437
        ShowCursor_(1);
438
        if (gWin32_hwnd) {
439
            dr_dprintf("Resizing main window...");
440
            SetWindowPos_(gWin32_hwnd, 0, -100, -100, 64, 64, 0x414u);
441
        }
442
        dr_dprintf("Servicing messages...");
443
        Win32PumpMessages();
444
        dr_dprintf("Sending WM_SHOWWINDOW broadcast message...");
445
        SendMessageA_(HWND_BROADCAST, 0x18u, 1u, 0);
446
        if (gWin32_show_fatal_error_message) {
447
            dr_dprintf("Displaying fatal error...");
448
            MessageBoxA_(0, gWin32_fatal_error_message, "Carmageddon Fatal Error", MB_ICONERROR);
449
        }
450
        if (gWin32_hwnd) {
451
            dr_dprintf("Destroying window...");
452
            DestroyWindow_(gWin32_hwnd);
453
            gWin32_hwnd = 0;
454
        }
455
        dr_dprintf("End of PDShutdownSystem().");
456
        CloseDiagnostics();
457
        ExitProcess_(gWin32_fatal_error_exit_code);
458
    }
459
    ExitProcess_(8u);
460
}
461
 
462
void PDSaveOriginalPalette(void) {
463
    LOG_TRACE("()");
464
    NOT_IMPLEMENTED();
465
}
466
 
467
void PDRevertPalette(void) {
468
    LOG_TRACE("()");
469
 
470
    // empty function
471
}
472
 
473
int PDInitScreenVars(int pArgc, char** pArgv) {
474
    // this codes from dossys
475
    gGraf_specs[gGraf_spec_index].phys_width = gGraf_specs[gGraf_spec_index].total_width;
476
    gGraf_specs[gGraf_spec_index].phys_height = gGraf_specs[gGraf_spec_index].total_height;
477
    return 1;
478
}
479
 
480
void PDInitScreen(void) {
481
    Win32InitScreen();
482
}
483
 
484
void Win32InitScreen(void) {
485
    // SWP_NOSENDCHANGING | SWP_NOACTIVATE | SWP_NOZORDER
486
    SetWindowPos_(gWin32_hwnd, 0, 0, 0, gGraf_specs[gReal_graf_data_index].total_width, gGraf_specs[gReal_graf_data_index].total_height, 0x414u);
487
    // get_window_rect(gWin32_hwnd);
488
    Win32PumpMessages();
489
}
490
 
491
void PDLockRealBackScreen(void) {
492
    LOG_TRACE("()");
493
 
494
    // no-op
495
}
496
 
497
void PDUnlockRealBackScreen(void) {
498
    LOG_TRACE("()");
499
 
500
    // no-op
501
}
502
 
503
void PDAllocateScreenAndBack(void) {
504
    // this is a mix of windows and dos code
505
 
506
    dr_dprintf("PDAllocateScreenAndBack() - START...");
507
    BrMaterialFindHook(PDMissingMaterial);
508
    BrTableFindHook(PDMissingTable);
509
    BrModelFindHook(PDMissingModel);
510
    BrMapFindHook(PDMissingMap);
511
 
512
    int row_bytes;
513
    SSDXInitDirectDraw(gGraf_specs[gGraf_spec_index].total_width, gGraf_specs[gGraf_spec_index].total_height, &row_bytes);
514
    gScreen = BrPixelmapAllocate(BR_PMT_INDEX_8, gGraf_specs[gGraf_spec_index].total_width, gGraf_specs[gGraf_spec_index].total_height, NULL, BR_PMAF_NORMAL);
515
 
516
    gScreen->origin_x = 0;
517
    gGfx_initialized = 1;
518
    gScreen->origin_y = 0;
519
    gBack_screen = BrPixelmapMatch(gScreen, BR_PMMATCH_OFFSCREEN);
520
    gBack_screen->origin_x = 0;
521
    gBack_screen->origin_y = 0;
522
    gTemp_screen = BrPixelmapMatch(gScreen, BR_PMMATCH_OFFSCREEN);
523
    gTemp_screen->origin_x = 0;
524
    gTemp_screen->origin_y = 0;
525
    dr_dprintf("PDAllocateScreenAndBack() - END.");
526
 
527
    // dr_dprintf("PDAllocateScreenAndBack() - START...");
528
    // BrMaterialFindHook(PDMissingMaterial);
529
    // BrTableFindHook(PDMissingTable);
530
    // BrModelFindHook(PDMissingModel);
531
    // BrMapFindHook(PDMissingMap);
532
    // dr_dprintf("Setting up DirectDraw stuff...");
533
    // SSDXStart(gWin32_hwnd, 0, 1);
534
    // dr_dprintf("Setting up DirectSound stuff...");
535
    // int result = SSDXStart(gWin32_hwnd, 0, 2);
536
    // int row_bytes;
537
    // if (result == 0) {
538
    //     result = SSDXInitDirectDraw(gGraf_specs[gGraf_spec_index].total_width, gGraf_specs[gGraf_spec_index].total_height, &row_bytes);
539
    // }
540
    // if (result) {
541
    //     SSDXHandleError(result);
542
    //     PDFatalError("Unable to setup DirectDraw - please check that DirectX is installed");
543
    // }
544
    // gGraf_specs[gGraf_spec_index].row_bytes = row_bytes;
545
    // gGraf_specs[gGraf_spec_index].phys_width = gGraf_specs[gGraf_spec_index].row_bytes;
546
    // gGraf_specs[gGraf_spec_index].phys_height = gGraf_specs[gGraf_spec_index].total_height;
547
    // dr_dprintf(
548
    //     "Graf spec info after screen allocation:\n"
549
    //     "                 Total width:   %d\n"
550
    //     "                 Total height:  %d\n"
551
    //     "                 Phys width:    %d\n"
552
    //     "                 Phys height:   %d\n"
553
    //     "                 Row bytes:     %d\n",
554
    //     gGraf_specs[gGraf_spec_index].total_width,
555
    //     gGraf_specs[gGraf_spec_index].total_height,
556
    //     gGraf_specs[gGraf_spec_index].phys_width,
557
    //     gGraf_specs[gGraf_spec_index].phys_height,
558
    //     gGraf_specs[gGraf_spec_index].row_bytes);
559
    // dr_dprintf("First attempt at locking surface...");
560
    // PDLockRealBackScreen();
561
    // dr_dprintf("Done first surface lock; gOffscreen_pixels 0x%p", gOffscreen_pixels);
562
    // gBack_screen = BrPixelmapAllocate(
563
    //     3u,
564
    //     gGraf_specs[gGraf_spec_index].phys_width,
565
    //     gGraf_specs[gGraf_spec_index].phys_height,
566
    //     gOffscreen_pixels,
567
    //     0);
568
    // gScreen = gBack_screen;
569
    // gBack_screen->origin_x = 0;
570
    // gBack_screen->origin_y = 0;
571
    // gBack_screen->base_x = 0;
572
    // gBack_screen->base_y = 0;
573
    // gBack_screen->row_bytes = gGraf_specs[gGraf_spec_index].row_bytes;
574
    // gTemp_screen = (br_pixelmap*)BrPixelmapMatch(gBack_screen, 0);
575
    // gGfx_initialized = 1;
576
    // memset(gBack_screen->pixels, 0, gBack_screen->row_bytes * gBack_screen->height);
577
    // PDScreenBufferSwap(0);
578
    // dr_dprintf("PDAllocateScreenAndBack() - END.");
579
}
580
 
581
void Copy8BitTo16BitPixelmap(br_pixelmap* pDst, br_pixelmap* pSrc, br_pixelmap* pPalette) {
582
    //int x; // Pierre-Marie Baty -- unused variable
583
    //int y; // Pierre-Marie Baty -- unused variable
584
    //tU8* src; // Pierre-Marie Baty -- unused variable
585
    //tU8 value; // Pierre-Marie Baty -- unused variable
586
    //tU8 red; // Pierre-Marie Baty -- unused variable
587
    //tU8 green; // Pierre-Marie Baty -- unused variable
588
    //tU8 blue; // Pierre-Marie Baty -- unused variable
589
    //tU16* dst; // Pierre-Marie Baty -- unused variable
590
    //tU16* palette_entry; // Pierre-Marie Baty -- unused variable
591
    LOG_TRACE("(%p, %p, %p)", pDst, pSrc, pPalette);
592
    NOT_IMPLEMENTED();
593
}
594
 
595
void Double8BitTo16BitPixelmap(br_pixelmap* pDst, br_pixelmap* pSrc, br_pixelmap* pPalette, tU16 pOff, tU16 pSrc_width, tU16 pSrc_height) {
596
    //int x; // Pierre-Marie Baty -- unused variable
597
    //int y; // Pierre-Marie Baty -- unused variable
598
    //tU8* src; // Pierre-Marie Baty -- unused variable
599
    //tU8 value; // Pierre-Marie Baty -- unused variable
600
    //tU8 red; // Pierre-Marie Baty -- unused variable
601
    //tU8 green; // Pierre-Marie Baty -- unused variable
602
    //tU8 blue; // Pierre-Marie Baty -- unused variable
603
    //tU16* dst0; // Pierre-Marie Baty -- unused variable
604
    //tU16* dst1; // Pierre-Marie Baty -- unused variable
605
    //tU16 sixteen; // Pierre-Marie Baty -- unused variable
606
    //tU16* palette_entry; // Pierre-Marie Baty -- unused variable
607
    LOG_TRACE("(%p, %p, %p, %d, %d, %d)", pDst, pSrc, pPalette, pOff, pSrc_width, pSrc_height);
608
    NOT_IMPLEMENTED();
609
}
610
 
611
br_pixelmap* PDInterfacePixelmap(void) {
612
    LOG_TRACE("()");
613
    NOT_IMPLEMENTED();
614
}
615
 
616
void SwapBackScreen(void) {
617
    LOG_TRACE("()");
618
    NOT_IMPLEMENTED();
619
}
620
 
621
void ReallyCopyBackScreen(int pRendering_area_only, int pClear_top_and_bottom) {
622
    LOG_TRACE("(%d, %d)", pRendering_area_only, pClear_top_and_bottom);
623
    NOT_IMPLEMENTED();
624
}
625
 
626
void CopyBackScreen(int pRendering_area_only) {
627
    LOG_TRACE("(%d)", pRendering_area_only);
628
    NOT_IMPLEMENTED();
629
}
630
 
631
void PDScreenBufferSwap(int pRendering_area_only) {
632
    // taken from dossys
633
 
634
    if (pRendering_area_only) {
635
        BrPixelmapRectangleCopy(gScreen, gY_offset, gX_offset, gRender_screen, 0, 0, gWidth, gHeight);
636
    } else {
637
        if (gReal_graf_data_index == gGraf_data_index) {
638
            BrPixelmapDoubleBuffer(gScreen, gBack_screen);
639
        } else {
640
            DRPixelmapDoubledCopy(gTemp_screen, gBack_screen, 320, 200, 0, 40);
641
            BrPixelmapDoubleBuffer(gScreen, gTemp_screen);
642
        }
643
    }
644
}
645
 
646
void PDPixelmapToScreenRectangleCopy(br_pixelmap* dst, br_int_16 dx, br_int_16 dy, br_pixelmap* src, br_int_16 sx, br_int_16 sy, br_uint_16 w, br_uint_16 h) {
647
    LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d)", dst, dx, dy, src, sx, sy, w, h);
648
    NOT_IMPLEMENTED();
649
}
650
 
651
void PDPixelmapHLineOnScreen(br_pixelmap* dst, br_int_16 x1, br_int_16 y1, br_int_16 x2, br_int_16 y2, br_uint_32 colour) {
652
    LOG_TRACE("(%p, %d, %d, %d, %d, %d)", dst, x1, y1, x2, y2, colour);
653
    NOT_IMPLEMENTED();
654
}
655
 
656
void PDPixelmapVLineOnScreen(br_pixelmap* dst, br_int_16 x1, br_int_16 y1, br_int_16 x2, br_int_16 y2, br_uint_32 colour) {
657
    LOG_TRACE("(%p, %d, %d, %d, %d, %d)", dst, x1, y1, x2, y2, colour);
658
    NOT_IMPLEMENTED();
659
}
660
 
661
void PDInstallErrorHandlers(void) {
662
    LOG_TRACE("()");
663
 
664
    gWin32_br_diaghandler.identifier = "LlantisilioBlahBlahBlahOgOgOch";
665
    gWin32_br_diaghandler.warning = Win32BRenderWarningFunc;
666
    gWin32_br_diaghandler.failure = Win32BRenderFailureFunc;
667
    BrDiagHandlerSet(&gWin32_br_diaghandler);
668
}
669
 
670
void PDSetFileVariables(void) {
671
 
672
    // Changed by dethrace for cross-platform
673
    // strcpy(gDir_separator, "\\");
674
    strcpy(gDir_separator, "/");
675
}
676
 
677
void PDBuildAppPath(char* pThe_path) {
678
    GetCurrentDirectoryA_(253, pThe_path);
679
    // GetShortPathNameA(pThe_path, pThe_path, 253);
680
    strcat(pThe_path, "/"); // original: strcat(pThe_path, "\\")
681
    dr_dprintf("Application path '%s'", pThe_path);
682
}
683
 
684
void PDForEveryFile(char* pThe_path, void (*pAction_routine)(char*)) {
685
    char found_path[256];        // [esp+Ch] [ebp-448h] BYREF
686
    WIN32_FIND_DATAA_ find_data; // [esp+10Ch] [ebp-348h] BYREF
687
    HANDLE_ hFindFile;           // [esp+24Ch] [ebp-208h]
688
    char file_filter[256];       // [esp+250h] [ebp-204h] BYREF
689
    char current_dir[260];       // [esp+350h] [ebp-104h] BYREF
690
 
691
    GetCurrentDirectoryA_(sizeof(current_dir), current_dir);
692
    if (SetCurrentDirectoryA_(pThe_path)) {
693
        strcpy(file_filter, "*.???");
694
        hFindFile = FindFirstFileA_(file_filter, &find_data);
695
        if (hFindFile != INVALID_HANDLE_VALUE) {
696
            do {
697
                PathCat(found_path, pThe_path, find_data.cFileName);
698
                pAction_routine(found_path);
699
            } while (FindNextFileA_(hFindFile, &find_data));
700
            FindClose_(hFindFile);
701
        }
702
        SetCurrentDirectoryA_(current_dir);
703
    }
704
}
705
 
706
void PDSetPalette(br_pixelmap* pThe_palette) {
707
    PDSetPaletteEntries(pThe_palette, 0, 256);
708
}
709
 
710
void Win32SetPaletteEntries(uint8_t* entries, int pFirst_colour, int pCount) {
711
    int i;           // [esp+Ch] [ebp-Ch]
712
    int last_colour; // [esp+14h] [ebp-4h]
713
 
714
    last_colour = pFirst_colour + pCount - 1;
715
    if (last_colour > 255) {
716
        last_colour = 255;
717
    }
718
    for (i = pFirst_colour; i <= last_colour; i++) {
719
        gWin32_palette[i].peFlags = 0;
720
        gWin32_palette[i].peRed = entries[i * 4 + 2];
721
        gWin32_palette[i].peGreen = entries[i * 4 + 1];
722
        gWin32_palette[i].peBlue = entries[i * 4];
723
    }
724
    SSDXSetPaleeteEntries(gWin32_palette, 0, 256);
725
}
726
 
727
void PDSetPaletteEntries(br_pixelmap* pPalette, int pFirst_colour, int pCount) {
728
    Win32SetPaletteEntries(pPalette->pixels, pFirst_colour, pCount);
729
}
730
 
731
void PDSwitchToRealResolution(void) {
732
    LOG_TRACE("()");
733
}
734
 
735
void PDSwitchToLoresMode(void) {
736
    LOG_TRACE("()");
737
}
738
 
739
void PDMouseButtons(int* pButton_1, int* pButton_2) {
740
    //br_uint_32 mouse_buttons; // Pierre-Marie Baty -- unused variable
741
    //br_int_32 mouse_x; // Pierre-Marie Baty -- unused variable
742
    //br_int_32 mouse_y; // Pierre-Marie Baty -- unused variable
743
    LOG_TRACE("(%p, %p)", pButton_1, pButton_2);
744
 
745
    // added by dethrace
746
    gHarness_platform.GetMouseButtons(pButton_1, pButton_2);
747
}
748
 
749
void PDGetMousePosition(int* pX_coord, int* pY_coord) {
750
    POINT_ p;
751
    LOG_TRACE("(%p, %p)", pX_coord, pY_coord);
752
 
753
    GetCursorPos_(&p);
754
    ScreenToClient_(gWin32_hwnd, &p);
755
    *pX_coord = p.x;
756
    *pY_coord = p.y;
757
}
758
 
759
int PDGetTotalTime(void) {
760
    return timeGetTime_();
761
}
762
 
763
int PDServiceSystem(tU32 pTime_since_last_call) {
764
    Win32PumpMessages();
765
    return 0;
766
}
767
 
768
void Win32InitInputDevice(void) {
769
    // do directinput stuff
770
    gJoystick_deadzone = 8000;
771
}
772
 
773
void Win32AllocateActionReplayBuffer(void) {
774
    MEMORYSTATUS_ mem_status; // [esp+Ch] [ebp-28h] BYREF
775
    size_t buf_size;          // [esp+2Ch] [ebp-8h]
776
    void* buf;                // [esp+30h] [ebp-4h]
777
 
778
    buf = 0;
779
    if (gWin32_action_replay_buffer_allocated) {
780
        return;
781
    }
782
    gWin32_action_replay_buffer_allocated = 1;
783
    mem_status.dwLength = sizeof(mem_status);
784
    GlobalMemoryStatus_(&mem_status);
785
    dr_dprintf(
786
        "Win32AllocateActionReplayBuffer(): Memory Status BEFORE Action Replay Allocation:\n"
787
        "             dwLength        %u\n"
788
        "             dwMemoryLoad    %u\n"
789
        "             dwTotalPhys     %u\n"
790
        "             dwAvailPhys     %u\n"
791
        "             dwTotalPageFile %u\n"
792
        "             dwAvailPageFile %u\n"
793
        "             dwTotalVirtual  %u\n"
794
        "             dwAvailVirtual  %u",
795
        mem_status.dwLength,
796
        mem_status.dwMemoryLoad,
797
        mem_status.dwTotalPhys,
798
        mem_status.dwAvailPhys,
799
        mem_status.dwTotalPageFile,
800
        mem_status.dwAvailPageFile,
801
        mem_status.dwTotalVirtual,
802
        mem_status.dwAvailVirtual);
803
 
804
    buf_size = 20000000;
805
 
806
    if (mem_status.dwTotalPhys < 16000000) {
807
        buf_size = 500000;
808
    }
809
    if (mem_status.dwTotalPhys < 24000000) {
810
        buf_size = 4000000;
811
    }
812
    if (mem_status.dwTotalPhys < 32000000) {
813
        buf_size = 6000000;
814
    }
815
    if (mem_status.dwTotalPhys < 48000000) {
816
        buf_size = 12000000;
817
    }
818
 
819
    dr_dprintf("Win32AllocateActionReplayBuffer(): We want %d bytes...", buf_size);
820
    if (mem_status.dwAvailPhys + mem_status.dwAvailPageFile < buf_size) {
821
        buf_size = mem_status.dwAvailPhys + mem_status.dwAvailPageFile - 1048576;
822
        dr_dprintf("Win32AllocateActionReplayBuffer(): ...but there's only %d bytes available...", buf_size);
823
    }
824
    if (buf_size < 65536) {
825
        buf_size = 65536;
826
        dr_dprintf("Win32AllocateActionReplayBuffer(): ...but we have to have a minimum size of %d bytes...", 65536);
827
    }
828
    while (buf_size >= 65536) {
829
        buf = malloc(buf_size);
830
        if (buf) {
831
            break;
832
        }
833
        buf_size -= 65536;
834
    }
835
    gWin32_action_replay_buffer = buf;
836
    if (buf) {
837
        gWin32_action_replay_buffer_size = buf_size;
838
        Sleep_(1000u);
839
    } else {
840
        gWin32_action_replay_buffer_size = 0;
841
    }
842
    dr_dprintf("Win32AllocateActionReplayBuffer(): Actually allocated %d bytes.", buf_size);
843
    GlobalMemoryStatus_(&mem_status);
844
    dr_dprintf(
845
        "Win32AllocateActionReplayBuffer(): Memory Status AFTER Action Replay Allocation:\n"
846
        "             dwLength        %u\n"
847
        "             dwMemoryLoad    %u\n"
848
        "             dwTotalPhys     %u\n"
849
        "             dwAvailPhys     %u\n"
850
        "             dwTotalPageFile %u\n"
851
        "             dwAvailPageFile %u\n"
852
        "             dwTotalVirtual  %u\n"
853
        "             dwAvailVirtual  %u",
854
        mem_status.dwLength,
855
        mem_status.dwMemoryLoad,
856
        mem_status.dwTotalPhys,
857
        mem_status.dwAvailPhys,
858
        mem_status.dwTotalPageFile,
859
        mem_status.dwAvailPageFile,
860
        mem_status.dwTotalVirtual,
861
        mem_status.dwAvailVirtual);
862
}
863
 
864
void PDAllocateActionReplayBuffer(char** pBuffer, tU32* pBuffer_size) {
865
    LOG_TRACE("(%p, %p)", pBuffer, pBuffer_size);
866
 
867
    Win32AllocateActionReplayBuffer();
868
    *pBuffer = gWin32_action_replay_buffer;
869
    *pBuffer_size = gWin32_action_replay_buffer_size;
870
}
871
 
872
void PDDisposeActionReplayBuffer(char* pBuffer) {
873
    LOG_TRACE("(\"%s\")", pBuffer);
874
}
875
 
876
// this function is taken from dossys
877
void Usage(char* pProgpath) {
878
    char base_name[256]; // fix: changed from 9 to avoid overflow on longer filenames
879
 
880
    _splitpath_(pProgpath, NULL, NULL, base_name, NULL);
881
 
882
    fprintf(stderr,
883
        "Usage: %s [%s] [%s YonFactor] [%s CarSimplificationLevel] [%s SoundDetailLevel] [%s] [%s] [%s] [%s] [%s] [%s]\nWhere YonFactor is between 0 and 1,\nCarSimplificationLevel is a whole number between 0 and %d,\nand SoundDetailLevel is a whole number.\n",
884
        base_name,
885
        "-hires",
886
        "-yon",
887
        "-simple",
888
        "-sound",
889
        "-robots",
890
        "-lomem",
891
        "-nosound",
892
        "-spamfritter",
893
        "-nocutscenes",
894
        "-noreplay",
895
        CAR_MAX_SIMPLIFICATION_LEVEL);
896
    exit(1);
897
}
898
 
899
// Renamed from "main" to "original_main" to allow for harness + unit testing
900
int original_main(int pArgc, char** pArgv) {
901
    int arg;
902
    int i;
903
    float f;
904
 
905
    for (i = 1; i < pArgc; i++) {
906
        if (strcasecmp(pArgv[i], "-hires") == 0) {
907
            gGraf_spec_index = 1;
908
        } else if (strcasecmp(pArgv[i], "-yon") == 0 && i < pArgc - 1) {
909
            i++;
910
            sscanf(pArgv[i], "%f", &f);
911
            if (f >= 0.0 && f <= 1.0f) {
912
                gYon_multiplier = f;
913
            }
914
        } else if (strcasecmp(pArgv[i], "-simple") == 0 && i < pArgc - 1) {
915
            i++;
916
            sscanf(pArgv[i], "%d", &arg);
917
            if (arg >= 0 && arg < 5) {
918
                gCar_simplification_level = arg;
919
            }
920
        } else if (strcasecmp(pArgv[i], "-sound") == 0 && i < pArgc - 1) {
921
            i++;
922
            sscanf(pArgv[i], "%d", &arg);
923
            gSound_detail_level = arg;
924
 
925
        } else if (strcasecmp(pArgv[i], "-robots") == 0 || strcasecmp(pArgv[i], "-german") == 0) {
926
            gSausage_override = 1;
927
        } else if (strcasecmp(pArgv[i], "-lomem") == 0) {
928
            gAustere_override = 1;
929
        } else if (strcasecmp(pArgv[i], "-nosound") == 0) {
930
            gSound_override = 1;
931
        } else if (strcasecmp(pArgv[i], "-spamfritter") == 0) {
932
            gExtra_mem = 2000000;
933
        } else if (strcasecmp(pArgv[i], "-nocutscenes") == 0) {
934
            gCut_scene_override = 1;
935
        } else if (strcasecmp(pArgv[i], "-noreplay") == 0) {
936
            gReplay_override = 1;
937
        } else {
938
            Usage(pArgv[0]);
939
        }
940
    }
941
 
942
    gNetwork_profile_fname[0] = 0;
943
    uint32_t len = GetCurrentDirectoryA_(240, gNetwork_profile_fname);
944
    if (len > 0 && len == strlen(gNetwork_profile_fname)) {
945
        gNetwork_profile_file_exists = 1;
946
        strcat(gNetwork_profile_fname, "/");
947
        strcat(gNetwork_profile_fname, "NETWORK.INI");
948
    }
949
 
950
    GameMain(pArgc, pArgv);
951
    return 0;
952
}
953
 
954
int OurGetChar(void) {
955
    //int key; // Pierre-Marie Baty -- unused variable
956
    LOG_TRACE("()");
957
    NOT_IMPLEMENTED();
958
}
959
 
960
void PDEnterDebugger(char* pStr) {
961
    //static unsigned char* save_it; // Pierre-Marie Baty -- unused variable
962
    LOG_TRACE("(\"%s\")", pStr);
963
 
964
    dr_dprintf("PDEnterDebugger(): %s", pStr);
965
    ShowCursor_(1);
966
#ifdef DETHRACE_FIX_BUGS
967
    if (strcmp(pStr, "Bet you weren't expecting this") != 0
968
#else
969
    if (pStr != "Bet you weren't expecting this"
970
#endif
971
        && _CrtDbgReport_(_CRT_ASSERT, "C:\\Msdev\\Projects\\DethRace\\Win95sys.c", 437, 0, 0) == 1) {
972
 
973
        abort(); // original: __debugbreak();
974
    }
975
    ShowCursor_(0);
976
}
977
 
978
// Added function
979
br_material* PDMissingMaterial(char* name) {
980
    LOG_TRACE("(\"%s\")", name);
981
 
982
    // FIXME: call functiont in harness
983
    dr_dprintf("SOMETHING MISSING WARNING - Can't find material '%s'", name);
984
    return NULL;
985
}
986
 
987
// Added function
988
br_pixelmap* PDMissingTable(char* name) {
989
    LOG_TRACE("(\"%s\")", name);
990
 
991
    // FIXME: call function in harness
992
    dr_dprintf("SOMETHING MISSING WARNING - Can't find shade table '%s'", name);
993
    return NULL;
994
}
995
 
996
// Added function
997
br_model* PDMissingModel(char* name) {
998
    LOG_TRACE("(\"%s\")", name);
999
 
1000
    // FIXME: call function in harness
1001
    dr_dprintf("SOMETHING MISSING WARNING - Can't find model '%s'", name);
1002
    return NULL;
1003
}
1004
 
1005
// Added function
1006
br_pixelmap* PDMissingMap(char* name) {
1007
    LOG_TRACE("(\"%s\")", name);
1008
 
1009
    // FIXME: call function in harness
1010
    dr_dprintf("SOMETHING MISSING WARNING - Can't find pixelmap '%s'", name);
1011
    return NULL;
1012
}
1013
 
1014
void PDEndItAllAndReRunTheBastard(void) {
1015
    LOG_TRACE("()");
1016
    NOT_IMPLEMENTED();
1017
}
1018
 
1019
int InitJoysticks(void) {
1020
    LOG_TRACE("()");
1021
    NOT_IMPLEMENTED();
1022
}
1023
 
1024
tU32 ReadJoystickAxis(int pBit) {
1025
    //tU32 val; // Pierre-Marie Baty -- unused variable
1026
    //tU32 count; // Pierre-Marie Baty -- unused variable
1027
    LOG_TRACE("(%d)", pBit);
1028
    NOT_IMPLEMENTED();
1029
}
1030
 
1031
void PDReadJoySticks(void) {
1032
    //tU32 temp1x; // Pierre-Marie Baty -- unused variable
1033
    //tU32 temp1y; // Pierre-Marie Baty -- unused variable
1034
    //tU32 temp2x; // Pierre-Marie Baty -- unused variable
1035
    //tU32 temp2y; // Pierre-Marie Baty -- unused variable
1036
    LOG_TRACE("()");
1037
    NOT_IMPLEMENTED();
1038
}
1039
 
1040
tS32 PDGetJoy1X(void) {
1041
    //tS32 joy; // Pierre-Marie Baty -- unused variable
1042
    LOG_TRACE("()");
1043
    NOT_IMPLEMENTED();
1044
}
1045
 
1046
tS32 PDGetJoy1Y(void) {
1047
    //tS32 joy; // Pierre-Marie Baty -- unused variable
1048
    LOG_TRACE("()");
1049
    NOT_IMPLEMENTED();
1050
}
1051
 
1052
tS32 PDGetJoy2X(void) {
1053
    //tS32 joy; // Pierre-Marie Baty -- unused variable
1054
    LOG_TRACE("()");
1055
    NOT_IMPLEMENTED();
1056
}
1057
 
1058
tS32 PDGetJoy2Y(void) {
1059
    //tS32 joy; // Pierre-Marie Baty -- unused variable
1060
    LOG_TRACE("()");
1061
    NOT_IMPLEMENTED();
1062
}
1063
 
1064
int PDGetJoy1Button1(void) {
1065
    LOG_TRACE("()");
1066
    NOT_IMPLEMENTED();
1067
}
1068
 
1069
int PDGetJoy1Button2(void) {
1070
    LOG_TRACE("()");
1071
    NOT_IMPLEMENTED();
1072
}
1073
 
1074
int PDGetJoy1Button3(void) {
1075
    LOG_TRACE("()");
1076
    NOT_IMPLEMENTED();
1077
}
1078
 
1079
int PDGetJoy1Button4(void) {
1080
    LOG_TRACE("()");
1081
    NOT_IMPLEMENTED();
1082
}
1083
 
1084
int PDGetJoy2Button1(void) {
1085
    LOG_TRACE("()");
1086
    NOT_IMPLEMENTED();
1087
}
1088
 
1089
int PDGetJoy2Button2(void) {
1090
    LOG_TRACE("()");
1091
    NOT_IMPLEMENTED();
1092
}
1093
 
1094
int PDGetJoy2Button3(void) {
1095
    LOG_TRACE("()");
1096
    NOT_IMPLEMENTED();
1097
}
1098
 
1099
int PDGetJoy2Button4(void) {
1100
    LOG_TRACE("()");
1101
    NOT_IMPLEMENTED();
1102
}
1103
 
1104
int PDFileUnlock(char* pThe_path) {
1105
    uint32_t dwFileAttributes; // [esp+Ch] [ebp-4h]
1106
    LOG_TRACE("(\"%s\")", pThe_path);
1107
 
1108
    dwFileAttributes = GetFileAttributesA_(pThe_path);
1109
    return dwFileAttributes != INVALID_FILE_ATTRIBUTES && SetFileAttributesA_(pThe_path, dwFileAttributes & ~FILE_ATTRIBUTE_READONLY);
1110
}
1111
 
1112
int PDCheckDriveExists2(char* pThe_path, char* pFile_name, tU32 pMin_size) {
1113
 
1114
    char the_path[256]; // [esp+Ch] [ebp-108h] BYREF
1115
    tU32 file_size;     // [esp+10Ch] [ebp-8h]
1116
    HANDLE_ hFile;      // [esp+110h] [ebp-4h]
1117
    LOG_TRACE("(\"%s\", \"%s\", %d)", pThe_path, pFile_name, pMin_size);
1118
 
1119
    file_size = 0;
1120
    if (pFile_name) {
1121
        PathCat(the_path, pThe_path, pFile_name);
1122
    } else {
1123
        strcpy(the_path, pThe_path);
1124
    }
1125
    if (the_path[0] && the_path[1] == ':' && !the_path[2]) {
1126
        strcat(the_path, gDir_separator);
1127
    }
1128
    if (GetFileAttributesA_(pThe_path) == INVALID_FILE_ATTRIBUTES) {
1129
        return 0;
1130
    }
1131
    hFile = CreateFileA_(the_path, GENERIC_READ, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
1132
    if (hFile != INVALID_HANDLE_VALUE) {
1133
        file_size = GetFileSize_(hFile, 0);
1134
        CloseHandle_(hFile);
1135
    }
1136
    return file_size >= pMin_size;
1137
}
1138
 
1139
int PDDoWeLeadAnAustereExistance(void) {
1140
    return 0;
1141
}
1142
 
1143
// Windows-specific functions. The only name we know for sure is "Win32AllocateReplayBuffer".
1144
 
1145
void Win32BRenderWarningFunc(char* msg) {
1146
    dr_dprintf("*******************************************************************************");
1147
    dr_dprintf("BRender WARNING: '%s'", msg);
1148
    dr_dprintf("*******************************************************************************");
1149
}
1150
 
1151
void Win32BRenderFailureFunc(char* msg) {
1152
    dr_dprintf("*******************************************************************************");
1153
    dr_dprintf("BRender FAILURE: '%s'", msg);
1154
    dr_dprintf("*******************************************************************************");
1155
    Win32FatalError("BRender error detected:", msg);
1156
}