2016-06-13 21:10:54 +00:00
|
|
|
// Emacs style mode select -*- C++ -*-
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// $Id:$
|
|
|
|
//
|
|
|
|
// Copyright (C) 1993-1996 by id Software, Inc.
|
|
|
|
//
|
|
|
|
// This source is available for distribution and/or modification
|
|
|
|
// only under the terms of the DOOM Source Code License as
|
|
|
|
// published by id Software. All rights reserved.
|
|
|
|
//
|
|
|
|
// The source is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the DOOM Source Code License
|
|
|
|
// for more details.
|
|
|
|
//
|
|
|
|
// DESCRIPTION:
|
|
|
|
// System specific interface stuff.
|
|
|
|
//
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef __R_DRAW_RGBA__
|
|
|
|
#define __R_DRAW_RGBA__
|
|
|
|
|
|
|
|
#include "r_draw.h"
|
2016-06-14 21:05:20 +00:00
|
|
|
#include "v_palette.h"
|
2016-10-07 02:01:38 +00:00
|
|
|
#include "r_thread.h"
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-30 11:45:06 +00:00
|
|
|
#ifndef NO_SSE
|
|
|
|
#include <immintrin.h>
|
|
|
|
#endif
|
|
|
|
|
2016-10-07 02:01:38 +00:00
|
|
|
struct FSpecialColormap;
|
|
|
|
|
|
|
|
EXTERN_CVAR(Bool, r_mipmap)
|
2016-11-05 15:12:59 +00:00
|
|
|
EXTERN_CVAR(Float, r_lod_bias)
|
2016-10-07 02:01:38 +00:00
|
|
|
|
2016-06-13 21:10:54 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Drawer functions:
|
|
|
|
|
|
|
|
void rt_initcols_rgba(BYTE *buffer);
|
|
|
|
void rt_span_coverage_rgba(int x, int start, int stop);
|
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void rt_copy1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_copy4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_shaded1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_shaded4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_map1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_add1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_addclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_subclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_revsubclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_tlate1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_tlateadd1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_tlateaddclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_tlatesubclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_tlaterevsubclamp1col_rgba(int hx, int sx, int yl, int yh);
|
|
|
|
void rt_map4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_add4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_addclamp4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_subclamp4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_revsubclamp4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_tlate4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_tlateadd4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_tlateaddclamp4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_tlatesubclamp4cols_rgba(int sx, int yl, int yh);
|
|
|
|
void rt_tlaterevsubclamp4cols_rgba(int sx, int yl, int yh);
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void R_DrawColumnHoriz_rgba();
|
|
|
|
void R_DrawColumn_rgba();
|
|
|
|
void R_DrawFuzzColumn_rgba();
|
|
|
|
void R_DrawTranslatedColumn_rgba();
|
|
|
|
void R_DrawShadedColumn_rgba();
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void R_FillColumn_rgba();
|
|
|
|
void R_FillAddColumn_rgba();
|
|
|
|
void R_FillAddClampColumn_rgba();
|
|
|
|
void R_FillSubClampColumn_rgba();
|
|
|
|
void R_FillRevSubClampColumn_rgba();
|
|
|
|
void R_DrawAddColumn_rgba();
|
|
|
|
void R_DrawTlatedAddColumn_rgba();
|
|
|
|
void R_DrawAddClampColumn_rgba();
|
|
|
|
void R_DrawAddClampTranslatedColumn_rgba();
|
|
|
|
void R_DrawSubClampColumn_rgba();
|
|
|
|
void R_DrawSubClampTranslatedColumn_rgba();
|
|
|
|
void R_DrawRevSubClampColumn_rgba();
|
|
|
|
void R_DrawRevSubClampTranslatedColumn_rgba();
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void R_DrawSpan_rgba(void);
|
|
|
|
void R_DrawSpanMasked_rgba(void);
|
|
|
|
void R_DrawSpanTranslucent_rgba();
|
|
|
|
void R_DrawSpanMaskedTranslucent_rgba();
|
|
|
|
void R_DrawSpanAddClamp_rgba();
|
|
|
|
void R_DrawSpanMaskedAddClamp_rgba();
|
|
|
|
void R_FillSpan_rgba();
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-08-08 23:17:45 +00:00
|
|
|
void R_DrawTiltedSpan_rgba(int y, int x1, int x2, const FVector3 &plane_sz, const FVector3 &plane_su, const FVector3 &plane_sv, bool plane_shade, int planeshade, float planelightfloat, fixed_t pviewx, fixed_t pviewy);
|
|
|
|
void R_DrawColoredSpan_rgba(int y, int x1, int x2);
|
|
|
|
|
2016-09-14 08:03:39 +00:00
|
|
|
void R_SetupDrawSlab_rgba(FSWColormap *base_colormap, float light, int shade);
|
2016-06-21 22:22:06 +00:00
|
|
|
void R_DrawSlab_rgba(int dx, fixed_t v, int dy, fixed_t vi, const BYTE *vptr, BYTE *p);
|
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void R_DrawFogBoundary_rgba(int x1, int x2, short *uclip, short *dclip);
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
DWORD vlinec1_rgba();
|
|
|
|
void vlinec4_rgba();
|
|
|
|
DWORD mvlinec1_rgba();
|
|
|
|
void mvlinec4_rgba();
|
|
|
|
fixed_t tmvline1_add_rgba();
|
|
|
|
void tmvline4_add_rgba();
|
|
|
|
fixed_t tmvline1_addclamp_rgba();
|
|
|
|
void tmvline4_addclamp_rgba();
|
|
|
|
fixed_t tmvline1_subclamp_rgba();
|
|
|
|
void tmvline4_subclamp_rgba();
|
|
|
|
fixed_t tmvline1_revsubclamp_rgba();
|
|
|
|
void tmvline4_revsubclamp_rgba();
|
2016-06-13 21:10:54 +00:00
|
|
|
|
2016-06-13 21:33:52 +00:00
|
|
|
void R_FillColumnHoriz_rgba();
|
|
|
|
void R_FillSpan_rgba();
|
2016-06-13 21:10:54 +00:00
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Drawer commands:
|
|
|
|
|
|
|
|
class ApplySpecialColormapRGBACommand : public DrawerCommand
|
|
|
|
{
|
|
|
|
BYTE *buffer;
|
|
|
|
int pitch;
|
|
|
|
int width;
|
|
|
|
int height;
|
|
|
|
int start_red;
|
|
|
|
int start_green;
|
|
|
|
int start_blue;
|
|
|
|
int end_red;
|
|
|
|
int end_green;
|
|
|
|
int end_blue;
|
|
|
|
|
|
|
|
public:
|
|
|
|
ApplySpecialColormapRGBACommand(FSpecialColormap *colormap, DFrameBuffer *screen);
|
|
|
|
void Execute(DrawerThread *thread) override;
|
2016-10-12 11:25:05 +00:00
|
|
|
FString DebugInfo() override { return "ApplySpecialColormapRGBACommand"; }
|
2016-06-13 21:10:54 +00:00
|
|
|
};
|
|
|
|
|
2016-06-26 19:23:32 +00:00
|
|
|
template<typename CommandType, typename BlendMode>
|
|
|
|
class DrawerBlendCommand : public CommandType
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void Execute(DrawerThread *thread) override
|
|
|
|
{
|
2016-06-30 11:56:53 +00:00
|
|
|
typename CommandType::LoopIterator loop(this, thread);
|
2016-06-26 19:23:32 +00:00
|
|
|
if (!loop) return;
|
|
|
|
BlendMode blend(*this, loop);
|
|
|
|
do
|
|
|
|
{
|
|
|
|
blend.Blend(*this, loop);
|
|
|
|
} while (loop.next());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-06-14 21:05:20 +00:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
2016-06-24 09:37:51 +00:00
|
|
|
// Pixel shading inline functions:
|
2016-06-14 21:05:20 +00:00
|
|
|
|
|
|
|
// Give the compiler a strong hint we want these functions inlined:
|
|
|
|
#ifndef FORCEINLINE
|
|
|
|
#if defined(_MSC_VER)
|
|
|
|
#define FORCEINLINE __forceinline
|
|
|
|
#elif defined(__GNUC__)
|
|
|
|
#define FORCEINLINE __attribute__((always_inline)) inline
|
|
|
|
#else
|
|
|
|
#define FORCEINLINE inline
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2016-06-18 09:17:59 +00:00
|
|
|
// Promise compiler we have no aliasing of this pointer
|
|
|
|
#ifndef RESTRICT
|
|
|
|
#if defined(_MSC_VER)
|
|
|
|
#define RESTRICT __restrict
|
|
|
|
#elif defined(__GNUC__)
|
|
|
|
#define RESTRICT __restrict__
|
|
|
|
#else
|
|
|
|
#define RESTRICT
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
class LightBgra
|
2016-06-14 21:05:20 +00:00
|
|
|
{
|
2016-06-24 09:37:51 +00:00
|
|
|
public:
|
|
|
|
// calculates the light constant passed to the shade_pal_index function
|
|
|
|
FORCEINLINE static uint32_t calc_light_multiplier(dsfixed_t light)
|
|
|
|
{
|
|
|
|
return 256 - (light >> (FRACBITS - 8));
|
|
|
|
}
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
// Calculates a ARGB8 color for the given palette index and light multiplier
|
|
|
|
FORCEINLINE static uint32_t shade_pal_index_simple(uint32_t index, uint32_t light)
|
2016-06-14 21:05:20 +00:00
|
|
|
{
|
2016-06-24 09:37:51 +00:00
|
|
|
const PalEntry &color = GPalette.BaseColors[index];
|
|
|
|
uint32_t red = color.r;
|
|
|
|
uint32_t green = color.g;
|
|
|
|
uint32_t blue = color.b;
|
|
|
|
|
2016-06-14 21:05:20 +00:00
|
|
|
red = red * light / 256;
|
|
|
|
green = green * light / 256;
|
|
|
|
blue = blue * light / 256;
|
2016-06-24 09:37:51 +00:00
|
|
|
|
|
|
|
return 0xff000000 | (red << 16) | (green << 8) | blue;
|
2016-06-14 21:05:20 +00:00
|
|
|
}
|
2016-06-24 09:37:51 +00:00
|
|
|
|
|
|
|
// Calculates a ARGB8 color for the given palette index, light multiplier and dynamic colormap
|
|
|
|
FORCEINLINE static uint32_t shade_pal_index(uint32_t index, uint32_t light, const ShadeConstants &constants)
|
2016-06-14 21:05:20 +00:00
|
|
|
{
|
2016-06-24 09:37:51 +00:00
|
|
|
const PalEntry &color = GPalette.BaseColors[index];
|
|
|
|
uint32_t alpha = color.d & 0xff000000;
|
|
|
|
uint32_t red = color.r;
|
|
|
|
uint32_t green = color.g;
|
|
|
|
uint32_t blue = color.b;
|
|
|
|
if (constants.simple_shade)
|
|
|
|
{
|
|
|
|
red = red * light / 256;
|
|
|
|
green = green * light / 256;
|
|
|
|
blue = blue * light / 256;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint32_t inv_light = 256 - light;
|
|
|
|
uint32_t inv_desaturate = 256 - constants.desaturate;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
uint32_t intensity = ((red * 77 + green * 143 + blue * 37) >> 8) * constants.desaturate;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (red * inv_desaturate + intensity) / 256;
|
|
|
|
green = (green * inv_desaturate + intensity) / 256;
|
|
|
|
blue = (blue * inv_desaturate + intensity) / 256;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (constants.fade_red * inv_light + red * light) / 256;
|
|
|
|
green = (constants.fade_green * inv_light + green * light) / 256;
|
|
|
|
blue = (constants.fade_blue * inv_light + blue * light) / 256;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (red * constants.light_red) / 256;
|
|
|
|
green = (green * constants.light_green) / 256;
|
|
|
|
blue = (blue * constants.light_blue) / 256;
|
|
|
|
}
|
|
|
|
return alpha | (red << 16) | (green << 8) | blue;
|
2016-06-14 21:05:20 +00:00
|
|
|
}
|
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
FORCEINLINE static uint32_t shade_bgra_simple(uint32_t color, uint32_t light)
|
2016-06-14 21:05:20 +00:00
|
|
|
{
|
2016-06-24 09:37:51 +00:00
|
|
|
uint32_t red = RPART(color) * light / 256;
|
|
|
|
uint32_t green = GPART(color) * light / 256;
|
|
|
|
uint32_t blue = BPART(color) * light / 256;
|
|
|
|
return 0xff000000 | (red << 16) | (green << 8) | blue;
|
2016-06-14 21:05:20 +00:00
|
|
|
}
|
2016-06-24 09:37:51 +00:00
|
|
|
|
|
|
|
FORCEINLINE static uint32_t shade_bgra(uint32_t color, uint32_t light, const ShadeConstants &constants)
|
2016-06-14 21:05:20 +00:00
|
|
|
{
|
2016-06-24 09:37:51 +00:00
|
|
|
uint32_t alpha = color & 0xff000000;
|
|
|
|
uint32_t red = (color >> 16) & 0xff;
|
|
|
|
uint32_t green = (color >> 8) & 0xff;
|
|
|
|
uint32_t blue = color & 0xff;
|
|
|
|
if (constants.simple_shade)
|
|
|
|
{
|
|
|
|
red = red * light / 256;
|
|
|
|
green = green * light / 256;
|
|
|
|
blue = blue * light / 256;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
uint32_t inv_light = 256 - light;
|
|
|
|
uint32_t inv_desaturate = 256 - constants.desaturate;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
uint32_t intensity = ((red * 77 + green * 143 + blue * 37) >> 8) * constants.desaturate;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (red * inv_desaturate + intensity) / 256;
|
|
|
|
green = (green * inv_desaturate + intensity) / 256;
|
|
|
|
blue = (blue * inv_desaturate + intensity) / 256;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (constants.fade_red * inv_light + red * light) / 256;
|
|
|
|
green = (constants.fade_green * inv_light + green * light) / 256;
|
|
|
|
blue = (constants.fade_blue * inv_light + blue * light) / 256;
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-24 09:37:51 +00:00
|
|
|
red = (red * constants.light_red) / 256;
|
|
|
|
green = (green * constants.light_green) / 256;
|
|
|
|
blue = (blue * constants.light_blue) / 256;
|
|
|
|
}
|
|
|
|
return alpha | (red << 16) | (green << 8) | blue;
|
2016-06-14 21:05:20 +00:00
|
|
|
}
|
2016-06-24 09:37:51 +00:00
|
|
|
};
|
2016-06-14 21:05:20 +00:00
|
|
|
|
2016-06-13 21:10:54 +00:00
|
|
|
#endif
|