doom3-bfg/neo/renderer/Image_intrinsic.cpp

896 lines
24 KiB
C++
Raw Normal View History

2012-11-26 18:58:24 +00:00
/*
===========================================================================
Doom 3 BFG Edition GPL Source Code
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
2015-12-26 14:20:19 +00:00
Copyright (C) 2013-2015 Robert Beckebans
2012-11-26 18:58:24 +00:00
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
2012-11-26 18:58:24 +00:00
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#pragma hdrstop
#include "precompiled.h"
2012-11-26 18:58:24 +00:00
#include "tr_local.h"
2015-12-26 14:20:19 +00:00
#include "SMAA/AreaTex.h"
#include "SMAA/SearchTex.h"
2012-11-26 18:58:24 +00:00
#define DEFAULT_SIZE 16
/*
==================
idImage::MakeDefault
the default image will be grey with a white box outline
to allow you to see the mapping coordinates on a surface
==================
*/
void idImage::MakeDefault()
{
2012-11-26 18:58:24 +00:00
int x, y;
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
if( com_developer.GetBool() )
{
2012-11-26 18:58:24 +00:00
// grey center
for( y = 0 ; y < DEFAULT_SIZE ; y++ )
{
for( x = 0 ; x < DEFAULT_SIZE ; x++ )
{
2012-11-26 18:58:24 +00:00
data[y][x][0] = 32;
data[y][x][1] = 32;
data[y][x][2] = 32;
data[y][x][3] = 255;
}
}
2012-11-26 18:58:24 +00:00
// white border
for( x = 0 ; x < DEFAULT_SIZE ; x++ )
{
2012-11-26 18:58:24 +00:00
data[0][x][0] =
data[0][x][1] =
data[0][x][2] =
data[0][x][3] = 255;
2012-11-26 18:58:24 +00:00
data[x][0][0] =
data[x][0][1] =
data[x][0][2] =
data[x][0][3] = 255;
data[DEFAULT_SIZE - 1][x][0] =
data[DEFAULT_SIZE - 1][x][1] =
data[DEFAULT_SIZE - 1][x][2] =
data[DEFAULT_SIZE - 1][x][3] = 255;
data[x][DEFAULT_SIZE - 1][0] =
data[x][DEFAULT_SIZE - 1][1] =
data[x][DEFAULT_SIZE - 1][2] =
data[x][DEFAULT_SIZE - 1][3] = 255;
2012-11-26 18:58:24 +00:00
}
}
else
{
for( y = 0 ; y < DEFAULT_SIZE ; y++ )
{
for( x = 0 ; x < DEFAULT_SIZE ; x++ )
{
2012-11-26 18:58:24 +00:00
data[y][x][0] = 0;
data[y][x][1] = 0;
data[y][x][2] = 0;
data[y][x][3] = 0;
}
}
}
GenerateImage( ( byte* )data,
DEFAULT_SIZE, DEFAULT_SIZE,
TF_DEFAULT, TR_REPEAT, TD_DEFAULT );
2012-11-26 18:58:24 +00:00
defaulted = true;
}
static void R_DefaultImage( idImage* image )
{
2012-11-26 18:58:24 +00:00
image->MakeDefault();
}
static void R_WhiteImage( idImage* image )
{
2012-11-26 18:58:24 +00:00
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
2012-11-26 18:58:24 +00:00
// solid white texture
memset( data, 255, sizeof( data ) );
image->GenerateImage( ( byte* )data, DEFAULT_SIZE, DEFAULT_SIZE,
TF_DEFAULT, TR_REPEAT, TD_DEFAULT );
2012-11-26 18:58:24 +00:00
}
static void R_BlackImage( idImage* image )
{
2012-11-26 18:58:24 +00:00
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
2012-11-26 18:58:24 +00:00
// solid black texture
memset( data, 0, sizeof( data ) );
image->GenerateImage( ( byte* )data, DEFAULT_SIZE, DEFAULT_SIZE,
TF_DEFAULT, TR_REPEAT, TD_DEFAULT );
2012-11-26 18:58:24 +00:00
}
static void R_RGBA8Image( idImage* image )
{
2012-11-26 18:58:24 +00:00
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
2012-11-26 18:58:24 +00:00
memset( data, 0, sizeof( data ) );
data[0][0][0] = 16;
data[0][0][1] = 32;
data[0][0][2] = 48;
data[0][0][3] = 96;
image->GenerateImage( ( byte* )data, DEFAULT_SIZE, DEFAULT_SIZE, TF_DEFAULT, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
2012-11-26 18:58:24 +00:00
}
2015-12-28 20:36:25 +00:00
static void R_RGBA8LinearImage( idImage* image )
{
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
memset( data, 0, sizeof( data ) );
data[0][0][0] = 16;
data[0][0][1] = 32;
data[0][0][2] = 48;
data[0][0][3] = 96;
image->GenerateImage( ( byte* )data, DEFAULT_SIZE, DEFAULT_SIZE, TF_LINEAR, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
static void R_DepthImage( idImage* image )
{
// RB: NULL data and MSAA support
#if defined(USE_HDR_MSAA)
int msaaSamples = glConfig.multisamples;
#else
int msaaSamples = 0;
#endif
image->GenerateImage( NULL, renderSystem->GetWidth(), renderSystem->GetHeight(), TF_NEAREST, TR_CLAMP, TD_DEPTH, msaaSamples );
// RB end
2012-11-26 18:58:24 +00:00
}
// RB begin
static void R_HDR_RGBA16FImage_ResNative( idImage* image )
{
#if defined(USE_HDR_MSAA)
int msaaSamples = glConfig.multisamples;
#else
int msaaSamples = 0;
#endif
image->GenerateImage( NULL, renderSystem->GetWidth(), renderSystem->GetHeight(), TF_NEAREST, TR_CLAMP, TD_RGBA16F, msaaSamples );
}
static void R_HDR_RGBA16FImage_ResNative_NoMSAA( idImage* image )
{
image->GenerateImage( NULL, renderSystem->GetWidth(), renderSystem->GetHeight(), TF_NEAREST, TR_CLAMP, TD_RGBA16F );
}
static void R_HDR_RGBA16FImage_ResQuarter( idImage* image )
{
image->GenerateImage( NULL, renderSystem->GetWidth() / 4, renderSystem->GetHeight() / 4, TF_NEAREST, TR_CLAMP, TD_RGBA16F );
}
static void R_HDR_RGBA16FImage_ResQuarter_Linear( idImage* image )
{
image->GenerateImage( NULL, renderSystem->GetWidth() / 4, renderSystem->GetHeight() / 4, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_RGBA );
}
static void R_HDR_RGBA16FImage_Res64( idImage* image )
{
image->GenerateImage( NULL, 64, 64, TF_NEAREST, TR_CLAMP, TD_RGBA16F );
}
2015-12-26 15:11:47 +00:00
static void R_SMAAImage_ResNative( idImage* image )
{
image->GenerateImage( NULL, renderSystem->GetWidth(), renderSystem->GetHeight(), TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_RGBA );
2015-12-26 15:11:47 +00:00
}
static void R_HierarchicalZBufferImage_ResNative( idImage* image )
{
image->GenerateImage( NULL, renderSystem->GetWidth(), renderSystem->GetHeight(), TF_NEAREST_MIPMAP, TR_CLAMP, TD_R32F );
}
// RB end
static void R_AlphaNotchImage( idImage* image )
{
2012-11-26 18:58:24 +00:00
byte data[2][4];
2012-11-26 18:58:24 +00:00
// this is used for alpha test clip planes
2012-11-26 18:58:24 +00:00
data[0][0] = data[0][1] = data[0][2] = 255;
data[0][3] = 0;
data[1][0] = data[1][1] = data[1][2] = 255;
data[1][3] = 255;
image->GenerateImage( ( byte* )data, 2, 1, TF_NEAREST, TR_CLAMP, TD_LOOKUP_TABLE_ALPHA );
2012-11-26 18:58:24 +00:00
}
static void R_FlatNormalImage( idImage* image )
{
2012-11-26 18:58:24 +00:00
byte data[DEFAULT_SIZE][DEFAULT_SIZE][4];
2012-11-26 18:58:24 +00:00
// flat normal map for default bunp mapping
for( int i = 0 ; i < 4 ; i++ )
{
2012-11-26 18:58:24 +00:00
data[0][i][0] = 128;
data[0][i][1] = 128;
data[0][i][2] = 255;
data[0][i][3] = 255;
}
image->GenerateImage( ( byte* )data, 2, 2, TF_DEFAULT, TR_REPEAT, TD_BUMP );
2012-11-26 18:58:24 +00:00
}
/*
================
R_CreateNoFalloffImage
This is a solid white texture that is zero clamped.
================
*/
static void R_CreateNoFalloffImage( idImage* image )
{
int x, y;
2012-11-26 18:58:24 +00:00
byte data[16][FALLOFF_TEXTURE_SIZE][4];
2012-11-26 18:58:24 +00:00
memset( data, 0, sizeof( data ) );
for( x = 1 ; x < FALLOFF_TEXTURE_SIZE - 1 ; x++ )
{
for( y = 1 ; y < 15 ; y++ )
{
2012-11-26 18:58:24 +00:00
data[y][x][0] = 255;
data[y][x][1] = 255;
data[y][x][2] = 255;
data[y][x][3] = 255;
}
}
image->GenerateImage( ( byte* )data, FALLOFF_TEXTURE_SIZE, 16, TF_DEFAULT, TR_CLAMP_TO_ZERO, TD_LOOKUP_TABLE_MONO );
2012-11-26 18:58:24 +00:00
}
/*
================
R_FogImage
We calculate distance correctly in two planes, but the
third will still be projection based
================
*/
const int FOG_SIZE = 128;
void R_FogImage( idImage* image )
{
int x, y;
2012-11-26 18:58:24 +00:00
byte data[FOG_SIZE][FOG_SIZE][4];
int b;
2012-11-26 18:58:24 +00:00
float step[256];
int i;
float remaining = 1.0;
for( i = 0 ; i < 256 ; i++ )
{
2012-11-26 18:58:24 +00:00
step[i] = remaining;
remaining *= 0.982f;
}
for( x = 0 ; x < FOG_SIZE ; x++ )
{
for( y = 0 ; y < FOG_SIZE ; y++ )
{
2012-11-26 18:58:24 +00:00
float d;
d = idMath::Sqrt( ( x - FOG_SIZE / 2 ) * ( x - FOG_SIZE / 2 )
+ ( y - FOG_SIZE / 2 ) * ( y - FOG_SIZE / 2 ) );
d /= FOG_SIZE / 2 - 1;
b = ( byte )( d * 255 );
if( b <= 0 )
{
2012-11-26 18:58:24 +00:00
b = 0;
}
else if( b > 255 )
{
2012-11-26 18:58:24 +00:00
b = 255;
}
b = ( byte )( 255 * ( 1.0 - step[b] ) );
if( x == 0 || x == FOG_SIZE - 1 || y == 0 || y == FOG_SIZE - 1 )
{
2012-11-26 18:58:24 +00:00
b = 255; // avoid clamping issues
}
data[y][x][0] =
data[y][x][1] =
data[y][x][2] = 255;
2012-11-26 18:58:24 +00:00
data[y][x][3] = b;
}
}
image->GenerateImage( ( byte* )data, FOG_SIZE, FOG_SIZE, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_ALPHA );
2012-11-26 18:58:24 +00:00
}
/*
================
FogFraction
Height values below zero are inside the fog volume
================
*/
static const float RAMP_RANGE = 8;
static const float DEEP_RANGE = -30;
static float FogFraction( float viewHeight, float targetHeight )
{
2012-11-26 18:58:24 +00:00
float total = idMath::Fabs( targetHeight - viewHeight );
2012-11-26 18:58:24 +00:00
// return targetHeight >= 0 ? 0 : 1.0;
// only ranges that cross the ramp range are special
if( targetHeight > 0 && viewHeight > 0 )
{
2012-11-26 18:58:24 +00:00
return 0.0;
}
if( targetHeight < -RAMP_RANGE && viewHeight < -RAMP_RANGE )
{
2012-11-26 18:58:24 +00:00
return 1.0;
}
2012-11-26 18:58:24 +00:00
float above;
if( targetHeight > 0 )
{
2012-11-26 18:58:24 +00:00
above = targetHeight;
}
else if( viewHeight > 0 )
{
2012-11-26 18:58:24 +00:00
above = viewHeight;
}
else
{
2012-11-26 18:58:24 +00:00
above = 0;
}
2012-11-26 18:58:24 +00:00
float rampTop, rampBottom;
if( viewHeight > targetHeight )
{
2012-11-26 18:58:24 +00:00
rampTop = viewHeight;
rampBottom = targetHeight;
}
else
{
2012-11-26 18:58:24 +00:00
rampTop = targetHeight;
rampBottom = viewHeight;
}
if( rampTop > 0 )
{
2012-11-26 18:58:24 +00:00
rampTop = 0;
}
if( rampBottom < -RAMP_RANGE )
{
2012-11-26 18:58:24 +00:00
rampBottom = -RAMP_RANGE;
}
2012-11-26 18:58:24 +00:00
float rampSlope = 1.0 / RAMP_RANGE;
if( !total )
{
2012-11-26 18:58:24 +00:00
return -viewHeight * rampSlope;
}
2012-11-26 18:58:24 +00:00
float ramp = ( 1.0 - ( rampTop * rampSlope + rampBottom * rampSlope ) * -0.5 ) * ( rampTop - rampBottom );
2012-11-26 18:58:24 +00:00
float frac = ( total - above - ramp ) / total;
2012-11-26 18:58:24 +00:00
// after it gets moderately deep, always use full value
float deepest = viewHeight < targetHeight ? viewHeight : targetHeight;
2012-11-26 18:58:24 +00:00
float deepFrac = deepest / DEEP_RANGE;
if( deepFrac >= 1.0 )
{
2012-11-26 18:58:24 +00:00
return 1.0;
}
2012-11-26 18:58:24 +00:00
frac = frac * ( 1.0 - deepFrac ) + deepFrac;
2012-11-26 18:58:24 +00:00
return frac;
}
/*
================
R_FogEnterImage
Modulate the fog alpha density based on the distance of the
start and end points to the terminator plane
================
*/
void R_FogEnterImage( idImage* image )
{
int x, y;
2012-11-26 18:58:24 +00:00
byte data[FOG_ENTER_SIZE][FOG_ENTER_SIZE][4];
int b;
for( x = 0 ; x < FOG_ENTER_SIZE ; x++ )
{
for( y = 0 ; y < FOG_ENTER_SIZE ; y++ )
{
2012-11-26 18:58:24 +00:00
float d;
d = FogFraction( x - ( FOG_ENTER_SIZE / 2 ), y - ( FOG_ENTER_SIZE / 2 ) );
b = ( byte )( d * 255 );
if( b <= 0 )
{
2012-11-26 18:58:24 +00:00
b = 0;
}
else if( b > 255 )
{
2012-11-26 18:58:24 +00:00
b = 255;
}
data[y][x][0] =
data[y][x][1] =
data[y][x][2] = 255;
2012-11-26 18:58:24 +00:00
data[y][x][3] = b;
}
}
2012-11-26 18:58:24 +00:00
// if mipmapped, acutely viewed surfaces fade wrong
image->GenerateImage( ( byte* )data, FOG_ENTER_SIZE, FOG_ENTER_SIZE, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_ALPHA );
2012-11-26 18:58:24 +00:00
}
/*
================
R_QuadraticImage
================
*/
static const int QUADRATIC_WIDTH = 32;
static const int QUADRATIC_HEIGHT = 4;
void R_QuadraticImage( idImage* image )
{
int x, y;
2012-11-26 18:58:24 +00:00
byte data[QUADRATIC_HEIGHT][QUADRATIC_WIDTH][4];
int b;
for( x = 0 ; x < QUADRATIC_WIDTH ; x++ )
{
for( y = 0 ; y < QUADRATIC_HEIGHT ; y++ )
{
2012-11-26 18:58:24 +00:00
float d;
d = x - ( QUADRATIC_WIDTH / 2 - 0.5 );
2012-11-26 18:58:24 +00:00
d = idMath::Fabs( d );
d -= 0.5;
d /= QUADRATIC_WIDTH / 2;
2012-11-26 18:58:24 +00:00
d = 1.0 - d;
d = d * d;
b = ( byte )( d * 255 );
if( b <= 0 )
{
2012-11-26 18:58:24 +00:00
b = 0;
}
else if( b > 255 )
{
2012-11-26 18:58:24 +00:00
b = 255;
}
data[y][x][0] =
data[y][x][1] =
data[y][x][2] = b;
2012-11-26 18:58:24 +00:00
data[y][x][3] = 255;
}
}
image->GenerateImage( ( byte* )data, QUADRATIC_WIDTH, QUADRATIC_HEIGHT, TF_DEFAULT, TR_CLAMP, TD_LOOKUP_TABLE_RGB1 );
2012-11-26 18:58:24 +00:00
}
// RB begin
static void R_CreateShadowMapImage_Res0( idImage* image )
{
int size = shadowMapResolutions[0];
image->GenerateShadowArray( size, size, TF_LINEAR, TR_CLAMP_TO_ZERO_ALPHA, TD_SHADOW_ARRAY );
}
static void R_CreateShadowMapImage_Res1( idImage* image )
{
int size = shadowMapResolutions[1];
image->GenerateShadowArray( size, size, TF_LINEAR, TR_CLAMP_TO_ZERO_ALPHA, TD_SHADOW_ARRAY );
}
static void R_CreateShadowMapImage_Res2( idImage* image )
{
int size = shadowMapResolutions[2];
image->GenerateShadowArray( size, size, TF_LINEAR, TR_CLAMP_TO_ZERO_ALPHA, TD_SHADOW_ARRAY );
}
static void R_CreateShadowMapImage_Res3( idImage* image )
{
int size = shadowMapResolutions[3];
image->GenerateShadowArray( size, size, TF_LINEAR, TR_CLAMP_TO_ZERO_ALPHA, TD_SHADOW_ARRAY );
}
static void R_CreateShadowMapImage_Res4( idImage* image )
{
int size = shadowMapResolutions[4];
image->GenerateShadowArray( size, size, TF_LINEAR, TR_CLAMP_TO_ZERO_ALPHA, TD_SHADOW_ARRAY );
}
const static int JITTER_SIZE = 128;
static void R_CreateJitterImage16( idImage* image )
{
static byte data[JITTER_SIZE][JITTER_SIZE * 16][4];
for( int i = 0 ; i < JITTER_SIZE ; i++ )
{
for( int s = 0 ; s < 16 ; s++ )
{
int sOfs = 64 * ( s & 3 );
int tOfs = 64 * ( ( s >> 2 ) & 3 );
for( int j = 0 ; j < JITTER_SIZE ; j++ )
{
data[i][s * JITTER_SIZE + j][0] = ( rand() & 63 ) | sOfs;
data[i][s * JITTER_SIZE + j][1] = ( rand() & 63 ) | tOfs;
data[i][s * JITTER_SIZE + j][2] = rand();
data[i][s * JITTER_SIZE + j][3] = 0;
}
}
}
image->GenerateImage( ( byte* )data, JITTER_SIZE * 16, JITTER_SIZE, TF_NEAREST, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
static void R_CreateJitterImage4( idImage* image )
{
byte data[JITTER_SIZE][JITTER_SIZE * 4][4];
for( int i = 0 ; i < JITTER_SIZE ; i++ )
{
for( int s = 0 ; s < 4 ; s++ )
{
int sOfs = 128 * ( s & 1 );
int tOfs = 128 * ( ( s >> 1 ) & 1 );
for( int j = 0 ; j < JITTER_SIZE ; j++ )
{
data[i][s * JITTER_SIZE + j][0] = ( rand() & 127 ) | sOfs;
data[i][s * JITTER_SIZE + j][1] = ( rand() & 127 ) | tOfs;
data[i][s * JITTER_SIZE + j][2] = rand();
data[i][s * JITTER_SIZE + j][3] = 0;
}
}
}
image->GenerateImage( ( byte* )data, JITTER_SIZE * 4, JITTER_SIZE, TF_NEAREST, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
static void R_CreateJitterImage1( idImage* image )
{
byte data[JITTER_SIZE][JITTER_SIZE][4];
for( int i = 0 ; i < JITTER_SIZE ; i++ )
{
for( int j = 0 ; j < JITTER_SIZE ; j++ )
{
data[i][j][0] = rand();
data[i][j][1] = rand();
data[i][j][2] = rand();
data[i][j][3] = 0;
}
}
image->GenerateImage( ( byte* )data, JITTER_SIZE, JITTER_SIZE, TF_NEAREST, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
static void R_CreateRandom256Image( idImage* image )
{
byte data[256][256][4];
for( int i = 0 ; i < 256 ; i++ )
{
for( int j = 0 ; j < 256 ; j++ )
{
data[i][j][0] = rand();
data[i][j][1] = rand();
data[i][j][2] = rand();
data[i][j][3] = rand();
}
}
image->GenerateImage( ( byte* )data, 256, 256, TF_NEAREST, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
2015-12-23 12:53:21 +00:00
static void R_CreateHeatmap5ColorsImage( idImage* image )
{
int x, y;
byte data[16][FALLOFF_TEXTURE_SIZE][4];
const int numColors = 5;
static idVec4 colors[numColors] = { colorBlue, colorCyan, colorGreen, colorYellow, colorRed };
memset( data, 0, sizeof( data ) );
for( x = 0 ; x < FALLOFF_TEXTURE_SIZE; x++ )
{
int index1, index2;
float value = x / ( float )FALLOFF_TEXTURE_SIZE;
float lerp = 0.0f;
if( value <= 0.0 )
{
index1 = index2 = 0;
}
else if( value >= 1.0f )
{
index1 = index2 = numColors - 1;
}
else
{
value = value * ( numColors - 1 );
index1 = idMath::Floor( value );
index2 = index1 + 1;
lerp = value - float( index1 );
}
idVec4 color( 0, 0, 0, 1 );
color.x = ( colors[index2].x - colors[index1].x ) * lerp + colors[index1].x;
color.y = ( colors[index2].y - colors[index1].y ) * lerp + colors[index1].y;
color.z = ( colors[index2].z - colors[index1].z ) * lerp + colors[index1].z;
for( y = 0 ; y < 16 ; y++ )
{
data[y][x][0] = color.x * 255;
data[y][x][1] = color.y * 255;
data[y][x][2] = color.z * 255;
data[y][x][3] = 255;
}
}
image->GenerateImage( ( byte* )data, FALLOFF_TEXTURE_SIZE, 16, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_RGBA );
}
static void R_CreateHeatmap7ColorsImage( idImage* image )
{
int x, y;
byte data[16][FALLOFF_TEXTURE_SIZE][4];
const int numColors = 7;
static idVec4 colors[numColors] = { colorBlack, colorBlue, colorCyan, colorGreen, colorYellow, colorRed, colorWhite };
memset( data, 0, sizeof( data ) );
for( x = 0 ; x < FALLOFF_TEXTURE_SIZE; x++ )
{
int index1, index2;
float value = x / ( float )FALLOFF_TEXTURE_SIZE;
float lerp = 0.0f;
if( value <= 0.0 )
{
index1 = index2 = 0;
}
else if( value >= 1.0f )
{
index1 = index2 = numColors - 1;
}
else
{
value = value * ( numColors - 1 );
index1 = idMath::Floor( value );
index2 = index1 + 1;
lerp = value - float( index1 );
}
idVec4 color( 0, 0, 0, 1 );
color.x = ( colors[index2].x - colors[index1].x ) * lerp + colors[index1].x;
color.y = ( colors[index2].y - colors[index1].y ) * lerp + colors[index1].y;
color.z = ( colors[index2].z - colors[index1].z ) * lerp + colors[index1].z;
for( y = 0 ; y < 16 ; y++ )
{
data[y][x][0] = color.x * 255;
data[y][x][1] = color.y * 255;
data[y][x][2] = color.z * 255;
data[y][x][3] = 255;
}
}
image->GenerateImage( ( byte* )data, FALLOFF_TEXTURE_SIZE, 16, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_RGBA );
}
2015-12-23 12:53:21 +00:00
static void R_CreateGrainImage1( idImage* image )
{
const static int GRAIN_SIZE = 128;
static byte data[GRAIN_SIZE][GRAIN_SIZE][4];
idRandom2 random( Sys_Milliseconds() );
for( int i = 0 ; i < GRAIN_SIZE ; i++ )
{
for( int j = 0 ; j < GRAIN_SIZE ; j++ )
{
#if 0
//int value = 127 - 8 + ( rand() & 15 ); //random.RandomInt( 127 );
int value = 127 - 8 + random.RandomInt( 15 );
data[i][j][0] = value;
data[i][j][1] = value;
data[i][j][2] = value;
data[i][j][3] = 0;
#else
data[i][j][0] = 127 - 8 + random.RandomInt( 15 );
data[i][j][1] = 127 - 8 + random.RandomInt( 15 );
data[i][j][2] = 127 - 8 + random.RandomInt( 15 );
data[i][j][3] = 0;
#endif
}
}
image->GenerateImage( ( byte* )data, GRAIN_SIZE, GRAIN_SIZE, TF_NEAREST, TR_REPEAT, TD_LOOKUP_TABLE_RGBA );
}
2015-12-26 14:20:19 +00:00
static void R_CreateSMAAAreaImage( idImage* image )
{
static byte data[AREATEX_HEIGHT][AREATEX_WIDTH][4];
idRandom2 random( Sys_Milliseconds() );
for( int x = 0; x < AREATEX_WIDTH; x++ )
{
for( int y = 0; y < AREATEX_HEIGHT; y++ )
{
2015-12-28 20:36:25 +00:00
#if 0
data[AREATEX_HEIGHT - y][x][0] = areaTexBytes[ y * AREATEX_PITCH + x * 2 + 0 ];
data[AREATEX_HEIGHT - y][x][1] = areaTexBytes[ y * AREATEX_PITCH + x * 2 + 1 ];
data[AREATEX_HEIGHT - y][x][2] = 0;
data[AREATEX_HEIGHT - y][x][3] = 1;
#else
2015-12-26 14:20:19 +00:00
data[y][x][0] = areaTexBytes[ y * AREATEX_PITCH + x * 2 + 0 ];
data[y][x][1] = areaTexBytes[ y * AREATEX_PITCH + x * 2 + 1 ];
data[y][x][2] = 0;
data[y][x][3] = 1;
2015-12-28 20:36:25 +00:00
#endif
2015-12-26 14:20:19 +00:00
}
}
2015-12-26 15:11:47 +00:00
image->GenerateImage( ( byte* )data, AREATEX_WIDTH, AREATEX_HEIGHT, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_RGBA );
2015-12-26 14:20:19 +00:00
}
static void R_CreateSMAASearchImage( idImage* image )
{
static byte data[SEARCHTEX_HEIGHT][SEARCHTEX_WIDTH][4];
idRandom2 random( Sys_Milliseconds() );
for( int x = 0; x < SEARCHTEX_WIDTH; x++ )
{
for( int y = 0; y < SEARCHTEX_HEIGHT; y++ )
{
2015-12-28 20:36:25 +00:00
#if 0
data[SEARCHTEX_HEIGHT - y][x][0] = searchTexBytes[ y * SEARCHTEX_PITCH + x ];
data[SEARCHTEX_HEIGHT - y][x][1] = 0;
data[SEARCHTEX_HEIGHT - y][x][2] = 0;
data[SEARCHTEX_HEIGHT - y][x][3] = 1;
#else
2015-12-26 14:20:19 +00:00
data[y][x][0] = searchTexBytes[ y * SEARCHTEX_PITCH + x ];
data[y][x][1] = 0;
data[y][x][2] = 0;
data[y][x][3] = 1;
2015-12-28 20:36:25 +00:00
#endif
2015-12-26 14:20:19 +00:00
}
}
image->GenerateImage( ( byte* )data, SEARCHTEX_WIDTH, SEARCHTEX_HEIGHT, TF_LINEAR, TR_CLAMP, TD_LOOKUP_TABLE_MONO );
2015-12-26 14:20:19 +00:00
}
// RB end
2012-11-26 18:58:24 +00:00
/*
================
idImageManager::CreateIntrinsicImages
================
*/
void idImageManager::CreateIntrinsicImages()
{
2012-11-26 18:58:24 +00:00
// create built in images
defaultImage = ImageFromFunction( "_default", R_DefaultImage );
whiteImage = ImageFromFunction( "_white", R_WhiteImage );
blackImage = ImageFromFunction( "_black", R_BlackImage );
flatNormalMap = ImageFromFunction( "_flat", R_FlatNormalImage );
alphaNotchImage = ImageFromFunction( "_alphaNotch", R_AlphaNotchImage );
fogImage = ImageFromFunction( "_fog", R_FogImage );
fogEnterImage = ImageFromFunction( "_fogEnter", R_FogEnterImage );
noFalloffImage = ImageFromFunction( "_noFalloff", R_CreateNoFalloffImage );
ImageFromFunction( "_quadratic", R_QuadraticImage );
// RB begin
shadowImage[0] = ImageFromFunction( va( "_shadowMapArray0_%i", shadowMapResolutions[0] ), R_CreateShadowMapImage_Res0 );
shadowImage[1] = ImageFromFunction( va( "_shadowMapArray1_%i", shadowMapResolutions[1] ), R_CreateShadowMapImage_Res1 );
shadowImage[2] = ImageFromFunction( va( "_shadowMapArray2_%i", shadowMapResolutions[2] ), R_CreateShadowMapImage_Res2 );
shadowImage[3] = ImageFromFunction( va( "_shadowMapArray3_%i", shadowMapResolutions[3] ), R_CreateShadowMapImage_Res3 );
shadowImage[4] = ImageFromFunction( va( "_shadowMapArray4_%i", shadowMapResolutions[4] ), R_CreateShadowMapImage_Res4 );
jitterImage1 = globalImages->ImageFromFunction( "_jitter1", R_CreateJitterImage1 );
jitterImage4 = globalImages->ImageFromFunction( "_jitter4", R_CreateJitterImage4 );
jitterImage16 = globalImages->ImageFromFunction( "_jitter16", R_CreateJitterImage16 );
randomImage256 = globalImages->ImageFromFunction( "_random256", R_CreateRandom256Image );
currentRenderHDRImage = globalImages->ImageFromFunction( "_currentRenderHDR", R_HDR_RGBA16FImage_ResNative );
#if defined(USE_HDR_MSAA)
currentRenderHDRImageNoMSAA = globalImages->ImageFromFunction( "_currentRenderHDRNoMSAA", R_HDR_RGBA16FImage_ResNative_NoMSAA );
#endif
currentRenderHDRImageQuarter = globalImages->ImageFromFunction( "_currentRenderHDRQuarter", R_HDR_RGBA16FImage_ResQuarter );
currentRenderHDRImage64 = globalImages->ImageFromFunction( "_currentRenderHDR64", R_HDR_RGBA16FImage_Res64 );
2016-01-06 21:37:16 +00:00
bloomRenderImage[0] = globalImages->ImageFromFunction( "_bloomRender0", R_HDR_RGBA16FImage_ResQuarter_Linear );
bloomRenderImage[1] = globalImages->ImageFromFunction( "_bloomRender1", R_HDR_RGBA16FImage_ResQuarter_Linear );
heatmap5Image = ImageFromFunction( "_heatmap5", R_CreateHeatmap5ColorsImage );
heatmap7Image = ImageFromFunction( "_heatmap7", R_CreateHeatmap7ColorsImage );
2015-12-23 12:53:21 +00:00
grainImage1 = globalImages->ImageFromFunction( "_grain1", R_CreateGrainImage1 );
2015-12-26 15:11:47 +00:00
2015-12-28 20:36:25 +00:00
smaaInputImage = ImageFromFunction( "_smaaInput", R_RGBA8LinearImage );
2015-12-26 14:20:19 +00:00
smaaAreaImage = globalImages->ImageFromFunction( "_smaaArea", R_CreateSMAAAreaImage );
smaaSearchImage = globalImages->ImageFromFunction( "_smaaSearch", R_CreateSMAASearchImage );
2015-12-26 15:11:47 +00:00
smaaEdgesImage = globalImages->ImageFromFunction( "_smaaEdges", R_SMAAImage_ResNative );
smaaBlendImage = globalImages->ImageFromFunction( "_smaaBlend", R_SMAAImage_ResNative );
2016-01-18 22:21:53 +00:00
currentNormalsImage = ImageFromFunction( "_currentNormals", R_SMAAImage_ResNative );
2016-01-06 21:37:16 +00:00
ambientOcclusionImage[0] = ImageFromFunction( "_ao0", R_SMAAImage_ResNative );
ambientOcclusionImage[1] = ImageFromFunction( "_ao1", R_SMAAImage_ResNative );
hierarchicalZbufferImage = ImageFromFunction( "_cszBuffer", R_HierarchicalZBufferImage_ResNative );
// RB end
2012-11-26 18:58:24 +00:00
// scratchImage is used for screen wipes/doublevision etc..
scratchImage = ImageFromFunction( "_scratch", R_RGBA8Image );
scratchImage2 = ImageFromFunction( "_scratch2", R_RGBA8Image );
accumImage = ImageFromFunction( "_accum", R_RGBA8Image );
currentRenderImage = ImageFromFunction( "_currentRender", R_RGBA8Image );
currentDepthImage = ImageFromFunction( "_currentDepth", R_DepthImage );
2012-11-26 18:58:24 +00:00
// save a copy of this for material comparison, because currentRenderImage may get
// reassigned during stereo rendering
originalCurrentRenderImage = currentRenderImage;
loadingIconImage = ImageFromFile( "textures/loadingicon2", TF_DEFAULT, TR_CLAMP, TD_DEFAULT, CF_2D );
hellLoadingIconImage = ImageFromFile( "textures/loadingicon3", TF_DEFAULT, TR_CLAMP, TD_DEFAULT, CF_2D );
2012-11-26 18:58:24 +00:00
release_assert( loadingIconImage->referencedOutsideLevelLoad );
release_assert( hellLoadingIconImage->referencedOutsideLevelLoad );
}