Created
August 20, 2021 11:07
-
-
Save xDShot/04f2149af9b267251d3f9e6e77006a31 to your computer and use it in GitHub Desktop.
quake64 loader
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "kexlib.h" | |
#include "quakeex.h" | |
/* | |
============================================================================= | |
DECOMPRESSION | |
============================================================================= | |
*/ | |
int wad_numlumps2; | |
lumpinfo_t* wad_lumps2; | |
byte* wad_base2; | |
/*=======*/ | |
/* TYPES */ | |
/*=======*/ | |
typedef struct { | |
int var0; | |
int var1; | |
int var2; | |
int var3; | |
kexArray<uint8_t> write; | |
byte *read; | |
byte *readPos; | |
} decoder_t; | |
/*=========*/ | |
/* GLOBALS */ | |
/*=========*/ | |
static short ShiftTable[6] = {4, 6, 8, 10, 12, 14}; // 8005D8A0 | |
static int tableVar01[18]; // 800B2250 | |
static short *PtrEvenTbl; // 800B2298 | |
static short *PtrOddTbl; // 800B229C | |
static short *PtrNumTbl1; // 800B22A0 | |
static short *PtrNumTbl2; // 800B22A4 | |
static short DecodeTable[2524]; // 800B22A8 | |
static short array01[1258]; // 800B3660 | |
static decoder_t decoder; // 800B4034 | |
static byte *allocPtr; // 800B4054 | |
static int OVERFLOW_READ; // 800B4058 | |
static int OVERFLOW_WRITE; // 800B405C | |
/* | |
============================================================================ | |
DECODE BASED ROUTINES | |
============================================================================ | |
*/ | |
/* | |
======================== | |
= | |
= GetDecodeByte | |
= | |
======================== | |
*/ | |
static byte GetDecodeByte(void) // 8002D1D0 | |
{ | |
if((int)(decoder.readPos - decoder.read) >= OVERFLOW_READ) | |
{ | |
return -1; | |
} | |
return *decoder.readPos++; | |
} | |
/* | |
======================== | |
= | |
= WriteOutput | |
= | |
======================== | |
*/ | |
static void WriteOutput(byte outByte) // 8002D214 | |
{ | |
decoder.write.Push(outByte); | |
} | |
/* | |
======================== | |
= | |
= WriteBinary | |
= routine required for encoding | |
= | |
======================== | |
*/ | |
static void WriteBinary(int binary) // 8002D288 | |
{ | |
decoder.var3 = (decoder.var3 << 1); | |
if(binary != 0) | |
{ | |
decoder.var3 = (decoder.var3 | 1); | |
} | |
decoder.var2 = (decoder.var2 + 1); | |
if(decoder.var2 == 8) | |
{ | |
WriteOutput((byte)decoder.var3); | |
decoder.var2 = 0; | |
} | |
} | |
/* | |
======================== | |
= | |
= DecodeScan | |
= | |
======================== | |
*/ | |
static int DecodeScan(void) // 8002D2F4 | |
{ | |
int resultbyte; | |
resultbyte = decoder.var0; | |
decoder.var0 = (resultbyte - 1); | |
if((resultbyte < 1)) | |
{ | |
resultbyte = GetDecodeByte(); | |
decoder.var1 = resultbyte; | |
decoder.var0 = 7; | |
} | |
resultbyte = (0 < (decoder.var1 & 0x80)); | |
decoder.var1 = (decoder.var1 << 1); | |
return resultbyte; | |
} | |
/* | |
======================== | |
= | |
= MakeExtraBinary | |
= routine required for encoding | |
= | |
======================== | |
*/ | |
static void MakeExtraBinary(int binary, int shift) // 8002D364 | |
{ | |
int i; | |
i = 0; | |
if(shift > 0) | |
{ | |
do | |
{ | |
WriteBinary(binary & 1); | |
binary = (binary >> 1); | |
} while(++i != shift); | |
} | |
} | |
/* | |
======================== | |
= | |
= RescanByte | |
= | |
======================== | |
*/ | |
static int RescanByte(int byte) // 8002D3B8 | |
{ | |
int shift; | |
int i; | |
int resultbyte; | |
resultbyte = 0; | |
i = 0; | |
shift = 1; | |
if(byte <= 0) | |
return resultbyte; | |
do | |
{ | |
if(DecodeScan() != 0) | |
{ | |
resultbyte |= shift; | |
} | |
i++; | |
shift = (shift << 1); | |
} while(i != byte); | |
return resultbyte; | |
} | |
/* | |
======================== | |
= | |
= WriteEndCode | |
= routine required for encoding | |
= | |
======================== | |
*/ | |
static void WriteEndCode(void) // 8002D424 | |
{ | |
if(decoder.var2 > 0) | |
{ | |
WriteOutput((byte)(decoder.var3 << (8 - decoder.var2)) & 0xff); | |
} | |
} | |
/* | |
======================== | |
= | |
= InitDecodeTable | |
= | |
======================== | |
*/ | |
static void InitDecodeTable(void) // 8002D468 | |
{ | |
int evenVal, oddVal, incrVal; | |
short *curArray; | |
short *incrTbl; | |
short *evenTbl; | |
short *oddTbl; | |
tableVar01[15] = 3; | |
tableVar01[16] = 0; | |
tableVar01[17] = 0; | |
decoder.var0 = 0; | |
decoder.var1 = 0; | |
decoder.var2 = 0; | |
decoder.var3 = 0; | |
curArray = &array01[2]; | |
incrTbl = &DecodeTable[0x4F2]; | |
incrVal = 2; | |
do | |
{ | |
if(incrVal < 0) | |
{ | |
*incrTbl = (short)((incrVal + 1) >> 1); | |
} | |
else | |
{ | |
*incrTbl = (short)(incrVal >> 1); | |
} | |
*curArray++ = 1; | |
incrTbl++; | |
} while(++incrVal < 1258); | |
oddTbl = &DecodeTable[0x279]; | |
evenTbl = &DecodeTable[1]; | |
evenVal = 2; | |
oddVal = 3; | |
do | |
{ | |
*oddTbl++ = (short)oddVal; | |
oddVal += 2; | |
*evenTbl++ = (short)evenVal; | |
evenVal += 2; | |
} while(oddVal < 1259); | |
tableVar01[0] = 0; | |
incrVal = (1 << ShiftTable[0]); | |
tableVar01[6] = (incrVal - 1); | |
tableVar01[1] = incrVal; | |
incrVal += (1 << ShiftTable[1]); | |
tableVar01[7] = (incrVal - 1); | |
tableVar01[2] = incrVal; | |
incrVal += (1 << ShiftTable[2]); | |
tableVar01[8] = (incrVal - 1); | |
tableVar01[3] = incrVal; | |
incrVal += (1 << ShiftTable[3]); | |
tableVar01[9] = (incrVal - 1); | |
tableVar01[4] = incrVal; | |
incrVal += (1 << ShiftTable[4]); | |
tableVar01[10] = (incrVal - 1); | |
tableVar01[5] = incrVal; | |
incrVal += (1 << ShiftTable[5]); | |
tableVar01[11] = (incrVal - 1); | |
tableVar01[12] = (incrVal - 1); | |
tableVar01[13] = tableVar01[12] + 64; | |
} | |
/* | |
======================== | |
= | |
= CheckTable | |
= | |
======================== | |
*/ | |
static void CheckTable(int a0,int a1,int a2) // 8002D624 | |
{ | |
int i; | |
int idByte1; | |
int idByte2; | |
short *curArray; | |
short *evenTbl; | |
short *oddTbl; | |
short *incrTbl; | |
i = 0; | |
evenTbl = &DecodeTable[0]; | |
oddTbl = &DecodeTable[0x278]; | |
incrTbl = &DecodeTable[0x4F0]; | |
idByte1 = a0; | |
do | |
{ | |
idByte2 = incrTbl[idByte1]; | |
array01[idByte2] = (array01[a1] + array01[a0]); | |
a0 = idByte2; | |
if(idByte2 != 1) | |
{ | |
idByte1 = incrTbl[idByte2]; | |
idByte2 = evenTbl[idByte1]; | |
a1 = idByte2; | |
if(a0 == idByte2) | |
{ | |
a1 = oddTbl[idByte1]; | |
} | |
} | |
idByte1 = a0; | |
}while(a0 != 1); | |
if(array01[1] != 0x7D0) | |
{ | |
return; | |
} | |
array01[1] >>= 1; | |
curArray = &array01[2]; | |
do | |
{ | |
curArray[3] >>= 1; | |
curArray[2] >>= 1; | |
curArray[1] >>= 1; | |
curArray[0] >>= 1; | |
curArray += 4; | |
i += 4; | |
} while(i != 1256); | |
} | |
/* | |
======================== | |
= | |
= DecodeByte | |
= | |
======================== | |
*/ | |
static void DecodeByte(int tblpos) // 8002D72C | |
{ | |
int incrIdx; | |
int evenVal; | |
int idByte1; | |
int idByte2; | |
int idByte3; | |
int idByte4; | |
short *evenTbl; | |
short *oddTbl; | |
short *incrTbl; | |
short *tmpIncrTbl; | |
evenTbl = &DecodeTable[0]; | |
oddTbl = &DecodeTable[0x278]; | |
incrTbl = &DecodeTable[0x4F0]; | |
idByte1 = (tblpos + 0x275); | |
array01[idByte1] += 1; | |
if(incrTbl[idByte1] != 1) | |
{ | |
tmpIncrTbl = &incrTbl[idByte1]; | |
idByte2 = *tmpIncrTbl; | |
if(idByte1 == evenTbl[idByte2]) | |
{ | |
CheckTable(idByte1, oddTbl[idByte2], idByte1); | |
} | |
else | |
{ | |
CheckTable(idByte1, evenTbl[idByte2], idByte1); | |
} | |
do | |
{ | |
incrIdx = incrTbl[idByte2]; | |
evenVal = evenTbl[incrIdx]; | |
if(idByte2 == evenVal) | |
{ | |
idByte3 = oddTbl[incrIdx]; | |
} | |
else | |
{ | |
idByte3 = evenVal; | |
} | |
if(array01[idByte3] < array01[idByte1]) | |
{ | |
if(idByte2 == evenVal) | |
{ | |
oddTbl[incrIdx] = (short)idByte1; | |
} | |
else | |
{ | |
evenTbl[incrIdx] = (short)idByte1; | |
} | |
evenVal = evenTbl[idByte2]; | |
if(idByte1 == evenVal) | |
{ | |
idByte4 = oddTbl[idByte2]; | |
evenTbl[idByte2] = (short)idByte3; | |
} | |
else | |
{ | |
idByte4 = evenVal; | |
oddTbl[idByte2] = (short)idByte3; | |
} | |
incrTbl[idByte3] = (short)idByte2; | |
*tmpIncrTbl = (short)incrIdx; | |
CheckTable(idByte3, idByte4, idByte4); | |
tmpIncrTbl = &incrTbl[idByte3]; | |
} | |
idByte1 = *tmpIncrTbl; | |
tmpIncrTbl = &incrTbl[idByte1]; | |
idByte2 = *tmpIncrTbl; | |
} while (idByte2 != 1); | |
} | |
} | |
/* | |
======================== | |
= | |
= StartDecodeByte | |
= | |
======================== | |
*/ | |
static int StartDecodeByte(void) // 8002D904 | |
{ | |
int lookup; | |
short *evenTbl; | |
short *oddTbl; | |
lookup = 1; | |
evenTbl = &DecodeTable[0]; | |
oddTbl = &DecodeTable[0x278]; | |
while(lookup < 0x275) | |
{ | |
if(DecodeScan() == 0) | |
{ | |
lookup = evenTbl[lookup]; | |
} | |
else | |
{ | |
lookup = oddTbl[lookup]; | |
} | |
} | |
lookup = (lookup + -0x275); | |
DecodeByte(lookup); | |
return lookup; | |
} | |
/* | |
======================== | |
= | |
= DecodeD64 | |
= | |
= Exclusive Doom 64 | |
= | |
======================== | |
*/ | |
void DecodeD64(uint8_t *input) // 8002DFA0 | |
{ | |
int copyPos, storePos; | |
int dec_byte, resc_byte; | |
int incrBit, copyCnt, shiftPos, j; | |
InitDecodeTable(); | |
OVERFLOW_READ = D_MAXINT; | |
OVERFLOW_WRITE = D_MAXINT; | |
incrBit = 0; | |
decoder.read = input; | |
decoder.readPos = input; | |
decoder.write.Empty(); | |
allocPtr = (byte*)Z_Malloc(tableVar01[13]); | |
dec_byte = StartDecodeByte(); | |
while(dec_byte != 256) | |
{ | |
if(dec_byte < 256) | |
{ | |
/* Decode the data directly using binary data code */ | |
WriteOutput((byte)(dec_byte & 0xff)); | |
allocPtr[incrBit] = (byte)dec_byte; | |
/* Resets the count once the memory limit is exceeded in allocPtr, | |
so to speak resets it at startup for reuse */ | |
incrBit += 1; | |
if(incrBit == tableVar01[13]) | |
{ | |
incrBit = 0; | |
} | |
} | |
else | |
{ | |
/* Decode the data using binary data code, | |
a count is obtained for the repeated data, | |
positioning itself in the root that is being stored in allocPtr previously. */ | |
/* A number is obtained from a range from 0 to 5, | |
necessary to obtain a shift value in the ShiftTable*/ | |
shiftPos = (dec_byte + -257) / 62; | |
/* get a count number for data to copy */ | |
copyCnt = (dec_byte - (shiftPos * 62)) + -254; | |
/* To start copying data, you receive a position number | |
that you must sum with the position of table tableVar01 */ | |
resc_byte = RescanByte(ShiftTable[shiftPos]); | |
/* with this formula the exact position is obtained | |
to start copying previously stored data */ | |
copyPos = incrBit - ((tableVar01[shiftPos] + resc_byte) + copyCnt); | |
if(copyPos < 0) | |
{ | |
copyPos += tableVar01[13]; | |
} | |
storePos = incrBit; | |
for(j = 0; j < copyCnt; j++) | |
{ | |
/* write the copied data */ | |
WriteOutput(allocPtr[copyPos]); | |
/* save copied data at current position in memory allocPtr */ | |
allocPtr[storePos] = allocPtr[copyPos]; | |
storePos++; /* advance to next allocPtr memory block to store */ | |
copyPos++; /* advance to next allocPtr memory block to copy */ | |
/* reset the position of storePos once the memory limit is exceeded */ | |
if(storePos == tableVar01[13]) | |
{ | |
storePos = 0; | |
} | |
/* reset the position of copyPos once the memory limit is exceeded */ | |
if(copyPos == tableVar01[13]) | |
{ | |
copyPos = 0; | |
} | |
} | |
/* Resets the count once the memory limit is exceeded in allocPtr, | |
so to speak resets it at startup for reuse */ | |
incrBit += copyCnt; | |
if(incrBit >= tableVar01[13]) | |
{ | |
incrBit -= tableVar01[13]; | |
} | |
} | |
dec_byte = StartDecodeByte(); | |
} | |
Z_Free(allocPtr); | |
} | |
/* | |
============= | |
W_GetLumpinfo2 | |
============= | |
*/ | |
lumpinfo_t* W_GetLumpinfo2(char* name) | |
{ | |
int i; | |
lumpinfo_t* lump_p; | |
char clean[16]; | |
W_CleanupName(name, clean); | |
for(lump_p=wad_lumps2, i=0 ; i<wad_numlumps2 ; i++,lump_p++) | |
{ | |
if(!strcmp(clean, lump_p->name)) | |
{ | |
return lump_p; | |
} | |
} | |
Sys_Error("W_GetLumpinfo2: %s not found", name); | |
return nullptr; | |
} | |
/* | |
============= | |
W_GetLumpName2 | |
============= | |
*/ | |
void* W_GetLumpName2(char* name) | |
{ | |
lumpinfo_t* lump; | |
lump = W_GetLumpinfo2(name); | |
return (void*)(wad_base2 + lump->filepos); | |
} | |
/* | |
============= | |
W_GetLumpNum2 | |
============= | |
*/ | |
void* W_GetLumpNum2(int num) | |
{ | |
lumpinfo_t* lump; | |
if(num < 0 || num > wad_numlumps2) | |
{ | |
Sys_Error("W_GetLumpNum2: bad number: %i", num); | |
} | |
lump = wad_lumps2 + num; | |
return (void*)(wad_base2 + lump->filepos); | |
} | |
/* | |
============= | |
W_CheckNumForName2 | |
Returns -1 if name not found. | |
============= | |
*/ | |
int W_CheckNumForName2(const char* name) | |
{ | |
int i = -1; | |
lumpinfo_t* lump_p; | |
char clean[16]; | |
W_CleanupName(name, clean); | |
for(lump_p=wad_lumps2, i=0 ; i<wad_numlumps2 ; i++,lump_p++) | |
{ | |
if(!kexStr::StrCaseCmp(clean, lump_p->name)) | |
{ | |
return i; | |
} | |
} | |
return -1; | |
} | |
/* | |
============= | |
W_GetNumForName2 | |
Calls W_CheckNumForName, but bombs out if not found. | |
============= | |
*/ | |
int W_GetNumForName2(const char* name) | |
{ | |
int i; | |
i = W_CheckNumForName2(name); | |
if(i == -1) | |
{ | |
kexError("W_GetNumForName2: %s not found!", name); | |
} | |
return i; | |
} | |
/* | |
==================== | |
W_LoadWadFile2 | |
==================== | |
*/ | |
void W_LoadWadFile2(const char* filename) | |
{ | |
lumpinfo_t* lump_p; | |
wadinfo_t* header; | |
int i; | |
int infotableofs; | |
kexPrintf("W_LoadWadFile2: Adding %s...\n", filename); | |
wad_base2 = COM_LoadHunkFile(filename); | |
if(!wad_base2) | |
{ | |
kexWarning("W_LoadWadFile2: couldn't load %s\n", filename); | |
return; | |
} | |
header = (wadinfo_t*)wad_base2; | |
if(header->identification[0] != 'W' | |
|| header->identification[1] != 'A' | |
|| header->identification[2] != 'D' | |
|| header->identification[3] != '2') | |
{ | |
kexWarning("Wad file %s doesn't have WAD2 id\n",filename); | |
return; | |
} | |
wad_numlumps2 = kexEndian::SwapLE32(header->numlumps); | |
infotableofs = kexEndian::SwapLE32(header->infotableofs); | |
wad_lumps2 = (lumpinfo_t*)(wad_base2 + infotableofs); | |
for(i=0, lump_p = wad_lumps ; i<wad_numlumps2 ; i++,lump_p++) | |
{ | |
lump_p->filepos = kexEndian::SwapLE32(lump_p->filepos); | |
lump_p->size = kexEndian::SwapLE32(lump_p->size); | |
lump_p->disksize = kexEndian::SwapLE32(lump_p->disksize); | |
W_CleanupName(lump_p->name, lump_p->name); | |
} | |
} | |
struct qTexInfoEntry_s | |
{ | |
qTexInfoEntry_s() : | |
index(-1), | |
n64texlump(-1), | |
dwOffsetToDataBlob(uint32_t(-1)), | |
dwFileSize(0) | |
{ | |
} | |
int32_t index; | |
int32_t n64texlump; | |
uint32_t dwOffsetToDataBlob; | |
size_t dwFileSize; | |
}; | |
static void DescrambleTexture(byte* pData, const int width, const int height) | |
{ | |
const int size = (width * height)/2; | |
constexpr int mask = 1; | |
byte* pRover = pData; | |
for(int i = 0; i < height; ++i) | |
{ | |
kexAssert(pRover < (pData + (width * height))); | |
if(i & mask) | |
{ | |
for(int x = 0; x < width; x += 4) | |
{ | |
int* pTmp = (int*)&pRover[x]; | |
*pTmp = kexEndian::SwapBE32(*pTmp); | |
} | |
} | |
pRover += width; | |
} | |
pRover = pData; | |
for(int i = 0; i < height; ++i) | |
{ | |
kexAssert(pRover < (pData + (width * height))); | |
if(i & mask) | |
{ | |
for(int x = 0; x < width; x += 2) | |
{ | |
int16_t* pTmp = (int16_t*)&pRover[x]; | |
*pTmp = kexEndian::SwapBE16(*pTmp); | |
} | |
} | |
pRover += width; | |
} | |
} | |
static void BuildQuake64BSPLevel(const kexStr& strOutputPath, const int lump) | |
{ | |
uint32_t dwDataPos[HEADER_LUMPS]; | |
uint32_t dwDataSize[HEADER_LUMPS]; | |
kexArray<byte> nLevelDataBlob; | |
kexArray<byte> nTexturesDataBlob; | |
kexBufferStreamGrowable cTexturesDataLump; | |
kexTMap<qTexInfoEntry_s> nTexEntryMap; | |
int32_t iCurrentTextureID; | |
kexMemclr(dwDataPos); | |
kexMemclr(dwDataSize); | |
const int t_start = W_GetNumForName2("t_start")+1; | |
const int scanLump = lump+1; | |
iCurrentTextureID = 0; | |
for(int i = scanLump; i < scanLump+14; ++i) | |
{ | |
lumpinfo_t* pLump = &wad_lumps2[i]; | |
DecodeD64((byte*)W_GetLumpNum2(i)); | |
kexArray<byte> nData = decoder.write; | |
int index = (i - lump)-1; | |
if(index < HEADER_LUMPS) | |
{ | |
if(index >= LUMP_TEXTURES) | |
{ | |
index++; | |
} | |
if(index == LUMP_TEXINFO) | |
{ | |
texinfo_t* pTexInfo = (texinfo_t*)nData.GetDataPtr(); | |
size_t count = pLump->size / sizeof(*pTexInfo); | |
for(int t = 0; t < count; ++t) | |
{ | |
int texid = pTexInfo[t].miptex; | |
if(texid == -1) | |
{ | |
continue; | |
} | |
lumpinfo_t* pTexLump = &wad_lumps2[t_start+texid]; | |
kexStr strTextureName = pTexLump->name; | |
qTexInfoEntry_s* pEntry = nTexEntryMap.FindAdd(strTextureName); | |
if(pEntry->index == -1) | |
{ | |
DecodeD64((byte*)W_GetLumpNum2(t_start+texid)); | |
kexArray<byte> nTexData = decoder.write; | |
byte* pTexData = nTexData.GetDataPtr(); | |
int16_t* pTmp = (int16_t*)pTexData; | |
int width = kexEndian::SwapBE16(pTmp[0]); | |
int height = kexEndian::SwapBE16(pTmp[1]); | |
int shift = kexEndian::SwapBE16(pTmp[2]); | |
int oldwidth = width; | |
int oldheight = height; | |
width >>= shift; | |
height >>= shift; | |
if(pTexLump->name[0] != '*') | |
{ | |
DescrambleTexture(pTexData+8, width, height); | |
} | |
kexImage cImage(pTexData+8, width, height, RTPF_R8); | |
const size_t dwMipTexSize = (44 + (width*height)); | |
pEntry->dwOffsetToDataBlob = nTexturesDataBlob.Length(); | |
nTexturesDataBlob.Resize(nTexturesDataBlob.Length() + dwMipTexSize); | |
kexBufferStreamDirect cTexWrite(&nTexturesDataBlob[pEntry->dwOffsetToDataBlob], dwMipTexSize); | |
for(size_t c = 0; c < strTextureName.Length(); ++c) | |
{ | |
cTexWrite.Write8(strTextureName[c]); | |
} | |
for(size_t c = strTextureName.Length(); c < strTextureName.Length()+(16-strTextureName.Length()); ++c) | |
{ | |
cTexWrite.Write8(0); | |
} | |
cTexWrite.Write32(width); | |
cTexWrite.Write32(height); | |
cTexWrite.Write32(shift); | |
cTexWrite.Write32(44); | |
cTexWrite.Write32(44); | |
cTexWrite.Write32(44); | |
cTexWrite.Write32(44); | |
for(int c = 0; c < (width*height); ++c) | |
{ | |
cTexWrite.Write8(cImage.Data()[c]); | |
} | |
pEntry->n64texlump = t_start+texid; | |
pEntry->index = iCurrentTextureID; | |
pEntry->dwFileSize = dwMipTexSize; | |
iCurrentTextureID++; | |
} | |
} | |
for(int t = 0; t < count; ++t) | |
{ | |
int texid = pTexInfo[t].miptex; | |
lumpinfo_t* pTexLump = &wad_lumps2[t_start+texid]; | |
kexStr strTextureName = pTexLump->name; | |
qTexInfoEntry_s* pEntry = nTexEntryMap.GetValue(strTextureName); | |
if(!pEntry || pEntry->index <= -1) | |
{ | |
pTexInfo[t].miptex = -1; | |
continue; | |
} | |
pTexInfo[t].miptex = pEntry->index; | |
} | |
cTexturesDataLump.Write32(iCurrentTextureID); | |
for(int t = 0; t < iCurrentTextureID; ++t) | |
{ | |
typename kexTMap<qTexInfoEntry_s>::Iterator cItr(nTexEntryMap); | |
typename kexTMap<qTexInfoEntry_s>::hashType_t* pKey = nullptr; | |
while((pKey = cItr.GetNext())) | |
{ | |
if(pKey->GetValue().index == t) | |
{ | |
cTexturesDataLump.WriteU32(pKey->GetValue().dwOffsetToDataBlob + (4*iCurrentTextureID) + 4); | |
break; | |
} | |
} | |
} | |
for(const auto& someByte : nTexturesDataBlob) | |
{ | |
cTexturesDataLump.Write8(someByte); | |
} | |
kexAutoFileWrite cTexturesWrite(kexStr::Format("%s/%sTEXTURES.lmp", strOutputPath.c_str(), wad_lumps2[lump].name)); | |
if(cTexturesWrite.IsValid()) | |
{ | |
cTexturesWrite->Write(cTexturesDataLump.Buffer(), cTexturesDataLump.BufferLength()); | |
} | |
} | |
dwDataPos[index] = nLevelDataBlob.Length() + 124; | |
dwDataSize[index] = pLump->size; | |
for(int j = 0; j < pLump->size; ++j) | |
{ | |
nLevelDataBlob.Push(nData[j]); | |
} | |
} | |
} | |
kexAutoFileWrite cBspWrite(kexStr::Format("%s/%s.bsp", strOutputPath.c_str(), wad_lumps2[lump].name)); | |
if(cBspWrite.IsValid()) | |
{ | |
int32_t iOffsetFromTextureData = 0; | |
int32_t iScratch = BSPVERSION_QUAKE64; | |
cBspWrite->Write((byte*)&iScratch, 4); | |
for(int j = 0; j < HEADER_LUMPS; ++j) | |
{ | |
if(j == LUMP_TEXTURES) | |
{ | |
iOffsetFromTextureData = cTexturesDataLump.BufferLength(); | |
iScratch = dwDataPos[j+1]; | |
cBspWrite->Write((byte*)&iScratch, 4); | |
iScratch -= 124; | |
nLevelDataBlob.Insert(iScratch, cTexturesDataLump.BufferLength()); | |
kexMemcpy(&nLevelDataBlob[iScratch], cTexturesDataLump.Buffer(), cTexturesDataLump.BufferLength()); | |
iScratch = (int32_t)cTexturesDataLump.BufferLength(); | |
cBspWrite->Write((byte*)&iScratch, 4); | |
continue; | |
} | |
iScratch = dwDataPos[j]; | |
iScratch += iOffsetFromTextureData; | |
cBspWrite->Write((byte*)&iScratch, 4); | |
cBspWrite->Write((byte*)&dwDataSize[j], 4); | |
} | |
cBspWrite->Write(nLevelDataBlob.GetDataPtr(), nLevelDataBlob.Length()); | |
} | |
} | |
COMMAND(loadquake64wad) | |
{ | |
W_LoadWadFile2("Quake64.wad"); | |
if(!wad_base2 || wad_numlumps2 <= 0) | |
{ | |
return; | |
} | |
kexStr strOutputPath = kexStr::FormatPath("q64output"); | |
if(!kexPlatform::cFile->MakeDirectory(strOutputPath.c_str())) | |
{ | |
return; | |
} | |
for(int i = 0; i< wad_numlumps2 ; i++) | |
{ | |
lumpinfo_t* lump = &wad_lumps2[i]; | |
if( !strcmp(lump->name, "NEND") || | |
(lump->name[0] == 'N' && | |
(lump->name[0] == 'N' && | |
lump->name[1] == 'E' && | |
(lump->name[2] >= '1' && lump->name[2] <= '9') && | |
lump->name[3] == 'M' && | |
(lump->name[4] >= '1' && lump->name[4] <= '9') && | |
lump->name[5] == '\0'))) | |
{ | |
BuildQuake64BSPLevel(strOutputPath, i); | |
continue; | |
} | |
else if(lump->size == 0) | |
{ | |
continue; | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment