1920 lines
57 KiB
C
1920 lines
57 KiB
C
#include "utility.h"
|
|
#include <stdlib.h>
|
|
|
|
#include "brender.h"
|
|
#include "constants.h"
|
|
#include "errors.h"
|
|
#include "globvars.h"
|
|
#include "globvrbm.h"
|
|
#include "globvrpb.h"
|
|
#include "graphics.h"
|
|
#include "harness/config.h"
|
|
#include "harness/hooks.h"
|
|
#include "harness/trace.h"
|
|
#include "input.h"
|
|
#include "loading.h"
|
|
#include "loadsave.h"
|
|
#include "main.h"
|
|
#include "mainmenu.h"
|
|
#include "network.h"
|
|
#include "pd/sys.h"
|
|
#include "sound.h"
|
|
#include "world.h"
|
|
|
|
#include <ctype.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
// Added >>
|
|
#define MIN_SERVICE_INTERVAL 200
|
|
// <<
|
|
|
|
int gIn_check_quit = 0;
|
|
tU32 gLost_time = 0;
|
|
#if BR_ENDIAN_BIG
|
|
tU32 gLong_key[4] = { 0x6c1b995f, 0xb9cd5f13, 0xcb04200e, 0x5e1ca10e };
|
|
tU32 gOther_long_key[4] = { 0x67a8d626, 0xb6dd451b, 0x327e2213, 0x15c29437 };
|
|
#else
|
|
tU32 gLong_key[4] = { 0x5f991b6c, 0x135fcdb9, 0x0e2004cb, 0x0ea11c5e };
|
|
tU32 gOther_long_key[4] = { 0x26d6a867, 0x1b45ddb6, 0x13227e32, 0x3794c215 };
|
|
#endif
|
|
int gEncryption_method = 0;
|
|
char* gMisc_strings[250];
|
|
br_pixelmap* g16bit_palette;
|
|
br_pixelmap* gSource_for_16bit_palette;
|
|
|
|
// IDA: int __cdecl CheckQuit()
|
|
int CheckQuit(void) {
|
|
LOG_TRACE8("()");
|
|
|
|
if (gIn_check_quit) {
|
|
return 0;
|
|
}
|
|
if (!KeyIsDown(KEYMAP_CTRL_QUIT) || !KeyIsDown(KEYMAP_CONTROL_ANY)) {
|
|
return 0;
|
|
}
|
|
gIn_check_quit = 1;
|
|
while (AnyKeyDown()) {
|
|
;
|
|
}
|
|
|
|
if (DoVerifyQuit(1)) {
|
|
QuitGame();
|
|
}
|
|
gIn_check_quit = 0;
|
|
return 1;
|
|
}
|
|
|
|
// IDA: double __cdecl sqr(double pN)
|
|
double sqr(double pN) {
|
|
|
|
return pN * pN;
|
|
}
|
|
|
|
// Added to handle demo-specific text file decryption behavior
|
|
void EncodeLine_DEMO(char* pS) {
|
|
int len;
|
|
int seed;
|
|
int i;
|
|
char* key;
|
|
unsigned char c;
|
|
FILE* test;
|
|
tPath_name the_path;
|
|
#if BR_ENDIAN_BIG
|
|
const tU32 gLong_key_DEMO[] = { 0x58503A76, 0xCBB68565, 0x15CD5B07, 0xB168DE3A };
|
|
#else
|
|
const tU32 gLong_key_DEMO[] = { 0x763A5058, 0x6585B6CB, 0x75BCD15, 0x3ADE68B1 };
|
|
#endif
|
|
|
|
len = strlen(pS);
|
|
key = (char*)gLong_key_DEMO;
|
|
|
|
while (len > 0 && (pS[len - 1] == '\r' || pS[len - 1] == '\n')) {
|
|
len--;
|
|
pS[len] = 0;
|
|
}
|
|
seed = len % 16;
|
|
for (i = 0; i < len; i++) {
|
|
c = pS[i];
|
|
if (c == '\t') {
|
|
c = 0x9F;
|
|
}
|
|
c = ((key[seed] ^ (c - 32)) & 0x7F) + 32;
|
|
seed = (seed + 7) % 16;
|
|
if (c == 0x9F) {
|
|
c = '\t';
|
|
}
|
|
pS[i] = c;
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall EncodeLine(char *pS@<EAX>)
|
|
void EncodeLine(char* pS) {
|
|
int len;
|
|
int seed;
|
|
int i;
|
|
char* key;
|
|
unsigned char c;
|
|
FILE* test;
|
|
tPath_name the_path;
|
|
char s[256];
|
|
|
|
// Demo has its own decryption key + behavior
|
|
if (harness_game_info.mode == eGame_carmageddon_demo) {
|
|
EncodeLine_DEMO(pS);
|
|
return;
|
|
}
|
|
|
|
len = strlen(pS);
|
|
key = (char*)gLong_key;
|
|
if (gEncryption_method == 0) {
|
|
PathCat(the_path, gApplication_path, "GENERAL.TXT");
|
|
|
|
test = fopen(the_path, "rt");
|
|
if (test != NULL) {
|
|
fgets(s, 256, test);
|
|
if (s[0] != '@') {
|
|
gEncryption_method = 2;
|
|
} else {
|
|
gEncryption_method = 1;
|
|
EncodeLine(&s[1]);
|
|
s[7] = '\0';
|
|
if (strcmp(&s[1], "0.01\t\t") != 0) {
|
|
gEncryption_method = 2;
|
|
}
|
|
}
|
|
fclose(test);
|
|
} else {
|
|
gEncryption_method = 2;
|
|
}
|
|
}
|
|
while (len > 0 && (pS[len - 1] == '\r' || pS[len - 1] == '\n')) {
|
|
len--;
|
|
pS[len] = '\0';
|
|
}
|
|
|
|
seed = len % 16;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
c = pS[i];
|
|
#if defined(DETHRACE_FIX_BUGS)
|
|
// When loading game data, Carmageddon does not switch the XOR cypher when the comments start.
|
|
if (i >= 2) {
|
|
if (pS[i - 1] == '/' && pS[i - 2] == '/') {
|
|
key = (char*)gOther_long_key;
|
|
}
|
|
}
|
|
#endif
|
|
if (gEncryption_method == 1) {
|
|
if (c == '\t') {
|
|
c = 0x9f;
|
|
}
|
|
|
|
c -= 0x20;
|
|
c ^= key[seed];
|
|
c &= 0x7f;
|
|
c += 0x20;
|
|
|
|
seed += 7;
|
|
seed %= 16;
|
|
|
|
if (c == 0x9f) {
|
|
c = '\t';
|
|
}
|
|
} else {
|
|
if (c == '\t') {
|
|
c = 0x80;
|
|
}
|
|
|
|
c -= 0x20;
|
|
if ((c & 0x80) == 0) {
|
|
c ^= key[seed] & 0x7f;
|
|
}
|
|
c += 0x20;
|
|
|
|
seed += 7;
|
|
seed %= 16;
|
|
|
|
if (c == 0x80) {
|
|
c = '\t';
|
|
}
|
|
}
|
|
pS[i] = c;
|
|
}
|
|
}
|
|
|
|
// IDA: int __usercall IRandomBetween@<EAX>(int pA@<EAX>, int pB@<EDX>)
|
|
int IRandomBetween(int pA, int pB) {
|
|
int num;
|
|
char s[32];
|
|
|
|
num = rand();
|
|
#if RAND_MAX == 0x7fff
|
|
// If RAND_MAX == 0x7fff, then `num` can be seen as a fixed point number with 15 fractional and 17 integral bits
|
|
return pA + ((num * (pB + 1 - pA)) >> 15);
|
|
#else
|
|
// If RAND_MAX != 0x7fff, then use floating numbers (alternative is using modulo)
|
|
return pA + (int)((pB + 1 - pA) * (num / ((float)RAND_MAX + 1)));
|
|
#endif
|
|
}
|
|
|
|
// IDA: int __usercall PercentageChance@<EAX>(int pC@<EAX>)
|
|
int PercentageChance(int pC) {
|
|
LOG_TRACE("(%d)", pC);
|
|
|
|
return IRandomBetween(0, 99) < pC;
|
|
}
|
|
|
|
// IDA: int __usercall IRandomPosNeg@<EAX>(int pN@<EAX>)
|
|
int IRandomPosNeg(int pN) {
|
|
LOG_TRACE("(%d)", pN);
|
|
|
|
return IRandomBetween(-pN, pN);
|
|
}
|
|
|
|
// IDA: float __cdecl FRandomBetween(float pA, float pB)
|
|
float FRandomBetween(float pA, float pB) {
|
|
LOG_TRACE8("(%f, %f)", pA, pB);
|
|
return (double)rand() * (pB - pA) / (double)RAND_MAX + pA;
|
|
}
|
|
|
|
// IDA: float __cdecl FRandomPosNeg(float pN)
|
|
float FRandomPosNeg(float pN) {
|
|
LOG_TRACE("(%f)", pN);
|
|
|
|
return FRandomBetween(-pN, pN);
|
|
}
|
|
|
|
// IDA: br_scalar __cdecl SRandomBetween(br_scalar pA, br_scalar pB)
|
|
br_scalar SRandomBetween(br_scalar pA, br_scalar pB) {
|
|
LOG_TRACE8("(%f, %f)", pA, pB);
|
|
|
|
return FRandomBetween(pA, pB);
|
|
}
|
|
|
|
// IDA: br_scalar __cdecl SRandomPosNeg(br_scalar pN)
|
|
br_scalar SRandomPosNeg(br_scalar pN) {
|
|
LOG_TRACE("(%f)", pN);
|
|
|
|
return SRandomBetween(-pN, pN);
|
|
}
|
|
|
|
// IDA: char* __usercall GetALineWithNoPossibleService@<EAX>(FILE *pF@<EAX>, unsigned char *pS@<EDX>)
|
|
char* GetALineWithNoPossibleService(FILE* pF, unsigned char* pS) {
|
|
// Jeff removed "signed' to avoid compiler warnings..
|
|
/*signed*/ char* result;
|
|
/*signed*/ char s[256];
|
|
int ch;
|
|
int len;
|
|
int i;
|
|
|
|
do {
|
|
result = fgets(s, 256, pF);
|
|
if (result == NULL) {
|
|
s[0] = '\0';
|
|
break;
|
|
}
|
|
if (s[0] == '@') {
|
|
EncodeLine(&s[1]);
|
|
len = strlen(s);
|
|
memmove(s, &s[1], len);
|
|
} else {
|
|
while (s[0] == ' ' || s[0] == '\t') {
|
|
len = strlen(s);
|
|
memmove(s, &s[1], len);
|
|
}
|
|
}
|
|
|
|
while (1) {
|
|
ch = fgetc(pF);
|
|
if (ch != '\r' && ch != '\n') {
|
|
break;
|
|
}
|
|
}
|
|
if (ch != -1) {
|
|
ungetc(ch, pF);
|
|
}
|
|
} while (!Harness_Hook_isalnum(s[0])
|
|
&& s[0] != '-'
|
|
&& s[0] != '.'
|
|
&& s[0] != '!'
|
|
&& s[0] != '&'
|
|
&& s[0] != '('
|
|
&& s[0] != '\''
|
|
&& s[0] != '\"'
|
|
&& s[0] >= 0);
|
|
|
|
if (result) {
|
|
len = strlen(result);
|
|
if (len != 0 && (result[len - 1] == '\r' || result[len - 1] == '\n')) {
|
|
result[len - 1] = 0;
|
|
}
|
|
if (len != 1 && (result[len - 2] == '\r' || result[len - 2] == '\n')) {
|
|
result[len - 2] = 0;
|
|
}
|
|
}
|
|
strcpy((char*)pS, s);
|
|
len = strlen(s);
|
|
for (i = 0; i < len; i++) {
|
|
if (pS[i] >= 0xe0) {
|
|
pS[i] -= 32;
|
|
}
|
|
}
|
|
// LOG_DEBUG("%s", result);
|
|
return result;
|
|
}
|
|
|
|
// IDA: char* __usercall GetALineAndDontArgue@<EAX>(FILE *pF@<EAX>, char *pS@<EDX>)
|
|
char* GetALineAndDontArgue(FILE* pF, char* pS) {
|
|
// LOG_TRACE10("(%p, \"%s\")", pF, pS);
|
|
|
|
PossibleService();
|
|
return GetALineWithNoPossibleService(pF, (unsigned char*)pS);
|
|
}
|
|
|
|
// IDA: void __usercall PathCat(char *pDestn_str@<EAX>, char *pStr_1@<EDX>, char *pStr_2@<EBX>)
|
|
void PathCat(char* pDestn_str, char* pStr_1, char* pStr_2) {
|
|
|
|
if (pDestn_str != pStr_1) { // Added to avoid strcpy overlap checks
|
|
strcpy(pDestn_str, pStr_1);
|
|
}
|
|
if (strlen(pStr_2) != 0) {
|
|
strcat(pDestn_str, gDir_separator);
|
|
strcat(pDestn_str, pStr_2);
|
|
}
|
|
}
|
|
|
|
// IDA: int __cdecl Chance(float pChance_per_second, int pPeriod)
|
|
int Chance(float pChance_per_second, int pPeriod) {
|
|
LOG_TRACE("(%f, %d)", pChance_per_second, pPeriod);
|
|
|
|
return FRandomBetween(0.f, 1.f) < (pPeriod * pChance_per_second / 1000.f);
|
|
}
|
|
|
|
// IDA: float __cdecl tandeg(float pAngle)
|
|
float tandeg(float pAngle) {
|
|
LOG_TRACE("(%f)", pAngle);
|
|
|
|
pAngle = DEG_TO_RAD(pAngle);
|
|
return sinf(pAngle) / cosf(pAngle);
|
|
}
|
|
|
|
// IDA: tU32 __usercall GetFileLength@<EAX>(FILE *pF@<EAX>)
|
|
tU32 GetFileLength(FILE* pF) {
|
|
tU32 the_size;
|
|
|
|
fseek(pF, 0, SEEK_END);
|
|
the_size = ftell(pF);
|
|
rewind(pF);
|
|
return the_size;
|
|
}
|
|
|
|
// IDA: int __usercall BooleanTo1Or0@<EAX>(int pB@<EAX>)
|
|
int BooleanTo1Or0(int pB) {
|
|
LOG_TRACE("(%d)", pB);
|
|
|
|
return pB != 0;
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall DRPixelmapAllocate@<EAX>(br_uint_8 pType@<EAX>, br_uint_16 pW@<EDX>, br_uint_16 pH@<EBX>, void *pPixels@<ECX>, int pFlags)
|
|
br_pixelmap* DRPixelmapAllocate(br_uint_8 pType, br_uint_16 pW, br_uint_16 pH, void* pPixels, int pFlags) {
|
|
br_pixelmap* the_map;
|
|
|
|
the_map = BrPixelmapAllocate(pType, pW, pH, pPixels, pFlags);
|
|
if (the_map != NULL) {
|
|
the_map->origin_y = 0;
|
|
the_map->origin_x = 0;
|
|
}
|
|
return the_map;
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall DRPixelmapAllocateSub@<EAX>(br_pixelmap *pPm@<EAX>, br_uint_16 pX@<EDX>, br_uint_16 pY@<EBX>, br_uint_16 pW@<ECX>, br_uint_16 pH)
|
|
br_pixelmap* DRPixelmapAllocateSub(br_pixelmap* pPm, br_uint_16 pX, br_uint_16 pY, br_uint_16 pW, br_uint_16 pH) {
|
|
br_pixelmap* the_map;
|
|
LOG_TRACE("(%p, %d, %d, %d, %d)", pPm, pX, pY, pW, pH);
|
|
|
|
the_map = BrPixelmapAllocateSub(pPm, pX, pY, pW, pH);
|
|
if (the_map != NULL) {
|
|
the_map->origin_y = 0;
|
|
the_map->origin_x = 0;
|
|
}
|
|
return the_map;
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall DRPixelmapMatchSized@<EAX>(br_pixelmap *pSrc@<EAX>, tU8 pMatch_type@<EDX>, tS32 pWidth@<EBX>, tS32 pHeight@<ECX>)
|
|
br_pixelmap* DRPixelmapMatchSized(br_pixelmap* pSrc, tU8 pMatch_type, tS32 pWidth, tS32 pHeight) {
|
|
br_pixelmap* result;
|
|
LOG_TRACE("(%p, %d, %d, %d)", pSrc, pMatch_type, pWidth, pHeight);
|
|
NOT_IMPLEMENTED();
|
|
}
|
|
|
|
// IDA: void __usercall CopyDoubled8BitTo16BitRectangle(br_pixelmap *pDst@<EAX>, br_pixelmap *pSrc@<EDX>, int pSrc_width@<EBX>, int pSrc_height@<ECX>, int pDst_x, int pDst_y, br_pixelmap *pPalette)
|
|
void CopyDoubled8BitTo16BitRectangle(br_pixelmap* pDst, br_pixelmap* pSrc, int pSrc_width, int pSrc_height, int pDst_x, int pDst_y, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start0;
|
|
tU16* dst_start1;
|
|
tU16* palette_entry;
|
|
LOG_TRACE("(%p, %p, %d, %d, %d, %d, %p)", pDst, pSrc, pSrc_width, pSrc_height, pDst_x, pDst_y, pPalette);
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
|
|
for (y = 0; y < pSrc_height; y++) {
|
|
src_start = ((tU8*)pSrc->pixels) + pSrc->row_bytes * y;
|
|
dst_start0 = (tU16*)(((tU8*)pDst->pixels) + pDst->row_bytes * pDst_y);
|
|
dst_start0 += pDst_x;
|
|
dst_start1 = (tU16*)(((tU8*)pDst->pixels) + pDst->row_bytes * (pDst_y + 1));
|
|
dst_start1 += pDst_x;
|
|
|
|
for (x = 0; x < pSrc_width; x++) {
|
|
dst_start0[0] = palette_entry[*src_start];
|
|
dst_start0[1] = palette_entry[*src_start];
|
|
dst_start1[0] = palette_entry[*src_start];
|
|
dst_start1[1] = palette_entry[*src_start];
|
|
src_start++;
|
|
dst_start0 += 2;
|
|
dst_start1 += 2;
|
|
}
|
|
pDst_y += 2;
|
|
}
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall Scale8BitPixelmap@<EAX>(br_pixelmap *pSrc@<EAX>, int pWidth@<EDX>, int pHeight@<EBX>)
|
|
br_pixelmap* Scale8BitPixelmap(br_pixelmap* pSrc, int pWidth, int pHeight) {
|
|
br_pixelmap* result;
|
|
int x;
|
|
int y;
|
|
tU8* src_pixels;
|
|
tU8* dst_pixels;
|
|
LOG_TRACE("(%p, %d, %d)", pSrc, pWidth, pHeight);
|
|
NOT_IMPLEMENTED();
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall Tile8BitPixelmap@<EAX>(br_pixelmap *pSrc@<EAX>, int pN@<EDX>)
|
|
br_pixelmap* Tile8BitPixelmap(br_pixelmap* pSrc, int pN) {
|
|
br_pixelmap* result;
|
|
int new_width;
|
|
int new_height;
|
|
int x;
|
|
int y2;
|
|
int y;
|
|
tU8* src_pixels;
|
|
tU8* dst_pixels;
|
|
LOG_TRACE("(%p, %d)", pSrc, pN);
|
|
NOT_IMPLEMENTED();
|
|
}
|
|
|
|
// IDA: tException_list __usercall FindExceptionInList@<EAX>(char *pName@<EAX>, tException_list pList@<EDX>)
|
|
tException_list FindExceptionInList(char* pName, tException_list pList) {
|
|
LOG_TRACE("(\"%s\", %d)", pName, pList);
|
|
|
|
while (pList) {
|
|
if (DRStricmp(pName, pList->name) == 0) {
|
|
return pList;
|
|
}
|
|
pList = pList->next;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall PurifiedPixelmap@<EAX>(br_pixelmap *pSrc@<EAX>)
|
|
br_pixelmap* PurifiedPixelmap(br_pixelmap* pSrc) {
|
|
br_pixelmap* intermediate;
|
|
br_pixelmap* result;
|
|
int new_width;
|
|
int new_height;
|
|
tException_list e;
|
|
LOG_TRACE("(%p)", pSrc);
|
|
|
|
// dethrace: added conditional to allow both software and 3dfx modes
|
|
if (!harness_game_config.opengl_3dfx_mode) {
|
|
return pSrc;
|
|
}
|
|
|
|
LOG_INFO("PurifiedPixelmap not implemented");
|
|
return pSrc;
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall DRPixelmapLoad@<EAX>(char *pFile_name@<EAX>)
|
|
br_pixelmap* DRPixelmapLoad(char* pFile_name) {
|
|
br_pixelmap* the_map;
|
|
br_int_8 lobyte;
|
|
LOG_TRACE("(\"%s\")", pFile_name);
|
|
|
|
the_map = BrPixelmapLoad(pFile_name);
|
|
if (the_map != NULL) {
|
|
the_map->origin_x = 0;
|
|
the_map->origin_y = 0;
|
|
the_map->row_bytes = (the_map->row_bytes + sizeof(int32_t) - 1) & ~(sizeof(int32_t) - 1);
|
|
}
|
|
return the_map;
|
|
}
|
|
|
|
// IDA: br_uint_32 __usercall DRPixelmapLoadMany@<EAX>(char *pFile_name@<EAX>, br_pixelmap **pPixelmaps@<EDX>, br_uint_16 pNum@<EBX>)
|
|
br_uint_32 DRPixelmapLoadMany(char* pFile_name, br_pixelmap** pPixelmaps, br_uint_16 pNum) {
|
|
br_pixelmap* the_map;
|
|
int number_loaded;
|
|
int i;
|
|
br_uint_8 lobyte;
|
|
LOG_TRACE("(\"%s\", %p, %d)", pFile_name, pPixelmaps, pNum);
|
|
number_loaded = BrPixelmapLoadMany(pFile_name, pPixelmaps, pNum);
|
|
for (i = 0; i < number_loaded; i++) {
|
|
the_map = pPixelmaps[i];
|
|
the_map->row_bytes = (the_map->row_bytes + sizeof(int32_t) - 1) & ~(sizeof(int32_t) - 1);
|
|
the_map->base_x = 0;
|
|
the_map->base_y = 0;
|
|
}
|
|
return number_loaded;
|
|
}
|
|
|
|
// IDA: void __usercall WaitFor(tU32 pDelay@<EAX>)
|
|
void WaitFor(tU32 pDelay) {
|
|
tU32 start_time;
|
|
LOG_TRACE("(%d)", pDelay);
|
|
|
|
start_time = PDGetTotalTime();
|
|
while (start_time + pDelay < PDGetTotalTime()) {
|
|
SoundService();
|
|
}
|
|
}
|
|
|
|
// IDA: br_uint_32 __usercall DRActorEnumRecurse@<EAX>(br_actor *pActor@<EAX>, br_actor_enum_cbfn *callback@<EDX>, void *arg@<EBX>)
|
|
br_uintptr_t DRActorEnumRecurse(br_actor* pActor, br_actor_enum_cbfn* callback, void* arg) {
|
|
br_uintptr_t result;
|
|
|
|
result = callback(pActor, arg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
for (pActor = pActor->children; pActor != NULL; pActor = pActor->next) {
|
|
result = DRActorEnumRecurse(pActor, callback, arg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// IDA: br_uint_32 __cdecl CompareActorID(br_actor *pActor, void *pArg)
|
|
br_uintptr_t CompareActorID(br_actor* pActor, void* pArg) {
|
|
LOG_TRACE("(%p, %p)", pActor, pArg);
|
|
|
|
if (pActor->identifier && !strcmp(pActor->identifier, (const char*)pArg)) {
|
|
return (intptr_t)pActor;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// IDA: br_actor* __usercall DRActorFindRecurse@<EAX>(br_actor *pSearch_root@<EAX>, char *pName@<EDX>)
|
|
br_actor* DRActorFindRecurse(br_actor* pSearch_root, char* pName) {
|
|
LOG_TRACE("(%p, \"%s\")", pSearch_root, pName);
|
|
|
|
return (br_actor*)DRActorEnumRecurse(pSearch_root, CompareActorID, pName);
|
|
}
|
|
|
|
// IDA: br_uint_32 __usercall DRActorEnumRecurseWithMat@<EAX>(br_actor *pActor@<EAX>, br_material *pMat@<EDX>, br_uint_32 (*pCall_back)(br_actor*, br_material*, void*)@<EBX>, void *pArg@<ECX>)
|
|
br_uint_32 DRActorEnumRecurseWithMat(br_actor* pActor, br_material* pMat, recurse_with_mat_cbfn* pCall_back, void* pArg) {
|
|
br_uint_32 result;
|
|
LOG_TRACE("(%p, %p, %p, %p)", pActor, pMat, pCall_back, pArg);
|
|
|
|
if (pActor->material != NULL) {
|
|
pMat = pActor->material;
|
|
}
|
|
result = pCall_back(pActor, pMat, pArg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
for (pActor = pActor->children; pActor != NULL; pActor = pActor->next) {
|
|
result = DRActorEnumRecurseWithMat(pActor, pMat, pCall_back, pArg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// IDA: br_uint_32 __usercall DRActorEnumRecurseWithTrans@<EAX>(br_actor *pActor@<EAX>, br_matrix34 *pMatrix@<EDX>, br_uint_32 (*pCall_back)(br_actor*, br_matrix34*, void*)@<EBX>, void *pArg@<ECX>)
|
|
br_uint_32 DRActorEnumRecurseWithTrans(br_actor* pActor, br_matrix34* pMatrix, br_uint_32 (*pCall_back)(br_actor*, br_matrix34*, void*), void* pArg) {
|
|
br_uint_32 result;
|
|
br_matrix34 combined_transform;
|
|
LOG_TRACE("(%p, %p, %p, %p)", pActor, pMatrix, pCall_back, pArg);
|
|
|
|
if (pMatrix == NULL) {
|
|
BrMatrix34Copy(&combined_transform, &pActor->t.t.mat);
|
|
} else {
|
|
BrMatrix34Mul(&combined_transform, pMatrix, &pActor->t.t.mat);
|
|
}
|
|
result = pCall_back(pActor, &combined_transform, pArg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
for (pActor = pActor->children; pActor != NULL; pActor = pActor->next) {
|
|
result = DRActorEnumRecurseWithTrans(pActor, &combined_transform, pCall_back, pArg);
|
|
if (result != 0) {
|
|
return result;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// IDA: int __usercall sign@<EAX>(int pNumber@<EAX>)
|
|
int sign(int pNumber) {
|
|
LOG_TRACE("(%d)", pNumber);
|
|
|
|
if (pNumber > 0) {
|
|
return 1;
|
|
} else if (pNumber < 0) {
|
|
return -1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// IDA: float __cdecl fsign(float pNumber)
|
|
float fsign(float pNumber) {
|
|
LOG_TRACE("(%f)", pNumber);
|
|
if (pNumber > 0.f) {
|
|
return 1;
|
|
} else if (pNumber < 0.f) {
|
|
return -1.f;
|
|
} else {
|
|
return 0.f;
|
|
}
|
|
}
|
|
|
|
// IDA: FILE* __usercall OpenUniqueFileB@<EAX>(char *pPrefix@<EAX>, char *pExtension@<EDX>)
|
|
FILE* OpenUniqueFileB(char* pPrefix, char* pExtension) {
|
|
int index;
|
|
FILE* f;
|
|
tPath_name the_path;
|
|
LOG_TRACE("(\"%s\", \"%s\")", pPrefix, pExtension);
|
|
|
|
for (index = 0; index < 10000; index++) {
|
|
PathCat(the_path, gApplication_path, pPrefix);
|
|
sprintf(the_path + strlen(the_path), "%04d.%s", index, pExtension);
|
|
f = DRfopen(the_path, "rt");
|
|
if (f == NULL) {
|
|
return DRfopen(the_path, "wb");
|
|
}
|
|
fclose(f);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// IDA: void __usercall PrintScreenFile(FILE *pF@<EAX>)
|
|
void PrintScreenFile(FILE* pF) {
|
|
int i;
|
|
int j;
|
|
int bit_map_size;
|
|
int offset;
|
|
tU8* pixel_ptr;
|
|
LOG_TRACE("(%p)", pF);
|
|
|
|
bit_map_size = gBack_screen->height * gBack_screen->row_bytes;
|
|
|
|
// 1. BMP Header
|
|
// 1. 'BM' Signature
|
|
WriteU8L(pF, 'B');
|
|
WriteU8L(pF, 'M');
|
|
// 2. File size in bytes (header = 0xe bytes; infoHeader = 0x28 bytes; colorTable = 0x400 bytes; pixelData = xxx)
|
|
WriteU32L(pF, bit_map_size + 0x436);
|
|
// 3. unused
|
|
WriteU16L(pF, 0);
|
|
// 4. unused
|
|
WriteU16L(pF, 0);
|
|
// 5. pixelData offset (from beginning of file)
|
|
WriteU32L(pF, 0x436);
|
|
|
|
// 2. Info Header
|
|
// 1. InfoHeader Size
|
|
WriteU32L(pF, 0x28);
|
|
// 2. Width of bitmap in pixels
|
|
WriteU32L(pF, gBack_screen->row_bytes);
|
|
// 3. Height of bitmap in pixels
|
|
WriteU32L(pF, gBack_screen->height);
|
|
// 4. Number of planes
|
|
WriteU16L(pF, 1);
|
|
// 5. Bits per pixels / palletization (8 -> 8bit palletized ==> #colors = 256)
|
|
WriteU16L(pF, 8);
|
|
// 6. Compression (0 = BI_RGB -> no compression)
|
|
WriteU32L(pF, 0);
|
|
// 7. Image Size (0 --> no compression)
|
|
WriteU32L(pF, 0);
|
|
// 8. Horizontal Pixels Per Meter
|
|
WriteU32L(pF, 0);
|
|
// 9. Vertical Pixels Per Meter
|
|
WriteU32L(pF, 0);
|
|
// 10. # Actually used colors
|
|
WriteU32L(pF, 0);
|
|
// 11. Number of important colors
|
|
WriteU32L(pF, 256);
|
|
|
|
// 3. Color table (=palette)
|
|
for (i = 0; i < 256; i++) {
|
|
// red, green, blue, unused
|
|
#if BR_ENDIAN_BIG
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i + 3]);
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i + 2]);
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i + 1]);
|
|
#else
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i]);
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i + 1]);
|
|
WriteU8L(pF, ((tU8*)gCurrent_palette->pixels)[4 * i + 2]);
|
|
#endif
|
|
WriteU8L(pF, 0);
|
|
}
|
|
|
|
// 4. Pixel Data (=LUT)
|
|
offset = bit_map_size - gBack_screen->row_bytes;
|
|
for (i = 0; i < gBack_screen->height; i++) {
|
|
for (j = 0; j < gBack_screen->row_bytes; j++) {
|
|
WriteU8L(pF, ((tU8*)gBack_screen->pixels)[offset]);
|
|
offset++;
|
|
}
|
|
offset -= 2 * gBack_screen->row_bytes;
|
|
}
|
|
WriteU16L(pF, 0);
|
|
}
|
|
|
|
// IDA: void __usercall PrintScreenFile16(FILE *pF@<EAX>)
|
|
void PrintScreenFile16(FILE* pF) {
|
|
int i;
|
|
int j;
|
|
int bit_map_size;
|
|
int offset;
|
|
tU8* pixel_ptr;
|
|
tU16 pixel;
|
|
LOG_TRACE("(%p)", pF);
|
|
NOT_IMPLEMENTED();
|
|
}
|
|
|
|
// IDA: void __cdecl PrintScreen()
|
|
void PrintScreen(void) {
|
|
FILE* f;
|
|
LOG_TRACE("()");
|
|
|
|
f = OpenUniqueFileB("DUMP", "BMP");
|
|
if (f == NULL) {
|
|
return;
|
|
}
|
|
#ifdef DETHRACE_3DFX_PATCH
|
|
if (gBack_screen->type == BR_PMT_RGB_565) {
|
|
PrintScreenFile16(f);
|
|
} else
|
|
#endif
|
|
{
|
|
PrintScreenFile(f);
|
|
}
|
|
fclose(f);
|
|
}
|
|
|
|
// IDA: tU32 __cdecl GetTotalTime()
|
|
tU32 GetTotalTime(void) {
|
|
LOG_TRACE9("()");
|
|
|
|
if (gAction_replay_mode) {
|
|
return gLast_replay_frame_time;
|
|
}
|
|
if (gNet_mode != eNet_mode_none) {
|
|
return PDGetTotalTime();
|
|
}
|
|
return PDGetTotalTime() - gLost_time;
|
|
}
|
|
|
|
// IDA: tU32 __cdecl GetRaceTime()
|
|
tU32 GetRaceTime(void) {
|
|
LOG_TRACE("()");
|
|
|
|
return GetTotalTime() - gRace_start;
|
|
}
|
|
|
|
// IDA: void __usercall AddLostTime(tU32 pLost_time@<EAX>)
|
|
void AddLostTime(tU32 pLost_time) {
|
|
|
|
gLost_time += pLost_time;
|
|
}
|
|
|
|
// IDA: void __usercall TimerString(tU32 pTime@<EAX>, char *pStr@<EDX>, int pFudge_colon@<EBX>, int pForce_colon@<ECX>)
|
|
void TimerString(tU32 pTime, char* pStr, int pFudge_colon, int pForce_colon) {
|
|
int seconds;
|
|
LOG_TRACE("(%d, \"%s\", %d, %d)", pTime, pStr, pFudge_colon, pForce_colon);
|
|
|
|
seconds = (pTime + 500) / 1000;
|
|
if (pForce_colon || seconds > 59) {
|
|
if (pFudge_colon) {
|
|
sprintf(pStr, "%d/%02d", seconds / 60, seconds % 60);
|
|
} else {
|
|
sprintf(pStr, "%d:%02d", seconds / 60, seconds % 60);
|
|
}
|
|
} else {
|
|
sprintf(pStr, "%d", seconds);
|
|
}
|
|
}
|
|
|
|
// IDA: char* __usercall GetMiscString@<EAX>(int pIndex@<EAX>)
|
|
char* GetMiscString(int pIndex) {
|
|
|
|
return gMisc_strings[pIndex];
|
|
}
|
|
|
|
// IDA: void __usercall GetCopyOfMiscString(int pIndex@<EAX>, char *pStr@<EDX>)
|
|
void GetCopyOfMiscString(int pIndex, char* pStr) {
|
|
LOG_TRACE("(%d, \"%s\")", pIndex, pStr);
|
|
|
|
strcpy(pStr, GetMiscString(pIndex));
|
|
}
|
|
|
|
// IDA: int __usercall Flash@<EAX>(tU32 pPeriod@<EAX>, tU32 *pLast_change@<EDX>, int *pCurrent_state@<EBX>)
|
|
int Flash(tU32 pPeriod, tU32* pLast_change, int* pCurrent_state) {
|
|
tU32 the_time;
|
|
LOG_TRACE("(%d, %p, %p)", pPeriod, pLast_change, pCurrent_state);
|
|
|
|
the_time = PDGetTotalTime();
|
|
if (the_time - *pLast_change > pPeriod) {
|
|
*pCurrent_state = !*pCurrent_state;
|
|
*pLast_change = the_time;
|
|
}
|
|
return *pCurrent_state;
|
|
}
|
|
|
|
// IDA: void __usercall MaterialCopy(br_material *pDst@<EAX>, br_material *pSrc@<EDX>)
|
|
void MaterialCopy(br_material* pDst, br_material* pSrc) {
|
|
LOG_TRACE("(%p, %p)", pDst, pSrc);
|
|
|
|
pDst->flags = pSrc->flags;
|
|
pDst->ka = pSrc->ka;
|
|
pDst->kd = pSrc->kd;
|
|
pDst->ks = pSrc->ks;
|
|
pDst->power = pSrc->power;
|
|
pDst->colour = pSrc->colour;
|
|
pDst->index_base = pSrc->index_base;
|
|
pDst->index_range = pSrc->index_range;
|
|
pDst->index_shade = pSrc->index_shade;
|
|
pDst->colour_map = pSrc->colour_map;
|
|
pDst->map_transform = pSrc->map_transform;
|
|
pDst->identifier = pSrc->identifier;
|
|
}
|
|
|
|
// IDA: double __usercall RGBDifferenceSqr@<ST0>(tRGB_colour *pColour_1@<EAX>, tRGB_colour *pColour_2@<EDX>)
|
|
double RGBDifferenceSqr(tRGB_colour* pColour_1, tRGB_colour* pColour_2) {
|
|
LOG_TRACE("(%p, %p)", pColour_1, pColour_2);
|
|
|
|
return ((pColour_1->red - pColour_2->red) * (pColour_1->red - pColour_2->red))
|
|
+ ((pColour_1->green - pColour_2->green) * (pColour_1->green - pColour_2->green))
|
|
+ ((pColour_1->blue - pColour_2->blue) * (pColour_1->blue - pColour_2->blue));
|
|
}
|
|
|
|
// IDA: int __usercall FindBestMatch@<EAX>(tRGB_colour *pRGB_colour@<EAX>, br_pixelmap *pPalette@<EDX>)
|
|
int FindBestMatch(tRGB_colour* pRGB_colour, br_pixelmap* pPalette) {
|
|
int n;
|
|
int near_c;
|
|
double min_d;
|
|
double d;
|
|
tRGB_colour trial_RGB;
|
|
br_colour* dp;
|
|
LOG_TRACE("(%p, %p)", pRGB_colour, pPalette);
|
|
|
|
near_c = 127;
|
|
min_d = 1.79769e+308; // max double
|
|
dp = pPalette->pixels;
|
|
for (n = 0; n < 256; n++) {
|
|
trial_RGB.red = (dp[n] >> 16) & 0xff;
|
|
trial_RGB.green = (dp[n] >> 8) & 0xff;
|
|
trial_RGB.blue = (dp[n] >> 0) & 0xff;
|
|
d = RGBDifferenceSqr(pRGB_colour, &trial_RGB);
|
|
if (d < min_d) {
|
|
min_d = d;
|
|
near_c = n;
|
|
}
|
|
}
|
|
return near_c;
|
|
}
|
|
|
|
// IDA: void __usercall BuildShadeTablePath(char *pThe_path@<EAX>, int pR@<EDX>, int pG@<EBX>, int pB@<ECX>)
|
|
void BuildShadeTablePath(char* pThe_path, int pR, int pG, int pB) {
|
|
char s[32];
|
|
LOG_TRACE("(\"%s\", %d, %d, %d)", pThe_path, pR, pG, pB);
|
|
|
|
s[0] = 's';
|
|
s[1] = 't';
|
|
s[2] = 'A' + ((pR & 0xf0) >> 4);
|
|
s[3] = 'A' + ((pR & 0x0f) >> 0);
|
|
s[4] = 'A' + ((pG & 0xf0) >> 4);
|
|
s[5] = 'A' + ((pG & 0x0f) >> 0);
|
|
s[6] = 'A' + ((pB & 0xf0) >> 4);
|
|
s[7] = 'A' + ((pB & 0x0f) >> 0);
|
|
s[8] = '\0';
|
|
strcat(s, ".TAB");
|
|
PathCat(pThe_path, gApplication_path, "SHADETAB");
|
|
PathCat(pThe_path, pThe_path, s);
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall LoadGeneratedShadeTable@<EAX>(int pR@<EAX>, int pG@<EDX>, int pB@<EBX>)
|
|
br_pixelmap* LoadGeneratedShadeTable(int pR, int pG, int pB) {
|
|
char the_path[256];
|
|
LOG_TRACE("(%d, %d, %d)", pR, pG, pB);
|
|
|
|
BuildShadeTablePath(the_path, pR, pG, pB);
|
|
return BrPixelmapLoad(the_path);
|
|
}
|
|
|
|
// IDA: void __usercall SaveGeneratedShadeTable(br_pixelmap *pThe_table@<EAX>, int pR@<EDX>, int pG@<EBX>, int pB@<ECX>)
|
|
void SaveGeneratedShadeTable(br_pixelmap* pThe_table, int pR, int pG, int pB) {
|
|
char the_path[256];
|
|
LOG_TRACE("(%p, %d, %d, %d)", pThe_table, pR, pG, pB);
|
|
|
|
BuildShadeTablePath(the_path, pR, pG, pB);
|
|
BrPixelmapSave(the_path, pThe_table);
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall GenerateShadeTable@<EAX>(int pHeight@<EAX>, br_pixelmap *pPalette@<EDX>, int pRed_mix@<EBX>, int pGreen_mix@<ECX>, int pBlue_mix, float pQuarter, float pHalf, float pThree_quarter)
|
|
br_pixelmap* GenerateShadeTable(int pHeight, br_pixelmap* pPalette, int pRed_mix, int pGreen_mix, int pBlue_mix, float pQuarter, float pHalf, float pThree_quarter) {
|
|
LOG_TRACE("(%d, %p, %d, %d, %d, %f, %f, %f)", pHeight, pPalette, pRed_mix, pGreen_mix, pBlue_mix, pQuarter, pHalf, pThree_quarter);
|
|
|
|
PossibleService();
|
|
return GenerateDarkenedShadeTable(
|
|
pHeight,
|
|
pPalette,
|
|
pRed_mix,
|
|
pGreen_mix,
|
|
pBlue_mix,
|
|
pQuarter,
|
|
pHalf,
|
|
pThree_quarter,
|
|
1.0);
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall GenerateDarkenedShadeTable@<EAX>(int pHeight@<EAX>, br_pixelmap *pPalette@<EDX>, int pRed_mix@<EBX>, int pGreen_mix@<ECX>, int pBlue_mix, float pQuarter, float pHalf, float pThree_quarter, br_scalar pDarken)
|
|
br_pixelmap* GenerateDarkenedShadeTable(int pHeight, br_pixelmap* pPalette, int pRed_mix, int pGreen_mix, int pBlue_mix, float pQuarter, float pHalf, float pThree_quarter, br_scalar pDarken) {
|
|
br_pixelmap* the_table;
|
|
tRGB_colour the_RGB;
|
|
tRGB_colour new_RGB;
|
|
tRGB_colour ref_col;
|
|
br_colour* cp;
|
|
char* tab_ptr;
|
|
char* shade_ptr;
|
|
double f_i;
|
|
double f_total_minus_1;
|
|
double ratio1;
|
|
double ratio2;
|
|
int i;
|
|
int c;
|
|
LOG_TRACE("(%d, %p, %d, %d, %d, %f, %f, %f, %f)", pHeight, pPalette, pRed_mix, pGreen_mix, pBlue_mix, pQuarter, pHalf, pThree_quarter, pDarken);
|
|
|
|
the_table = LoadGeneratedShadeTable(pRed_mix, pGreen_mix, pBlue_mix);
|
|
if (the_table == NULL) {
|
|
the_table = BrPixelmapAllocate(BR_PMT_INDEX_8, 256, pHeight, NULL, 0);
|
|
if (the_table == NULL) {
|
|
FatalError(kFatalError_LoadGeneratedShadeTable);
|
|
}
|
|
cp = pPalette->pixels;
|
|
|
|
ref_col.red = pRed_mix;
|
|
ref_col.green = pGreen_mix;
|
|
ref_col.blue = pBlue_mix;
|
|
|
|
for (c = 0, tab_ptr = the_table->pixels; c < 256; c++, tab_ptr++) {
|
|
the_RGB.red = ((cp[c] >> 16) & 0xff) * pDarken;
|
|
the_RGB.green = ((cp[c] >> 8) & 0xff) * pDarken;
|
|
the_RGB.blue = ((cp[c] >> 0) & 0xff) * pDarken;
|
|
|
|
if (pHeight == 1) {
|
|
f_total_minus_1 = 1.;
|
|
} else {
|
|
f_total_minus_1 = pHeight - 1;
|
|
}
|
|
shade_ptr = tab_ptr;
|
|
for (i = 0, shade_ptr = tab_ptr; i < pHeight; i++, shade_ptr += 0x100) {
|
|
f_i = i;
|
|
ratio1 = f_i / f_total_minus_1;
|
|
if (ratio1 < .5) {
|
|
if (ratio1 < .25) {
|
|
ratio2 = pQuarter * ratio1 * 4.;
|
|
} else {
|
|
ratio2 = (ratio1 - .25) * (pHalf - pQuarter) * 4. + pQuarter;
|
|
}
|
|
} else {
|
|
if (ratio1 < 0.75) {
|
|
ratio2 = (ratio1 - .5) * (pThree_quarter - pHalf) * 4. + pHalf;
|
|
} else {
|
|
ratio2 = 1. - (1. - pThree_quarter) * (1. - ratio1) * 4.;
|
|
}
|
|
}
|
|
new_RGB.red = ref_col.red * ratio2 + the_RGB.red * (1. - ratio2);
|
|
new_RGB.green = ref_col.green * ratio2 + the_RGB.green * (1. - ratio2);
|
|
new_RGB.blue = ref_col.blue * ratio2 + the_RGB.blue * (1. - ratio2);
|
|
*shade_ptr = FindBestMatch(&new_RGB, pPalette);
|
|
}
|
|
}
|
|
SaveGeneratedShadeTable(the_table, pRed_mix, pGreen_mix, pBlue_mix);
|
|
}
|
|
BrTableAdd(the_table);
|
|
return the_table;
|
|
}
|
|
|
|
// IDA: void __cdecl PossibleService()
|
|
void PossibleService(void) {
|
|
tU32 time;
|
|
static tU32 last_service = 0;
|
|
|
|
time = PDGetTotalTime();
|
|
if (time - last_service > MIN_SERVICE_INTERVAL && !gProgram_state.racing) {
|
|
SoundService();
|
|
NetService(gProgram_state.racing);
|
|
last_service = time;
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall DRMatrix34TApplyP(br_vector3 *pA@<EAX>, br_vector3 *pB@<EDX>, br_matrix34 *pC@<EBX>)
|
|
void DRMatrix34TApplyP(br_vector3* pA, br_vector3* pB, br_matrix34* pC) {
|
|
br_scalar t1;
|
|
br_scalar t2;
|
|
br_scalar t3;
|
|
LOG_TRACE("(%p, %p, %p)", pA, pB, pC);
|
|
|
|
t1 = pB->v[0] - pC->m[3][0];
|
|
t2 = pB->v[1] - pC->m[3][1];
|
|
t3 = pB->v[2] - pC->m[3][2];
|
|
pA->v[0] = pC->m[0][0] * t1 + pC->m[0][1] * t2 + pC->m[0][2] * t3;
|
|
pA->v[1] = pC->m[1][0] * t1 + pC->m[1][1] * t2 + pC->m[1][2] * t3;
|
|
pA->v[2] = pC->m[2][0] * t1 + pC->m[2][1] * t2 + pC->m[2][2] * t3;
|
|
}
|
|
|
|
// IDA: tU16 __usercall PaletteEntry16Bit@<AX>(br_pixelmap *pPal@<EAX>, int pEntry@<EDX>)
|
|
tU16 PaletteEntry16Bit(br_pixelmap* pPal, int pEntry) {
|
|
tU32* src_entry;
|
|
int red;
|
|
int green;
|
|
int blue;
|
|
LOG_TRACE("(%p, %d)", pPal, pEntry);
|
|
|
|
src_entry = pPal->pixels;
|
|
return ((tU8)src_entry[pEntry] >> 3) | (((src_entry[pEntry] >> 19) & 0x1F) << 11) | (32 * ((tU16)src_entry[pEntry] >> 10));
|
|
}
|
|
|
|
// IDA: br_pixelmap* __usercall PaletteOf16Bits@<EAX>(br_pixelmap *pSrc@<EAX>)
|
|
br_pixelmap* PaletteOf16Bits(br_pixelmap* pSrc) {
|
|
tU16* dst_entry;
|
|
int value;
|
|
LOG_TRACE("(%p)", pSrc);
|
|
|
|
if (g16bit_palette == NULL) {
|
|
g16bit_palette = BrPixelmapAllocate(BR_PMT_RGB_565, 1, 256, g16bit_palette, 0);
|
|
if (g16bit_palette == NULL) {
|
|
FatalError(94, "16-bit palette");
|
|
}
|
|
}
|
|
if (!g16bit_palette_valid || pSrc != gSource_for_16bit_palette) {
|
|
value = 0;
|
|
dst_entry = g16bit_palette->pixels;
|
|
for (value = 0; value < 256; value++) {
|
|
*dst_entry = PaletteEntry16Bit(pSrc, value);
|
|
dst_entry++;
|
|
}
|
|
gSource_for_16bit_palette = pSrc;
|
|
g16bit_palette_valid = 1;
|
|
}
|
|
return g16bit_palette;
|
|
}
|
|
|
|
// IDA: void __usercall Copy8BitTo16Bit(br_pixelmap *pDst@<EAX>, br_pixelmap *pSrc@<EDX>, br_pixelmap *pPalette@<EBX>)
|
|
void Copy8BitTo16Bit(br_pixelmap* pDst, br_pixelmap* pSrc, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start;
|
|
tU16* palette_entry;
|
|
LOG_TRACE("(%p, %p, %p)", pDst, pSrc, pPalette);
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
for (y = 0; y < pDst->height; y++) {
|
|
src_start = (tU8*)pSrc->pixels + pSrc->row_bytes * y;
|
|
dst_start = (tU16*)((tU8*)pDst->pixels + pDst->row_bytes * y);
|
|
for (x = 0; x < pDst->width; x++) {
|
|
*dst_start = palette_entry[*src_start];
|
|
src_start++;
|
|
dst_start++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall Copy8BitTo16BitRectangle(br_pixelmap *pDst@<EAX>, tS16 pDst_x@<EDX>, tS16 pDst_y@<EBX>, br_pixelmap *pSrc@<ECX>, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap *pPalette)
|
|
void Copy8BitTo16BitRectangle(br_pixelmap* pDst, tS16 pDst_x, tS16 pDst_y, br_pixelmap* pSrc, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start;
|
|
tU16* palette_entry;
|
|
LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d, %p)", pDst, pDst_x, pDst_y, pSrc, pSrc_x, pSrc_y, pWidth, pHeight, pPalette);
|
|
|
|
if (pSrc_x < 0) {
|
|
pWidth = pSrc_x + pWidth;
|
|
pDst_x = pDst_x - pSrc_x;
|
|
pSrc_x = 0;
|
|
}
|
|
if (pDst_x < 0) {
|
|
pWidth += pDst_x;
|
|
pSrc_x -= pDst_x;
|
|
pDst_x = 0;
|
|
}
|
|
if (pSrc_y < 0) {
|
|
pHeight = pSrc_y + pHeight;
|
|
pDst_y = pDst_y - pSrc_y;
|
|
pSrc_y = 0;
|
|
}
|
|
if (pDst_y < 0) {
|
|
pHeight += pDst_y;
|
|
pSrc_y -= pDst_y;
|
|
pDst_y = 0;
|
|
}
|
|
|
|
if (pSrc_x + pWidth > pSrc->width) {
|
|
pWidth = pSrc->width - pSrc_x;
|
|
}
|
|
if (pSrc_y + pHeight > pSrc->height) {
|
|
pHeight = pSrc->height - pSrc_y;
|
|
}
|
|
|
|
if (pDst_x + pWidth > pDst->width) {
|
|
pWidth = pDst->width - pDst_x;
|
|
}
|
|
if (pDst_y + pHeight > pDst->height) {
|
|
pHeight = pDst->height - pDst_y;
|
|
}
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
for (y = 0; y < pHeight; y++) {
|
|
src_start = (tU8*)pSrc->pixels + (pSrc->row_bytes * (pSrc_y + y));
|
|
src_start += pSrc_x;
|
|
dst_start = (tU16*)((tU8*)pDst->pixels + (pDst->row_bytes * (pDst_y + y)));
|
|
dst_start += pDst_x;
|
|
for (x = 0; x < pWidth; x++) {
|
|
// even though we have a specific `WithTransparency` version of this function, this one also handles transparency!
|
|
if (*src_start != 0) {
|
|
*dst_start = palette_entry[*src_start];
|
|
}
|
|
src_start++;
|
|
dst_start++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall Copy8BitTo16BitRectangleWithTransparency(br_pixelmap *pDst@<EAX>, tS16 pDst_x@<EDX>, tS16 pDst_y@<EBX>, br_pixelmap *pSrc@<ECX>, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap *pPalette)
|
|
void Copy8BitTo16BitRectangleWithTransparency(br_pixelmap* pDst, tS16 pDst_x, tS16 pDst_y, br_pixelmap* pSrc, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start;
|
|
tU16* palette_entry;
|
|
LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d, %p)", pDst, pDst_x, pDst_y, pSrc, pSrc_x, pSrc_y, pWidth, pHeight, pPalette);
|
|
|
|
if (pSrc_x < 0) {
|
|
pWidth = pSrc_x + pWidth;
|
|
pDst_x = pDst_x - pSrc_x;
|
|
pSrc_x = 0;
|
|
}
|
|
if (pDst_x < 0) {
|
|
pWidth += pDst_x;
|
|
pSrc_x -= pDst_x;
|
|
pDst_x = 0;
|
|
}
|
|
if (pSrc_y < 0) {
|
|
pHeight = pSrc_y + pHeight;
|
|
pDst_y = pDst_y - pSrc_y;
|
|
pSrc_y = 0;
|
|
}
|
|
if (pDst_y < 0) {
|
|
pHeight += pDst_y;
|
|
pSrc_y -= pDst_y;
|
|
pDst_y = 0;
|
|
}
|
|
|
|
if (pSrc_x + pWidth > pSrc->width) {
|
|
pWidth = pSrc->width - pSrc_x;
|
|
}
|
|
if (pSrc_y + pHeight > pSrc->height) {
|
|
pHeight = pSrc->height - pSrc_y;
|
|
}
|
|
|
|
if (pDst_x + pWidth > pDst->width) {
|
|
pWidth = pDst->width - pDst_x;
|
|
}
|
|
if (pDst_y + pHeight > pDst->height) {
|
|
pHeight = pDst->height - pDst_y;
|
|
}
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
for (y = 0; y < pHeight; y++) {
|
|
src_start = (tU8*)pSrc->pixels + (pSrc->row_bytes * (pSrc_y + y)) + pSrc_x;
|
|
dst_start = (tU16*)((tU8*)pDst->pixels + (pDst->row_bytes * (pDst_y + y)));
|
|
dst_start += pDst_x;
|
|
for (x = 0; x < pWidth; x++) {
|
|
if (*src_start != 0) {
|
|
*dst_start = palette_entry[*src_start];
|
|
}
|
|
src_start++;
|
|
dst_start++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall Copy8BitToOnscreen16BitRectangleWithTransparency(br_pixelmap *pDst@<EAX>, tS16 pDst_x@<EDX>, tS16 pDst_y@<EBX>, br_pixelmap *pSrc@<ECX>, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap *pPalette)
|
|
void Copy8BitToOnscreen16BitRectangleWithTransparency(br_pixelmap* pDst, tS16 pDst_x, tS16 pDst_y, br_pixelmap* pSrc, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start;
|
|
tU16* palette_entry;
|
|
LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d, %p)", pDst, pDst_x, pDst_y, pSrc, pSrc_x, pSrc_y, pWidth, pHeight, pPalette);
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
for (y = 0; y < pHeight; y++) {
|
|
src_start = (tU8*)pSrc->pixels + (pSrc->row_bytes * (pSrc_y + y)) + pSrc_x;
|
|
dst_start = (tU16*)((tU8*)pDst->pixels + (pDst->row_bytes * (pDst_y + y)));
|
|
dst_start += pDst_x;
|
|
for (x = 0; x < pWidth; x++) {
|
|
if (*src_start != 0) {
|
|
*dst_start = palette_entry[*src_start];
|
|
}
|
|
src_start++;
|
|
dst_start++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall Copy8BitRectangleTo16BitRhombusWithTransparency(br_pixelmap *pDst@<EAX>, tS16 pDst_x@<EDX>, tS16 pDst_y@<EBX>, br_pixelmap *pSrc@<ECX>, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, tX1616 pShear, br_pixelmap *pPalette)
|
|
void Copy8BitRectangleTo16BitRhombusWithTransparency(br_pixelmap* pDst, tS16 pDst_x, tS16 pDst_y, br_pixelmap* pSrc, tS16 pSrc_x, tS16 pSrc_y, tS16 pWidth, tS16 pHeight, tX1616 pShear, br_pixelmap* pPalette) {
|
|
int x;
|
|
int y;
|
|
tU8* src_start;
|
|
tU16* dst_start;
|
|
tU16* palette_entry;
|
|
tX1616 total_shear;
|
|
tS16 sheared_x;
|
|
tS16 clipped_src_x;
|
|
tS16 clipped_width;
|
|
LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d, %d, %p)", pDst, pDst_x, pDst_y, pSrc, pSrc_x, pSrc_y, pWidth, pHeight, pShear, pPalette);
|
|
|
|
palette_entry = PaletteOf16Bits(pPalette)->pixels;
|
|
total_shear = 0;
|
|
if (pSrc_y + pSrc->origin_y < 0) {
|
|
pHeight += pSrc_y + pSrc->origin_y;
|
|
pDst_y -= (pSrc_y + pSrc->origin_y);
|
|
pSrc_y = 0;
|
|
}
|
|
if (pDst_y + pDst->origin_y < 0) {
|
|
pHeight += pDst_y + pDst->origin_y;
|
|
pSrc_y -= (pDst_y + pDst->origin_y);
|
|
pDst_y = 0;
|
|
}
|
|
|
|
if (pSrc_y + pSrc->origin_y + pHeight > pSrc->height) {
|
|
pHeight = pSrc->height - (pSrc_y + pSrc->origin_y);
|
|
}
|
|
|
|
if (pDst_y + pDst->origin_y + pHeight > pDst->height) {
|
|
pHeight = pDst->height - pDst_y + pDst->origin_y;
|
|
}
|
|
if (pHeight > 0) {
|
|
if (pSrc_x + pSrc->origin_x < 0) {
|
|
pWidth += pSrc_x + pSrc->origin_x;
|
|
pDst_x -= (pSrc_x + pSrc->origin_x);
|
|
pSrc_x = 0;
|
|
}
|
|
|
|
if (pSrc_x + pSrc->origin_x + pWidth > pSrc->width) {
|
|
pWidth = pSrc->width - (pSrc_x + pSrc->origin_x);
|
|
}
|
|
for (y = 0; y < pHeight; y++) {
|
|
clipped_src_x = pSrc_x + pSrc->origin_x;
|
|
sheared_x = pDst_x + pDst->origin_x + (total_shear >> 16);
|
|
clipped_width = pWidth;
|
|
if ((sheared_x & 0x8000u) != 0) {
|
|
clipped_width = pWidth + sheared_x;
|
|
clipped_src_x = (pSrc_x + pSrc->origin_x) - sheared_x;
|
|
sheared_x = 0;
|
|
}
|
|
|
|
if (sheared_x + clipped_width > pDst->width) {
|
|
clipped_width = pDst->width - sheared_x;
|
|
}
|
|
if (clipped_width > 0) {
|
|
src_start = ((tU8*)pSrc->pixels) + (y + pSrc_y + pSrc->origin_y) * pSrc->row_bytes + clipped_src_x;
|
|
dst_start = (tU16*)((tU8*)pDst->pixels + 2 * sheared_x + (y + pDst_y + pDst->origin_y) * pDst->row_bytes);
|
|
|
|
for (x = clipped_width; x > 0; x--) {
|
|
if (*src_start) {
|
|
*dst_start = palette_entry[*src_start];
|
|
}
|
|
src_start++;
|
|
dst_start++;
|
|
}
|
|
}
|
|
total_shear += pShear;
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall DRPixelmapRectangleCopy(br_pixelmap *dst@<EAX>, br_int_16 dx@<EDX>, br_int_16 dy@<EBX>, br_pixelmap *src@<ECX>, br_int_16 sx, br_int_16 sy, br_uint_16 w, br_uint_16 h)
|
|
void DRPixelmapRectangleCopy(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) {
|
|
LOG_TRACE("(%p, %d, %d, %p, %d, %d, %d, %d)", dst, dx, dy, src, sx, sy, w, h);
|
|
|
|
#ifdef DETHRACE_3DFX_PATCH
|
|
if (dst->type == src->type) {
|
|
BrPixelmapRectangleCopy(dst, dx, dy, src, sx, sy, w, h);
|
|
} else if (dst->type == BR_PMT_RGB_565 && src->type == BR_PMT_INDEX_8) {
|
|
Copy8BitTo16BitRectangle(dst, dx, dy, src, sx, sy, w, h, gCurrent_palette);
|
|
}
|
|
#else
|
|
BrPixelmapRectangleCopy(dst, dx, dy, src, sx, sy, w, h);
|
|
#endif
|
|
}
|
|
|
|
// IDA: void __usercall DRPixelmapCopy(br_pixelmap *dst@<EAX>, br_pixelmap *src@<EDX>)
|
|
void DRPixelmapCopy(br_pixelmap* dst, br_pixelmap* src) {
|
|
LOG_TRACE("(%p, %p)", dst, src);
|
|
|
|
#ifdef DETHRACE_3DFX_PATCH
|
|
if (dst->type == src->type) {
|
|
BrPixelmapCopy(dst, src);
|
|
} else if (dst->type == BR_PMT_RGB_565 && src->type == BR_PMT_INDEX_8) {
|
|
Copy8BitTo16Bit(dst, src, gCurrent_palette);
|
|
}
|
|
#else
|
|
BrPixelmapCopy(dst, src);
|
|
#endif
|
|
}
|
|
|
|
// IDA: void __usercall DRPixelmapRectangleFill(br_pixelmap *dst@<EAX>, br_int_16 x@<EDX>, br_int_16 y@<EBX>, br_uint_16 w@<ECX>, br_uint_16 h, br_uint_32 colour)
|
|
void DRPixelmapRectangleFill(br_pixelmap* dst, br_int_16 x, br_int_16 y, br_uint_16 w, br_uint_16 h, br_uint_32 colour) {
|
|
LOG_TRACE("(%p, %d, %d, %d, %d, %d)", dst, x, y, w, h, colour);
|
|
|
|
BrPixelmapRectangleFill(dst, x, y, w, h, colour);
|
|
}
|
|
|
|
// IDA: int __usercall NormalSideOfPlane@<EAX>(br_vector3 *pPoint@<EAX>, br_vector3 *pNormal@<EDX>, br_scalar pD)
|
|
int NormalSideOfPlane(br_vector3* pPoint, br_vector3* pNormal, br_scalar pD) {
|
|
br_scalar numer;
|
|
br_scalar denom;
|
|
LOG_TRACE("(%p, %p, %f)", pPoint, pNormal, pD);
|
|
|
|
return (BrVector3Dot(pNormal, pPoint) - pD) >= 0.f;
|
|
}
|
|
|
|
// IDA: br_material* __usercall DRMaterialClone@<EAX>(br_material *pMaterial@<EAX>)
|
|
br_material* DRMaterialClone(br_material* pMaterial) {
|
|
br_material* the_material;
|
|
char s[256];
|
|
static int name_suffix = 0;
|
|
LOG_TRACE("(%p)", pMaterial);
|
|
|
|
the_material = BrMaterialAllocate(NULL);
|
|
the_material->flags = pMaterial->flags;
|
|
the_material->ka = pMaterial->ka;
|
|
the_material->kd = pMaterial->kd;
|
|
the_material->ks = pMaterial->ks;
|
|
the_material->power = pMaterial->power;
|
|
the_material->colour = pMaterial->colour;
|
|
the_material->index_base = pMaterial->index_base;
|
|
the_material->index_range = pMaterial->index_range;
|
|
the_material->index_shade = pMaterial->index_shade;
|
|
the_material->index_blend = pMaterial->index_blend;
|
|
the_material->colour_map = pMaterial->colour_map;
|
|
memcpy(&the_material->map_transform, &pMaterial->map_transform, sizeof(the_material->map_transform));
|
|
sprintf(s, "%s(%d)", pMaterial->identifier, name_suffix);
|
|
name_suffix++;
|
|
the_material->identifier = BrResAllocate(the_material, strlen(s) + 1, BR_MEMORY_STRING);
|
|
strcpy(the_material->identifier, s);
|
|
BrMaterialAdd(the_material);
|
|
return the_material;
|
|
}
|
|
|
|
// IDA: void __usercall StripCR(char *s@<EAX>)
|
|
void StripCR(char* s) {
|
|
char* pos;
|
|
|
|
pos = s;
|
|
while (*pos != '\0') {
|
|
if (*pos == '\r' || *pos == '\n') {
|
|
*pos = '\0';
|
|
break;
|
|
}
|
|
pos++;
|
|
}
|
|
}
|
|
|
|
// IDA: void __cdecl SubsStringJob(char *pStr, ...)
|
|
void SubsStringJob(char* pStr, ...) {
|
|
char* sub_str;
|
|
char temp_str[256];
|
|
char* sub_pt;
|
|
va_list ap;
|
|
LOG_TRACE("(\"%s\")", pStr);
|
|
|
|
va_start(ap, pStr);
|
|
for (;;) {
|
|
sub_pt = strchr(pStr, '%');
|
|
if (sub_pt == NULL) {
|
|
va_end(ap);
|
|
return;
|
|
}
|
|
sub_str = va_arg(ap, char*);
|
|
StripCR(sub_str);
|
|
strcpy(temp_str, &sub_pt[1]);
|
|
strcpy(sub_pt, sub_str);
|
|
strcat(pStr, temp_str);
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall DecodeLine2(char *pS@<EAX>)
|
|
void DecodeLine2(char* pS) {
|
|
int len;
|
|
int seed;
|
|
int i;
|
|
unsigned char c;
|
|
char* key;
|
|
|
|
len = strlen(pS);
|
|
key = (char*)gLong_key;
|
|
while (len > 0 && (pS[len - 1] == '\r' || pS[len - 1] == '\n')) {
|
|
len--;
|
|
pS[len] = '\0';
|
|
}
|
|
seed = len % 16;
|
|
for (i = 0; i < len; i++) {
|
|
c = pS[i];
|
|
if (i >= 2) {
|
|
if (pS[i - 1] == '/' && pS[i - 2] == '/') {
|
|
key = (char*)gOther_long_key;
|
|
}
|
|
}
|
|
if (gEncryption_method == 1) {
|
|
if (c == '\t') {
|
|
c = 0x9f;
|
|
}
|
|
|
|
c -= 0x20;
|
|
c ^= key[seed];
|
|
c &= 0x7f;
|
|
c += 0x20;
|
|
|
|
seed += 7;
|
|
seed %= 16;
|
|
|
|
if (c == 0x9f) {
|
|
c = '\t';
|
|
}
|
|
} else {
|
|
if (c == '\t') {
|
|
c = 0x80;
|
|
}
|
|
c -= 0x20;
|
|
if ((c & 0x80) == 0) {
|
|
c ^= key[seed] & 0x7f;
|
|
}
|
|
c += 0x20;
|
|
|
|
seed += 7;
|
|
seed %= 16;
|
|
|
|
if (c == 0x80) {
|
|
c = '\t';
|
|
}
|
|
}
|
|
pS[i] = c;
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall EncodeLine2(char *pS@<EAX>)
|
|
void EncodeLine2(char* pS) {
|
|
int len;
|
|
int seed;
|
|
int i;
|
|
int count;
|
|
unsigned char c;
|
|
char* key;
|
|
|
|
len = strlen(pS);
|
|
count = 0;
|
|
key = (char*)gLong_key;
|
|
while (len > 0 && (pS[len - 1] == '\r' || pS[len - 1] == '\n')) {
|
|
len--;
|
|
pS[len] = '\0';
|
|
}
|
|
|
|
seed = len % 16;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
if (count == 2) {
|
|
key = (char*)gOther_long_key;
|
|
}
|
|
if (pS[i] == '/') {
|
|
count++;
|
|
} else {
|
|
count = 0;
|
|
}
|
|
if (pS[i] == '\t') {
|
|
pS[i] = 0x80;
|
|
}
|
|
|
|
c = pS[i] - 0x20;
|
|
if ((c & 0x80) == 0) {
|
|
c ^= key[seed] & 0x7f;
|
|
}
|
|
c += 0x20;
|
|
|
|
seed += 7;
|
|
seed %= 16;
|
|
|
|
if (c == 0x80) {
|
|
c = '\t';
|
|
}
|
|
pS[i] = c;
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall EncodeFile(char *pThe_path@<EAX>)
|
|
void EncodeFile(char* pThe_path) {
|
|
FILE* f;
|
|
FILE* d;
|
|
char line[257];
|
|
char new_file[256];
|
|
char* s;
|
|
char* result;
|
|
int ch;
|
|
int decode;
|
|
int len;
|
|
int count;
|
|
LOG_TRACE("(\"%s\")", pThe_path);
|
|
|
|
len = strlen(pThe_path);
|
|
strcpy(new_file, pThe_path);
|
|
strcpy(&new_file[len - 3], "ENC");
|
|
|
|
f = fopen(pThe_path, "rt");
|
|
if (f == NULL) {
|
|
FatalError(kFatalError_Open_S, pThe_path);
|
|
}
|
|
|
|
ch = fgetc(f);
|
|
ungetc(ch, f);
|
|
|
|
if (gDecode_thing == '@' && gDecode_thing == (char)ch) {
|
|
fclose(f);
|
|
return;
|
|
}
|
|
|
|
d = fopen(new_file, "wb");
|
|
if (d == NULL) {
|
|
FatalError(kFatalError_Open_S, new_file);
|
|
}
|
|
|
|
result = &line[1];
|
|
|
|
while (!feof(f)) {
|
|
s = fgets(result, 256, f);
|
|
|
|
if (s == NULL) {
|
|
continue;
|
|
}
|
|
|
|
if (result[0] == '@') {
|
|
decode = 1;
|
|
} else {
|
|
decode = 0;
|
|
// Strip leading whitespace
|
|
while (result[0] == ' ' || result[0] == '\t') {
|
|
memmove(result, &result[1], strlen(result));
|
|
}
|
|
}
|
|
|
|
if (decode) {
|
|
DecodeLine2(&result[decode]);
|
|
} else {
|
|
EncodeLine2(&result[decode]);
|
|
}
|
|
|
|
line[0] = '@';
|
|
fputs(&line[decode * 2], d);
|
|
count = -1;
|
|
while (1) {
|
|
count++;
|
|
ch = fgetc(f);
|
|
if (ch != '\r' && ch != '\n') {
|
|
break;
|
|
}
|
|
}
|
|
if (count > 2) {
|
|
fputc('\r', d);
|
|
fputc('\n', d);
|
|
}
|
|
fputc('\r', d);
|
|
fputc('\n', d);
|
|
|
|
if (ch != -1) {
|
|
ungetc(ch, f);
|
|
}
|
|
}
|
|
fclose(f);
|
|
fclose(d);
|
|
|
|
PDFileUnlock(pThe_path);
|
|
remove(pThe_path);
|
|
rename(new_file, pThe_path);
|
|
}
|
|
|
|
// IDA: void __usercall EncodeFileWrapper(char *pThe_path@<EAX>)
|
|
void EncodeFileWrapper(char* pThe_path) {
|
|
int len;
|
|
LOG_TRACE("(\"%s\")", pThe_path);
|
|
|
|
len = strlen(pThe_path);
|
|
|
|
// if file doesn't end in .txt, bail out
|
|
if (STR_ENDS_WITH(pThe_path, ".TXT") != 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "DKEYMAP0.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "DKEYMAP1.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "DKEYMAP2.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "DKEYMAP3.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYMAP_0.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYMAP_1.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYMAP_2.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYMAP_3.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "OPTIONS.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYNAMES.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "KEYMAP.TXT") == 0) {
|
|
return;
|
|
}
|
|
if (STR_ENDS_WITH(pThe_path, "PATHS.TXT") == 0) {
|
|
return;
|
|
}
|
|
|
|
EncodeFile(pThe_path);
|
|
}
|
|
|
|
// IDA: void __usercall EncodeAllFilesInDirectory(char *pThe_path@<EAX>)
|
|
void EncodeAllFilesInDirectory(char* pThe_path) {
|
|
char s[256];
|
|
LOG_TRACE("(\"%s\")", pThe_path);
|
|
|
|
PathCat(s, gApplication_path, pThe_path);
|
|
PDForEveryFile(s, EncodeFileWrapper);
|
|
}
|
|
|
|
// IDA: void __usercall SkipNLines(FILE *pF@<EAX>)
|
|
void SkipNLines(FILE* pF) {
|
|
int i;
|
|
int count;
|
|
char s[256];
|
|
LOG_TRACE("(%p)", pF);
|
|
|
|
count = GetAnInt(pF);
|
|
for (i = 0; i < count; i++) {
|
|
GetALineAndDontArgue(pF, s);
|
|
}
|
|
}
|
|
|
|
// IDA: int __usercall DRStricmp@<EAX>(char *p1@<EAX>, char *p2@<EDX>)
|
|
int DRStricmp(char* p1, char* p2) {
|
|
int val;
|
|
while (p1) {
|
|
val = tolower(*p1) - tolower(*p2);
|
|
if (val != 0) {
|
|
return val;
|
|
}
|
|
p1++;
|
|
p2++;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
// IDA: void __usercall GlorifyMaterial(br_material **pArray@<EAX>, int pCount@<EDX>)
|
|
void GlorifyMaterial(br_material** pArray, int pCount) {
|
|
int i;
|
|
int c;
|
|
br_pixelmap* big_tile;
|
|
tException_list e;
|
|
LOG_TRACE("(%p, %d)", pArray, pCount);
|
|
|
|
// Added by dethrace.
|
|
// `GlorifyMaterial` is only present in the 3dfx patch.
|
|
// If software mode, don't glorify, otherwise it puts the software renderer into lit mode
|
|
// See `WhitenVertexRGB` for a similar check that is present in the original code
|
|
if (!harness_game_config.opengl_3dfx_mode) {
|
|
return;
|
|
}
|
|
// <<<
|
|
|
|
for (i = 0; i < pCount; i++) {
|
|
if (pArray[i]->colour_map != NULL) {
|
|
e = FindExceptionInList(pArray[i]->colour_map->identifier, gExceptions);
|
|
|
|
if (gInterpolate_textures) {
|
|
// use linear texture filtering unless we have a "nobilinear" flag or the texture has transparent parts
|
|
if ((e == NULL || (e->flags & ExceptionFlag_NoBilinear) == 0)
|
|
&& !DRPixelmapHasZeros(pArray[i]->colour_map)) {
|
|
pArray[i]->flags |= BR_MATF_MAP_INTERPOLATION;
|
|
}
|
|
}
|
|
if (gUse_mip_maps) {
|
|
pArray[i]->flags |= BR_MATF_MAP_ANTIALIASING;
|
|
}
|
|
if (gPerspective_is_fast) {
|
|
pArray[i]->flags |= BR_MATF_PERSPECTIVE;
|
|
}
|
|
if (e && (e->flags & ExceptionFlag_Double)) {
|
|
pArray[i]->map_transform.m[0][0] = 0.5f;
|
|
pArray[i]->map_transform.m[1][1] = 0.5f;
|
|
} else if (e && (e->flags & ExceptionFlag_Quadruple)) {
|
|
pArray[i]->map_transform.m[0][0] = 0.25f;
|
|
pArray[i]->map_transform.m[1][1] = 0.25f;
|
|
}
|
|
} else {
|
|
c = pArray[i]->index_base + pArray[i]->index_range / 2;
|
|
pArray[i]->colour = ((br_colour*)gRender_palette->pixels)[c];
|
|
}
|
|
pArray[i]->ka = 1.0f;
|
|
pArray[i]->kd = 0.0f;
|
|
pArray[i]->ks = 0.0f;
|
|
pArray[i]->flags &= ~BR_MATF_PRELIT;
|
|
pArray[i]->flags |= BR_MATF_LIGHT;
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall WhitenVertexRGB(br_model **pArray@<EAX>, int pN@<EDX>)
|
|
void WhitenVertexRGB(br_model** pArray, int pN) {
|
|
int m;
|
|
int v;
|
|
br_vertex* vertex;
|
|
LOG_TRACE("(%p, %d)", pArray, pN);
|
|
|
|
if (gScreen && gScreen->type != BR_PMT_INDEX_8 && pN > 0) {
|
|
for (m = 0; m < pN; m++) {
|
|
vertex = pArray[m]->vertices;
|
|
for (v = 0; v < pArray[m]->nvertices; v++, vertex++) {
|
|
vertex->red = 255;
|
|
vertex->grn = 255;
|
|
vertex->blu = 255;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall NobbleNonzeroBlacks(br_pixelmap *pPalette@<EAX>)
|
|
void NobbleNonzeroBlacks(br_pixelmap* pPalette) {
|
|
tU32 red;
|
|
tU32 green;
|
|
tU32 blue;
|
|
tU32 value;
|
|
tU32* palette_entry;
|
|
tU32 frobbed;
|
|
LOG_TRACE("(%p)", pPalette);
|
|
|
|
int i;
|
|
|
|
palette_entry = pPalette->pixels;
|
|
frobbed = 0;
|
|
if (*palette_entry != 0) {
|
|
*palette_entry = 0;
|
|
frobbed = 1;
|
|
}
|
|
palette_entry++;
|
|
for (i = 1; i < 256; i++) {
|
|
blue = (*palette_entry >> 16) & 0xff;
|
|
green = (*palette_entry >> 8) & 0xff;
|
|
red = (*palette_entry) & 0xff;
|
|
if (blue == 0 && green == 0 && red == 0) {
|
|
frobbed = 1;
|
|
*palette_entry = 0x010101;
|
|
}
|
|
palette_entry++;
|
|
}
|
|
if (frobbed) {
|
|
BrMapUpdate(pPalette, BR_MAPU_ALL);
|
|
}
|
|
}
|
|
|
|
// IDA: int __usercall PDCheckDriveExists@<EAX>(char *pThe_path@<EAX>)
|
|
int PDCheckDriveExists(char* pThe_path) {
|
|
LOG_TRACE9("(\"%s\")", pThe_path);
|
|
|
|
return PDCheckDriveExists2(pThe_path, NULL, 0);
|
|
}
|
|
|
|
// IDA: int __usercall OpacityInPrims@<EAX>(br_token_value *pPrims@<EAX>)
|
|
int OpacityInPrims(br_token_value* pPrims) {
|
|
LOG_TRACE("(%p)", pPrims);
|
|
|
|
for (; pPrims->t != 0 && pPrims->t != BRT_OPACITY_X; pPrims++) {
|
|
}
|
|
return pPrims->t != 0;
|
|
}
|
|
|
|
// IDA: int __usercall AlreadyBlended@<EAX>(br_material *pMaterial@<EAX>)
|
|
int AlreadyBlended(br_material* pMaterial) {
|
|
LOG_TRACE("(%p)", pMaterial);
|
|
|
|
if (pMaterial->index_blend != NULL) {
|
|
return 1;
|
|
}
|
|
if (pMaterial->extra_prim == NULL) {
|
|
return 0;
|
|
}
|
|
return OpacityInPrims(pMaterial->extra_prim);
|
|
}
|
|
|
|
// IDA: void __usercall BlendifyMaterialTablishly(br_material *pMaterial@<EAX>, int pPercent@<EDX>)
|
|
void BlendifyMaterialTablishly(br_material* pMaterial, int pPercent) {
|
|
char* s = NULL;
|
|
LOG_TRACE("(%p, %d)", pMaterial, pPercent);
|
|
|
|
switch (pPercent) {
|
|
case 25:
|
|
s = "BLEND75.TAB";
|
|
break;
|
|
case 50:
|
|
s = "BLEND50.TAB";
|
|
break;
|
|
case 75:
|
|
s = "BLEND25.TAB";
|
|
break;
|
|
default:
|
|
PDFatalError("Invalid alpha");
|
|
break;
|
|
}
|
|
pMaterial->index_blend = BrTableFind(s);
|
|
if (pMaterial->index_blend == NULL) {
|
|
pMaterial->index_blend = LoadSingleShadeTable(&gTrack_storage_space, s);
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall BlendifyMaterialPrimitively(br_material *pMaterial@<EAX>, int pPercent@<EDX>)
|
|
void BlendifyMaterialPrimitively(br_material* pMaterial, int pPercent) {
|
|
|
|
static br_token_value alpha25[3] = {
|
|
{ BRT_BLEND_B, { .b = 1 } },
|
|
{ BRT_OPACITY_X, { .x = 0x400000 } },
|
|
{ 0 },
|
|
};
|
|
static br_token_value alpha50[3] = {
|
|
{ BRT_BLEND_B, { .b = 1 } },
|
|
{ BRT_OPACITY_X, { .x = 0x800000 } },
|
|
{ 0 },
|
|
};
|
|
static br_token_value alpha75[3] = {
|
|
{ BRT_BLEND_B, { .b = 1 } },
|
|
{ BRT_OPACITY_X, { .x = 0xc00000 } },
|
|
{ 0 },
|
|
};
|
|
LOG_TRACE("(%p, %d)", pMaterial, pPercent);
|
|
|
|
switch (pPercent) {
|
|
case 25:
|
|
pMaterial->extra_prim = alpha25;
|
|
break;
|
|
case 50:
|
|
pMaterial->extra_prim = alpha50;
|
|
break;
|
|
case 75:
|
|
pMaterial->extra_prim = alpha75;
|
|
break;
|
|
default:
|
|
PDFatalError("Invalid alpha");
|
|
}
|
|
}
|
|
|
|
// IDA: void __usercall BlendifyMaterial(br_material *pMaterial@<EAX>, int pPercent@<EDX>)
|
|
void BlendifyMaterial(br_material* pMaterial, int pPercent) {
|
|
LOG_TRACE("(%p, %d)", pMaterial, pPercent);
|
|
|
|
if (gScreen->type == BR_PMT_INDEX_8) {
|
|
BlendifyMaterialTablishly(pMaterial, pPercent);
|
|
} else {
|
|
BlendifyMaterialPrimitively(pMaterial, pPercent);
|
|
}
|
|
}
|