2
0
mirror of https://github.com/rehlds/rehlds.git synced 2025-01-07 12:35:33 +03:00
rehlds/rehlds/engine/cmodel.cpp
2015-05-04 22:25:41 +04:00

289 lines
6.2 KiB
C++

/*
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* In addition, as a special exception, the author gives permission to
* link the code of this program with the Half-Life Game Engine ("HL
* Engine") and Modified Game Libraries ("MODs") developed by Valve,
* L.L.C ("Valve"). You must obey the GNU General Public License in all
* respects for all of the code used other than the HL Engine and MODs
* from Valve. If you modify this file, you may extend this exception
* to your version of the file, but you are not obligated to do so. If
* you do not wish to do so, delete this exception statement from your
* version.
*
*/
#include "precompiled.h"
unsigned char *gPAS;
unsigned char *gPVS;
int gPVSRowBytes;
unsigned char mod_novis[MODEL_MAX_PVS];
/* <812a> ../engine/cmodel.c:23 */
void Mod_Init(void)
{
SW_Mod_Init();
Q_memset(mod_novis, 255, MODEL_MAX_PVS);
}
/* <813f> ../engine/cmodel.c:37 */
unsigned char *Mod_DecompressVis(unsigned char *in, model_t *model)
{
static unsigned char decompressed[MODEL_MAX_PVS];
if (in == NULL)
{
return mod_novis;
}
int row = (model->numleafs + 7) / 8;
// TODO: Move to model loading code
if (row < 0 || row > MODEL_MAX_PVS)
{
Sys_Error(__FUNCTION__ ": oversized model->numleafs: %i", model->numleafs);
}
CM_DecompressPVS(in, decompressed, row);
return decompressed;
}
/* <824c> ../engine/cmodel.c:54 */
unsigned char *Mod_LeafPVS(mleaf_t *leaf, model_t *model)
{
if (leaf == model->leafs)
{
return mod_novis;
}
if (!gPVS)
{
return Mod_DecompressVis(leaf->compressed_vis, model);
}
int leafnum = leaf - model->leafs;
return CM_LeafPVS(leafnum);
}
/* <8358> ../engine/cmodel.c:68 */
void CM_DecompressPVS(unsigned char *in, unsigned char *decompressed, int byteCount)
{
int c;
unsigned char *out;
if (in == NULL)
{
// Make all visible
Q_memcpy(decompressed, mod_novis, byteCount);
return;
}
out = decompressed;
while (out < decompressed + byteCount)
{
// Non zero is not copmpressed
if (*in)
{
*out++ = *in++;
continue;
}
c = in[1]; // TODO: Check that input buffer is correct (last byte on the buffer could be zero and we will go out of the buffer - check the model)
in += 2;
// Prevent buffer overrun
if (c > decompressed + byteCount - out)
{
c = decompressed + byteCount - out;
}
// Unpack zeros
Q_memset(out, 0, c);
out += c;
}
}
/* <8106> ../engine/cmodel.c:100 */
unsigned char *CM_LeafPVS(int leafnum)
{
if (gPVS)
{
return &gPVS[gPVSRowBytes * leafnum];
}
return mod_novis;
}
/* <83b7> ../engine/cmodel.c:109 */
unsigned char *CM_LeafPAS(int leafnum)
{
if (gPAS)
{
return &gPAS[gPVSRowBytes * leafnum];
}
return mod_novis;
}
/* <83ec> ../engine/cmodel.c:118 */
void CM_FreePAS(void)
{
if (gPAS)
Mem_Free(gPAS);
if (gPVS)
Mem_Free(gPVS);
gPAS = 0;
gPVS = 0;
}
/* <83fd> ../engine/cmodel.c:139 */
void CM_CalcPAS(model_t *pModel)
{
int rows, rowwords;
int actualRowBytes;
int i, j, k, l;
int index;
int bitbyte;
unsigned int *dest, *src;
unsigned char *scan;
int count, vcount, acount;
Con_DPrintf("Building PAS...\n");
CM_FreePAS();
// Calculate memory: matrix of each to each leaf visibility
rows = (pModel->numleafs + 7) / 8;
count = pModel->numleafs + 1;
actualRowBytes = (rows + 3) & 0xFFFFFFFC; // 4-byte align
rowwords = actualRowBytes / 4;
gPVSRowBytes = actualRowBytes;
// Alloc PVS
gPVS = (byte *)Mem_Calloc(gPVSRowBytes, count);
// Decompress visibility data
scan = gPVS;
vcount = 0;
for (i = 0; i < count; i++, scan += gPVSRowBytes)
{
CM_DecompressPVS(pModel->leafs[i].compressed_vis, scan, rows);
if (i == 0)
{
continue;
}
for (j = 0; j < count; j++)
{
if (scan[j >> 3] & (1 << (j & 7)))
{
++vcount;
}
}
}
// Alloc PAS
gPAS = (byte *)Mem_Calloc(gPVSRowBytes, count);
// Build PAS
acount = 0;
scan = gPVS;
dest = (unsigned int *)gPAS;
for (i = 0; i < count; i++, scan += gPVSRowBytes, dest += rowwords)
{
Q_memcpy(dest, scan, gPVSRowBytes);
for (j = 0; j < gPVSRowBytes; j++) // bytes
{
bitbyte = scan[j];
if (bitbyte == 0)
{
continue;
}
for (k = 0; k < 8; k++) // bits
{
if (!(bitbyte & (1 << k)))
{
continue;
}
index = j * 8 + k + 1; // bit index
if (index >= count)
{
continue;
}
src = (unsigned int *)&gPVS[index * gPVSRowBytes];
for (l = 0; l < rowwords; l++)
{
dest[l] |= src[l];
}
}
}
if (i == 0)
{
continue;
}
for (j = 0; j < count; j++)
{
if (((byte *)dest)[j >> 3] & (1 << (j & 7)))
{
++acount;
}
}
}
Con_DPrintf("Average leaves visible / audible / total: %i / %i / %i\n", vcount / count, acount / count, count);
}
/* <858a> ../engine/cmodel.c:218 */
qboolean CM_HeadnodeVisible(mnode_t *node, unsigned char *visbits, int *first_visible_leafnum)
{
int leafnum;
mleaf_t *leaf;
leaf = (mleaf_t *)node;
while (leaf && leaf->contents != CONTENTS_SOLID)
{
if (leaf->contents < 0)
{
leafnum = leaf - g_psv.worldmodel->leafs - 1;
if ((visbits[leafnum >> 3] & (1 << (leafnum & 7))) == 0)
{
return 0;
}
if (first_visible_leafnum)
{
*first_visible_leafnum = leafnum;
}
return 1;
}
if (CM_HeadnodeVisible(((mnode_t *)leaf)->children[0], visbits, first_visible_leafnum))
{
return 1;
}
leaf = (mleaf_t *)((mnode_t *)leaf)->children[1];
}
return 0;
}