2016-12-07 10:40:59 +00:00
|
|
|
/*
|
|
|
|
** r_draw.cpp
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 1998-2016 Randy Heit
|
|
|
|
** Copyright 2016 Magnus Norddahl
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
#include "templates.h"
|
|
|
|
#include "doomdef.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "v_video.h"
|
|
|
|
#include "doomstat.h"
|
|
|
|
#include "st_stuff.h"
|
|
|
|
#include "g_game.h"
|
|
|
|
#include "g_level.h"
|
|
|
|
#include "r_data/r_translate.h"
|
|
|
|
#include "v_palette.h"
|
|
|
|
#include "r_data/colormaps.h"
|
2016-12-07 08:34:49 +00:00
|
|
|
#include "r_draw.h"
|
2016-12-07 17:28:06 +00:00
|
|
|
#include "r_draw_rgba.h"
|
2016-12-07 08:34:49 +00:00
|
|
|
#include "r_draw_pal.h"
|
|
|
|
#include "r_thread.h"
|
2016-12-28 00:56:37 +00:00
|
|
|
#include "swrenderer/r_main.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-23 22:44:52 +00:00
|
|
|
CVAR(Bool, r_dynlights, 1, CVAR_ARCHIVE | CVAR_GLOBALCONFIG);
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
namespace swrenderer
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
double dc_texturemid;
|
2016-12-21 06:33:28 +00:00
|
|
|
FLightNode *dc_light_list;
|
2016-12-07 08:34:49 +00:00
|
|
|
|
|
|
|
int ylookup[MAXHEIGHT];
|
|
|
|
uint8_t shadetables[NUMCOLORMAPS * 16 * 256];
|
|
|
|
FDynamicColormap ShadeFakeColormap[16];
|
|
|
|
uint8_t identitymap[256];
|
|
|
|
FDynamicColormap identitycolormap;
|
|
|
|
int fuzzoffset[FUZZTABLE + 1];
|
|
|
|
int fuzzpos;
|
|
|
|
int fuzzviewheight;
|
|
|
|
|
2016-12-27 22:53:47 +00:00
|
|
|
DrawerFunc colfunc;
|
|
|
|
DrawerFunc basecolfunc;
|
|
|
|
DrawerFunc fuzzcolfunc;
|
|
|
|
DrawerFunc transcolfunc;
|
|
|
|
DrawerFunc spanfunc;
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
namespace drawerargs
|
|
|
|
{
|
|
|
|
int dc_pitch;
|
|
|
|
lighttable_t *dc_colormap;
|
2016-12-07 17:28:06 +00:00
|
|
|
FSWColormap *dc_fcolormap;
|
|
|
|
ShadeConstants dc_shade_constants;
|
|
|
|
fixed_t dc_light;
|
2016-12-07 08:34:49 +00:00
|
|
|
int dc_x;
|
|
|
|
int dc_yl;
|
|
|
|
int dc_yh;
|
|
|
|
fixed_t dc_iscale;
|
|
|
|
fixed_t dc_texturefrac;
|
|
|
|
uint32_t dc_textureheight;
|
|
|
|
int dc_color;
|
|
|
|
uint32_t dc_srccolor;
|
|
|
|
uint32_t dc_srccolor_bgra;
|
|
|
|
uint32_t *dc_srcblend;
|
|
|
|
uint32_t *dc_destblend;
|
|
|
|
fixed_t dc_srcalpha;
|
|
|
|
fixed_t dc_destalpha;
|
|
|
|
const uint8_t *dc_source;
|
|
|
|
const uint8_t *dc_source2;
|
|
|
|
uint32_t dc_texturefracx;
|
|
|
|
uint8_t *dc_translation;
|
|
|
|
uint8_t *dc_dest;
|
|
|
|
uint8_t *dc_destorg;
|
|
|
|
int dc_destheight;
|
|
|
|
int dc_count;
|
2017-01-02 05:52:50 +00:00
|
|
|
FVector3 dc_normal;
|
2016-12-21 06:33:28 +00:00
|
|
|
FVector3 dc_viewpos;
|
|
|
|
FVector3 dc_viewpos_step;
|
|
|
|
TriLight *dc_lights;
|
|
|
|
int dc_num_lights;
|
2016-12-19 20:13:29 +00:00
|
|
|
uint32_t dc_wall_texturefrac[4];
|
|
|
|
uint32_t dc_wall_iscale[4];
|
|
|
|
uint8_t *dc_wall_colormap[4];
|
|
|
|
fixed_t dc_wall_light[4];
|
|
|
|
const uint8_t *dc_wall_source[4];
|
|
|
|
const uint8_t *dc_wall_source2[4];
|
|
|
|
uint32_t dc_wall_texturefracx[4];
|
|
|
|
uint32_t dc_wall_sourceheight[4];
|
|
|
|
int dc_wall_fracbits;
|
2016-12-07 08:34:49 +00:00
|
|
|
int ds_y;
|
|
|
|
int ds_x1;
|
|
|
|
int ds_x2;
|
|
|
|
lighttable_t * ds_colormap;
|
2016-12-07 17:28:06 +00:00
|
|
|
FSWColormap *ds_fcolormap;
|
|
|
|
ShadeConstants ds_shade_constants;
|
2016-12-07 08:34:49 +00:00
|
|
|
dsfixed_t ds_light;
|
|
|
|
dsfixed_t ds_xfrac;
|
|
|
|
dsfixed_t ds_yfrac;
|
|
|
|
dsfixed_t ds_xstep;
|
|
|
|
dsfixed_t ds_ystep;
|
|
|
|
int ds_xbits;
|
|
|
|
int ds_ybits;
|
|
|
|
fixed_t ds_alpha;
|
|
|
|
double ds_lod;
|
|
|
|
const uint8_t *ds_source;
|
2016-12-07 17:28:06 +00:00
|
|
|
bool ds_source_mipmapped;
|
2016-12-07 08:34:49 +00:00
|
|
|
int ds_color;
|
2016-12-07 17:28:06 +00:00
|
|
|
bool drawer_needs_pal_input;
|
2016-12-07 08:34:49 +00:00
|
|
|
unsigned int dc_tspans[4][MAXHEIGHT];
|
|
|
|
unsigned int *dc_ctspan[4];
|
|
|
|
unsigned int *horizspan[4];
|
|
|
|
}
|
|
|
|
|
2016-12-27 07:47:51 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
SWPixelFormatDrawers *active_drawers;
|
|
|
|
SWPalDrawers pal_drawers;
|
|
|
|
SWTruecolorDrawers tc_drawers;
|
|
|
|
}
|
|
|
|
|
2016-12-27 07:54:37 +00:00
|
|
|
SWPixelFormatDrawers *R_Drawers()
|
2016-12-27 07:47:51 +00:00
|
|
|
{
|
|
|
|
return active_drawers;
|
|
|
|
}
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
void R_InitColumnDrawers()
|
|
|
|
{
|
2016-12-27 07:47:51 +00:00
|
|
|
if (r_swtruecolor)
|
|
|
|
active_drawers = &tc_drawers;
|
|
|
|
else
|
|
|
|
active_drawers = &pal_drawers;
|
|
|
|
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = basecolfunc = &SWPixelFormatDrawers::DrawColumn;
|
|
|
|
fuzzcolfunc = &SWPixelFormatDrawers::DrawFuzzColumn;
|
|
|
|
transcolfunc = &SWPixelFormatDrawers::DrawTranslatedColumn;
|
|
|
|
spanfunc = &SWPixelFormatDrawers::DrawSpan;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void R_InitShadeMaps()
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
// set up shading tables for shaded columns
|
|
|
|
// 16 colormap sets, progressing from full alpha to minimum visible alpha
|
|
|
|
|
|
|
|
uint8_t *table = shadetables;
|
|
|
|
|
|
|
|
// Full alpha
|
|
|
|
for (i = 0; i < 16; ++i)
|
|
|
|
{
|
|
|
|
ShadeFakeColormap[i].Color = ~0u;
|
|
|
|
ShadeFakeColormap[i].Desaturate = ~0u;
|
|
|
|
ShadeFakeColormap[i].Next = NULL;
|
|
|
|
ShadeFakeColormap[i].Maps = table;
|
|
|
|
|
|
|
|
for (j = 0; j < NUMCOLORMAPS; ++j)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
int a = (NUMCOLORMAPS - j) * 256 / NUMCOLORMAPS * (16 - i);
|
|
|
|
for (int k = 0; k < 256; ++k)
|
|
|
|
{
|
|
|
|
uint8_t v = (((k + 2) * a) + 256) >> 14;
|
|
|
|
table[k] = MIN<uint8_t>(v, 64);
|
|
|
|
}
|
|
|
|
table += 256;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-07 08:34:49 +00:00
|
|
|
for (i = 0; i < NUMCOLORMAPS * 16 * 256; ++i)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
assert(shadetables[i] <= 64);
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
// Set up a guaranteed identity map
|
|
|
|
for (i = 0; i < 256; ++i)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
identitymap[i] = i;
|
|
|
|
}
|
2016-12-15 16:37:05 +00:00
|
|
|
identitycolormap.Maps = identitymap;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
void R_InitFuzzTable(int fuzzoff)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
/*
|
|
|
|
FUZZOFF,-FUZZOFF,FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,
|
|
|
|
FUZZOFF,FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,
|
|
|
|
FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,-FUZZOFF,-FUZZOFF,-FUZZOFF,
|
|
|
|
FUZZOFF,-FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,
|
|
|
|
FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,-FUZZOFF,FUZZOFF,
|
|
|
|
FUZZOFF,-FUZZOFF,-FUZZOFF,-FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,
|
|
|
|
FUZZOFF,FUZZOFF,-FUZZOFF,FUZZOFF,FUZZOFF,-FUZZOFF,FUZZOFF
|
|
|
|
*/
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
static const int8_t fuzzinit[FUZZTABLE] = {
|
|
|
|
1,-1, 1,-1, 1, 1,-1,
|
|
|
|
1, 1,-1, 1, 1, 1,-1,
|
|
|
|
1, 1, 1,-1,-1,-1,-1,
|
|
|
|
1,-1,-1, 1, 1, 1, 1,-1,
|
|
|
|
1,-1, 1, 1,-1,-1, 1,
|
|
|
|
1,-1,-1,-1,-1, 1, 1,
|
|
|
|
1, 1,-1, 1, 1,-1, 1
|
|
|
|
};
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
for (int i = 0; i < FUZZTABLE; i++)
|
|
|
|
{
|
|
|
|
fuzzoffset[i] = fuzzinit[i] * fuzzoff;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
namespace
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
bool R_SetBlendFunc(int op, fixed_t fglevel, fixed_t bglevel, int flags)
|
|
|
|
{
|
|
|
|
using namespace drawerargs;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
// r_drawtrans is a seriously bad thing to turn off. I wonder if I should
|
|
|
|
// just remove it completely.
|
|
|
|
if (!r_drawtrans || (op == STYLEOP_Add && fglevel == FRACUNIT && bglevel == 0 && !(flags & STYLEF_InvertSource)))
|
|
|
|
{
|
|
|
|
if (flags & STYLEF_ColorIsFixed)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::FillColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (dc_translation == NULL)
|
|
|
|
{
|
|
|
|
colfunc = basecolfunc;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
colfunc = transcolfunc;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (flags & STYLEF_InvertSource)
|
|
|
|
{
|
|
|
|
dc_srcblend = Col2RGB8_Inverse[fglevel >> 10];
|
|
|
|
dc_destblend = Col2RGB8_LessPrecision[bglevel >> 10];
|
|
|
|
dc_srcalpha = fglevel;
|
|
|
|
dc_destalpha = bglevel;
|
|
|
|
}
|
|
|
|
else if (op == STYLEOP_Add && fglevel + bglevel <= FRACUNIT)
|
|
|
|
{
|
|
|
|
dc_srcblend = Col2RGB8[fglevel >> 10];
|
|
|
|
dc_destblend = Col2RGB8[bglevel >> 10];
|
|
|
|
dc_srcalpha = fglevel;
|
|
|
|
dc_destalpha = bglevel;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dc_srcblend = Col2RGB8_LessPrecision[fglevel >> 10];
|
|
|
|
dc_destblend = Col2RGB8_LessPrecision[bglevel >> 10];
|
|
|
|
dc_srcalpha = fglevel;
|
|
|
|
dc_destalpha = bglevel;
|
|
|
|
}
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
case STYLEOP_Add:
|
|
|
|
if (fglevel == 0 && bglevel == FRACUNIT)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (fglevel + bglevel <= FRACUNIT)
|
|
|
|
{ // Colors won't overflow when added
|
|
|
|
if (flags & STYLEF_ColorIsFixed)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::FillAddColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (dc_translation == NULL)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawAddColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawTranslatedAddColumn;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Colors might overflow when added
|
|
|
|
if (flags & STYLEF_ColorIsFixed)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::FillAddClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (dc_translation == NULL)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawAddClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawAddClampTranslatedColumn;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
case STYLEOP_Sub:
|
|
|
|
if (flags & STYLEF_ColorIsFixed)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::FillSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (dc_translation == NULL)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawSubClampTranslatedColumn;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
return true;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
case STYLEOP_RevSub:
|
|
|
|
if (fglevel == 0 && bglevel == FRACUNIT)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (flags & STYLEF_ColorIsFixed)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::FillRevSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (dc_translation == NULL)
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawRevSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawRevSubClampTranslatedColumn;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
return true;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
fixed_t GetAlpha(int type, fixed_t alpha)
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case STYLEALPHA_Zero: return 0;
|
|
|
|
case STYLEALPHA_One: return OPAQUE;
|
|
|
|
case STYLEALPHA_Src: return alpha;
|
|
|
|
case STYLEALPHA_InvSrc: return OPAQUE - alpha;
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
FDynamicColormap *basecolormapsave;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-25 04:15:23 +00:00
|
|
|
bool R_SetPatchStyle(FRenderStyle style, fixed_t alpha, int translation, uint32_t color)
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
|
|
|
using namespace drawerargs;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
fixed_t fglevel, bglevel;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = false;
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
style.CheckFuzz();
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
if (style.BlendOp == STYLEOP_Shadow)
|
|
|
|
{
|
|
|
|
style = LegacyRenderStyles[STYLE_TranslucentStencil];
|
|
|
|
alpha = TRANSLUC33;
|
|
|
|
color = 0;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-26 20:37:22 +00:00
|
|
|
if (style.Flags & STYLEF_ForceAlpha)
|
|
|
|
{
|
|
|
|
alpha = clamp<fixed_t>(alpha, 0, OPAQUE);
|
|
|
|
}
|
|
|
|
else if (style.Flags & STYLEF_TransSoulsAlpha)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
alpha = fixed_t(transsouls * OPAQUE);
|
|
|
|
}
|
|
|
|
else if (style.Flags & STYLEF_Alpha1)
|
|
|
|
{
|
|
|
|
alpha = FRACUNIT;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
alpha = clamp<fixed_t>(alpha, 0, OPAQUE);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2016-12-07 08:34:49 +00:00
|
|
|
|
|
|
|
if (translation != -1)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
dc_translation = NULL;
|
|
|
|
if (translation != 0)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
FRemapTable *table = TranslationToTable(translation);
|
|
|
|
if (table != NULL && !table->Inactive)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
if (r_swtruecolor)
|
|
|
|
dc_translation = (uint8_t*)table->Palette;
|
|
|
|
else
|
|
|
|
dc_translation = table->Remap;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
basecolormapsave = basecolormap;
|
|
|
|
|
|
|
|
// Check for special modes
|
|
|
|
if (style.BlendOp == STYLEOP_Fuzz)
|
|
|
|
{
|
|
|
|
colfunc = fuzzcolfunc;
|
2016-12-25 04:15:23 +00:00
|
|
|
return true;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
else if (style == LegacyRenderStyles[STYLE_Shaded])
|
|
|
|
{
|
|
|
|
// Shaded drawer only gets 16 levels of alpha because it saves memory.
|
|
|
|
if ((alpha >>= 12) == 0)
|
2016-12-25 04:15:23 +00:00
|
|
|
return false;
|
2016-12-27 22:53:47 +00:00
|
|
|
colfunc = &SWPixelFormatDrawers::DrawShadedColumn;
|
2016-12-07 17:28:06 +00:00
|
|
|
drawer_needs_pal_input = true;
|
|
|
|
dc_color = fixedcolormap ? fixedcolormap->Maps[APART(color)] : basecolormap->Maps[APART(color)];
|
|
|
|
basecolormap = &ShadeFakeColormap[16 - alpha];
|
2016-12-07 08:34:49 +00:00
|
|
|
if (fixedlightlev >= 0 && fixedcolormap == NULL)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
R_SetColorMapLight(basecolormap, 0, FIXEDLIGHT2SHADE(fixedlightlev));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
R_SetColorMapLight(basecolormap, 0, 0);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2016-12-25 04:15:23 +00:00
|
|
|
return true;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
fglevel = GetAlpha(style.SrcAlpha, alpha);
|
|
|
|
bglevel = GetAlpha(style.DestAlpha, alpha);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
if (style.Flags & STYLEF_ColorIsFixed)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
uint32_t x = fglevel >> 10;
|
|
|
|
uint32_t r = RPART(color);
|
|
|
|
uint32_t g = GPART(color);
|
|
|
|
uint32_t b = BPART(color);
|
|
|
|
// dc_color is used by the rt_* routines. It is indexed into dc_srcblend.
|
2016-12-17 09:11:52 +00:00
|
|
|
dc_color = RGB256k.RGB[r >> 2][g >> 2][b >> 2];
|
2016-12-07 08:34:49 +00:00
|
|
|
if (style.Flags & STYLEF_InvertSource)
|
|
|
|
{
|
|
|
|
r = 255 - r;
|
|
|
|
g = 255 - g;
|
|
|
|
b = 255 - b;
|
|
|
|
}
|
|
|
|
uint32_t alpha = clamp(fglevel >> (FRACBITS - 8), 0, 255);
|
|
|
|
dc_srccolor_bgra = (alpha << 24) | (r << 16) | (g << 8) | b;
|
|
|
|
// dc_srccolor is used by the R_Fill* routines. It is premultiplied
|
|
|
|
// with the alpha.
|
|
|
|
dc_srccolor = ((((r*x) >> 4) << 20) | ((g*x) >> 4) | ((((b)*x) >> 4) << 10)) & 0x3feffbff;
|
2016-12-07 17:28:06 +00:00
|
|
|
R_SetColorMapLight(&identitycolormap, 0, 0);
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
if (!R_SetBlendFunc(style.BlendOp, fglevel, bglevel, style.Flags))
|
|
|
|
{
|
2016-12-25 04:15:23 +00:00
|
|
|
return false;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-25 04:15:23 +00:00
|
|
|
return true;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-25 04:15:23 +00:00
|
|
|
bool R_SetPatchStyle(FRenderStyle style, float alpha, int translation, uint32_t color)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
return R_SetPatchStyle(style, FLOAT2FIXED(alpha), translation, color);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
void R_FinishSetPatchStyle()
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
basecolormap = basecolormapsave;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-27 22:53:47 +00:00
|
|
|
DrawerFunc R_GetTransMaskDrawer()
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
if (colfunc == &SWPixelFormatDrawers::DrawAddColumn)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
return &SWPixelFormatDrawers::DrawWallAddColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-27 22:53:47 +00:00
|
|
|
if (colfunc == &SWPixelFormatDrawers::DrawAddClampColumn)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
return &SWPixelFormatDrawers::DrawWallAddClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-27 22:53:47 +00:00
|
|
|
if (colfunc == &SWPixelFormatDrawers::DrawSubClampColumn)
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
return &SWPixelFormatDrawers::DrawWallSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-27 22:53:47 +00:00
|
|
|
if (colfunc == &SWPixelFormatDrawers::DrawRevSubClampColumn)
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
2016-12-27 22:53:47 +00:00
|
|
|
return &SWPixelFormatDrawers::DrawWallRevSubClampColumn;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-27 22:53:47 +00:00
|
|
|
return nullptr;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
void R_SetColorMapLight(FSWColormap *base_colormap, float light, int shade)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
using namespace drawerargs;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
dc_fcolormap = base_colormap;
|
|
|
|
if (r_swtruecolor)
|
|
|
|
{
|
|
|
|
dc_shade_constants.light_red = dc_fcolormap->Color.r * 256 / 255;
|
|
|
|
dc_shade_constants.light_green = dc_fcolormap->Color.g * 256 / 255;
|
|
|
|
dc_shade_constants.light_blue = dc_fcolormap->Color.b * 256 / 255;
|
|
|
|
dc_shade_constants.light_alpha = dc_fcolormap->Color.a * 256 / 255;
|
|
|
|
dc_shade_constants.fade_red = dc_fcolormap->Fade.r;
|
|
|
|
dc_shade_constants.fade_green = dc_fcolormap->Fade.g;
|
|
|
|
dc_shade_constants.fade_blue = dc_fcolormap->Fade.b;
|
|
|
|
dc_shade_constants.fade_alpha = dc_fcolormap->Fade.a;
|
|
|
|
dc_shade_constants.desaturate = MIN(abs(dc_fcolormap->Desaturate), 255) * 255 / 256;
|
|
|
|
dc_shade_constants.simple_shade = (dc_fcolormap->Color.d == 0x00ffffff && dc_fcolormap->Fade.d == 0x00000000 && dc_fcolormap->Desaturate == 0);
|
|
|
|
dc_colormap = base_colormap->Maps;
|
|
|
|
dc_light = LIGHTSCALE(light, shade);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
dc_colormap = base_colormap->Maps + (GETPALOOKUP(light, shade) << COLORMAPSHIFT);
|
|
|
|
}
|
2016-12-17 21:34:36 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
void R_SetDSColorMapLight(FSWColormap *base_colormap, float light, int shade)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
using namespace drawerargs;
|
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
ds_fcolormap = base_colormap;
|
|
|
|
if (r_swtruecolor)
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
ds_shade_constants.light_red = ds_fcolormap->Color.r * 256 / 255;
|
|
|
|
ds_shade_constants.light_green = ds_fcolormap->Color.g * 256 / 255;
|
|
|
|
ds_shade_constants.light_blue = ds_fcolormap->Color.b * 256 / 255;
|
|
|
|
ds_shade_constants.light_alpha = ds_fcolormap->Color.a * 256 / 255;
|
|
|
|
ds_shade_constants.fade_red = ds_fcolormap->Fade.r;
|
|
|
|
ds_shade_constants.fade_green = ds_fcolormap->Fade.g;
|
|
|
|
ds_shade_constants.fade_blue = ds_fcolormap->Fade.b;
|
|
|
|
ds_shade_constants.fade_alpha = ds_fcolormap->Fade.a;
|
|
|
|
ds_shade_constants.desaturate = MIN(abs(ds_fcolormap->Desaturate), 255) * 255 / 256;
|
|
|
|
ds_shade_constants.simple_shade = (ds_fcolormap->Color.d == 0x00ffffff && ds_fcolormap->Fade.d == 0x00000000 && ds_fcolormap->Desaturate == 0);
|
|
|
|
ds_colormap = base_colormap->Maps;
|
|
|
|
ds_light = LIGHTSCALE(light, shade);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ds_colormap = base_colormap->Maps + (GETPALOOKUP(light, shade) << COLORMAPSHIFT);
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 08:34:49 +00:00
|
|
|
void R_SetTranslationMap(lighttable_t *translation)
|
|
|
|
{
|
|
|
|
using namespace drawerargs;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
if (r_swtruecolor)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
dc_fcolormap = nullptr;
|
|
|
|
dc_colormap = nullptr;
|
|
|
|
dc_translation = translation;
|
|
|
|
dc_shade_constants.light_red = 256;
|
|
|
|
dc_shade_constants.light_green = 256;
|
|
|
|
dc_shade_constants.light_blue = 256;
|
|
|
|
dc_shade_constants.light_alpha = 256;
|
|
|
|
dc_shade_constants.fade_red = 0;
|
|
|
|
dc_shade_constants.fade_green = 0;
|
|
|
|
dc_shade_constants.fade_blue = 0;
|
|
|
|
dc_shade_constants.fade_alpha = 256;
|
|
|
|
dc_shade_constants.desaturate = 0;
|
|
|
|
dc_shade_constants.simple_shade = true;
|
|
|
|
dc_light = 0;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
2016-12-07 17:28:06 +00:00
|
|
|
else
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
dc_fcolormap = nullptr;
|
|
|
|
dc_colormap = translation;
|
2016-12-07 08:34:49 +00:00
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-27 06:30:29 +00:00
|
|
|
void R_SetSpanTexture(FTexture *tex)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
using namespace drawerargs;
|
|
|
|
|
|
|
|
tex->GetWidth();
|
|
|
|
ds_xbits = tex->WidthBits;
|
|
|
|
ds_ybits = tex->HeightBits;
|
|
|
|
if ((1 << ds_xbits) > tex->GetWidth())
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
ds_xbits--;
|
|
|
|
}
|
|
|
|
if ((1 << ds_ybits) > tex->GetHeight())
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 08:34:49 +00:00
|
|
|
ds_ybits--;
|
|
|
|
}
|
2016-12-27 06:30:29 +00:00
|
|
|
|
|
|
|
ds_source = r_swtruecolor ? (const uint8_t*)tex->GetPixelsBgra() : tex->GetPixels();
|
|
|
|
ds_source_mipmapped = tex->Mipmapped() && tex->GetWidth() > 1 && tex->GetHeight() > 1;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-07 17:28:06 +00:00
|
|
|
void R_SetSpanColormap(FDynamicColormap *colormap, int shade)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-12-07 17:28:06 +00:00
|
|
|
R_SetDSColorMapLight(colormap, 0, shade);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-12-27 06:45:10 +00:00
|
|
|
void R_UpdateFuzzPos()
|
2016-12-07 08:34:49 +00:00
|
|
|
{
|
|
|
|
using namespace drawerargs;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-12-27 06:45:10 +00:00
|
|
|
dc_yl = MAX(dc_yl, 1);
|
|
|
|
dc_yh = MIN(dc_yh, fuzzviewheight);
|
|
|
|
if (dc_yl <= dc_yh)
|
|
|
|
fuzzpos = (fuzzpos + dc_yh - dc_yl + 1) % FUZZTABLE;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|