mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-07 09:40:43 +00:00
dc60c7f72e
Due to the blending they are essentially inactive on translucent content and Duke Nukem heavily abuses this.
337 lines
9.3 KiB
C++
337 lines
9.3 KiB
C++
/*
|
|
** gl_palmanager.cpp
|
|
** Palette management
|
|
**
|
|
**---------------------------------------------------------------------------
|
|
** Copyright 2019 Christoph Oelckers
|
|
** 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 <memory>
|
|
#include "m_crc32.h"
|
|
#include "glbackend.h"
|
|
|
|
#include "baselayer.h"
|
|
#include "resourcefile.h"
|
|
#include "imagehelpers.h"
|
|
#include "v_font.h"
|
|
#include "palette.h"
|
|
|
|
//===========================================================================
|
|
//
|
|
// The palette manager will contain all palettes being used for texture
|
|
// creation. It is also responsible for creating palette textures for indexed
|
|
// rendering.
|
|
//
|
|
//===========================================================================
|
|
|
|
PaletteManager::~PaletteManager()
|
|
{
|
|
DeleteAll();
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
//
|
|
//
|
|
//===========================================================================
|
|
|
|
void PaletteManager::DeleteAllTextures()
|
|
{
|
|
for (auto& pal : palettes)
|
|
{
|
|
if (pal.paltexture) delete pal.paltexture;
|
|
pal.paltexture = nullptr;
|
|
}
|
|
for (auto& pal : palswaps)
|
|
{
|
|
if (pal.swaptexture) delete pal.swaptexture;
|
|
pal.swaptexture = nullptr;
|
|
}
|
|
if (palswapTexture) delete palswapTexture;
|
|
palswapTexture = nullptr;
|
|
}
|
|
|
|
void PaletteManager::DeleteAll()
|
|
{
|
|
DeleteAllTextures();
|
|
palettes.Reset();
|
|
palswaps.Reset();
|
|
lastindex = ~0u;
|
|
lastsindex = ~0u;
|
|
memset(palettemap, 0, sizeof(palettemap));
|
|
memset(palswapmap, 0, sizeof(palswapmap));
|
|
memset(addshade, 0, sizeof(addshade));
|
|
memset(mulshade, 0, sizeof(mulshade));
|
|
numshades = 1;
|
|
|
|
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
// Adds a new palette while looking for duplicates.
|
|
//
|
|
//===========================================================================
|
|
|
|
unsigned PaletteManager::FindPalette(const uint8_t *paldata)
|
|
{
|
|
auto crc32 = CalcCRC32(paldata, 1024);
|
|
for (unsigned int i = 0; i< palettes.Size(); i++)
|
|
{
|
|
if (crc32 == palettes[i].crc32)
|
|
{
|
|
if (!memcmp(paldata, palettes[i].colors, 1024))
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
PaletteData pd;
|
|
memcpy(pd.colors, paldata, 1024);
|
|
pd.crc32 = crc32;
|
|
pd.paltexture = nullptr;
|
|
pd.shadesdone = false;
|
|
return palettes.Push(pd);
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
// Adds a new palette while looking for duplicates.
|
|
//
|
|
//===========================================================================
|
|
|
|
unsigned PaletteManager::FindPalswap(const uint8_t* paldata, palette_t &fadecolor)
|
|
{
|
|
if (paldata == nullptr) return 0;
|
|
auto crc32 = CalcCRC32(paldata, 256 * numshades);
|
|
for (unsigned int i = 0; i < palswaps.Size(); i++)
|
|
{
|
|
if (crc32 == palswaps[i].crc32)
|
|
{
|
|
if (!memcmp(paldata, palswaps[i].lookup, 256 * numshades))
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
PalswapData pd;
|
|
pd.lookup = paldata;
|
|
pd.crc32 = crc32;
|
|
pd.swaptexture = nullptr;
|
|
memset(pd.brightcolors, 0, 256);
|
|
pd.isbright = false;
|
|
|
|
for (int i = 0; i < 255; i++)
|
|
{
|
|
int firstmap = paldata[i];
|
|
int lastmap = paldata[i + 256 * (numshades - 2)];
|
|
|
|
PalEntry color1 = palettes[palettemap[0]].colors[firstmap];
|
|
PalEntry color2 = palettes[palettemap[0]].colors[lastmap];
|
|
int lum1 = color1.Amplitude();
|
|
int lum2 = color2.Amplitude();
|
|
if (lum1 > 40 && lum2 * 10 >= lum1 * 9)
|
|
{
|
|
pd.brightcolors[i] = 255;
|
|
pd.isbright = true;
|
|
}
|
|
}
|
|
|
|
|
|
if (fadecolor.f == 0)
|
|
{
|
|
// Find what index maps to black (or the darkest available color)
|
|
int found = -1;
|
|
PalEntry foundColor = 0xffffffff;
|
|
for (int i = 0; i < 255; i++)
|
|
{
|
|
int map = paldata[i];
|
|
PalEntry color = palettes[palettemap[0]].colors[map];
|
|
if (color.Luminance() < foundColor.Luminance())
|
|
{
|
|
foundColor = color;
|
|
found = i;
|
|
}
|
|
}
|
|
|
|
// Determine the fade color. We pick what black, or the darkest color, maps to in the lowest shade level.
|
|
int map = paldata[(numshades - 2) * 256 + found]; // do not look in the latest shade level because it doesn't always contain useful data for this.
|
|
pd.fadeColor = palettes[palettemap[0]].colors[map];
|
|
if (pd.fadeColor.Luminance() < 10) pd.fadeColor = 0; // Account for the inability to check the last fade level by using a higher threshold for determining black fog.
|
|
}
|
|
else
|
|
{
|
|
pd.fadeColor = PalEntry(fadecolor.r, fadecolor.g, fadecolor.b);
|
|
}
|
|
|
|
return palswaps.Push(pd);
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
//
|
|
//
|
|
//===========================================================================
|
|
|
|
void PaletteManager::SetPalette(int index, const uint8_t* data)
|
|
{
|
|
// New palettes may only be added if declared transient or on startup.
|
|
// Otherwise this would require a renderer reset to flush out the textures affected by the change.
|
|
|
|
if (index < 0 || index > 255) return; // invalid index - ignore.
|
|
palettemap[index] = FindPalette(data);
|
|
if (index == 0)
|
|
{
|
|
ImageHelpers::SetPalette((PalEntry*)data); // Palette 0 is always the reference for downconverting images
|
|
}
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
//
|
|
//
|
|
//===========================================================================
|
|
|
|
void PaletteManager::BindPalette(int index)
|
|
{
|
|
if (palettemap[index] < palettes.Size())
|
|
{
|
|
auto uindex = palettemap[index];
|
|
if (uindex != lastindex)
|
|
{
|
|
lastindex = uindex;
|
|
if (palettes[uindex].paltexture == nullptr)
|
|
{
|
|
auto p = GLInterface.NewTexture();
|
|
p->CreateTexture(256, 1, FHardwareTexture::TrueColor, false);
|
|
p->LoadTexture((uint8_t*)palettes[uindex].colors);
|
|
p->SetSampler(SamplerNoFilterClampXY);
|
|
palettes[uindex].paltexture = p;
|
|
}
|
|
inst->BindTexture(2, palettes[uindex].paltexture);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
//
|
|
//
|
|
//===========================================================================
|
|
|
|
void PaletteManager::SetPalswapData(int index, const uint8_t* data, int numshades_, palette_t &fadecolor)
|
|
{
|
|
if (index < 0 || index > 255) return; // invalid index - ignore.
|
|
numshades = numshades_;
|
|
palswapmap[index] = FindPalswap(data, fadecolor);
|
|
}
|
|
|
|
//===========================================================================
|
|
//
|
|
//
|
|
//
|
|
//===========================================================================
|
|
|
|
void PaletteManager::BindPalswap(int index)
|
|
{
|
|
if (palswapmap[index] < palswaps.Size())
|
|
{
|
|
auto uindex = palswapmap[index];
|
|
if (uindex != lastsindex)
|
|
{
|
|
lastsindex = uindex;
|
|
auto& ps = palswaps[uindex];
|
|
if (ps.swaptexture == nullptr)
|
|
{
|
|
auto p = GLInterface.NewTexture();
|
|
p->CreateTexture(256, numshades, FHardwareTexture::Indexed, false);
|
|
p->LoadTexture((uint8_t*)ps.lookup);
|
|
p->SetSampler(SamplerNoFilterClampXY);
|
|
ps.swaptexture = p;
|
|
}
|
|
inst->BindTexture(1, ps.swaptexture);
|
|
inst->SetFadeColor(ps.fadeColor);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
int PaletteManager::LookupPalette(int palette, int palswap, bool brightmap)
|
|
{
|
|
int realpal = palettemap[palette];
|
|
int realswap = palswapmap[palswap];
|
|
int combined = (brightmap? 0x1000000 : 0) + realpal * 0x10000 + realswap;
|
|
int* combinedindex = swappedpalmap.CheckKey(combined);
|
|
if (combinedindex) return *combinedindex;
|
|
|
|
PaletteData* paldata = &palettes[realpal];
|
|
PalswapData* swapdata = &palswaps[realswap];
|
|
PalEntry swappedpalette[256];
|
|
if (!brightmap)
|
|
{
|
|
for (int i = 0; i < 255; i++)
|
|
{
|
|
int swapi = swapdata->lookup[i];
|
|
swappedpalette[i] = paldata->colors[swapi];
|
|
swappedpalette[i].a = 255;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!swapdata->isbright)
|
|
{
|
|
swappedpalmap.Insert(combined, -1);
|
|
return -1;
|
|
}
|
|
|
|
bool found = false;
|
|
memset(swappedpalette, 0, sizeof(swappedpalette));
|
|
for (int i = 0; i < 255; i++)
|
|
{
|
|
int swapi = swapdata->brightcolors[i];
|
|
if (swapi)
|
|
{
|
|
found = true;
|
|
swappedpalette[i] = 0xffffffff;
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
swappedpalmap.Insert(combined, -1);
|
|
return -1;
|
|
}
|
|
}
|
|
swappedpalette[255] = 0;
|
|
int palid = FindPalette((uint8_t*)swappedpalette);
|
|
swappedpalmap.Insert(combined, palid);
|
|
return palid;
|
|
}
|