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"
|
|
|
|
|
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
|
|
|
|
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,
|
|
|
|
sw_Mod_LoadSkin,
|
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
|
|
|
|
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)
|
|
|
|
{
|
2022-03-09 13:53:15 +00:00
|
|
|
if (r_numsurfs->int_val) {
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r_numedges->int_val) {
|
|
|
|
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
|
|
|
{
|
2022-03-24 06:45:01 +00:00
|
|
|
if (scr_fisheye->int_val) {
|
|
|
|
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)
|
|
|
|
{
|
2022-03-17 08:50:38 +00:00
|
|
|
if (r_reportsurfout->int_val && r_outofsurfaces)
|
|
|
|
Sys_Printf ("Short %d surfaces\n", r_outofsurfaces);
|
|
|
|
|
|
|
|
if (r_reportedgeout->int_val && r_outofedges)
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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 };
|
|
|
|
R_SetVrect (&r, &r_refdef.vrect, 0);
|
|
|
|
R_ViewChanged ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-25 15:29:34 +00:00
|
|
|
static void
|
|
|
|
sw_set_viewport (const vrect_t *view)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
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,
|
|
|
|
|
2022-03-07 14:32:44 +00:00
|
|
|
sw_SCR_CaptureBGR,
|
2012-02-22 12:53:17 +00:00
|
|
|
|
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,
|
|
|
|
R_NewMap,
|
|
|
|
R_LineGraph,
|
2012-04-11 13:45:23 +00:00
|
|
|
R_ViewChanged,
|
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-24 03:22:27 +00:00
|
|
|
|
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;
|
|
|
|
}
|