mirror of
https://github.com/w23/xash3d-fwgs
synced 2025-01-05 16:35:56 +01:00
5e0a0765ce
The `.editorconfig` file in this repo is configured to trim all trailing whitespace regardless of whether the line is modified. Trims all trailing whitespace in the repository to make the codebase easier to work with in editors that respect `.editorconfig`. `git blame` becomes less useful on these lines but it already isn't very useful. Commands: ``` find . -type f -name '*.h' -exec sed --in-place 's/[[:space:]]\+$//' {} \+ find . -type f -name '*.c' -exec sed --in-place 's/[[:space:]]\+$//' {} \+ ```
406 lines
9.3 KiB
C
406 lines
9.3 KiB
C
/*
|
|
Copyright (C) 1997-2001 Id Software, Inc.
|
|
|
|
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.
|
|
|
|
*/
|
|
// r_misc.c
|
|
|
|
#include "r_local.h"
|
|
|
|
#define NUM_MIPS 4
|
|
|
|
cvar_t *sw_mipcap;
|
|
cvar_t *sw_mipscale;
|
|
|
|
surfcache_t *d_initial_rover;
|
|
qboolean d_roverwrapped;
|
|
int d_minmip;
|
|
float d_scalemip[NUM_MIPS-1];
|
|
|
|
static float basemip[NUM_MIPS-1] = {1.0, 0.5*0.8, 0.25*0.8};
|
|
|
|
|
|
//int d_vrectx, d_vrecty, d_vrectright_particle, d_vrectbottom_particle;
|
|
|
|
//int d_pix_min, d_pix_max, d_pix_shift;
|
|
|
|
int d_scantable[MAXHEIGHT];
|
|
short *zspantable[MAXHEIGHT];
|
|
struct qfrustum_s qfrustum;
|
|
/*
|
|
================
|
|
D_Patch
|
|
================
|
|
*/
|
|
void D_Patch (void)
|
|
{
|
|
#if id386
|
|
extern void D_Aff8Patch( void );
|
|
static qboolean protectset8 = false;
|
|
extern void D_PolysetAff8Start( void );
|
|
|
|
if (!protectset8)
|
|
{
|
|
Sys_MakeCodeWriteable ((int)D_PolysetAff8Start,
|
|
(int)D_Aff8Patch - (int)D_PolysetAff8Start);
|
|
Sys_MakeCodeWriteable ((long)R_Surf8Start,
|
|
(long)R_Surf8End - (long)R_Surf8Start);
|
|
protectset8 = true;
|
|
}
|
|
colormap = vid.colormap;
|
|
|
|
R_Surf8Patch ();
|
|
D_Aff8Patch();
|
|
#endif
|
|
}
|
|
/*
|
|
================
|
|
D_ViewChanged
|
|
================
|
|
*/
|
|
|
|
void D_ViewChanged (void)
|
|
{
|
|
int i;
|
|
|
|
scale_for_mip = xscale;
|
|
if (yscale > xscale)
|
|
scale_for_mip = yscale;
|
|
|
|
d_zrowbytes = vid.width * 2;
|
|
d_zwidth = vid.width;
|
|
|
|
/*d_pix_min = gpGlobals->width / 320;
|
|
if (d_pix_min < 1)
|
|
d_pix_min = 1;
|
|
|
|
d_pix_max = (int)((float)gpGlobals->height / (320.0 / 4.0) + 0.5);
|
|
d_pix_shift = 8 - (int)((float)gpGlobals->height / 320.0 + 0.5);
|
|
if (d_pix_max < 1)
|
|
d_pix_max = 1;*/
|
|
|
|
//d_vrectx = RI.vrect.x;
|
|
//d_vrecty = RI.vrect.y;
|
|
//d_vrectright_particle = gpGlobals->width - d_pix_max;
|
|
//d_vrectbottom_particle =
|
|
// gpGlobals->height - d_pix_max;
|
|
|
|
for (i=0 ; i<vid.height; i++)
|
|
{
|
|
d_scantable[i] = i*r_screenwidth;
|
|
zspantable[i] = d_pzbuffer + i*d_zwidth;
|
|
}
|
|
|
|
/*
|
|
** clear Z-buffer and color-buffers if we're doing the gallery
|
|
*/
|
|
if ( !RI.drawWorld )
|
|
{
|
|
memset( d_pzbuffer, 0xff, vid.width * vid.height * sizeof( d_pzbuffer[0] ) );
|
|
}
|
|
|
|
D_Patch ();
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
===================
|
|
R_TransformFrustum
|
|
===================
|
|
*/
|
|
void R_TransformFrustum (void)
|
|
{
|
|
int i;
|
|
vec3_t v, v2;
|
|
|
|
for (i=0 ; i<4 ; i++)
|
|
{
|
|
v[0] = qfrustum.screenedge[i].normal[2];
|
|
v[1] = -qfrustum.screenedge[i].normal[0];
|
|
v[2] = qfrustum.screenedge[i].normal[1];
|
|
|
|
v2[0] = v[1]*RI.vright[0] + v[2]*RI.vup[0] + v[0]*RI.vforward[0];
|
|
v2[1] = v[1]*RI.vright[1] + v[2]*RI.vup[1] + v[0]*RI.vforward[1];
|
|
v2[2] = v[1]*RI.vright[2] + v[2]*RI.vup[2] + v[0]*RI.vforward[2];
|
|
|
|
VectorCopy (v2, qfrustum.view_clipplanes[i].normal);
|
|
|
|
qfrustum.view_clipplanes[i].dist = DotProduct (tr.modelorg, v2);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
TransformVector
|
|
================
|
|
*/
|
|
void TransformVector (vec3_t in, vec3_t out)
|
|
{
|
|
out[0] = DotProduct(in,RI.vright);
|
|
out[1] = DotProduct(in,RI.vup);
|
|
out[2] = DotProduct(in,RI.vforward);
|
|
}
|
|
|
|
/*
|
|
================
|
|
R_TransformPlane
|
|
================
|
|
*/
|
|
void R_TransformPlane (mplane_t *p, float *normal, float *dist)
|
|
{
|
|
float d;
|
|
|
|
d = DotProduct (RI.vieworg, p->normal);
|
|
*dist = p->dist - d;
|
|
// TODO: when we have rotating entities, this will need to use the view matrix
|
|
TransformVector (p->normal, normal);
|
|
}
|
|
|
|
|
|
/*
|
|
===============
|
|
R_SetUpFrustumIndexes
|
|
===============
|
|
*/
|
|
void R_SetUpFrustumIndexes (void)
|
|
{
|
|
int i, j, *pindex;
|
|
|
|
pindex = qfrustum.frustum_indexes;
|
|
|
|
for (i=0 ; i<4 ; i++)
|
|
{
|
|
for (j=0 ; j<3 ; j++)
|
|
{
|
|
if (qfrustum.view_clipplanes[i].normal[j] < 0)
|
|
{
|
|
pindex[j] = j;
|
|
pindex[j+3] = j+3;
|
|
}
|
|
else
|
|
{
|
|
pindex[j] = j+3;
|
|
pindex[j+3] = j;
|
|
}
|
|
}
|
|
|
|
// FIXME: do just once at start
|
|
qfrustum.pfrustum_indexes[i] = pindex;
|
|
pindex += 6;
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
R_ViewChanged
|
|
|
|
Called every time the vid structure or r_refdef changes.
|
|
Guaranteed to be called before the first refresh
|
|
===============
|
|
*/
|
|
void R_ViewChanged (vrect_t *vr)
|
|
{
|
|
int i;
|
|
float verticalFieldOfView, horizontalFieldOfView, xOrigin, yOrigin;
|
|
|
|
RI.vrect = *vr;
|
|
|
|
horizontalFieldOfView = 2*tan((float)RI.fov_x/360.0f * M_PI_F);
|
|
verticalFieldOfView = 2*tan((float)RI.fov_y/360.0f * M_PI_F);
|
|
|
|
RI.fvrectx = (float)RI.vrect.x;
|
|
RI.fvrectx_adj = (float)RI.vrect.x - 0.5f;
|
|
RI.vrect_x_adj_shift20 = (RI.vrect.x<<20) + (1<<19) - 1;
|
|
RI.fvrecty = (float)RI.vrect.y;
|
|
RI.fvrecty_adj = (float)RI.vrect.y - 0.5f;
|
|
RI.vrectright = RI.vrect.x + RI.vrect.width;
|
|
RI.vrectright_adj_shift20 = (RI.vrectright<<20) + (1<<19) - 1;
|
|
RI.fvrectright = (float)RI.vrectright;
|
|
RI.fvrectright_adj = (float)RI.vrectright - 0.5f;
|
|
RI.vrectrightedge = (float)RI.vrectright - 0.99f;
|
|
RI.vrectbottom = RI.vrect.y + RI.vrect.height;
|
|
RI.fvrectbottom = (float)RI.vrectbottom;
|
|
RI.fvrectbottom_adj = (float)RI.vrectbottom - 0.5f;
|
|
|
|
RI.aliasvrect.x = (int)(RI.vrect.x * r_aliasuvscale);
|
|
RI.aliasvrect.y = (int)(RI.vrect.y * r_aliasuvscale);
|
|
RI.aliasvrect.width = (int)(RI.vrect.width * r_aliasuvscale);
|
|
RI.aliasvrect.height = (int)(RI.vrect.height * r_aliasuvscale);
|
|
RI.aliasvrectright = RI.aliasvrect.x +
|
|
RI.aliasvrect.width;
|
|
RI.aliasvrectbottom = RI.aliasvrect.y +
|
|
RI.aliasvrect.height;
|
|
|
|
xOrigin = XCENTERING;
|
|
yOrigin = YCENTERING;
|
|
#define PLANE_ANYZ 5
|
|
// values for perspective projection
|
|
// if math were exact, the values would range from 0.5 to to range+0.5
|
|
// hopefully they wll be in the 0.000001 to range+.999999 and truncate
|
|
// the polygon rasterization will never render in the first row or column
|
|
// but will definately render in the [range] row and column, so adjust the
|
|
// buffer origin to get an exact edge to edge fill
|
|
xcenter = ((float)RI.vrect.width * XCENTERING) +
|
|
RI.vrect.x - 0.5f;
|
|
aliasxcenter = xcenter * r_aliasuvscale;
|
|
ycenter = ((float)RI.vrect.height * YCENTERING) +
|
|
RI.vrect.y - 0.5f;
|
|
aliasycenter = ycenter * r_aliasuvscale;
|
|
|
|
xscale = RI.vrect.width / horizontalFieldOfView;
|
|
aliasxscale = xscale * r_aliasuvscale;
|
|
xscaleinv = 1.0f / xscale;
|
|
|
|
yscale = xscale;
|
|
aliasyscale = yscale * r_aliasuvscale;
|
|
yscaleinv = 1.0f / yscale;
|
|
//xscaleshrink = (RI.vrect.width-6)/RI.horizontalFieldOfView;
|
|
//yscaleshrink = xscaleshrink;
|
|
|
|
// left side clip
|
|
qfrustum.screenedge[0].normal[0] = -1.0f / (xOrigin*horizontalFieldOfView);
|
|
qfrustum.screenedge[0].normal[1] = 0;
|
|
qfrustum.screenedge[0].normal[2] = 1;
|
|
qfrustum.screenedge[0].type = PLANE_ANYZ;
|
|
|
|
// right side clip
|
|
qfrustum.screenedge[1].normal[0] =
|
|
1.0f / ((1.0f-xOrigin)*horizontalFieldOfView);
|
|
qfrustum.screenedge[1].normal[1] = 0;
|
|
qfrustum.screenedge[1].normal[2] = 1;
|
|
qfrustum.screenedge[1].type = PLANE_ANYZ;
|
|
|
|
// top side clip
|
|
qfrustum.screenedge[2].normal[0] = 0;
|
|
qfrustum.screenedge[2].normal[1] = -1.0f / (yOrigin*verticalFieldOfView);
|
|
qfrustum.screenedge[2].normal[2] = 1;
|
|
qfrustum.screenedge[2].type = PLANE_ANYZ;
|
|
|
|
// bottom side clip
|
|
qfrustum.screenedge[3].normal[0] = 0;
|
|
qfrustum.screenedge[3].normal[1] = 1.0f / ((1.0f-yOrigin)*verticalFieldOfView);
|
|
qfrustum.screenedge[3].normal[2] = 1;
|
|
qfrustum.screenedge[3].type = PLANE_ANYZ;
|
|
|
|
for (i=0 ; i<4 ; i++)
|
|
VectorNormalize (qfrustum.screenedge[i].normal);
|
|
|
|
D_ViewChanged ();
|
|
}
|
|
|
|
|
|
/*
|
|
===============
|
|
R_SetupFrame
|
|
===============
|
|
*/
|
|
void R_SetupFrameQ (void)
|
|
{
|
|
int i;
|
|
vrect_t vrect;
|
|
|
|
if (r_fullbright->flags & FCVAR_CHANGED)
|
|
{
|
|
r_fullbright->flags &= ~FCVAR_CHANGED;
|
|
D_FlushCaches( ); // so all lighting changes
|
|
}
|
|
|
|
//tr.framecount++;
|
|
|
|
|
|
// build the transformation matrix for the given view angles
|
|
VectorCopy (RI.vieworg, tr.modelorg);
|
|
|
|
//AngleVectors (RI.viewangles, RI.vforward, RI.vright, RI.vup);
|
|
|
|
// current viewleaf
|
|
if ( RI.drawWorld )
|
|
{
|
|
RI.viewleaf = gEngfuncs.Mod_PointInLeaf (RI.vieworg, WORLDMODEL->nodes);
|
|
r_viewcluster = RI.viewleaf->cluster;
|
|
}
|
|
|
|
// if (sw_waterwarp->value && (r_newrefdef.rdflags & RDF_UNDERWATER) )
|
|
// r_dowarp = true;
|
|
// else
|
|
|
|
/*vrect.x = 0;//r_newrefdef.x;
|
|
vrect.y = 0;//r_newrefdef.y;
|
|
vrect.width = gpGlobals->width;
|
|
vrect.height = gpGlobals->height;*/
|
|
vrect.x = RI.viewport[0];
|
|
vrect.y = RI.viewport[1];
|
|
vrect.width = RI.viewport[2];
|
|
vrect.height = RI.viewport[3];
|
|
|
|
d_viewbuffer = (void *)vid.buffer;
|
|
r_screenwidth = vid.rowbytes;
|
|
|
|
R_ViewChanged (&vrect);
|
|
|
|
// start off with just the four screen edge clip planes
|
|
R_TransformFrustum ();
|
|
R_SetUpFrustumIndexes ();
|
|
|
|
// save base values
|
|
VectorCopy (RI.vforward, RI.base_vpn);
|
|
VectorCopy (RI.vright, RI.base_vright);
|
|
VectorCopy (RI.vup, RI.base_vup);
|
|
|
|
// clear frame counts
|
|
/* c_faceclip = 0;
|
|
d_spanpixcount = 0;
|
|
r_polycount = 0;
|
|
r_drawnpolycount = 0;
|
|
r_wholepolycount = 0;
|
|
r_amodels_drawn = 0;
|
|
r_outofsurfaces = 0;
|
|
r_outofedges = 0;*/
|
|
|
|
// d_setup
|
|
d_roverwrapped = false;
|
|
d_initial_rover = sc_rover;
|
|
|
|
d_minmip = sw_mipcap->value;
|
|
if (d_minmip > 3)
|
|
d_minmip = 3;
|
|
else if (d_minmip < 0)
|
|
d_minmip = 0;
|
|
|
|
for (i=0 ; i<(NUM_MIPS-1) ; i++)
|
|
d_scalemip[i] = basemip[i] * sw_mipscale->value;
|
|
|
|
//d_aflatcolor = 0;
|
|
}
|
|
|
|
|
|
#if !id386
|
|
|
|
/*
|
|
================
|
|
R_SurfacePatch
|
|
================
|
|
*/
|
|
/*void R_SurfacePatch (void)
|
|
{
|
|
// we only patch code on Intel
|
|
}
|
|
*/
|
|
#endif // !id386
|