2001-02-19 21:15:25 +00:00
|
|
|
/*
|
|
|
|
r_light.c
|
|
|
|
|
2001-08-25 23:23:14 +00:00
|
|
|
common lightmap code.
|
2001-02-19 21:15:25 +00:00
|
|
|
|
|
|
|
Copyright (C) 1996-1997 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:
|
|
|
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
59 Temple Place - Suite 330
|
|
|
|
Boston, MA 02111-1307, USA
|
|
|
|
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
2003-01-15 15:31:36 +00:00
|
|
|
|
2001-08-25 23:23:14 +00:00
|
|
|
#ifdef HAVE_STRING_H
|
|
|
|
# include <string.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_STRINGS_H
|
|
|
|
# include <strings.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdio.h>
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2013-01-22 05:09:41 +00:00
|
|
|
#include "qfalloca.h"
|
|
|
|
|
2001-08-25 23:23:14 +00:00
|
|
|
#include "QF/cvar.h"
|
2001-05-20 05:42:52 +00:00
|
|
|
#include "QF/render.h"
|
|
|
|
|
2001-10-09 20:35:17 +00:00
|
|
|
#include "compat.h"
|
2012-02-14 08:28:09 +00:00
|
|
|
#include "r_internal.h"
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2001-10-09 20:35:17 +00:00
|
|
|
dlight_t *r_dlights;
|
2002-08-12 04:42:29 +00:00
|
|
|
vec3_t ambientcolor;
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2001-10-09 20:35:17 +00:00
|
|
|
unsigned int r_maxdlights;
|
2022-05-10 08:03:14 +00:00
|
|
|
static int r_dlightframecount;
|
2001-10-09 20:35:17 +00:00
|
|
|
|
2012-05-11 12:53:16 +00:00
|
|
|
void
|
2022-03-30 14:50:12 +00:00
|
|
|
R_FindNearLights (vec4f_t pos, int count, dlight_t **lights)
|
2012-05-11 05:28:16 +00:00
|
|
|
{
|
2012-05-11 12:53:16 +00:00
|
|
|
float *scores = alloca (count * sizeof (float));
|
2012-05-11 05:28:16 +00:00
|
|
|
float score;
|
|
|
|
dlight_t *dl;
|
|
|
|
unsigned i;
|
|
|
|
int num = 0, j;
|
|
|
|
vec3_t d;
|
|
|
|
|
2012-05-11 06:39:41 +00:00
|
|
|
dl = r_dlights;
|
2012-05-11 05:28:16 +00:00
|
|
|
for (i = 0; i < r_maxdlights; i++, dl++) {
|
2022-03-07 04:40:04 +00:00
|
|
|
if (dl->die < r_data->realtime || !dl->radius)
|
2012-05-11 05:28:16 +00:00
|
|
|
continue;
|
|
|
|
VectorSubtract (dl->origin, pos, d);
|
|
|
|
score = DotProduct (d, d) / dl->radius;
|
|
|
|
if (!num) {
|
|
|
|
scores[0] = score;
|
|
|
|
lights[0] = dl;
|
|
|
|
num = 1;
|
|
|
|
} else if (score <= scores[0]) {
|
|
|
|
memmove (&lights[1], &lights[0],
|
|
|
|
(count - 1) * sizeof (dlight_t *));
|
|
|
|
memmove (&scores[1], &scores[0], (count - 1) * sizeof (float));
|
|
|
|
scores[0] = score;
|
|
|
|
lights[0] = dl;
|
|
|
|
if (num < count)
|
|
|
|
num++;
|
|
|
|
} else if (score > scores[num - 1]) {
|
|
|
|
if (num < count) {
|
|
|
|
scores[num] = score;
|
|
|
|
lights[num] = dl;
|
|
|
|
num++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (j = num - 1; j > 0; j--) {
|
|
|
|
if (score > scores[j - 1]) {
|
|
|
|
memmove (&lights[j + 1], &lights[j],
|
|
|
|
(count - j) * sizeof (dlight_t *));
|
|
|
|
memmove (&scores[j + 1], &scores[j],
|
|
|
|
(count - j) * sizeof (float));
|
|
|
|
scores[j] = score;
|
|
|
|
lights[j] = dl;
|
|
|
|
if (num < count)
|
|
|
|
num++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-05-11 12:53:16 +00:00
|
|
|
for (j = num; j < count; j++)
|
|
|
|
lights[j] = 0;
|
2012-05-11 05:28:16 +00:00
|
|
|
}
|
2001-10-09 20:35:17 +00:00
|
|
|
|
|
|
|
void
|
[cvar] Make cvars properly typed
This is an extremely extensive patch as it hits every cvar, and every
usage of the cvars. Cvars no longer store the value they control,
instead, they use a cexpr value object to reference the value and
specify the value's type (currently, a null type is used for strings).
Non-string cvars are passed through cexpr, allowing expressions in the
cvars' settings. Also, cvars have returned to an enhanced version of the
original (id quake) registration scheme.
As a minor benefit, relevant code having direct access to the
cvar-controlled variables is probably a slight optimization as it
removed a pointer dereference, and the variables can be located for data
locality.
The static cvar descriptors are made private as an additional safety
layer, though there's nothing stopping external modification via
Cvar_FindVar (which is needed for adding listeners).
While not used yet (partly due to working out the design), cvars can
have a validation function.
Registering a cvar allows a primary listener (and its data) to be
specified: it will always be called first when the cvar is modified. The
combination of proper listeners and direct access to the controlled
variable greatly simplifies the more complex cvar interactions as much
less null checking is required, and there's no need for one cvar's
callback to call another's.
nq-x11 is known to work at least well enough for the demos. More testing
will come.
2022-04-23 03:22:45 +00:00
|
|
|
R_MaxDlightsCheck (int max_dlights)
|
2001-10-09 20:35:17 +00:00
|
|
|
{
|
[cvar] Make cvars properly typed
This is an extremely extensive patch as it hits every cvar, and every
usage of the cvars. Cvars no longer store the value they control,
instead, they use a cexpr value object to reference the value and
specify the value's type (currently, a null type is used for strings).
Non-string cvars are passed through cexpr, allowing expressions in the
cvars' settings. Also, cvars have returned to an enhanced version of the
original (id quake) registration scheme.
As a minor benefit, relevant code having direct access to the
cvar-controlled variables is probably a slight optimization as it
removed a pointer dereference, and the variables can be located for data
locality.
The static cvar descriptors are made private as an additional safety
layer, though there's nothing stopping external modification via
Cvar_FindVar (which is needed for adding listeners).
While not used yet (partly due to working out the design), cvars can
have a validation function.
Registering a cvar allows a primary listener (and its data) to be
specified: it will always be called first when the cvar is modified. The
combination of proper listeners and direct access to the controlled
variable greatly simplifies the more complex cvar interactions as much
less null checking is required, and there's no need for one cvar's
callback to call another's.
nq-x11 is known to work at least well enough for the demos. More testing
will come.
2022-04-23 03:22:45 +00:00
|
|
|
r_maxdlights = bound (0, max_dlights, MAX_DLIGHTS);
|
2001-10-09 20:35:17 +00:00
|
|
|
|
2001-10-09 21:50:51 +00:00
|
|
|
if (r_dlights)
|
|
|
|
free (r_dlights);
|
2001-10-09 21:57:49 +00:00
|
|
|
|
2001-10-09 21:50:51 +00:00
|
|
|
r_dlights=0;
|
2001-10-09 20:35:17 +00:00
|
|
|
|
2001-10-09 21:50:51 +00:00
|
|
|
if (r_maxdlights)
|
|
|
|
r_dlights = (dlight_t *) calloc (r_maxdlights, sizeof (dlight_t));
|
2001-10-09 20:35:17 +00:00
|
|
|
|
|
|
|
R_ClearDlights();
|
|
|
|
}
|
|
|
|
|
2001-02-26 06:48:02 +00:00
|
|
|
void
|
|
|
|
R_AnimateLight (void)
|
2001-02-19 21:15:25 +00:00
|
|
|
{
|
2001-02-26 06:48:02 +00:00
|
|
|
int i, j, k;
|
|
|
|
|
2022-04-07 13:30:19 +00:00
|
|
|
if (!r_data->lightstyle) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-05-15 21:13:07 +00:00
|
|
|
// light animations
|
|
|
|
// 'm' is normal light, 'a' is no light, 'z' is double bright
|
2022-03-07 04:40:04 +00:00
|
|
|
i = (int) (r_data->realtime * 10);
|
2001-02-26 06:48:02 +00:00
|
|
|
for (j = 0; j < MAX_LIGHTSTYLES; j++) {
|
2022-03-07 04:40:04 +00:00
|
|
|
if (!r_data->lightstyle[j].length) {
|
2001-02-19 21:15:25 +00:00
|
|
|
d_lightstylevalue[j] = 256;
|
|
|
|
continue;
|
|
|
|
}
|
[cvar] Make cvars properly typed
This is an extremely extensive patch as it hits every cvar, and every
usage of the cvars. Cvars no longer store the value they control,
instead, they use a cexpr value object to reference the value and
specify the value's type (currently, a null type is used for strings).
Non-string cvars are passed through cexpr, allowing expressions in the
cvars' settings. Also, cvars have returned to an enhanced version of the
original (id quake) registration scheme.
As a minor benefit, relevant code having direct access to the
cvar-controlled variables is probably a slight optimization as it
removed a pointer dereference, and the variables can be located for data
locality.
The static cvar descriptors are made private as an additional safety
layer, though there's nothing stopping external modification via
Cvar_FindVar (which is needed for adding listeners).
While not used yet (partly due to working out the design), cvars can
have a validation function.
Registering a cvar allows a primary listener (and its data) to be
specified: it will always be called first when the cvar is modified. The
combination of proper listeners and direct access to the controlled
variable greatly simplifies the more complex cvar interactions as much
less null checking is required, and there's no need for one cvar's
callback to call another's.
nq-x11 is known to work at least well enough for the demos. More testing
will come.
2022-04-23 03:22:45 +00:00
|
|
|
if (r_flatlightstyles == 2) {
|
2022-03-07 04:40:04 +00:00
|
|
|
k = r_data->lightstyle[j].peak - 'a';
|
[cvar] Make cvars properly typed
This is an extremely extensive patch as it hits every cvar, and every
usage of the cvars. Cvars no longer store the value they control,
instead, they use a cexpr value object to reference the value and
specify the value's type (currently, a null type is used for strings).
Non-string cvars are passed through cexpr, allowing expressions in the
cvars' settings. Also, cvars have returned to an enhanced version of the
original (id quake) registration scheme.
As a minor benefit, relevant code having direct access to the
cvar-controlled variables is probably a slight optimization as it
removed a pointer dereference, and the variables can be located for data
locality.
The static cvar descriptors are made private as an additional safety
layer, though there's nothing stopping external modification via
Cvar_FindVar (which is needed for adding listeners).
While not used yet (partly due to working out the design), cvars can
have a validation function.
Registering a cvar allows a primary listener (and its data) to be
specified: it will always be called first when the cvar is modified. The
combination of proper listeners and direct access to the controlled
variable greatly simplifies the more complex cvar interactions as much
less null checking is required, and there's no need for one cvar's
callback to call another's.
nq-x11 is known to work at least well enough for the demos. More testing
will come.
2022-04-23 03:22:45 +00:00
|
|
|
} else if (r_flatlightstyles == 1) {
|
2022-03-07 04:40:04 +00:00
|
|
|
k = r_data->lightstyle[j].average - 'a';
|
2012-05-26 02:06:21 +00:00
|
|
|
} else {
|
2022-03-07 04:40:04 +00:00
|
|
|
k = i % r_data->lightstyle[j].length;
|
|
|
|
k = r_data->lightstyle[j].map[k] - 'a';
|
2012-05-26 02:06:21 +00:00
|
|
|
}
|
|
|
|
d_lightstylevalue[j] = k * 22;
|
2001-02-26 06:48:02 +00:00
|
|
|
}
|
2001-02-19 21:15:25 +00:00
|
|
|
}
|
|
|
|
|
2003-08-11 06:05:07 +00:00
|
|
|
static inline void
|
2022-05-22 02:18:32 +00:00
|
|
|
real_mark_surfaces (float dist, msurface_t *surf, vec4f_t lightorigin,
|
2012-07-21 04:58:54 +00:00
|
|
|
dlight_t *light, unsigned lightnum)
|
2001-08-07 17:29:21 +00:00
|
|
|
{
|
2004-03-17 04:53:10 +00:00
|
|
|
float dist2, is, it;
|
2001-08-07 17:29:21 +00:00
|
|
|
float maxdist = light->radius * light->radius;
|
|
|
|
vec3_t impact;
|
2012-07-21 04:58:54 +00:00
|
|
|
unsigned ind, bit;
|
2001-08-07 17:29:21 +00:00
|
|
|
|
2004-03-17 04:53:10 +00:00
|
|
|
dist2 = maxdist - dist * dist;
|
2003-08-08 15:25:53 +00:00
|
|
|
VectorMultSub (light->origin, dist, surf->plane->normal, impact);
|
2001-08-07 17:29:21 +00:00
|
|
|
|
2004-03-17 04:53:10 +00:00
|
|
|
is = DotProduct (impact, surf->texinfo->vecs[0])
|
|
|
|
+ surf->texinfo->vecs[0][3] - surf->texturemins[0];
|
|
|
|
it = DotProduct (impact, surf->texinfo->vecs[1])
|
2001-08-07 17:29:21 +00:00
|
|
|
+ surf->texinfo->vecs[1][3] - surf->texturemins[1];
|
2004-03-17 04:53:10 +00:00
|
|
|
|
|
|
|
// compress the square to a point
|
|
|
|
if (is > surf->extents[0])
|
|
|
|
is -= surf->extents[0];
|
|
|
|
else if (is > 0)
|
|
|
|
is = 0;
|
|
|
|
if (it > surf->extents[1])
|
|
|
|
it -= surf->extents[1];
|
|
|
|
else if (it > 0)
|
|
|
|
it = 0;
|
|
|
|
if (is * is + it * it > dist2)
|
|
|
|
return;
|
2001-08-29 04:53:59 +00:00
|
|
|
|
2022-05-10 08:03:14 +00:00
|
|
|
if (surf->dlightframe != r_dlightframecount) {
|
2012-07-21 04:58:54 +00:00
|
|
|
memset (surf->dlightbits, 0, sizeof (surf->dlightbits));
|
2022-05-10 08:03:14 +00:00
|
|
|
surf->dlightframe = r_dlightframecount;
|
2001-08-07 17:29:21 +00:00
|
|
|
}
|
2012-07-21 04:58:54 +00:00
|
|
|
ind = lightnum / 32;
|
|
|
|
bit = 1 << (lightnum % 32);
|
|
|
|
surf->dlightbits[ind] |= bit;
|
2001-08-07 17:29:21 +00:00
|
|
|
}
|
|
|
|
|
2003-08-07 19:58:39 +00:00
|
|
|
static inline void
|
2022-05-22 02:18:32 +00:00
|
|
|
mark_surfaces (msurface_t *surf, vec4f_t lightorigin, dlight_t *light,
|
2012-07-21 04:58:54 +00:00
|
|
|
int lightnum)
|
2003-08-07 19:58:39 +00:00
|
|
|
{
|
|
|
|
float dist;
|
|
|
|
|
|
|
|
dist = PlaneDiff(lightorigin, surf->plane);
|
|
|
|
if (surf->flags & SURF_PLANEBACK)
|
|
|
|
dist = -dist;
|
2004-03-17 04:53:10 +00:00
|
|
|
if ((dist < 0 && !(surf->flags & SURF_LIGHTBOTHSIDES))
|
2003-08-07 19:58:39 +00:00
|
|
|
|| dist > light->radius)
|
|
|
|
return;
|
|
|
|
|
2012-07-21 04:58:54 +00:00
|
|
|
real_mark_surfaces (dist, surf, lightorigin, light, lightnum);
|
2003-08-07 19:58:39 +00:00
|
|
|
}
|
|
|
|
|
2004-02-21 04:23:00 +00:00
|
|
|
// LordHavoc: heavily modified, to eliminate unnecessary texture uploads,
|
|
|
|
// and support bmodel lighting better
|
|
|
|
void
|
2022-05-22 02:18:32 +00:00
|
|
|
R_RecursiveMarkLights (mod_brush_t *brush, vec4f_t lightorigin,
|
|
|
|
dlight_t *light, int lightnum, int node_id)
|
2004-02-21 04:23:00 +00:00
|
|
|
{
|
2012-09-07 07:09:24 +00:00
|
|
|
unsigned i;
|
2004-02-21 04:23:00 +00:00
|
|
|
float ndist, maxdist;
|
|
|
|
msurface_t *surf;
|
|
|
|
|
|
|
|
maxdist = light->radius;
|
|
|
|
|
|
|
|
loc0:
|
2022-05-22 02:18:32 +00:00
|
|
|
if (node_id < 0)
|
2004-02-21 04:23:00 +00:00
|
|
|
return;
|
|
|
|
|
2022-05-22 02:18:32 +00:00
|
|
|
mnode_t *node = brush->nodes + node_id;
|
|
|
|
ndist = dotf (lightorigin, node->plane)[0];
|
2004-02-21 04:23:00 +00:00
|
|
|
|
|
|
|
if (ndist > maxdist * maxdist) {
|
|
|
|
// Save time by not pushing another stack frame.
|
2022-05-22 02:18:32 +00:00
|
|
|
if (node->children[0] >= 0) {
|
|
|
|
node_id = node->children[0];
|
2004-02-21 04:23:00 +00:00
|
|
|
goto loc0;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (ndist < -maxdist * maxdist) {
|
2022-05-22 02:18:32 +00:00
|
|
|
if (node->children[1] >= 0) {
|
|
|
|
node_id = node->children[1];
|
2004-02-21 04:23:00 +00:00
|
|
|
goto loc0;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// mark the polygons
|
2021-02-01 10:31:11 +00:00
|
|
|
surf = brush->surfaces + node->firstsurface;
|
2004-02-21 04:23:00 +00:00
|
|
|
for (i = 0; i < node->numsurfaces; i++, surf++) {
|
2012-07-21 04:58:54 +00:00
|
|
|
mark_surfaces (surf, lightorigin, light, lightnum);
|
2004-02-21 04:23:00 +00:00
|
|
|
}
|
|
|
|
|
2022-05-22 02:18:32 +00:00
|
|
|
if (node->children[0] >= 0) {
|
|
|
|
if (node->children[1] >= 0)
|
2021-02-01 10:31:11 +00:00
|
|
|
R_RecursiveMarkLights (brush, lightorigin, light, lightnum,
|
2012-07-21 04:58:54 +00:00
|
|
|
node->children[1]);
|
2022-05-22 02:18:32 +00:00
|
|
|
node_id = node->children[0];
|
2004-02-21 04:23:00 +00:00
|
|
|
goto loc0;
|
2022-05-22 02:18:32 +00:00
|
|
|
} else if (node->children[1] >= 0) {
|
|
|
|
node_id = node->children[1];
|
2004-02-21 04:23:00 +00:00
|
|
|
goto loc0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-05-10 08:03:14 +00:00
|
|
|
static void
|
2022-05-22 02:18:32 +00:00
|
|
|
R_MarkLights (vec4f_t lightorigin, dlight_t *light, int lightnum,
|
2002-01-03 05:29:38 +00:00
|
|
|
model_t *model)
|
2001-08-07 17:29:21 +00:00
|
|
|
{
|
2021-02-01 10:31:11 +00:00
|
|
|
mod_brush_t *brush = &model->brush;
|
2001-08-07 17:29:21 +00:00
|
|
|
mleaf_t *pvsleaf = Mod_PointInLeaf (lightorigin, model);
|
|
|
|
|
|
|
|
if (!pvsleaf->compressed_vis) {
|
2022-05-22 02:18:32 +00:00
|
|
|
int node_id = brush->hulls[0].firstclipnode;
|
|
|
|
R_RecursiveMarkLights (brush, lightorigin, light, lightnum, node_id);
|
2001-08-07 17:29:21 +00:00
|
|
|
} else {
|
|
|
|
float radius = light->radius;
|
|
|
|
vec3_t mins, maxs;
|
2021-07-26 02:15:51 +00:00
|
|
|
unsigned leafnum = 0;
|
2001-08-07 17:29:21 +00:00
|
|
|
byte *in = pvsleaf->compressed_vis;
|
|
|
|
byte vis_bits;
|
|
|
|
|
|
|
|
mins[0] = lightorigin[0] - radius;
|
|
|
|
mins[1] = lightorigin[1] - radius;
|
|
|
|
mins[2] = lightorigin[2] - radius;
|
|
|
|
maxs[0] = lightorigin[0] + radius;
|
|
|
|
maxs[1] = lightorigin[1] + radius;
|
|
|
|
maxs[2] = lightorigin[2] + radius;
|
2021-07-27 03:32:40 +00:00
|
|
|
while (leafnum < brush->visleafs) {
|
2004-02-23 07:17:41 +00:00
|
|
|
int b;
|
2001-08-07 17:29:21 +00:00
|
|
|
if (!(vis_bits = *in++)) {
|
|
|
|
leafnum += (*in++) * 8;
|
|
|
|
continue;
|
|
|
|
}
|
2021-07-27 03:32:40 +00:00
|
|
|
for (b = 1; b < 256 && leafnum < brush->visleafs;
|
2004-02-23 07:17:41 +00:00
|
|
|
b <<= 1, leafnum++) {
|
2001-08-07 17:29:21 +00:00
|
|
|
int m;
|
2021-02-01 10:31:11 +00:00
|
|
|
mleaf_t *leaf = &brush->leafs[leafnum + 1];
|
2004-02-23 07:17:41 +00:00
|
|
|
if (!(vis_bits & b))
|
2001-08-07 17:29:21 +00:00
|
|
|
continue;
|
2022-05-22 05:43:07 +00:00
|
|
|
if (r_leaf_visframes[leafnum + 1] != r_visframecount)
|
2001-08-07 17:29:21 +00:00
|
|
|
continue;
|
|
|
|
if (leaf->mins[0] > maxs[0] || leaf->maxs[0] < mins[0]
|
|
|
|
|| leaf->mins[1] > maxs[1] || leaf->maxs[1] < mins[1]
|
|
|
|
|| leaf->mins[2] > maxs[2] || leaf->maxs[2] < mins[2])
|
|
|
|
continue;
|
2022-05-10 05:30:48 +00:00
|
|
|
msurface_t **msurf = brush->marksurfaces + leaf->firstmarksurface;
|
2001-08-07 17:29:21 +00:00
|
|
|
for (m = 0; m < leaf->nummarksurfaces; m++) {
|
2022-05-10 05:30:48 +00:00
|
|
|
msurface_t *surf = *msurf++;
|
2022-05-22 07:31:24 +00:00
|
|
|
int surf_id = surf - brush->surfaces;
|
|
|
|
if (r_face_visframes[surf_id] != r_visframecount)
|
2001-08-07 17:29:21 +00:00
|
|
|
continue;
|
2012-07-21 04:58:54 +00:00
|
|
|
mark_surfaces (surf, lightorigin, light, lightnum);
|
2001-08-07 17:29:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-02-19 21:15:25 +00:00
|
|
|
}
|
|
|
|
|
2001-02-26 06:48:02 +00:00
|
|
|
void
|
2002-01-03 05:29:38 +00:00
|
|
|
R_PushDlights (const vec3_t entorigin)
|
2001-02-19 21:15:25 +00:00
|
|
|
{
|
2003-04-17 00:01:48 +00:00
|
|
|
unsigned int i;
|
2001-02-26 06:48:02 +00:00
|
|
|
dlight_t *l;
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2023-01-15 15:38:43 +00:00
|
|
|
r_dlightframecount = r_framecount;
|
2022-05-10 08:03:14 +00:00
|
|
|
|
[cvar] Make cvars properly typed
This is an extremely extensive patch as it hits every cvar, and every
usage of the cvars. Cvars no longer store the value they control,
instead, they use a cexpr value object to reference the value and
specify the value's type (currently, a null type is used for strings).
Non-string cvars are passed through cexpr, allowing expressions in the
cvars' settings. Also, cvars have returned to an enhanced version of the
original (id quake) registration scheme.
As a minor benefit, relevant code having direct access to the
cvar-controlled variables is probably a slight optimization as it
removed a pointer dereference, and the variables can be located for data
locality.
The static cvar descriptors are made private as an additional safety
layer, though there's nothing stopping external modification via
Cvar_FindVar (which is needed for adding listeners).
While not used yet (partly due to working out the design), cvars can
have a validation function.
Registering a cvar allows a primary listener (and its data) to be
specified: it will always be called first when the cvar is modified. The
combination of proper listeners and direct access to the controlled
variable greatly simplifies the more complex cvar interactions as much
less null checking is required, and there's no need for one cvar's
callback to call another's.
nq-x11 is known to work at least well enough for the demos. More testing
will come.
2022-04-23 03:22:45 +00:00
|
|
|
if (!r_dlight_lightmap)
|
2001-08-25 23:23:14 +00:00
|
|
|
return;
|
|
|
|
|
2001-05-20 05:42:52 +00:00
|
|
|
l = r_dlights;
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2001-10-09 20:35:17 +00:00
|
|
|
for (i = 0; i < r_maxdlights; i++, l++) {
|
2022-03-07 04:40:04 +00:00
|
|
|
if (l->die < r_data->realtime || !l->radius)
|
2001-02-19 21:15:25 +00:00
|
|
|
continue;
|
2022-05-22 02:18:32 +00:00
|
|
|
vec4f_t lightorigin;
|
2001-02-26 06:48:02 +00:00
|
|
|
VectorSubtract (l->origin, entorigin, lightorigin);
|
2022-05-22 02:18:32 +00:00
|
|
|
lightorigin[3] = 1;
|
2022-03-14 06:27:43 +00:00
|
|
|
R_MarkLights (lightorigin, l, i, r_refdef.worldmodel);
|
2001-02-19 21:15:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-09-04 10:32:51 +00:00
|
|
|
/* LIGHT SAMPLING */
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2001-08-25 23:23:14 +00:00
|
|
|
vec3_t lightspot;
|
2001-05-15 21:13:07 +00:00
|
|
|
|
2002-08-12 04:42:29 +00:00
|
|
|
static int
|
|
|
|
calc_lighting_1 (msurface_t *surf, int ds, int dt)
|
|
|
|
{
|
|
|
|
int se_s = ((surf->extents[0] >> 4) + 1);
|
|
|
|
int se_t = ((surf->extents[0] >> 4) + 1);
|
|
|
|
int se_size = se_s * se_t;
|
|
|
|
int r = 0, maps;
|
|
|
|
byte *lightmap;
|
|
|
|
unsigned int scale;
|
|
|
|
|
|
|
|
ds >>= 4;
|
|
|
|
dt >>= 4;
|
|
|
|
|
|
|
|
lightmap = surf->samples;
|
|
|
|
if (lightmap) {
|
|
|
|
lightmap += dt * se_s + ds;
|
|
|
|
|
|
|
|
for (maps = 0; maps < MAXLIGHTMAPS && surf->styles[maps] != 255;
|
|
|
|
maps++) {
|
|
|
|
scale = d_lightstylevalue[surf->styles[maps]];
|
|
|
|
r += *lightmap * scale;
|
|
|
|
lightmap += se_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
r >>= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
ambientcolor[2] = ambientcolor[1] = ambientcolor[0] = r;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
calc_lighting_3 (msurface_t *surf, int ds, int dt)
|
|
|
|
{
|
|
|
|
int se_s = ((surf->extents[0] >> 4) + 1);
|
|
|
|
int se_t = ((surf->extents[0] >> 4) + 1);
|
|
|
|
int se_size = se_s * se_t * 3;
|
|
|
|
int r = 0, maps;
|
|
|
|
byte *lightmap;
|
|
|
|
float scale;
|
|
|
|
|
|
|
|
ds >>= 4;
|
|
|
|
dt >>= 4;
|
|
|
|
|
|
|
|
VectorZero (ambientcolor);
|
|
|
|
|
|
|
|
lightmap = surf->samples;
|
|
|
|
if (lightmap) {
|
2002-08-12 06:14:55 +00:00
|
|
|
lightmap += (dt * se_s + ds) * 3;
|
2002-08-12 04:42:29 +00:00
|
|
|
|
|
|
|
for (maps = 0; maps < MAXLIGHTMAPS && surf->styles[maps] != 255;
|
|
|
|
maps++) {
|
|
|
|
scale = d_lightstylevalue[surf->styles[maps]] / 256.0;
|
2003-08-08 15:25:53 +00:00
|
|
|
VectorMultAdd (ambientcolor, scale, lightmap, ambientcolor);
|
2002-08-12 04:42:29 +00:00
|
|
|
lightmap += se_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r = (ambientcolor[0] + ambientcolor[1] + ambientcolor[2]) / 3;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2003-01-06 18:28:13 +00:00
|
|
|
static int
|
2022-05-22 02:18:32 +00:00
|
|
|
RecursiveLightPoint (mod_brush_t *brush, int node_id, vec4f_t start,
|
2022-03-30 14:50:12 +00:00
|
|
|
vec4f_t end)
|
2001-02-19 21:15:25 +00:00
|
|
|
{
|
2012-09-07 07:09:24 +00:00
|
|
|
unsigned i;
|
|
|
|
int r, s, t, ds, dt, side;
|
2011-11-14 02:18:22 +00:00
|
|
|
float front, back, frac;
|
|
|
|
msurface_t *surf;
|
|
|
|
mtexinfo_t *tex;
|
2001-12-03 08:46:56 +00:00
|
|
|
loop:
|
2022-05-22 02:18:32 +00:00
|
|
|
if (node_id < 0)
|
2001-02-26 06:48:02 +00:00
|
|
|
return -1; // didn't hit anything
|
|
|
|
|
2001-08-27 01:00:03 +00:00
|
|
|
// calculate mid point
|
2022-05-22 02:18:32 +00:00
|
|
|
mnode_t *node = brush->nodes + node_id;
|
|
|
|
front = dotf (start, node->plane)[0];
|
|
|
|
back = dotf (end, node->plane)[0];
|
2001-02-19 21:15:25 +00:00
|
|
|
side = front < 0;
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-12-03 08:46:56 +00:00
|
|
|
if ((back < 0) == side) {
|
2022-05-22 02:18:32 +00:00
|
|
|
node_id = node->children[side];
|
2001-12-03 08:46:56 +00:00
|
|
|
goto loop;
|
|
|
|
}
|
2001-02-26 06:48:02 +00:00
|
|
|
|
|
|
|
frac = front / (front - back);
|
2022-03-30 14:50:12 +00:00
|
|
|
vec4f_t mid = start + (end - start) * frac;
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2012-05-21 23:23:22 +00:00
|
|
|
// go down front side
|
2021-02-01 10:31:11 +00:00
|
|
|
r = RecursiveLightPoint (brush, node->children[side], start, mid);
|
2001-02-19 21:15:25 +00:00
|
|
|
if (r >= 0)
|
2001-02-26 06:48:02 +00:00
|
|
|
return r; // hit something
|
|
|
|
|
|
|
|
if ((back < 0) == side)
|
2001-05-15 04:50:53 +00:00
|
|
|
return -1; // didn't hit anything
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-08-27 01:00:03 +00:00
|
|
|
// check for impact on this node
|
2001-08-25 23:23:14 +00:00
|
|
|
VectorCopy (mid, lightspot);
|
|
|
|
|
2021-02-01 10:31:11 +00:00
|
|
|
surf = brush->surfaces + node->firstsurface;
|
2001-02-26 06:48:02 +00:00
|
|
|
for (i = 0; i < node->numsurfaces; i++, surf++) {
|
2001-02-19 21:15:25 +00:00
|
|
|
if (surf->flags & SURF_DRAWTILED)
|
2001-02-26 06:48:02 +00:00
|
|
|
continue; // no lightmaps
|
2001-02-19 21:15:25 +00:00
|
|
|
|
|
|
|
tex = surf->texinfo;
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-02-19 21:15:25 +00:00
|
|
|
s = DotProduct (mid, tex->vecs[0]) + tex->vecs[0][3];
|
2002-11-20 21:44:04 +00:00
|
|
|
t = DotProduct (mid, tex->vecs[1]) + tex->vecs[1][3];
|
2001-02-19 21:15:25 +00:00
|
|
|
|
2001-02-26 06:48:02 +00:00
|
|
|
if (s < surf->texturemins[0] || t < surf->texturemins[1])
|
2001-02-19 21:15:25 +00:00
|
|
|
continue;
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-02-19 21:15:25 +00:00
|
|
|
ds = s - surf->texturemins[0];
|
|
|
|
dt = t - surf->texturemins[1];
|
2001-02-26 06:48:02 +00:00
|
|
|
|
|
|
|
if (ds > surf->extents[0] || dt > surf->extents[1])
|
2001-02-19 21:15:25 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!surf->samples)
|
|
|
|
return 0;
|
|
|
|
|
2002-09-18 14:38:41 +00:00
|
|
|
if (mod_lightmap_bytes == 1)
|
2002-08-12 04:42:29 +00:00
|
|
|
return calc_lighting_1 (surf, ds, dt);
|
|
|
|
else
|
|
|
|
return calc_lighting_3 (surf, ds, dt);
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-02-19 21:15:25 +00:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2001-08-27 01:00:03 +00:00
|
|
|
// go down back side
|
2022-05-22 02:18:32 +00:00
|
|
|
return RecursiveLightPoint (brush, node->children[side ^ 1], mid, end);
|
2001-02-19 21:15:25 +00:00
|
|
|
}
|
|
|
|
|
2001-02-26 06:48:02 +00:00
|
|
|
int
|
2022-03-30 14:50:12 +00:00
|
|
|
R_LightPoint (mod_brush_t *brush, vec4f_t p)
|
2001-02-19 21:15:25 +00:00
|
|
|
{
|
2021-02-01 10:31:11 +00:00
|
|
|
if (!brush->lightdata) {
|
2012-05-15 11:24:02 +00:00
|
|
|
// allow dlights to have some effect, so don't go /quite/ fullbright
|
2004-05-02 21:21:00 +00:00
|
|
|
ambientcolor[2] = ambientcolor[1] = ambientcolor[0] = 200;
|
|
|
|
return 200;
|
2004-05-02 20:40:09 +00:00
|
|
|
}
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2022-03-30 14:50:12 +00:00
|
|
|
vec4f_t end = p - (vec4f_t) { 0, 0, 2048, 0 };
|
2022-05-22 02:18:32 +00:00
|
|
|
int r = RecursiveLightPoint (brush, 0, p, end);
|
2001-02-26 06:48:02 +00:00
|
|
|
|
2001-02-19 21:15:25 +00:00
|
|
|
if (r == -1)
|
|
|
|
r = 0;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
2001-09-06 03:53:13 +00:00
|
|
|
|
|
|
|
dlight_t *
|
|
|
|
R_AllocDlight (int key)
|
|
|
|
{
|
2003-04-17 00:01:48 +00:00
|
|
|
unsigned int i;
|
2001-09-06 03:53:13 +00:00
|
|
|
dlight_t *dl;
|
2001-10-09 21:50:51 +00:00
|
|
|
|
|
|
|
if (!r_maxdlights) {
|
2002-06-05 22:07:38 +00:00
|
|
|
return NULL;
|
2001-10-09 21:50:51 +00:00
|
|
|
}
|
2001-09-06 03:53:13 +00:00
|
|
|
|
|
|
|
// first look for an exact key match
|
|
|
|
if (key) {
|
|
|
|
dl = r_dlights;
|
2001-10-09 20:35:17 +00:00
|
|
|
for (i = 0; i < r_maxdlights; i++, dl++) {
|
2001-09-06 03:53:13 +00:00
|
|
|
if (dl->key == key) {
|
|
|
|
memset (dl, 0, sizeof (*dl));
|
|
|
|
dl->key = key;
|
|
|
|
dl->color[0] = dl->color[1] = dl->color[2] = 1;
|
|
|
|
return dl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// then look for anything else
|
|
|
|
dl = r_dlights;
|
2001-10-09 20:35:17 +00:00
|
|
|
for (i = 0; i < r_maxdlights; i++, dl++) {
|
2022-03-07 04:40:04 +00:00
|
|
|
if (dl->die < r_data->realtime) {
|
2001-09-06 03:53:13 +00:00
|
|
|
memset (dl, 0, sizeof (*dl));
|
|
|
|
dl->key = key;
|
|
|
|
dl->color[0] = dl->color[1] = dl->color[2] = 1;
|
|
|
|
return dl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dl = &r_dlights[0];
|
|
|
|
memset (dl, 0, sizeof (*dl));
|
|
|
|
dl->key = key;
|
|
|
|
return dl;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
R_DecayLights (double frametime)
|
|
|
|
{
|
2003-04-17 00:01:48 +00:00
|
|
|
unsigned int i;
|
2001-09-06 03:53:13 +00:00
|
|
|
dlight_t *dl;
|
|
|
|
|
|
|
|
dl = r_dlights;
|
2001-10-09 20:35:17 +00:00
|
|
|
for (i = 0; i < r_maxdlights; i++, dl++) {
|
2022-03-07 04:40:04 +00:00
|
|
|
if (dl->die < r_data->realtime || !dl->radius)
|
2001-09-06 03:53:13 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
dl->radius -= frametime * dl->decay;
|
|
|
|
if (dl->radius < 0)
|
|
|
|
dl->radius = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
R_ClearDlights (void)
|
|
|
|
{
|
2001-10-09 21:50:51 +00:00
|
|
|
if (r_maxdlights)
|
|
|
|
memset (r_dlights, 0, r_maxdlights * sizeof (dlight_t));
|
2001-09-06 03:53:13 +00:00
|
|
|
}
|