mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-21 08:31:00 +00:00
8cc4c8a192
# Conflicts: # src/r_draw_pal.cpp
2550 lines
71 KiB
C++
2550 lines
71 KiB
C++
/*
|
|
** r_draw_pal.cpp
|
|
**
|
|
**---------------------------------------------------------------------------
|
|
** Copyright 1998-2016 Randy Heit
|
|
** Copyright 2016 Magnus Norddahl
|
|
** Copyright 2016 Rachael Alexanderson
|
|
** 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.
|
|
**---------------------------------------------------------------------------
|
|
**
|
|
*/
|
|
|
|
#include "templates.h"
|
|
#include "doomtype.h"
|
|
#include "doomdef.h"
|
|
#include "r_defs.h"
|
|
#include "r_draw.h"
|
|
#include "r_main.h"
|
|
#include "r_things.h"
|
|
#include "v_video.h"
|
|
#include "r_draw_pal.h"
|
|
|
|
/*
|
|
[RH] This translucency algorithm is based on DOSDoom 0.65's, but uses
|
|
a 32k RGB table instead of an 8k one. At least on my machine, it's
|
|
slightly faster (probably because it uses only one shift instead of
|
|
two), and it looks considerably less green at the ends of the
|
|
translucency range. The extra size doesn't appear to be an issue.
|
|
|
|
The following note is from DOSDoom 0.65:
|
|
|
|
New translucency algorithm, by Erik Sandberg:
|
|
|
|
Basically, we compute the red, green and blue values for each pixel, and
|
|
then use a RGB table to check which one of the palette colours that best
|
|
represents those RGB values. The RGB table is 8k big, with 4 R-bits,
|
|
5 G-bits and 4 B-bits. A 4k table gives a bit too bad precision, and a 32k
|
|
table takes up more memory and results in more cache misses, so an 8k
|
|
table seemed to be quite ultimate.
|
|
|
|
The computation of the RGB for each pixel is accelerated by using two
|
|
1k tables for each translucency level.
|
|
The xth element of one of these tables contains the r, g and b values for
|
|
the colour x, weighted for the current translucency level (for example,
|
|
the weighted rgb values for background colour at 75% translucency are 1/4
|
|
of the original rgb values). The rgb values are stored as three
|
|
low-precision fixed point values, packed into one long per colour:
|
|
Bit 0-4: Frac part of blue (5 bits)
|
|
Bit 5-8: Int part of blue (4 bits)
|
|
Bit 9-13: Frac part of red (5 bits)
|
|
Bit 14-17: Int part of red (4 bits)
|
|
Bit 18-22: Frac part of green (5 bits)
|
|
Bit 23-27: Int part of green (5 bits)
|
|
Bit 28-31: All zeros (4 bits)
|
|
|
|
The point of this format is that the two colours now can be added, and
|
|
then be converted to a RGB table index very easily: First, we just set
|
|
all the frac bits and the four upper zero bits to 1. It's now possible
|
|
to get the RGB table index by anding the current value >> 5 with the
|
|
current value >> 19. When asm-optimised, this should be the fastest
|
|
algorithm that uses RGB tables.
|
|
*/
|
|
|
|
namespace swrenderer
|
|
{
|
|
PalWall1Command::PalWall1Command()
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_iscale = dc_iscale;
|
|
_texturefrac = dc_texturefrac;
|
|
_colormap = dc_colormap;
|
|
_count = dc_count;
|
|
_source = dc_source;
|
|
_dest = dc_dest;
|
|
_fracbits = dc_wall_fracbits;
|
|
_pitch = dc_pitch;
|
|
_srcblend = dc_srcblend;
|
|
_destblend = dc_destblend;
|
|
}
|
|
|
|
PalWall4Command::PalWall4Command()
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_dest = dc_dest;
|
|
_count = dc_count;
|
|
_pitch = dc_pitch;
|
|
_fracbits = dc_wall_fracbits;
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
_colormap[col] = dc_wall_colormap[col];
|
|
_source[col] = dc_wall_source[col];
|
|
_iscale[col] = dc_wall_iscale[col];
|
|
_texturefrac[col] = dc_wall_texturefrac[col];
|
|
}
|
|
_srcblend = dc_srcblend;
|
|
_destblend = dc_destblend;
|
|
}
|
|
|
|
void DrawWall1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
*dest = colormap[source[frac >> bits]];
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWall4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
uint32_t place;
|
|
auto pal0 = _colormap[0];
|
|
auto pal1 = _colormap[1];
|
|
auto pal2 = _colormap[2];
|
|
auto pal3 = _colormap[3];
|
|
auto buf0 = _source[0];
|
|
auto buf1 = _source[1];
|
|
auto buf2 = _source[2];
|
|
auto buf3 = _source[3];
|
|
auto dc_wall_iscale0 = _iscale[0];
|
|
auto dc_wall_iscale1 = _iscale[1];
|
|
auto dc_wall_iscale2 = _iscale[2];
|
|
auto dc_wall_iscale3 = _iscale[3];
|
|
auto dc_wall_texturefrac0 = _texturefrac[0];
|
|
auto dc_wall_texturefrac1 = _texturefrac[1];
|
|
auto dc_wall_texturefrac2 = _texturefrac[2];
|
|
auto dc_wall_texturefrac3 = _texturefrac[3];
|
|
auto pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
dc_wall_texturefrac0 += dc_wall_iscale0 * skipped;
|
|
dc_wall_texturefrac1 += dc_wall_iscale1 * skipped;
|
|
dc_wall_texturefrac2 += dc_wall_iscale2 * skipped;
|
|
dc_wall_texturefrac3 += dc_wall_iscale3 * skipped;
|
|
dc_wall_iscale0 *= thread->num_cores;
|
|
dc_wall_iscale1 *= thread->num_cores;
|
|
dc_wall_iscale2 *= thread->num_cores;
|
|
dc_wall_iscale3 *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
dest[0] = pal0[buf0[(place = dc_wall_texturefrac0) >> bits]]; dc_wall_texturefrac0 = place + dc_wall_iscale0;
|
|
dest[1] = pal1[buf1[(place = dc_wall_texturefrac1) >> bits]]; dc_wall_texturefrac1 = place + dc_wall_iscale1;
|
|
dest[2] = pal2[buf2[(place = dc_wall_texturefrac2) >> bits]]; dc_wall_texturefrac2 = place + dc_wall_iscale2;
|
|
dest[3] = pal3[buf3[(place = dc_wall_texturefrac3) >> bits]]; dc_wall_texturefrac3 = place + dc_wall_iscale3;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallMasked1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix = source[frac >> bits];
|
|
if (pix != 0)
|
|
{
|
|
*dest = colormap[pix];
|
|
}
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallMasked4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
uint32_t place;
|
|
auto pal0 = _colormap[0];
|
|
auto pal1 = _colormap[1];
|
|
auto pal2 = _colormap[2];
|
|
auto pal3 = _colormap[3];
|
|
auto buf0 = _source[0];
|
|
auto buf1 = _source[1];
|
|
auto buf2 = _source[2];
|
|
auto buf3 = _source[3];
|
|
auto dc_wall_iscale0 = _iscale[0];
|
|
auto dc_wall_iscale1 = _iscale[1];
|
|
auto dc_wall_iscale2 = _iscale[2];
|
|
auto dc_wall_iscale3 = _iscale[3];
|
|
auto dc_wall_texturefrac0 = _texturefrac[0];
|
|
auto dc_wall_texturefrac1 = _texturefrac[1];
|
|
auto dc_wall_texturefrac2 = _texturefrac[2];
|
|
auto dc_wall_texturefrac3 = _texturefrac[3];
|
|
auto pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
dc_wall_texturefrac0 += dc_wall_iscale0 * skipped;
|
|
dc_wall_texturefrac1 += dc_wall_iscale1 * skipped;
|
|
dc_wall_texturefrac2 += dc_wall_iscale2 * skipped;
|
|
dc_wall_texturefrac3 += dc_wall_iscale3 * skipped;
|
|
dc_wall_iscale0 *= thread->num_cores;
|
|
dc_wall_iscale1 *= thread->num_cores;
|
|
dc_wall_iscale2 *= thread->num_cores;
|
|
dc_wall_iscale3 *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix;
|
|
|
|
pix = buf0[(place = dc_wall_texturefrac0) >> bits]; if (pix) dest[0] = pal0[pix]; dc_wall_texturefrac0 = place + dc_wall_iscale0;
|
|
pix = buf1[(place = dc_wall_texturefrac1) >> bits]; if (pix) dest[1] = pal1[pix]; dc_wall_texturefrac1 = place + dc_wall_iscale1;
|
|
pix = buf2[(place = dc_wall_texturefrac2) >> bits]; if (pix) dest[2] = pal2[pix]; dc_wall_texturefrac2 = place + dc_wall_iscale2;
|
|
pix = buf3[(place = dc_wall_texturefrac3) >> bits]; if (pix) dest[3] = pal3[pix]; dc_wall_texturefrac3 = place + dc_wall_iscale3;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallAdd1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix = source[frac >> bits];
|
|
if (pix != 0)
|
|
{
|
|
uint32_t r = MIN(GPalette.BaseColors[colormap[pix]].r + GPalette.BaseColors[*dest].r, 255);
|
|
uint32_t g = MIN(GPalette.BaseColors[colormap[pix]].g + GPalette.BaseColors[*dest].g, 255);
|
|
uint32_t b = MIN(GPalette.BaseColors[colormap[pix]].b + GPalette.BaseColors[*dest].b, 255);
|
|
*dest = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallAdd4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
|
|
uint32_t dc_wall_texturefrac[4] = { _texturefrac[0], _texturefrac[1], _texturefrac[2], _texturefrac[3] };
|
|
uint32_t dc_wall_iscale[4] = { _iscale[0], _iscale[1], _iscale[2], _iscale[3] };
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i] * skipped;
|
|
dc_wall_iscale[i] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
for (int i = 0; i < 4; ++i)
|
|
{
|
|
uint8_t pix = _source[i][dc_wall_texturefrac[i] >> bits];
|
|
if (pix != 0)
|
|
{
|
|
uint32_t r = MIN(GPalette.BaseColors[_colormap[i][pix]].r + GPalette.BaseColors[dest[i]].r, 255);
|
|
uint32_t g = MIN(GPalette.BaseColors[_colormap[i][pix]].g + GPalette.BaseColors[dest[i]].g, 255);
|
|
uint32_t b = MIN(GPalette.BaseColors[_colormap[i][pix]].b + GPalette.BaseColors[dest[i]].b, 255);
|
|
dest[i] = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i];
|
|
}
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallAddClamp1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix = source[frac >> bits];
|
|
if (pix != 0)
|
|
{
|
|
uint32_t r = MIN(GPalette.BaseColors[colormap[pix]].r + GPalette.BaseColors[*dest].r, 255);
|
|
uint32_t g = MIN(GPalette.BaseColors[colormap[pix]].g + GPalette.BaseColors[*dest].g, 255);
|
|
uint32_t b = MIN(GPalette.BaseColors[colormap[pix]].b + GPalette.BaseColors[*dest].b, 255);
|
|
*dest = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallAddClamp4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
|
|
uint32_t dc_wall_texturefrac[4] = { _texturefrac[0], _texturefrac[1], _texturefrac[2], _texturefrac[3] };
|
|
uint32_t dc_wall_iscale[4] = { _iscale[0], _iscale[1], _iscale[2], _iscale[3] };
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i] * skipped;
|
|
dc_wall_iscale[i] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
for (int i = 0; i < 4; ++i)
|
|
{
|
|
uint8_t pix = _source[i][dc_wall_texturefrac[i] >> bits];
|
|
if (pix != 0)
|
|
{
|
|
uint32_t r = MIN(GPalette.BaseColors[_colormap[i][pix]].r + GPalette.BaseColors[dest[i]].r, 255);
|
|
uint32_t g = MIN(GPalette.BaseColors[_colormap[i][pix]].g + GPalette.BaseColors[dest[i]].g, 255);
|
|
uint32_t b = MIN(GPalette.BaseColors[_colormap[i][pix]].b + GPalette.BaseColors[dest[i]].b, 255);
|
|
dest[i] = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i];
|
|
}
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallSubClamp1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix = source[frac >> bits];
|
|
if (pix != 0)
|
|
{
|
|
int r = clamp(-GPalette.BaseColors[colormap[pix]].r + GPalette.BaseColors[*dest].r, 0, 255);
|
|
int g = clamp(-GPalette.BaseColors[colormap[pix]].g + GPalette.BaseColors[*dest].g, 0, 255);
|
|
int b = clamp(-GPalette.BaseColors[colormap[pix]].b + GPalette.BaseColors[*dest].b, 0, 255);
|
|
*dest = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallSubClamp4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
|
|
uint32_t dc_wall_texturefrac[4] = { _texturefrac[0], _texturefrac[1], _texturefrac[2], _texturefrac[3] };
|
|
uint32_t dc_wall_iscale[4] = { _iscale[0], _iscale[1], _iscale[2], _iscale[3] };
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i] * skipped;
|
|
dc_wall_iscale[i] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
for (int i = 0; i < 4; ++i)
|
|
{
|
|
uint8_t pix = _source[i][dc_wall_texturefrac[i] >> bits];
|
|
if (pix != 0)
|
|
{
|
|
int r = clamp(-GPalette.BaseColors[_colormap[i][pix]].r + GPalette.BaseColors[dest[i]].r, 0, 255);
|
|
int g = clamp(-GPalette.BaseColors[_colormap[i][pix]].g + GPalette.BaseColors[dest[i]].g, 0, 255);
|
|
int b = clamp(-GPalette.BaseColors[_colormap[i][pix]].b + GPalette.BaseColors[dest[i]].b, 0, 255);
|
|
dest[i] = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i];
|
|
}
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallRevSubClamp1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint32_t fracstep = _iscale;
|
|
uint32_t frac = _texturefrac;
|
|
uint8_t *colormap = _colormap;
|
|
int count = _count;
|
|
const uint8_t *source = _source;
|
|
uint8_t *dest = _dest;
|
|
int bits = _fracbits;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
uint8_t pix = source[frac >> bits];
|
|
if (pix != 0)
|
|
{
|
|
int r = clamp(GPalette.BaseColors[colormap[pix]].r - GPalette.BaseColors[*dest].r, 0, 255);
|
|
int g = clamp(GPalette.BaseColors[colormap[pix]].g - GPalette.BaseColors[*dest].g, 0, 255);
|
|
int b = clamp(GPalette.BaseColors[colormap[pix]].b - GPalette.BaseColors[*dest].b, 0, 255);
|
|
*dest = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawWallRevSubClamp4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int bits = _fracbits;
|
|
|
|
uint32_t dc_wall_texturefrac[4] = { _texturefrac[0], _texturefrac[1], _texturefrac[2], _texturefrac[3] };
|
|
uint32_t dc_wall_iscale[4] = { _iscale[0], _iscale[1], _iscale[2], _iscale[3] };
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i] * skipped;
|
|
dc_wall_iscale[i] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
|
|
do
|
|
{
|
|
for (int i = 0; i < 4; ++i)
|
|
{
|
|
uint8_t pix = _source[i][dc_wall_texturefrac[i] >> bits];
|
|
if (pix != 0)
|
|
{
|
|
uint32_t r = clamp(GPalette.BaseColors[_colormap[i][pix]].r - GPalette.BaseColors[dest[i]].r, 0, 255);
|
|
uint32_t g = clamp(GPalette.BaseColors[_colormap[i][pix]].g - GPalette.BaseColors[dest[i]].g, 0, 255);
|
|
uint32_t b = clamp(GPalette.BaseColors[_colormap[i][pix]].b - GPalette.BaseColors[dest[i]].b, 0, 255);
|
|
dest[i] = RGB256k.RGB[r>>2][g>>2][b>>2];
|
|
}
|
|
dc_wall_texturefrac[i] += dc_wall_iscale[i];
|
|
}
|
|
dest += _pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
PalSkyCommand::PalSkyCommand(uint32_t solid_top, uint32_t solid_bottom) : solid_top(solid_top), solid_bottom(solid_bottom)
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_dest = dc_dest;
|
|
_count = dc_count;
|
|
_pitch = dc_pitch;
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
_source[col] = dc_wall_source[col];
|
|
_source2[col] = dc_wall_source2[col];
|
|
_sourceheight[col] = dc_wall_sourceheight[col];
|
|
_iscale[col] = dc_wall_iscale[col];
|
|
_texturefrac[col] = dc_wall_texturefrac[col];
|
|
}
|
|
}
|
|
|
|
void DrawSingleSky1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int pitch = _pitch;
|
|
const uint8_t *source0 = _source[0];
|
|
int textureheight0 = _sourceheight[0];
|
|
|
|
int32_t frac = _texturefrac[0];
|
|
int32_t fracstep = _iscale[0];
|
|
|
|
int start_fade = 2; // How fast it should fade out
|
|
|
|
int solid_top_r = RPART(solid_top);
|
|
int solid_top_g = GPART(solid_top);
|
|
int solid_top_b = BPART(solid_top);
|
|
int solid_bottom_r = RPART(solid_bottom);
|
|
int solid_bottom_g = GPART(solid_bottom);
|
|
int solid_bottom_b = BPART(solid_bottom);
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * skipped;
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
for (int index = 0; index < count; index++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[sample_index];
|
|
|
|
int alpha_top = MAX(MIN(frac >> (16 - start_fade), 256), 0);
|
|
int alpha_bottom = MAX(MIN(((2 << 24) - frac) >> (16 - start_fade), 256), 0);
|
|
|
|
if (alpha_top == 256 && alpha_bottom == 256)
|
|
{
|
|
*dest = fg;
|
|
}
|
|
else
|
|
{
|
|
int inv_alpha_top = 256 - alpha_top;
|
|
int inv_alpha_bottom = 256 - alpha_bottom;
|
|
|
|
const auto &c = GPalette.BaseColors[fg];
|
|
int c_red = c.r;
|
|
int c_green = c.g;
|
|
int c_blue = c.b;
|
|
c_red = (c_red * alpha_top + solid_top_r * inv_alpha_top) >> 8;
|
|
c_green = (c_green * alpha_top + solid_top_g * inv_alpha_top) >> 8;
|
|
c_blue = (c_blue * alpha_top + solid_top_b * inv_alpha_top) >> 8;
|
|
c_red = (c_red * alpha_bottom + solid_bottom_r * inv_alpha_bottom) >> 8;
|
|
c_green = (c_green * alpha_bottom + solid_bottom_g * inv_alpha_bottom) >> 8;
|
|
c_blue = (c_blue * alpha_bottom + solid_bottom_b * inv_alpha_bottom) >> 8;
|
|
*dest = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
}
|
|
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
}
|
|
}
|
|
|
|
void DrawSingleSky4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int pitch = _pitch;
|
|
const uint8_t *source0[4] = { _source[0], _source[1], _source[2], _source[3] };
|
|
int textureheight0 = _sourceheight[0];
|
|
const uint32_t *palette = (const uint32_t *)GPalette.BaseColors;
|
|
int32_t frac[4] = { (int32_t)_texturefrac[0], (int32_t)_texturefrac[1], (int32_t)_texturefrac[2], (int32_t)_texturefrac[3] };
|
|
int32_t fracstep[4] = { (int32_t)_iscale[0], (int32_t)_iscale[1], (int32_t)_iscale[2], (int32_t)_iscale[3] };
|
|
uint8_t output[4];
|
|
|
|
int start_fade = 2; // How fast it should fade out
|
|
|
|
int solid_top_r = RPART(solid_top);
|
|
int solid_top_g = GPART(solid_top);
|
|
int solid_top_b = BPART(solid_top);
|
|
int solid_bottom_r = RPART(solid_bottom);
|
|
int solid_bottom_g = GPART(solid_bottom);
|
|
int solid_bottom_b = BPART(solid_bottom);
|
|
uint32_t solid_top_fill = RGB256k.RGB[(solid_top_r >> 2)][(solid_top_g >> 2)][(solid_top_b >> 2)];
|
|
uint32_t solid_bottom_fill = RGB256k.RGB[(solid_bottom_r >> 2)][(solid_bottom_g >> 2)][(solid_bottom_b >> 2)];
|
|
solid_top_fill = (solid_top_fill << 24) | (solid_top_fill << 16) | (solid_top_fill << 8) | solid_top_fill;
|
|
solid_bottom_fill = (solid_bottom_fill << 24) | (solid_bottom_fill << 16) | (solid_bottom_fill << 8) | solid_bottom_fill;
|
|
|
|
// Find bands for top solid color, top fade, center textured, bottom fade, bottom solid color:
|
|
int fade_length = (1 << (24 - start_fade));
|
|
int start_fadetop_y = (-frac[0]) / fracstep[0];
|
|
int end_fadetop_y = (fade_length - frac[0]) / fracstep[0];
|
|
int start_fadebottom_y = ((2 << 24) - fade_length - frac[0]) / fracstep[0];
|
|
int end_fadebottom_y = ((2 << 24) - frac[0]) / fracstep[0];
|
|
for (int col = 1; col < 4; col++)
|
|
{
|
|
start_fadetop_y = MIN(start_fadetop_y, (-frac[0]) / fracstep[0]);
|
|
end_fadetop_y = MAX(end_fadetop_y, (fade_length - frac[0]) / fracstep[0]);
|
|
start_fadebottom_y = MIN(start_fadebottom_y, ((2 << 24) - fade_length - frac[0]) / fracstep[0]);
|
|
end_fadebottom_y = MAX(end_fadebottom_y, ((2 << 24) - frac[0]) / fracstep[0]);
|
|
}
|
|
start_fadetop_y = clamp(start_fadetop_y, 0, count);
|
|
end_fadetop_y = clamp(end_fadetop_y, 0, count);
|
|
start_fadebottom_y = clamp(start_fadebottom_y, 0, count);
|
|
end_fadebottom_y = clamp(end_fadebottom_y, 0, count);
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
frac[col] += fracstep[col] * skipped;
|
|
fracstep[col] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
int num_cores = thread->num_cores;
|
|
int index = skipped;
|
|
|
|
// Top solid color:
|
|
while (index < start_fadetop_y)
|
|
{
|
|
*((uint32_t*)dest) = solid_top_fill;
|
|
dest += pitch;
|
|
for (int col = 0; col < 4; col++)
|
|
frac[col] += fracstep[col];
|
|
index += num_cores;
|
|
}
|
|
|
|
// Top fade:
|
|
while (index < end_fadetop_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[col][sample_index];
|
|
|
|
uint32_t c = palette[fg];
|
|
int alpha_top = MAX(MIN(frac[col] >> (16 - start_fade), 256), 0);
|
|
int inv_alpha_top = 256 - alpha_top;
|
|
int c_red = RPART(c);
|
|
int c_green = GPART(c);
|
|
int c_blue = BPART(c);
|
|
c_red = (c_red * alpha_top + solid_top_r * inv_alpha_top) >> 8;
|
|
c_green = (c_green * alpha_top + solid_top_g * inv_alpha_top) >> 8;
|
|
c_blue = (c_blue * alpha_top + solid_top_b * inv_alpha_top) >> 8;
|
|
output[col] = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
frac[col] += fracstep[col];
|
|
}
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Textured center:
|
|
while (index < start_fadebottom_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
output[col] = source0[col][sample_index];
|
|
|
|
frac[col] += fracstep[col];
|
|
}
|
|
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Fade bottom:
|
|
while (index < end_fadebottom_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[col][sample_index];
|
|
|
|
uint32_t c = palette[fg];
|
|
int alpha_bottom = MAX(MIN(((2 << 24) - frac[col]) >> (16 - start_fade), 256), 0);
|
|
int inv_alpha_bottom = 256 - alpha_bottom;
|
|
int c_red = RPART(c);
|
|
int c_green = GPART(c);
|
|
int c_blue = BPART(c);
|
|
c_red = (c_red * alpha_bottom + solid_bottom_r * inv_alpha_bottom) >> 8;
|
|
c_green = (c_green * alpha_bottom + solid_bottom_g * inv_alpha_bottom) >> 8;
|
|
c_blue = (c_blue * alpha_bottom + solid_bottom_b * inv_alpha_bottom) >> 8;
|
|
output[col] = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
|
|
frac[col] += fracstep[col];
|
|
}
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Bottom solid color:
|
|
while (index < count)
|
|
{
|
|
*((uint32_t*)dest) = solid_bottom_fill;
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
}
|
|
|
|
void DrawDoubleSky1PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int pitch = _pitch;
|
|
const uint8_t *source0 = _source[0];
|
|
const uint8_t *source1 = _source2[0];
|
|
int textureheight0 = _sourceheight[0];
|
|
uint32_t maxtextureheight1 = _sourceheight[1] - 1;
|
|
|
|
int32_t frac = _texturefrac[0];
|
|
int32_t fracstep = _iscale[0];
|
|
|
|
int start_fade = 2; // How fast it should fade out
|
|
|
|
int solid_top_r = RPART(solid_top);
|
|
int solid_top_g = GPART(solid_top);
|
|
int solid_top_b = BPART(solid_top);
|
|
int solid_bottom_r = RPART(solid_bottom);
|
|
int solid_bottom_g = GPART(solid_bottom);
|
|
int solid_bottom_b = BPART(solid_bottom);
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * skipped;
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
for (int index = 0; index < count; index++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[sample_index];
|
|
if (fg == 0)
|
|
{
|
|
uint32_t sample_index2 = MIN(sample_index, maxtextureheight1);
|
|
fg = source1[sample_index2];
|
|
}
|
|
|
|
int alpha_top = MAX(MIN(frac >> (16 - start_fade), 256), 0);
|
|
int alpha_bottom = MAX(MIN(((2 << 24) - frac) >> (16 - start_fade), 256), 0);
|
|
|
|
if (alpha_top == 256 && alpha_bottom == 256)
|
|
{
|
|
*dest = fg;
|
|
}
|
|
else
|
|
{
|
|
int inv_alpha_top = 256 - alpha_top;
|
|
int inv_alpha_bottom = 256 - alpha_bottom;
|
|
|
|
const auto &c = GPalette.BaseColors[fg];
|
|
int c_red = c.r;
|
|
int c_green = c.g;
|
|
int c_blue = c.b;
|
|
c_red = (c_red * alpha_top + solid_top_r * inv_alpha_top) >> 8;
|
|
c_green = (c_green * alpha_top + solid_top_g * inv_alpha_top) >> 8;
|
|
c_blue = (c_blue * alpha_top + solid_top_b * inv_alpha_top) >> 8;
|
|
c_red = (c_red * alpha_bottom + solid_bottom_r * inv_alpha_bottom) >> 8;
|
|
c_green = (c_green * alpha_bottom + solid_bottom_g * inv_alpha_bottom) >> 8;
|
|
c_blue = (c_blue * alpha_bottom + solid_bottom_b * inv_alpha_bottom) >> 8;
|
|
*dest = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
}
|
|
|
|
frac += fracstep;
|
|
dest += pitch;
|
|
}
|
|
}
|
|
|
|
void DrawDoubleSky4PalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
uint8_t *dest = _dest;
|
|
int count = _count;
|
|
int pitch = _pitch;
|
|
const uint8_t *source0[4] = { _source[0], _source[1], _source[2], _source[3] };
|
|
const uint8_t *source1[4] = { _source2[0], _source2[1], _source2[2], _source2[3] };
|
|
int textureheight0 = _sourceheight[0];
|
|
uint32_t maxtextureheight1 = _sourceheight[1] - 1;
|
|
const uint32_t *palette = (const uint32_t *)GPalette.BaseColors;
|
|
int32_t frac[4] = { (int32_t)_texturefrac[0], (int32_t)_texturefrac[1], (int32_t)_texturefrac[2], (int32_t)_texturefrac[3] };
|
|
int32_t fracstep[4] = { (int32_t)_iscale[0], (int32_t)_iscale[1], (int32_t)_iscale[2], (int32_t)_iscale[3] };
|
|
uint8_t output[4];
|
|
|
|
int start_fade = 2; // How fast it should fade out
|
|
|
|
int solid_top_r = RPART(solid_top);
|
|
int solid_top_g = GPART(solid_top);
|
|
int solid_top_b = BPART(solid_top);
|
|
int solid_bottom_r = RPART(solid_bottom);
|
|
int solid_bottom_g = GPART(solid_bottom);
|
|
int solid_bottom_b = BPART(solid_bottom);
|
|
uint32_t solid_top_fill = RGB256k.RGB[(solid_top_r >> 2)][(solid_top_g >> 2)][(solid_top_b >> 2)];
|
|
uint32_t solid_bottom_fill = RGB256k.RGB[(solid_bottom_r >> 2)][(solid_bottom_g >> 2)][(solid_bottom_b >> 2)];
|
|
solid_top_fill = (solid_top_fill << 24) | (solid_top_fill << 16) | (solid_top_fill << 8) | solid_top_fill;
|
|
solid_bottom_fill = (solid_bottom_fill << 24) | (solid_bottom_fill << 16) | (solid_bottom_fill << 8) | solid_bottom_fill;
|
|
|
|
// Find bands for top solid color, top fade, center textured, bottom fade, bottom solid color:
|
|
int fade_length = (1 << (24 - start_fade));
|
|
int start_fadetop_y = (-frac[0]) / fracstep[0];
|
|
int end_fadetop_y = (fade_length - frac[0]) / fracstep[0];
|
|
int start_fadebottom_y = ((2 << 24) - fade_length - frac[0]) / fracstep[0];
|
|
int end_fadebottom_y = ((2 << 24) - frac[0]) / fracstep[0];
|
|
for (int col = 1; col < 4; col++)
|
|
{
|
|
start_fadetop_y = MIN(start_fadetop_y, (-frac[0]) / fracstep[0]);
|
|
end_fadetop_y = MAX(end_fadetop_y, (fade_length - frac[0]) / fracstep[0]);
|
|
start_fadebottom_y = MIN(start_fadebottom_y, ((2 << 24) - fade_length - frac[0]) / fracstep[0]);
|
|
end_fadebottom_y = MAX(end_fadebottom_y, ((2 << 24) - frac[0]) / fracstep[0]);
|
|
}
|
|
start_fadetop_y = clamp(start_fadetop_y, 0, count);
|
|
end_fadetop_y = clamp(end_fadetop_y, 0, count);
|
|
start_fadebottom_y = clamp(start_fadebottom_y, 0, count);
|
|
end_fadebottom_y = clamp(end_fadebottom_y, 0, count);
|
|
|
|
int skipped = thread->skipped_by_thread(_dest_y);
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
frac[col] += fracstep[col] * skipped;
|
|
fracstep[col] *= thread->num_cores;
|
|
}
|
|
pitch *= thread->num_cores;
|
|
int num_cores = thread->num_cores;
|
|
int index = skipped;
|
|
|
|
// Top solid color:
|
|
while (index < start_fadetop_y)
|
|
{
|
|
*((uint32_t*)dest) = solid_top_fill;
|
|
dest += pitch;
|
|
for (int col = 0; col < 4; col++)
|
|
frac[col] += fracstep[col];
|
|
index += num_cores;
|
|
}
|
|
|
|
// Top fade:
|
|
while (index < end_fadetop_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[col][sample_index];
|
|
if (fg == 0)
|
|
{
|
|
uint32_t sample_index2 = MIN(sample_index, maxtextureheight1);
|
|
fg = source1[col][sample_index2];
|
|
}
|
|
output[col] = fg;
|
|
|
|
uint32_t c = palette[fg];
|
|
int alpha_top = MAX(MIN(frac[col] >> (16 - start_fade), 256), 0);
|
|
int inv_alpha_top = 256 - alpha_top;
|
|
int c_red = RPART(c);
|
|
int c_green = GPART(c);
|
|
int c_blue = BPART(c);
|
|
c_red = (c_red * alpha_top + solid_top_r * inv_alpha_top) >> 8;
|
|
c_green = (c_green * alpha_top + solid_top_g * inv_alpha_top) >> 8;
|
|
c_blue = (c_blue * alpha_top + solid_top_b * inv_alpha_top) >> 8;
|
|
output[col] = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
|
|
frac[col] += fracstep[col];
|
|
}
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Textured center:
|
|
while (index < start_fadebottom_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[col][sample_index];
|
|
if (fg == 0)
|
|
{
|
|
uint32_t sample_index2 = MIN(sample_index, maxtextureheight1);
|
|
fg = source1[col][sample_index2];
|
|
}
|
|
output[col] = fg;
|
|
|
|
frac[col] += fracstep[col];
|
|
}
|
|
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Fade bottom:
|
|
while (index < end_fadebottom_y)
|
|
{
|
|
for (int col = 0; col < 4; col++)
|
|
{
|
|
uint32_t sample_index = (((((uint32_t)frac[col]) << 8) >> FRACBITS) * textureheight0) >> FRACBITS;
|
|
uint8_t fg = source0[col][sample_index];
|
|
if (fg == 0)
|
|
{
|
|
uint32_t sample_index2 = MIN(sample_index, maxtextureheight1);
|
|
fg = source1[col][sample_index2];
|
|
}
|
|
output[col] = fg;
|
|
|
|
uint32_t c = palette[fg];
|
|
int alpha_bottom = MAX(MIN(((2 << 24) - frac[col]) >> (16 - start_fade), 256), 0);
|
|
int inv_alpha_bottom = 256 - alpha_bottom;
|
|
int c_red = RPART(c);
|
|
int c_green = GPART(c);
|
|
int c_blue = BPART(c);
|
|
c_red = (c_red * alpha_bottom + solid_bottom_r * inv_alpha_bottom) >> 8;
|
|
c_green = (c_green * alpha_bottom + solid_bottom_g * inv_alpha_bottom) >> 8;
|
|
c_blue = (c_blue * alpha_bottom + solid_bottom_b * inv_alpha_bottom) >> 8;
|
|
output[col] = RGB256k.RGB[(c_red >> 2)][(c_green >> 2)][(c_blue >> 2)];
|
|
|
|
frac[col] += fracstep[col];
|
|
}
|
|
*((uint32_t*)dest) = *((uint32_t*)output);
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
|
|
// Bottom solid color:
|
|
while (index < count)
|
|
{
|
|
*((uint32_t*)dest) = solid_bottom_fill;
|
|
dest += pitch;
|
|
index += num_cores;
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
PalColumnCommand::PalColumnCommand()
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_count = dc_count;
|
|
_dest = dc_dest;
|
|
_pitch = dc_pitch;
|
|
_iscale = dc_iscale;
|
|
_texturefrac = dc_texturefrac;
|
|
_colormap = dc_colormap;
|
|
_source = dc_source;
|
|
_translation = dc_translation;
|
|
_color = dc_color;
|
|
_srcblend = dc_srcblend;
|
|
_destblend = dc_destblend;
|
|
_srccolor = dc_srccolor;
|
|
_srcalpha = dc_srcalpha;
|
|
_destalpha = dc_destalpha;
|
|
}
|
|
|
|
void DrawColumnPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
|
|
// Framebuffer destination address.
|
|
dest = _dest;
|
|
|
|
// Determine scaling,
|
|
// which is the only mapping to be done.
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
// [RH] Get local copies of these variables so that the compiler
|
|
// has a better chance of optimizing this well.
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
|
|
// Inner loop that does the actual texture mapping,
|
|
// e.g. a DDA-lile scaling.
|
|
// This is as fast as it gets.
|
|
do
|
|
{
|
|
// Re-map color indices from wall texture column
|
|
// using a lighting/special effects LUT.
|
|
*dest = colormap[source[frac >> FRACBITS]];
|
|
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
|
|
} while (--count);
|
|
}
|
|
|
|
void FillColumnPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
pitch *= thread->num_cores;
|
|
|
|
uint8_t color = _color;
|
|
do
|
|
{
|
|
*dest = color;
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void FillColumnAddPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
uint32_t *bg2rgb;
|
|
uint32_t fg;
|
|
|
|
bg2rgb = _destblend;
|
|
fg = _srccolor;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
pitch *= thread->num_cores;
|
|
|
|
const PalEntry* pal = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int src_r = ((_srccolor >> 16) & 0xff) * _srcalpha;
|
|
int src_g = ((_srccolor >> 0) & 0xff) * _srcalpha;
|
|
int src_b = ((_srccolor >> 8) & 0xff) * _srcalpha;
|
|
int r = clamp((src_r + pal[*dest].r * _destalpha)>>18, 0, 255);
|
|
int g = clamp((src_g + pal[*dest].g * _destalpha)>>18, 0, 255);
|
|
int b = clamp((src_b + pal[*dest].b * _destalpha)>>18, 0, 255);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
} while (--count);
|
|
|
|
}
|
|
|
|
void FillColumnAddClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
|
|
count = _count;
|
|
|
|
dest = _dest;
|
|
uint32_t *bg2rgb;
|
|
uint32_t fg;
|
|
|
|
bg2rgb = _destblend;
|
|
fg = _srccolor;
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
pitch *= thread->num_cores;
|
|
|
|
const PalEntry* pal = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int src_r = ((_srccolor >> 16) & 0xff) * _srcalpha;
|
|
int src_g = ((_srccolor >> 0) & 0xff) * _srcalpha;
|
|
int src_b = ((_srccolor >> 8) & 0xff) * _srcalpha;
|
|
int r = clamp((src_r + pal[*dest].r * _destalpha)>>18, 0, 255);
|
|
int g = clamp((src_g + pal[*dest].g * _destalpha)>>18, 0, 255);
|
|
int b = clamp((src_b + pal[*dest].b * _destalpha)>>18, 0, 255);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void FillColumnSubClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
|
|
count = _count;
|
|
|
|
dest = _dest;
|
|
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
pitch *= thread->num_cores;
|
|
|
|
const PalEntry* palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int src_r = ((_srccolor >> 16) & 0xff) * _srcalpha;
|
|
int src_g = ((_srccolor >> 0) & 0xff) * _srcalpha;
|
|
int src_b = ((_srccolor >> 8) & 0xff) * _srcalpha;
|
|
int bg = *dest;
|
|
int r = MAX((src_r * _srcalpha - palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((src_g * _srcalpha - palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((src_b * _srcalpha - palette[bg].b * _destalpha)>>18, 0);
|
|
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void FillColumnRevSubClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
|
|
count = _count;
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = _dest;
|
|
|
|
int pitch = _pitch;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
pitch *= thread->num_cores;
|
|
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int src_r = ((_srccolor >> 16) & 0xff) * _srcalpha;
|
|
int src_g = ((_srccolor >> 0) & 0xff) * _srcalpha;
|
|
int src_b = ((_srccolor >> 8) & 0xff) * _srcalpha;
|
|
int bg = *dest;
|
|
int r = MAX((src_r * _srcalpha - palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((src_g * _srcalpha - palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((src_b * _srcalpha - palette[bg].b * _destalpha)>>18, 0);
|
|
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnAddPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
uint32_t fg = colormap[source[frac >> FRACBITS]];
|
|
uint32_t bg = *dest;
|
|
uint32_t r = MIN((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 63);
|
|
uint32_t g = MIN((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 63);
|
|
uint32_t b = MIN((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 63);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnTranslatedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t* dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
// [RH] Local copies of global vars to improve compiler optimizations
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *translation = _translation;
|
|
const uint8_t *source = _source;
|
|
|
|
do
|
|
{
|
|
*dest = colormap[translation[source[frac >> FRACBITS]]];
|
|
dest += pitch;
|
|
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnTlatedAddPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *translation = _translation;
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
uint32_t fg = colormap[translation[source[frac >> FRACBITS]]];
|
|
uint32_t bg = *dest;
|
|
uint32_t r = MIN((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 63);
|
|
uint32_t g = MIN((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 63);
|
|
uint32_t b = MIN((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 63);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnShadedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac, fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
uint32_t val = source[frac >> FRACBITS];
|
|
|
|
int r = (palette[*dest].r * (255-val) + palette[_color].r * val) >> 10;
|
|
int g = (palette[*dest].g * (255-val) + palette[_color].g * val) >> 10;
|
|
int b = (palette[*dest].b * (255-val) + palette[_color].b * val) >> 10;
|
|
*dest = RGB256k.RGB[clamp(r,0,63)][clamp(g,0,63)][clamp(b,0,63)];
|
|
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnAddClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[source[frac >> FRACBITS]];
|
|
int bg = *dest;
|
|
int r = MIN((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 63);
|
|
int g = MIN((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 63);
|
|
int b = MIN((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 63);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnAddClampTranslatedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *translation = _translation;
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[translation[source[frac >> FRACBITS]]];
|
|
int bg = *dest;
|
|
int r = MIN((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 63);
|
|
int g = MIN((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 63);
|
|
int b = MIN((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 63);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnSubClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[source[frac >> FRACBITS]];
|
|
int bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha - palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha - palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha - palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnSubClampTranslatedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *translation = _translation;
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[translation[source[frac >> FRACBITS]]];
|
|
int bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha - palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha - palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha - palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnRevSubClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[source[frac >> FRACBITS]];
|
|
int bg = *dest;
|
|
int r = MAX((-palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((-palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((-palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
void DrawColumnRevSubClampTranslatedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count;
|
|
uint8_t *dest;
|
|
fixed_t frac;
|
|
fixed_t fracstep;
|
|
|
|
count = _count;
|
|
dest = _dest;
|
|
|
|
fracstep = _iscale;
|
|
frac = _texturefrac;
|
|
|
|
count = thread->count_for_thread(_dest_y, count);
|
|
if (count <= 0)
|
|
return;
|
|
|
|
int pitch = _pitch;
|
|
dest = thread->dest_for_thread(_dest_y, pitch, dest);
|
|
frac += fracstep * thread->skipped_by_thread(_dest_y);
|
|
fracstep *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
const uint8_t *translation = _translation;
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _source;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
do
|
|
{
|
|
int fg = colormap[translation[source[frac >> FRACBITS]]];
|
|
int bg = *dest;
|
|
int r = MAX((-palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((-palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((-palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
dest += pitch;
|
|
frac += fracstep;
|
|
} while (--count);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
DrawFuzzColumnPalCommand::DrawFuzzColumnPalCommand()
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_yl = dc_yl;
|
|
_yh = dc_yh;
|
|
_x = dc_x;
|
|
_destorg = dc_destorg;
|
|
_pitch = dc_pitch;
|
|
_fuzzpos = fuzzpos;
|
|
_fuzzviewheight = fuzzviewheight;
|
|
}
|
|
|
|
void DrawFuzzColumnPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int yl = MAX(_yl, 1);
|
|
int yh = MIN(_yh, _fuzzviewheight);
|
|
|
|
int count = thread->count_for_thread(yl, yh - yl + 1);
|
|
|
|
// Zero length.
|
|
if (count <= 0)
|
|
return;
|
|
|
|
uint8_t *map = &NormalLight.Maps[6 * 256];
|
|
|
|
uint8_t *dest = thread->dest_for_thread(yl, _pitch, ylookup[yl] + _x + _destorg);
|
|
|
|
int pitch = _pitch * thread->num_cores;
|
|
int fuzzstep = thread->num_cores;
|
|
int fuzz = (_fuzzpos + thread->skipped_by_thread(yl)) % FUZZTABLE;
|
|
|
|
yl += thread->skipped_by_thread(yl);
|
|
|
|
// Handle the case where we would go out of bounds at the top:
|
|
if (yl < fuzzstep)
|
|
{
|
|
uint8_t *srcdest = dest + fuzzoffset[fuzz] * fuzzstep + pitch;
|
|
//assert(static_cast<int>((srcdest - (uint8_t*)dc_destorg) / (_pitch)) < viewheight);
|
|
|
|
*dest = map[*srcdest];
|
|
dest += pitch;
|
|
fuzz += fuzzstep;
|
|
fuzz %= FUZZTABLE;
|
|
|
|
count--;
|
|
if (count == 0)
|
|
return;
|
|
}
|
|
|
|
bool lowerbounds = (yl + (count + fuzzstep - 1) * fuzzstep > _fuzzviewheight);
|
|
if (lowerbounds)
|
|
count--;
|
|
|
|
// Fuzz where fuzzoffset stays within bounds
|
|
while (count > 0)
|
|
{
|
|
int available = (FUZZTABLE - fuzz);
|
|
int next_wrap = available / fuzzstep;
|
|
if (available % fuzzstep != 0)
|
|
next_wrap++;
|
|
|
|
int cnt = MIN(count, next_wrap);
|
|
count -= cnt;
|
|
do
|
|
{
|
|
uint8_t *srcdest = dest + fuzzoffset[fuzz] * fuzzstep;
|
|
//assert(static_cast<int>((srcdest - (uint8_t*)dc_destorg) / (_pitch)) < viewheight);
|
|
|
|
*dest = map[*srcdest];
|
|
dest += pitch;
|
|
fuzz += fuzzstep;
|
|
} while (--cnt);
|
|
|
|
fuzz %= FUZZTABLE;
|
|
}
|
|
|
|
// Handle the case where we would go out of bounds at the bottom
|
|
if (lowerbounds)
|
|
{
|
|
uint8_t *srcdest = dest + fuzzoffset[fuzz] * fuzzstep - pitch;
|
|
//assert(static_cast<int>((srcdest - (uint8_t*)dc_destorg) / (_pitch)) < viewheight);
|
|
|
|
*dest = map[*srcdest];
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
PalSpanCommand::PalSpanCommand()
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_source = ds_source;
|
|
_colormap = ds_colormap;
|
|
_xfrac = ds_xfrac;
|
|
_yfrac = ds_yfrac;
|
|
_y = ds_y;
|
|
_x1 = ds_x1;
|
|
_x2 = ds_x2;
|
|
_destorg = dc_destorg;
|
|
_xstep = ds_xstep;
|
|
_ystep = ds_ystep;
|
|
_xbits = ds_xbits;
|
|
_ybits = ds_ybits;
|
|
_srcblend = dc_srcblend;
|
|
_destblend = dc_destblend;
|
|
_color = ds_color;
|
|
_srcalpha = dc_srcalpha;
|
|
_destalpha = dc_destalpha;
|
|
}
|
|
|
|
void DrawSpanPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
// Current texture index in u,v.
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
|
|
// Lookup pixel from flat texture tile,
|
|
// re-index using light/colormap.
|
|
*dest++ = colormap[source[spot]];
|
|
|
|
// Next step in u,v.
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
|
|
do
|
|
{
|
|
// Current texture index in u,v.
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
|
|
// Lookup pixel from flat texture tile,
|
|
// re-index using light/colormap.
|
|
*dest++ = colormap[source[spot]];
|
|
|
|
// Next step in u,v.
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void DrawSpanMaskedPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
int texdata;
|
|
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
*dest = colormap[texdata];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
do
|
|
{
|
|
int texdata;
|
|
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
*dest = colormap[texdata];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void DrawSpanTranslucentPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
uint32_t fg = colormap[source[spot]];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest++ = RGB256k.RGB[r][g][b];
|
|
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
do
|
|
{
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
uint32_t fg = colormap[source[spot]];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest++ = RGB256k.RGB[r][g][b];
|
|
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void DrawSpanMaskedTranslucentPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
uint8_t texdata;
|
|
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
uint32_t fg = colormap[texdata];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
do
|
|
{
|
|
uint8_t texdata;
|
|
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
uint32_t fg = colormap[texdata];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void DrawSpanAddClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
uint32_t fg = colormap[source[spot]];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest++ = RGB256k.RGB[r][g][b];
|
|
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
do
|
|
{
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
uint32_t fg = colormap[source[spot]];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest++ = RGB256k.RGB[r][g][b];
|
|
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void DrawSpanMaskedAddClampPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
dsfixed_t xfrac;
|
|
dsfixed_t yfrac;
|
|
dsfixed_t xstep;
|
|
dsfixed_t ystep;
|
|
uint8_t *dest;
|
|
const uint8_t *source = _source;
|
|
const uint8_t *colormap = _colormap;
|
|
int count;
|
|
int spot;
|
|
const PalEntry *palette = GPalette.BaseColors;
|
|
|
|
xfrac = _xfrac;
|
|
yfrac = _yfrac;
|
|
|
|
dest = ylookup[_y] + _x1 + _destorg;
|
|
|
|
count = _x2 - _x1 + 1;
|
|
|
|
xstep = _xstep;
|
|
ystep = _ystep;
|
|
|
|
if (_xbits == 6 && _ybits == 6)
|
|
{
|
|
// 64x64 is the most common case by far, so special case it.
|
|
do
|
|
{
|
|
uint8_t texdata;
|
|
|
|
spot = ((xfrac >> (32 - 6 - 6))&(63 * 64)) + (yfrac >> (32 - 6));
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
uint32_t fg = colormap[texdata];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
else
|
|
{
|
|
uint8_t yshift = 32 - _ybits;
|
|
uint8_t xshift = yshift - _xbits;
|
|
int xmask = ((1 << _xbits) - 1) << _ybits;
|
|
do
|
|
{
|
|
uint8_t texdata;
|
|
|
|
spot = ((xfrac >> xshift) & xmask) + (yfrac >> yshift);
|
|
texdata = source[spot];
|
|
if (texdata != 0)
|
|
{
|
|
uint32_t fg = colormap[texdata];
|
|
uint32_t bg = *dest;
|
|
int r = MAX((palette[fg].r * _srcalpha + palette[bg].r * _destalpha)>>18, 0);
|
|
int g = MAX((palette[fg].g * _srcalpha + palette[bg].g * _destalpha)>>18, 0);
|
|
int b = MAX((palette[fg].b * _srcalpha + palette[bg].b * _destalpha)>>18, 0);
|
|
*dest = RGB256k.RGB[r][g][b];
|
|
}
|
|
dest++;
|
|
xfrac += xstep;
|
|
yfrac += ystep;
|
|
} while (--count);
|
|
}
|
|
}
|
|
|
|
void FillSpanPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(_y))
|
|
return;
|
|
|
|
memset(ylookup[_y] + _x1 + _destorg, _color, _x2 - _x1 + 1);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
DrawTiltedSpanPalCommand::DrawTiltedSpanPalCommand(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)
|
|
: y(y), x1(x1), x2(x2), plane_sz(plane_sz), plane_su(plane_su), plane_sv(plane_sv), plane_shade(plane_shade), planeshade(planeshade), planelightfloat(planelightfloat), pviewx(pviewx), pviewy(pviewy)
|
|
{
|
|
using namespace drawerargs;
|
|
|
|
_colormap = ds_colormap;
|
|
_destorg = dc_destorg;
|
|
_ybits = ds_ybits;
|
|
_xbits = ds_xbits;
|
|
_source = ds_source;
|
|
basecolormapdata = basecolormap->Maps;
|
|
}
|
|
|
|
void DrawTiltedSpanPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(y))
|
|
return;
|
|
|
|
const uint8_t **tiltlighting = thread->tiltlighting;
|
|
|
|
int width = x2 - x1;
|
|
double iz, uz, vz;
|
|
uint8_t *fb;
|
|
uint32_t u, v;
|
|
int i;
|
|
|
|
iz = plane_sz[2] + plane_sz[1] * (centery - y) + plane_sz[0] * (x1 - centerx);
|
|
|
|
// Lighting is simple. It's just linear interpolation from start to end
|
|
if (plane_shade)
|
|
{
|
|
uz = (iz + plane_sz[0] * width) * planelightfloat;
|
|
vz = iz * planelightfloat;
|
|
CalcTiltedLighting(vz, uz, width, thread);
|
|
}
|
|
else
|
|
{
|
|
for (int i = 0; i < width; ++i)
|
|
{
|
|
tiltlighting[i] = _colormap;
|
|
}
|
|
}
|
|
|
|
uz = plane_su[2] + plane_su[1] * (centery - y) + plane_su[0] * (x1 - centerx);
|
|
vz = plane_sv[2] + plane_sv[1] * (centery - y) + plane_sv[0] * (x1 - centerx);
|
|
|
|
fb = ylookup[y] + x1 + _destorg;
|
|
|
|
uint8_t vshift = 32 - _ybits;
|
|
uint8_t ushift = vshift - _xbits;
|
|
int umask = ((1 << _xbits) - 1) << _ybits;
|
|
|
|
#if 0
|
|
// The "perfect" reference version of this routine. Pretty slow.
|
|
// Use it only to see how things are supposed to look.
|
|
i = 0;
|
|
do
|
|
{
|
|
double z = 1.f / iz;
|
|
|
|
u = int64_t(uz*z) + pviewx;
|
|
v = int64_t(vz*z) + pviewy;
|
|
R_SetDSColorMapLight(tiltlighting[i], 0, 0);
|
|
fb[i++] = ds_colormap[ds_source[(v >> vshift) | ((u >> ushift) & umask)]];
|
|
iz += plane_sz[0];
|
|
uz += plane_su[0];
|
|
vz += plane_sv[0];
|
|
} while (--width >= 0);
|
|
#else
|
|
//#define SPANSIZE 32
|
|
//#define INVSPAN 0.03125f
|
|
//#define SPANSIZE 8
|
|
//#define INVSPAN 0.125f
|
|
#define SPANSIZE 16
|
|
#define INVSPAN 0.0625f
|
|
|
|
double startz = 1.f / iz;
|
|
double startu = uz*startz;
|
|
double startv = vz*startz;
|
|
double izstep, uzstep, vzstep;
|
|
|
|
izstep = plane_sz[0] * SPANSIZE;
|
|
uzstep = plane_su[0] * SPANSIZE;
|
|
vzstep = plane_sv[0] * SPANSIZE;
|
|
x1 = 0;
|
|
width++;
|
|
|
|
while (width >= SPANSIZE)
|
|
{
|
|
iz += izstep;
|
|
uz += uzstep;
|
|
vz += vzstep;
|
|
|
|
double endz = 1.f / iz;
|
|
double endu = uz*endz;
|
|
double endv = vz*endz;
|
|
uint32_t stepu = (uint32_t)int64_t((endu - startu) * INVSPAN);
|
|
uint32_t stepv = (uint32_t)int64_t((endv - startv) * INVSPAN);
|
|
u = (uint32_t)(int64_t(startu) + pviewx);
|
|
v = (uint32_t)(int64_t(startv) + pviewy);
|
|
|
|
for (i = SPANSIZE - 1; i >= 0; i--)
|
|
{
|
|
fb[x1] = *(tiltlighting[x1] + _source[(v >> vshift) | ((u >> ushift) & umask)]);
|
|
x1++;
|
|
u += stepu;
|
|
v += stepv;
|
|
}
|
|
startu = endu;
|
|
startv = endv;
|
|
width -= SPANSIZE;
|
|
}
|
|
if (width > 0)
|
|
{
|
|
if (width == 1)
|
|
{
|
|
u = (uint32_t)int64_t(startu);
|
|
v = (uint32_t)int64_t(startv);
|
|
fb[x1] = *(tiltlighting[x1] + _source[(v >> vshift) | ((u >> ushift) & umask)]);
|
|
}
|
|
else
|
|
{
|
|
double left = width;
|
|
iz += plane_sz[0] * left;
|
|
uz += plane_su[0] * left;
|
|
vz += plane_sv[0] * left;
|
|
|
|
double endz = 1.f / iz;
|
|
double endu = uz*endz;
|
|
double endv = vz*endz;
|
|
left = 1.f / left;
|
|
uint32_t stepu = (uint32_t)int64_t((endu - startu) * left);
|
|
uint32_t stepv = (uint32_t)int64_t((endv - startv) * left);
|
|
u = (uint32_t)(int64_t(startu) + pviewx);
|
|
v = (uint32_t)(int64_t(startv) + pviewy);
|
|
|
|
for (; width != 0; width--)
|
|
{
|
|
fb[x1] = *(tiltlighting[x1] + _source[(v >> vshift) | ((u >> ushift) & umask)]);
|
|
x1++;
|
|
u += stepu;
|
|
v += stepv;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// Calculates the lighting for one row of a tilted plane. If the definition
|
|
// of GETPALOOKUP changes, this needs to change, too.
|
|
void DrawTiltedSpanPalCommand::CalcTiltedLighting(double lval, double lend, int width, DrawerThread *thread)
|
|
{
|
|
const uint8_t **tiltlighting = thread->tiltlighting;
|
|
|
|
double lstep;
|
|
uint8_t *lightfiller;
|
|
int i = 0;
|
|
|
|
if (width == 0 || lval == lend)
|
|
{ // Constant lighting
|
|
lightfiller = basecolormapdata + (GETPALOOKUP(lval, planeshade) << COLORMAPSHIFT);
|
|
}
|
|
else
|
|
{
|
|
lstep = (lend - lval) / width;
|
|
if (lval >= MAXLIGHTVIS)
|
|
{ // lval starts "too bright".
|
|
lightfiller = basecolormapdata + (GETPALOOKUP(lval, planeshade) << COLORMAPSHIFT);
|
|
for (; i <= width && lval >= MAXLIGHTVIS; ++i)
|
|
{
|
|
tiltlighting[i] = lightfiller;
|
|
lval += lstep;
|
|
}
|
|
}
|
|
if (lend >= MAXLIGHTVIS)
|
|
{ // lend ends "too bright".
|
|
lightfiller = basecolormapdata + (GETPALOOKUP(lend, planeshade) << COLORMAPSHIFT);
|
|
for (; width > i && lend >= MAXLIGHTVIS; --width)
|
|
{
|
|
tiltlighting[width] = lightfiller;
|
|
lend -= lstep;
|
|
}
|
|
}
|
|
if (width > 0)
|
|
{
|
|
lval = FIXED2DBL(planeshade) - lval;
|
|
lend = FIXED2DBL(planeshade) - lend;
|
|
lstep = (lend - lval) / width;
|
|
if (lstep < 0)
|
|
{ // Going from dark to light
|
|
if (lval < 1.)
|
|
{ // All bright
|
|
lightfiller = basecolormapdata;
|
|
}
|
|
else
|
|
{
|
|
if (lval >= NUMCOLORMAPS)
|
|
{ // Starts beyond the dark end
|
|
uint8_t *clight = basecolormapdata + ((NUMCOLORMAPS - 1) << COLORMAPSHIFT);
|
|
while (lval >= NUMCOLORMAPS && i <= width)
|
|
{
|
|
tiltlighting[i++] = clight;
|
|
lval += lstep;
|
|
}
|
|
if (i > width)
|
|
return;
|
|
}
|
|
while (i <= width && lval >= 0)
|
|
{
|
|
tiltlighting[i++] = basecolormapdata + (xs_ToInt(lval) << COLORMAPSHIFT);
|
|
lval += lstep;
|
|
}
|
|
lightfiller = basecolormapdata;
|
|
}
|
|
}
|
|
else
|
|
{ // Going from light to dark
|
|
if (lval >= (NUMCOLORMAPS - 1))
|
|
{ // All dark
|
|
lightfiller = basecolormapdata + ((NUMCOLORMAPS - 1) << COLORMAPSHIFT);
|
|
}
|
|
else
|
|
{
|
|
while (lval < 0 && i <= width)
|
|
{
|
|
tiltlighting[i++] = basecolormapdata;
|
|
lval += lstep;
|
|
}
|
|
if (i > width)
|
|
return;
|
|
while (i <= width && lval < (NUMCOLORMAPS - 1))
|
|
{
|
|
tiltlighting[i++] = basecolormapdata + (xs_ToInt(lval) << COLORMAPSHIFT);
|
|
lval += lstep;
|
|
}
|
|
lightfiller = basecolormapdata + ((NUMCOLORMAPS - 1) << COLORMAPSHIFT);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (; i <= width; i++)
|
|
{
|
|
tiltlighting[i] = lightfiller;
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
DrawColoredSpanPalCommand::DrawColoredSpanPalCommand(int y, int x1, int x2) : y(y), x1(x1), x2(x2)
|
|
{
|
|
using namespace drawerargs;
|
|
color = ds_color;
|
|
destorg = dc_destorg;
|
|
}
|
|
|
|
void DrawColoredSpanPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(y))
|
|
return;
|
|
|
|
memset(ylookup[y] + x1 + destorg, color, x2 - x1 + 1);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
DrawSlabPalCommand::DrawSlabPalCommand(int dx, fixed_t v, int dy, fixed_t vi, const uint8_t *vptr, uint8_t *p, const uint8_t *colormap)
|
|
: _dx(dx), _v(v), _dy(dy), _vi(vi), _vvptr(vptr), _p(p), _colormap(colormap)
|
|
{
|
|
using namespace drawerargs;
|
|
_pitch = dc_pitch;
|
|
_start_y = static_cast<int>((p - dc_destorg) / dc_pitch);
|
|
}
|
|
|
|
void DrawSlabPalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
int count = _dy;
|
|
uint8_t *dest = _p;
|
|
int pitch = _pitch;
|
|
int width = _dx;
|
|
const uint8_t *colormap = _colormap;
|
|
const uint8_t *source = _vvptr;
|
|
fixed_t fracpos = _v;
|
|
fixed_t iscale = _vi;
|
|
|
|
count = thread->count_for_thread(_start_y, count);
|
|
dest = thread->dest_for_thread(_start_y, pitch, dest);
|
|
fracpos += iscale * thread->skipped_by_thread(_start_y);
|
|
iscale *= thread->num_cores;
|
|
pitch *= thread->num_cores;
|
|
|
|
while (count > 0)
|
|
{
|
|
uint8_t color = colormap[source[fracpos >> FRACBITS]];
|
|
|
|
for (int x = 0; x < width; x++)
|
|
dest[x] = color;
|
|
|
|
dest += pitch;
|
|
fracpos += iscale;
|
|
count--;
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
DrawFogBoundaryLinePalCommand::DrawFogBoundaryLinePalCommand(int y, int x1, int x2) : y(y), x1(x1), x2(x2)
|
|
{
|
|
using namespace drawerargs;
|
|
_colormap = dc_colormap;
|
|
_destorg = dc_destorg;
|
|
}
|
|
|
|
void DrawFogBoundaryLinePalCommand::Execute(DrawerThread *thread)
|
|
{
|
|
if (thread->line_skipped_by_thread(y))
|
|
return;
|
|
|
|
const uint8_t *colormap = _colormap;
|
|
uint8_t *dest = ylookup[y] + _destorg;
|
|
int x = x1;
|
|
do
|
|
{
|
|
dest[x] = colormap[dest[x]];
|
|
} while (++x <= x2);
|
|
}
|
|
}
|