2012-02-14 13:19:17 +00:00
|
|
|
/*
|
2019-07-08 16:00:47 +00:00
|
|
|
vid_render_sw.c
|
2012-02-14 13:19:17 +00:00
|
|
|
|
|
|
|
SW version of the renderer
|
|
|
|
|
|
|
|
Copyright (C) 2012 Bill Currie <bill@taniwha.org>
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2022-03-09 13:53:15 +00:00
|
|
|
#include "QF/cvar.h"
|
2022-03-31 10:07:15 +00:00
|
|
|
#include "QF/image.h"
|
2022-03-09 13:53:15 +00:00
|
|
|
|
2012-02-14 13:19:17 +00:00
|
|
|
#include "QF/plugin/general.h"
|
|
|
|
#include "QF/plugin/vid_render.h"
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
#include "QF/ui/view.h"
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
#include "d_local.h"
|
2012-02-14 13:19:17 +00:00
|
|
|
#include "mod_internal.h"
|
|
|
|
#include "r_internal.h"
|
2019-07-08 16:00:47 +00:00
|
|
|
#include "vid_internal.h"
|
|
|
|
#include "vid_sw.h"
|
|
|
|
|
|
|
|
sw_ctx_t *sw_ctx;
|
2012-02-14 13:19:17 +00:00
|
|
|
|
2022-03-30 05:55:32 +00:00
|
|
|
static float r_aliasuvscale = 1.0;
|
|
|
|
|
2021-06-26 06:43:17 +00:00
|
|
|
static void
|
2021-07-10 15:09:41 +00:00
|
|
|
sw_vid_render_choose_visual (void *data)
|
2021-06-26 06:43:17 +00:00
|
|
|
{
|
|
|
|
sw_ctx->choose_visual (sw_ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2021-07-10 15:09:41 +00:00
|
|
|
sw_vid_render_create_context (void *data)
|
2021-06-26 06:43:17 +00:00
|
|
|
{
|
|
|
|
sw_ctx->create_context (sw_ctx);
|
|
|
|
}
|
|
|
|
|
2021-07-10 15:09:41 +00:00
|
|
|
static void
|
|
|
|
sw_vid_render_set_palette (void *data, const byte *palette)
|
|
|
|
{
|
|
|
|
sw_ctx->set_palette (sw_ctx, palette);
|
|
|
|
}
|
|
|
|
|
2022-03-29 04:08:58 +00:00
|
|
|
static void
|
|
|
|
sw_vid_render_set_colormap (void *data, const byte *colormap)
|
|
|
|
{
|
|
|
|
R_SetColormap (colormap);
|
|
|
|
}
|
|
|
|
|
2012-02-22 12:53:17 +00:00
|
|
|
static vid_model_funcs_t model_funcs = {
|
2021-01-19 04:05:39 +00:00
|
|
|
0,
|
2012-02-23 03:55:50 +00:00
|
|
|
sw_Mod_LoadLighting,
|
2021-01-18 23:26:53 +00:00
|
|
|
0,//Mod_SubdivideSurface,
|
|
|
|
0,//Mod_ProcessTexture,
|
2012-02-23 03:55:50 +00:00
|
|
|
|
2012-04-27 05:40:06 +00:00
|
|
|
Mod_LoadIQM,
|
2012-02-22 12:53:17 +00:00
|
|
|
Mod_LoadAliasModel,
|
|
|
|
Mod_LoadSpriteModel,
|
|
|
|
|
2012-02-23 03:55:50 +00:00
|
|
|
sw_Mod_MakeAliasModelDisplayLists,
|
[model] Make alias skin loading a batch operation
Really, this won't make all that much difference because alias models
with more than one skin are quite rare, and those with animated skin
groups are even rarer. However, for those models that do have more than
one skin, it will allow for reduced allocation overheads, and when
supported (glsl, vulkan, maybe gl), loading all the skins into an array
texture (since all skins are the same size, though external skins may
vary), but that's not implemented yet, this just wraps the old one skin
at a time code.
2022-04-04 06:38:27 +00:00
|
|
|
sw_Mod_LoadAllSkins,
|
2021-02-01 12:11:45 +00:00
|
|
|
0,
|
|
|
|
0,
|
2012-05-17 12:17:48 +00:00
|
|
|
sw_Mod_IQMFinish,
|
2012-02-23 03:55:50 +00:00
|
|
|
1,
|
2021-12-13 23:36:19 +00:00
|
|
|
sw_Mod_SpriteLoadFrames,
|
2012-02-23 03:55:50 +00:00
|
|
|
|
2022-05-12 13:46:31 +00:00
|
|
|
Skin_Free,
|
2012-02-22 12:53:17 +00:00
|
|
|
Skin_SetColormap,
|
|
|
|
Skin_SetSkin,
|
2012-02-23 03:09:19 +00:00
|
|
|
sw_Skin_SetupSkin,
|
2012-02-22 12:53:17 +00:00
|
|
|
Skin_SetTranslation,
|
2012-02-23 03:09:19 +00:00
|
|
|
sw_Skin_ProcessTranslation,
|
|
|
|
sw_Skin_InitTranslations,
|
2012-02-22 12:53:17 +00:00
|
|
|
};
|
|
|
|
|
2021-06-26 06:43:17 +00:00
|
|
|
static void
|
|
|
|
sw_vid_render_init (void)
|
|
|
|
{
|
|
|
|
if (!vr_data.vid->vid_internal->sw_context) {
|
|
|
|
Sys_Error ("Sorry, software rendering not supported by this program.");
|
|
|
|
}
|
|
|
|
sw_ctx = vr_data.vid->vid_internal->sw_context ();
|
|
|
|
|
2021-07-10 15:09:41 +00:00
|
|
|
vr_data.vid->vid_internal->data = sw_ctx;
|
|
|
|
vr_data.vid->vid_internal->set_palette = sw_vid_render_set_palette;
|
2022-03-29 04:08:58 +00:00
|
|
|
vr_data.vid->vid_internal->set_colormap = sw_vid_render_set_colormap;
|
2021-06-26 06:43:17 +00:00
|
|
|
vr_data.vid->vid_internal->choose_visual = sw_vid_render_choose_visual;
|
|
|
|
vr_data.vid->vid_internal->create_context = sw_vid_render_create_context;
|
|
|
|
|
|
|
|
vr_funcs = &sw_vid_render_funcs;
|
|
|
|
m_funcs = &model_funcs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
sw_vid_render_shutdown (void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
static void sw_bind_framebuffer (framebuffer_t *framebuffer);
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
static void
|
|
|
|
sw_begin_frame (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
|
|
|
if (r_numsurfs) {
|
2022-03-09 13:53:15 +00:00
|
|
|
int surfcount = surface_p - surfaces;
|
|
|
|
int max_surfs = surf_max - surfaces;
|
|
|
|
if (surfcount > r_maxsurfsseen)
|
|
|
|
r_maxsurfsseen = surfcount;
|
|
|
|
|
|
|
|
Sys_Printf ("Used %d of %d surfs; %d max\n",
|
|
|
|
surfcount, max_surfs, r_maxsurfsseen);
|
|
|
|
}
|
|
|
|
|
[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_numedges) {
|
2022-03-09 13:53:15 +00:00
|
|
|
int edgecount = edge_p - r_edges;
|
|
|
|
|
|
|
|
if (edgecount > r_maxedgesseen)
|
|
|
|
r_maxedgesseen = edgecount;
|
|
|
|
|
|
|
|
Sys_Printf ("Used %d of %d edges; %d max\n", edgecount,
|
|
|
|
r_numallocatededges, r_maxedgesseen);
|
|
|
|
}
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
sw_bind_framebuffer (0);
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
// do 3D refresh drawing, and then update the screen
|
|
|
|
if (vr_data.scr_fullupdate++ < vid.numpages) {
|
|
|
|
vr_data.scr_copyeverything = 1;
|
|
|
|
Draw_TileClear (0, 0, vid.width, vid.height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
sw_render_view (void)
|
|
|
|
{
|
|
|
|
R_RenderView ();
|
|
|
|
}
|
|
|
|
|
2022-03-17 15:14:48 +00:00
|
|
|
static void
|
|
|
|
sw_draw_transparent (void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
static void
|
2022-03-24 06:45:01 +00:00
|
|
|
sw_post_process (framebuffer_t *src)
|
2022-03-07 14:32:44 +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 (scr_fisheye) {
|
2022-03-24 06:45:01 +00:00
|
|
|
R_RenderFisheye (src);
|
|
|
|
} else if (r_dowarp) {
|
2022-03-25 15:29:34 +00:00
|
|
|
D_WarpScreen (src);
|
2022-03-24 03:22:27 +00:00
|
|
|
}
|
2022-03-07 14:32:44 +00:00
|
|
|
}
|
|
|
|
|
2022-03-24 06:45:01 +00:00
|
|
|
static void
|
|
|
|
sw_set_2d (int scaled)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
static void
|
|
|
|
sw_end_frame (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
|
|
|
if (r_reportsurfout && r_outofsurfaces)
|
2022-03-17 08:50:38 +00:00
|
|
|
Sys_Printf ("Short %d surfaces\n", r_outofsurfaces);
|
|
|
|
|
[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_reportedgeout && r_outofedges)
|
2022-03-17 08:50:38 +00:00
|
|
|
Sys_Printf ("Short roughly %d edges\n", r_outofedges * 2 / 3);
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
// update one of three areas
|
|
|
|
vrect_t vrect;
|
|
|
|
if (vr_data.scr_copyeverything) {
|
|
|
|
vrect.x = 0;
|
|
|
|
vrect.y = 0;
|
|
|
|
vrect.width = vid.width;
|
|
|
|
vrect.height = vid.height;
|
|
|
|
vrect.next = 0;
|
|
|
|
} else if (scr_copytop) {
|
|
|
|
vrect.x = 0;
|
|
|
|
vrect.y = 0;
|
|
|
|
vrect.width = vid.width;
|
|
|
|
vrect.height = vid.height - vr_data.lineadj;
|
|
|
|
vrect.next = 0;
|
|
|
|
} else {
|
|
|
|
vrect.x = vr_data.scr_view->xpos;
|
|
|
|
vrect.y = vr_data.scr_view->ypos;
|
|
|
|
vrect.width = vr_data.scr_view->xlen;
|
|
|
|
vrect.height = vr_data.scr_view->ylen;
|
|
|
|
vrect.next = 0;
|
|
|
|
}
|
|
|
|
sw_ctx->update (sw_ctx, &vrect);
|
|
|
|
}
|
|
|
|
|
2022-03-24 06:45:01 +00:00
|
|
|
static framebuffer_t *
|
|
|
|
sw_create_cube_map (int side)
|
|
|
|
{
|
|
|
|
size_t pixels = side * side; // per face
|
|
|
|
size_t size = sizeof (framebuffer_t) * 6;
|
|
|
|
size += sizeof (sw_framebuffer_t) * 6;
|
|
|
|
size += pixels * 6; // color buffer
|
|
|
|
// depth buffer, scan table and zspantable are shared between cube faces
|
|
|
|
// FIXME need *6 depth and zspan for multi-threaded
|
|
|
|
size += pixels * sizeof (short); // depth buffer
|
|
|
|
|
|
|
|
framebuffer_t *cube = malloc (size);
|
|
|
|
__auto_type buffer_base = (sw_framebuffer_t *) &cube[6];
|
|
|
|
byte *color_base = (byte *) &buffer_base[6];
|
|
|
|
short *depth_base = (short *) (color_base + 6 * pixels);
|
|
|
|
for (int i = 0; i < 6; i++) {
|
|
|
|
cube[i].width = side;
|
|
|
|
cube[i].height = side;
|
|
|
|
__auto_type buffer = buffer_base + i;
|
|
|
|
cube[i].buffer = buffer;
|
|
|
|
buffer->color = color_base + i * pixels;
|
|
|
|
buffer->depth = depth_base;
|
|
|
|
buffer->rowbytes = side;
|
|
|
|
}
|
|
|
|
return cube;
|
|
|
|
}
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
static framebuffer_t *
|
|
|
|
sw_create_frame_buffer (int width, int height)
|
|
|
|
{
|
|
|
|
size_t pixels = width * height;
|
|
|
|
size_t size = sizeof (framebuffer_t);
|
|
|
|
size += sizeof (sw_framebuffer_t);
|
|
|
|
size += pixels; // color buffer
|
|
|
|
size += pixels * sizeof (short); // depth buffer
|
|
|
|
|
|
|
|
framebuffer_t *fb = malloc (size);
|
|
|
|
fb->width = width;
|
|
|
|
fb->height = height;
|
|
|
|
__auto_type buffer = (sw_framebuffer_t *) &fb[1];
|
|
|
|
fb->buffer = buffer;
|
|
|
|
buffer->color = (byte *) &buffer[1];
|
|
|
|
buffer->depth = (short *) (buffer->color + pixels);
|
|
|
|
buffer->rowbytes = width;
|
|
|
|
return fb;
|
|
|
|
}
|
|
|
|
|
2022-03-30 05:55:32 +00:00
|
|
|
static void sw_set_viewport (const vrect_t *view);
|
|
|
|
|
2022-03-24 03:22:27 +00:00
|
|
|
static void
|
|
|
|
sw_bind_framebuffer (framebuffer_t *framebuffer)
|
|
|
|
{
|
|
|
|
int changed = 0;
|
|
|
|
|
|
|
|
if (!framebuffer) {
|
|
|
|
framebuffer = sw_ctx->framebuffer;
|
|
|
|
}
|
|
|
|
sw_framebuffer_t *fb = framebuffer->buffer;
|
|
|
|
|
|
|
|
if (!fb->depth) {
|
|
|
|
fb->depth = malloc (framebuffer->width * framebuffer->height
|
|
|
|
* sizeof (short));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (d_zbuffer != fb->depth
|
|
|
|
|| d_zwidth != framebuffer->width || d_height != framebuffer->height) {
|
|
|
|
d_zwidth = framebuffer->width;
|
|
|
|
d_zrowbytes = d_zwidth * sizeof (short);
|
|
|
|
for (unsigned i = 0; i < framebuffer->height; i++) {
|
|
|
|
zspantable[i] = fb->depth + i * d_zwidth;
|
|
|
|
}
|
|
|
|
changed = 1;
|
|
|
|
}
|
|
|
|
if (d_rowbytes != fb->rowbytes || d_height != framebuffer->height) {
|
|
|
|
d_rowbytes = fb->rowbytes;
|
|
|
|
d_height = framebuffer->height;
|
|
|
|
for (unsigned i = 0; i < framebuffer->height; i++) {
|
|
|
|
d_scantable[i] = i * d_rowbytes;
|
|
|
|
}
|
|
|
|
changed = 1;
|
|
|
|
}
|
|
|
|
d_viewbuffer = fb->color;
|
|
|
|
d_zbuffer = fb->depth;
|
|
|
|
|
|
|
|
if (changed) {
|
|
|
|
vrect_t r = { 0, 0, framebuffer->width, framebuffer->height };
|
2022-03-30 05:55:32 +00:00
|
|
|
sw_set_viewport (&r);
|
2022-03-24 03:22:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-25 15:29:34 +00:00
|
|
|
static void
|
|
|
|
sw_set_viewport (const vrect_t *view)
|
|
|
|
{
|
2022-03-30 05:55:32 +00:00
|
|
|
#define SHIFT20(x) (((x) << 20) + (1 << 19) - 1)
|
|
|
|
r_refdef.vrectright = view->x + view->width;
|
|
|
|
r_refdef.vrectbottom = view->y + view->height;
|
|
|
|
r_refdef.vrectx_adj_shift20 = SHIFT20 (view->x);
|
|
|
|
r_refdef.vrectright_adj_shift20 = SHIFT20 (r_refdef.vrectright);
|
|
|
|
|
|
|
|
r_refdef.fvrectx = (float) view->x;
|
|
|
|
r_refdef.fvrecty = (float) view->y;
|
|
|
|
r_refdef.fvrectright = (float) r_refdef.vrectright;
|
|
|
|
r_refdef.fvrectbottom = (float) r_refdef.vrectbottom;
|
|
|
|
|
|
|
|
r_refdef.fvrectx_adj = (float) view->x - 0.5;
|
|
|
|
r_refdef.fvrecty_adj = (float) view->y - 0.5;
|
|
|
|
r_refdef.fvrectright_adj = (float) r_refdef.vrectright - 0.5;
|
|
|
|
r_refdef.fvrectbottom_adj = (float) r_refdef.vrectbottom - 0.5;
|
|
|
|
|
|
|
|
int aleft = view->x * r_aliasuvscale;
|
|
|
|
int atop = view->y * r_aliasuvscale;
|
|
|
|
int awidth = view->width * r_aliasuvscale;
|
|
|
|
int aheight = view->height * r_aliasuvscale;
|
|
|
|
r_refdef.aliasvrectleft = aleft;
|
|
|
|
r_refdef.aliasvrecttop = atop;
|
|
|
|
r_refdef.aliasvrectright = aleft + awidth;
|
|
|
|
r_refdef.aliasvrectbottom = atop + aheight;
|
|
|
|
|
|
|
|
// 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 = view->width * XCENTERING + view->x - 0.5;
|
|
|
|
ycenter = view->height * YCENTERING + view->y - 0.5;
|
|
|
|
aliasxcenter = xcenter * r_aliasuvscale;
|
|
|
|
aliasycenter = ycenter * r_aliasuvscale;
|
|
|
|
|
|
|
|
r_refdef.vrect.x = view->x;
|
|
|
|
r_refdef.vrect.y = view->y;
|
|
|
|
r_refdef.vrect.width = view->width;
|
|
|
|
r_refdef.vrect.height = view->height;
|
|
|
|
|
|
|
|
D_ViewChanged ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
sw_set_fov (float x, float y)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
float res_scale;
|
|
|
|
|
|
|
|
r_viewchanged = true;
|
|
|
|
|
|
|
|
// 320*200 1.0 pixelAspect = 1.6 aspect
|
|
|
|
// 320*240 1.0 pixelAspect = 1.3333 aspect
|
|
|
|
// proper 320*200 pixelAspect = 0.8333333
|
|
|
|
pixelAspect = 1;//FIXME vid.aspect;
|
|
|
|
|
|
|
|
float hFOV = 2 * x;
|
|
|
|
float vFOV = 2 * y * pixelAspect;
|
|
|
|
|
|
|
|
// general perspective scaling
|
|
|
|
xscale = r_refdef.vrect.width / hFOV;
|
|
|
|
yscale = xscale * pixelAspect;
|
|
|
|
xscaleinv = 1.0 / xscale;
|
|
|
|
yscaleinv = 1.0 / yscale;
|
|
|
|
// perspective scaling for alias models
|
|
|
|
aliasxscale = xscale * r_aliasuvscale;
|
|
|
|
aliasyscale = yscale * r_aliasuvscale;
|
|
|
|
// perspective scaling for paricle position
|
|
|
|
xscaleshrink = (r_refdef.vrect.width - 6) / hFOV;
|
|
|
|
yscaleshrink = xscaleshrink * pixelAspect;
|
|
|
|
|
|
|
|
// left side clip
|
|
|
|
screenedge[0].normal[0] = -1.0 / (XCENTERING * hFOV);
|
|
|
|
screenedge[0].normal[1] = 0;
|
|
|
|
screenedge[0].normal[2] = 1;
|
|
|
|
screenedge[0].type = PLANE_ANYZ;
|
|
|
|
|
|
|
|
// right side clip
|
|
|
|
screenedge[1].normal[0] = 1.0 / ((1.0 - XCENTERING) * hFOV);
|
|
|
|
screenedge[1].normal[1] = 0;
|
|
|
|
screenedge[1].normal[2] = 1;
|
|
|
|
screenedge[1].type = PLANE_ANYZ;
|
|
|
|
|
|
|
|
// top side clip
|
|
|
|
screenedge[2].normal[0] = 0;
|
|
|
|
screenedge[2].normal[1] = -1.0 / (YCENTERING * vFOV);
|
|
|
|
screenedge[2].normal[2] = 1;
|
|
|
|
screenedge[2].type = PLANE_ANYZ;
|
|
|
|
|
|
|
|
// bottom side clip
|
|
|
|
screenedge[3].normal[0] = 0;
|
|
|
|
screenedge[3].normal[1] = 1.0 / ((1.0 - YCENTERING) * vFOV);
|
|
|
|
screenedge[3].normal[2] = 1;
|
|
|
|
screenedge[3].type = PLANE_ANYZ;
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
VectorNormalize (screenedge[i].normal);
|
|
|
|
|
|
|
|
res_scale = sqrt ((double) (r_refdef.vrect.width * r_refdef.vrect.height) /
|
|
|
|
(320.0 * 152.0)) * (2.0 / hFOV);
|
[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_aliastransition = r_aliastransbase * res_scale;
|
|
|
|
r_resfudge = r_aliastransadj * res_scale;
|
2022-03-25 15:29:34 +00:00
|
|
|
}
|
|
|
|
|
2022-03-31 10:07:15 +00:00
|
|
|
static void
|
|
|
|
sw_capture_screen (capfunc_t callback, void *data)
|
|
|
|
{
|
|
|
|
int count, x, y;
|
|
|
|
tex_t *tex;
|
|
|
|
const byte *src;
|
|
|
|
byte *dst;
|
|
|
|
framebuffer_t *fb = sw_ctx->framebuffer;
|
|
|
|
|
|
|
|
count = fb->width * fb->height;
|
|
|
|
tex = malloc (sizeof (tex_t) + count * 3);
|
|
|
|
if (tex) {
|
|
|
|
tex->data = (byte *) (tex + 1);
|
|
|
|
tex->width = fb->width;
|
|
|
|
tex->height = fb->height;
|
|
|
|
tex->format = tex_rgb;
|
|
|
|
tex->palette = 0;
|
2022-03-31 11:00:13 +00:00
|
|
|
tex->flagbits = 0;
|
|
|
|
tex->loaded = 1;
|
2022-03-31 10:07:15 +00:00
|
|
|
src = ((sw_framebuffer_t *) fb->buffer)->color;
|
|
|
|
int rowbytes = ((sw_framebuffer_t *) fb->buffer)->rowbytes;
|
|
|
|
for (y = 0; y < tex->height; y++) {
|
2022-03-31 11:00:13 +00:00
|
|
|
dst = tex->data + y * tex->width * 3;
|
2022-03-31 10:07:15 +00:00
|
|
|
for (x = 0; x < tex->width; x++) {
|
|
|
|
byte c = src[x];
|
2022-03-31 11:00:13 +00:00
|
|
|
*dst++ = vid.basepal[c * 3 + 0];
|
|
|
|
*dst++ = vid.basepal[c * 3 + 1];
|
|
|
|
*dst++ = vid.basepal[c * 3 + 2];
|
2022-03-31 10:07:15 +00:00
|
|
|
}
|
|
|
|
src += rowbytes;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
callback (tex, data);
|
|
|
|
}
|
|
|
|
|
2012-02-22 12:53:17 +00:00
|
|
|
vid_render_funcs_t sw_vid_render_funcs = {
|
2021-06-26 06:43:17 +00:00
|
|
|
sw_vid_render_init,
|
2012-02-22 12:53:17 +00:00
|
|
|
Draw_Character,
|
|
|
|
Draw_String,
|
|
|
|
Draw_nString,
|
|
|
|
Draw_AltString,
|
|
|
|
Draw_ConsoleBackground,
|
|
|
|
Draw_Crosshair,
|
|
|
|
Draw_CrosshairAt,
|
|
|
|
Draw_TileClear,
|
|
|
|
Draw_Fill,
|
|
|
|
Draw_TextBox,
|
|
|
|
Draw_FadeScreen,
|
|
|
|
Draw_BlendScreen,
|
|
|
|
Draw_CachePic,
|
|
|
|
Draw_UncachePic,
|
|
|
|
Draw_MakePic,
|
|
|
|
Draw_DestroyPic,
|
|
|
|
Draw_PicFromWad,
|
|
|
|
Draw_Pic,
|
|
|
|
Draw_Picf,
|
|
|
|
Draw_SubPic,
|
|
|
|
|
2021-12-19 05:47:25 +00:00
|
|
|
sw_ParticleSystem,
|
2012-04-10 04:27:53 +00:00
|
|
|
sw_R_Init,
|
2012-02-22 12:53:17 +00:00
|
|
|
R_ClearState,
|
|
|
|
R_LoadSkys,
|
2022-05-05 05:41:46 +00:00
|
|
|
R_NewScene,
|
2012-02-22 12:53:17 +00:00
|
|
|
R_LineGraph,
|
2022-03-07 14:32:44 +00:00
|
|
|
sw_begin_frame,
|
|
|
|
sw_render_view,
|
2022-03-17 08:50:38 +00:00
|
|
|
R_DrawEntitiesOnList,
|
|
|
|
R_DrawParticles,
|
2022-03-17 15:14:48 +00:00
|
|
|
sw_draw_transparent,
|
2022-03-24 06:45:01 +00:00
|
|
|
sw_post_process,
|
2022-03-07 14:32:44 +00:00
|
|
|
sw_set_2d,
|
|
|
|
sw_end_frame,
|
2022-03-24 03:22:27 +00:00
|
|
|
|
2022-03-24 06:45:01 +00:00
|
|
|
sw_create_cube_map,
|
2022-03-24 03:22:27 +00:00
|
|
|
sw_create_frame_buffer,
|
|
|
|
sw_bind_framebuffer,
|
2022-03-25 15:29:34 +00:00
|
|
|
sw_set_viewport,
|
2022-03-30 05:55:32 +00:00
|
|
|
sw_set_fov,
|
2022-03-24 03:22:27 +00:00
|
|
|
|
2022-03-31 10:07:15 +00:00
|
|
|
sw_capture_screen,
|
|
|
|
|
2012-02-22 12:53:17 +00:00
|
|
|
&model_funcs
|
|
|
|
};
|
|
|
|
|
2012-02-14 13:19:17 +00:00
|
|
|
static general_funcs_t plugin_info_general_funcs = {
|
2021-06-26 06:43:17 +00:00
|
|
|
.shutdown = sw_vid_render_shutdown,
|
2012-02-14 13:19:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static general_data_t plugin_info_general_data;
|
|
|
|
|
|
|
|
static plugin_funcs_t plugin_info_funcs = {
|
2021-06-26 06:43:17 +00:00
|
|
|
.general = &plugin_info_general_funcs,
|
|
|
|
.vid_render = &sw_vid_render_funcs,
|
2012-02-14 13:19:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static plugin_data_t plugin_info_data = {
|
2021-06-26 06:43:17 +00:00
|
|
|
.general = &plugin_info_general_data,
|
|
|
|
.vid_render = &vid_render_data,
|
2012-02-14 13:19:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static plugin_t plugin_info = {
|
2019-07-07 05:57:56 +00:00
|
|
|
qfp_vid_render,
|
2012-02-14 13:19:17 +00:00
|
|
|
0,
|
|
|
|
QFPLUGIN_VERSION,
|
|
|
|
"0.1",
|
|
|
|
"SW Renderer",
|
|
|
|
"Copyright (C) 1996-1997 Id Software, Inc.\n"
|
|
|
|
"Copyright (C) 1999-2012 contributors of the QuakeForge project\n"
|
|
|
|
"Please see the file \"AUTHORS\" for a list of contributors",
|
|
|
|
&plugin_info_funcs,
|
|
|
|
&plugin_info_data,
|
|
|
|
};
|
|
|
|
|
|
|
|
PLUGIN_INFO(vid_render, sw)
|
|
|
|
{
|
|
|
|
return &plugin_info;
|
|
|
|
}
|