2008-09-15 01:07:45 -05:00
//========== Copyright <20> <> 2005, Valve Corporation, All rights reserved. ==========
//
// Purpose:
//
//==============================================================================
# include "cbase.h"
# include "materialsystem/imaterialsystem.h"
# include "materialsystem/itexture.h"
# include "materialsystem/imaterialvar.h"
# include "materialsystem/imaterialsystemhardwareconfig.h"
# include "materialsystem/materialsystem_config.h"
# include "tier1/callqueue.h"
# include "colorcorrectionmgr.h"
# include "view_scene.h"
# include "c_world.h"
# include "ProxyEntity.h"
//-----------------------------------------------------------------------------
// Globals
//-----------------------------------------------------------------------------
// mapmaker controlled autoexposure
bool g_bUseCustomAutoExposureMin = false ;
bool g_bUseCustomAutoExposureMax = false ;
bool g_bUseCustomBloomScale = false ;
float g_flCustomAutoExposureMin = 0 ;
float g_flCustomAutoExposureMax = 0 ;
float g_flCustomBloomScale = 0.0f ;
float g_flCustomBloomScaleMinimum = 0.0f ;
bool g_bFlashlightIsOn = false ;
// hdr parameters
ConVar mat_bloomscale ( " mat_bloomscale " , " 1 " ) ;
ConVar mat_hdr_level ( " mat_hdr_level " , " 2 " ) ;
ConVar mat_bloomamount_rate ( " mat_bloomamount_rate " , " 0.05f " , FCVAR_CHEAT ) ;
static ConVar debug_postproc ( " mat_debug_postprocessing_effects " , " 0 " , FCVAR_NONE , " 0 = off, 1 = show post-processing passes in quadrants of the screen, 2 = only apply post-processing to the centre of the screen " ) ;
static ConVar split_postproc ( " mat_debug_process_halfscreen " , " 0 " , FCVAR_CHEAT ) ;
static ConVar mat_postprocessing_combine ( " mat_postprocessing_combine " , " 1 " , FCVAR_NONE , " Combine bloom, software anti-aliasing and color correction into one post-processing pass " ) ;
static ConVar mat_dynamic_tonemapping ( " mat_dynamic_tonemapping " , " 1 " , FCVAR_CHEAT ) ;
static ConVar mat_show_ab_hdr ( " mat_show_ab_hdr " , " 0 " ) ;
static ConVar mat_tonemapping_occlusion_use_stencil ( " mat_tonemapping_occlusion_use_stencil " , " 0 " ) ;
ConVar mat_debug_autoexposure ( " mat_debug_autoexposure " , " 0 " , FCVAR_CHEAT ) ;
static ConVar mat_autoexposure_max ( " mat_autoexposure_max " , " 2 " ) ;
static ConVar mat_autoexposure_min ( " mat_autoexposure_min " , " 0.5 " ) ;
static ConVar mat_show_histogram ( " mat_show_histogram " , " 0 " ) ;
ConVar mat_hdr_tonemapscale ( " mat_hdr_tonemapscale " , " 1.0 " , FCVAR_CHEAT ) ;
ConVar mat_hdr_uncapexposure ( " mat_hdr_uncapexposure " , " 0 " , FCVAR_CHEAT ) ;
ConVar mat_force_bloom ( " mat_force_bloom " , " 0 " , FCVAR_CHEAT ) ;
ConVar mat_disable_bloom ( " mat_disable_bloom " , " 0 " ) ;
ConVar mat_debug_bloom ( " mat_debug_bloom " , " 0 " , FCVAR_CHEAT ) ;
ConVar mat_colorcorrection ( " mat_colorcorrection " , " 0 " ) ;
ConVar mat_accelerate_adjust_exposure_down ( " mat_accelerate_adjust_exposure_down " , " 3.0 " , FCVAR_CHEAT ) ;
ConVar mat_hdr_manual_tonemap_rate ( " mat_hdr_manual_tonemap_rate " , " 1.0 " ) ;
// fudge factor to make non-hdr bloom more closely match hdr bloom. Because of auto-exposure, high
// bloomscales don't blow out as much in hdr. this factor was derived by comparing images in a
// reference scene.
ConVar mat_non_hdr_bloom_scalefactor ( " mat_non_hdr_bloom_scalefactor " , " .3 " ) ;
// Apply addition scale to the final bloom scale
static ConVar mat_bloom_scalefactor_scalar ( " mat_bloom_scalefactor_scalar " , " 1.0 " ) ;
//ConVar mat_exposure_center_region_x( "mat_exposure_center_region_x","0.75", FCVAR_CHEAT );
//ConVar mat_exposure_center_region_y( "mat_exposure_center_region_y","0.80", FCVAR_CHEAT );
//ConVar mat_exposure_center_region_x_flashlight( "mat_exposure_center_region_x_flashlight","0.33", FCVAR_CHEAT );
//ConVar mat_exposure_center_region_y_flashlight( "mat_exposure_center_region_y_flashlight","0.33", FCVAR_CHEAT );
ConVar mat_exposure_center_region_x ( " mat_exposure_center_region_x " , " 0.9 " , FCVAR_CHEAT ) ;
ConVar mat_exposure_center_region_y ( " mat_exposure_center_region_y " , " 0.85 " , FCVAR_CHEAT ) ;
ConVar mat_exposure_center_region_x_flashlight ( " mat_exposure_center_region_x_flashlight " , " 0.9 " , FCVAR_CHEAT ) ;
ConVar mat_exposure_center_region_y_flashlight ( " mat_exposure_center_region_y_flashlight " , " 0.85 " , FCVAR_CHEAT ) ;
ConVar mat_tonemap_algorithm ( " mat_tonemap_algorithm " , " 1 " , FCVAR_CHEAT , " 0 = Original Algorithm 1 = New Algorithm " ) ;
ConVar mat_tonemap_percent_target ( " mat_tonemap_percent_target " , " 60.0 " , FCVAR_CHEAT ) ;
ConVar mat_tonemap_percent_bright_pixels ( " mat_tonemap_percent_bright_pixels " , " 2.0 " , FCVAR_CHEAT ) ;
ConVar mat_tonemap_min_avglum ( " mat_tonemap_min_avglum " , " 3.0 " , FCVAR_CHEAT ) ;
ConVar mat_fullbright ( " mat_fullbright " , " 0 " , FCVAR_CHEAT ) ;
enum PostProcessingCondition {
PPP_ALWAYS ,
PPP_IF_COND_VAR ,
PPP_IF_NOT_COND_VAR
} ;
struct PostProcessingPass {
PostProcessingCondition ppp_test ;
ConVar const * cvar_to_test ;
char const * material_name ; // terminate list with null
char const * dest_rendering_target ;
char const * src_rendering_target ; // can be null. needed for source scaling
int xdest_scale , ydest_scale ; // allows scaling down
int xsrc_scale , ysrc_scale ; // allows scaling down
CMaterialReference m_mat_ref ; // so we don't have to keep searching
} ;
# define PPP_PROCESS_PARTIAL_SRC(srcmatname,dest_rt_name,src_tname,scale) \
{ PPP_ALWAYS , 0 , srcmatname , dest_rt_name , src_tname , 1 , 1 , scale , scale }
# define PPP_PROCESS_PARTIAL_DEST(srcmatname,dest_rt_name,src_tname,scale) \
{ PPP_ALWAYS , 0 , srcmatname , dest_rt_name , src_tname , scale , scale , 1 , 1 }
# define PPP_PROCESS_PARTIAL_SRC_PARTIAL_DEST(srcmatname,dest_rt_name,src_tname,srcscale,destscale) \
{ PPP_ALWAYS , 0 , srcmatname , dest_rt_name , src_tname , destscale , destscale , srcscale , srcscale }
# define PPP_END {PPP_ALWAYS,0,NULL,NULL,0,0,0,0,0}
# define PPP_PROCESS(srcmatname,dest_rt_name) {PPP_ALWAYS,0,srcmatname,dest_rt_name,0,1,1,1,1}
# define PPP_PROCESS_IF_CVAR(cvarptr,srcmatname,dest_rt_name) \
{ PPP_IF_COND_VAR , cvarptr , srcmatname , dest_rt_name , 0 , 1 , 1 , 1 , 1 }
# define PPP_PROCESS_IF_NOT_CVAR(cvarptr,srcmatname,dest_rt_name) \
{ PPP_IF_NOT_COND_VAR , cvarptr , srcmatname , dest_rt_name , 0 , 1 , 1 , 1 , 1 }
# define PPP_PROCESS_IF_NOT_CVAR_SRCTEXTURE(cvarptr,srcmatname,src_tname,dest_rt_name) \
{ PPP_IF_NOT_COND_VAR , cvarptr , srcmatname , dest_rt_name , src_tname , 1 , 1 , 1 , 1 }
# define PPP_PROCESS_IF_CVAR_SRCTEXTURE(cvarptr,srcmatname,src_txtrname,dest_rt_name) \
{ PPP_IF_COND_VAR , cvarptr , srcmatname , dest_rt_name , src_txtrname , 1 , 1 , 1 , 1 }
# define PPP_PROCESS_SRCTEXTURE(srcmatname,src_tname,dest_rt_name) \
{ PPP_ALWAYS , 0 , srcmatname , dest_rt_name , src_tname , 1 , 1 , 1 , 1 }
struct ClipBox
{
int m_minx , m_miny ;
int m_maxx , m_maxy ;
} ;
static void DrawClippedScreenSpaceRectangle (
IMaterial * pMaterial ,
int destx , int desty ,
int width , int height ,
float src_texture_x0 , float src_texture_y0 , // which texel you want to appear at
// destx/y
float src_texture_x1 , float src_texture_y1 , // which texel you want to appear at
// destx+width-1, desty+height-1
int src_texture_width , int src_texture_height , // needed for fixup
ClipBox const * clipbox ,
void * pClientRenderable = NULL )
{
if ( clipbox )
{
if ( ( destx > clipbox - > m_maxx ) | | ( desty > clipbox - > m_maxy ) )
return ;
if ( ( destx + width - 1 < clipbox - > m_minx ) | | ( desty + height - 1 < clipbox - > m_miny ) )
return ;
// left clip
if ( destx < clipbox - > m_minx )
{
src_texture_x0 = FLerp ( src_texture_x0 , src_texture_x1 , destx , destx + width - 1 , clipbox - > m_minx ) ;
width - = ( clipbox - > m_minx - destx ) ;
destx = clipbox - > m_minx ;
}
// top clip
if ( desty < clipbox - > m_miny )
{
src_texture_y0 = FLerp ( src_texture_y0 , src_texture_y1 , desty , desty + height - 1 , clipbox - > m_miny ) ;
height - = ( clipbox - > m_miny - desty ) ;
desty = clipbox - > m_miny ;
}
// right clip
if ( destx + width - 1 > clipbox - > m_maxx )
{
src_texture_x1 = FLerp ( src_texture_x0 , src_texture_x1 , destx , destx + width - 1 , clipbox - > m_maxx ) ;
width = clipbox - > m_maxx - destx ;
}
// bottom clip
if ( desty + height - 1 > clipbox - > m_maxy )
{
src_texture_y1 = FLerp ( src_texture_y0 , src_texture_y1 , desty , desty + height - 1 , clipbox - > m_maxy ) ;
height = clipbox - > m_maxy - desty ;
}
}
CMatRenderContextPtr pRenderContext ( materials ) ;
pRenderContext - > DrawScreenSpaceRectangle ( pMaterial , destx , desty , width , height , src_texture_x0 ,
src_texture_y0 , src_texture_x1 , src_texture_y1 ,
src_texture_width , src_texture_height , pClientRenderable ) ;
}
void ApplyPostProcessingPasses ( PostProcessingPass * pass_list , // table of effects to apply
ClipBox const * clipbox = 0 , // clipping box for these effects
ClipBox * dest_coords_out = 0 ) // receives dest coords of last blit
{
CMatRenderContextPtr pRenderContext ( materials ) ;
ITexture * pSaveRenderTarget = pRenderContext - > GetRenderTarget ( ) ;
int pcount = 0 ;
if ( debug_postproc . GetInt ( ) = = 1 )
{
pRenderContext - > SetRenderTarget ( NULL ) ;
int dest_width , dest_height ;
pRenderContext - > GetRenderTargetDimensions ( dest_width , dest_height ) ;
pRenderContext - > Viewport ( 0 , 0 , dest_width , dest_height ) ;
pRenderContext - > ClearColor3ub ( 255 , 0 , 0 ) ;
// pRenderContext->ClearBuffers(true,true);
}
while ( pass_list - > material_name )
{
bool do_it = true ;
switch ( pass_list - > ppp_test )
{
case PPP_IF_COND_VAR :
do_it = ( pass_list - > cvar_to_test ) - > GetBool ( ) ;
break ;
case PPP_IF_NOT_COND_VAR :
do_it = ! ( ( pass_list - > cvar_to_test ) - > GetBool ( ) ) ;
break ;
}
if ( ( pass_list - > dest_rendering_target = = 0 ) & & ( debug_postproc . GetInt ( ) = = 1 ) )
do_it = 0 ;
if ( do_it )
{
ClipBox const * cb = 0 ;
if ( pass_list - > dest_rendering_target = = 0 )
{
cb = clipbox ;
}
IMaterial * src_mat = pass_list - > m_mat_ref ;
if ( ! src_mat )
{
src_mat = materials - > FindMaterial ( pass_list - > material_name ,
TEXTURE_GROUP_OTHER , true ) ;
if ( src_mat )
{
pass_list - > m_mat_ref . Init ( src_mat ) ;
}
}
if ( pass_list - > dest_rendering_target )
{
ITexture * dest_rt = materials - > FindTexture ( pass_list - > dest_rendering_target ,
TEXTURE_GROUP_RENDER_TARGET ) ;
pRenderContext - > SetRenderTarget ( dest_rt ) ;
}
else
{
pRenderContext - > SetRenderTarget ( NULL ) ;
}
int dest_width , dest_height ;
pRenderContext - > GetRenderTargetDimensions ( dest_width , dest_height ) ;
pRenderContext - > Viewport ( 0 , 0 , dest_width , dest_height ) ;
dest_width / = pass_list - > xdest_scale ;
dest_height / = pass_list - > ydest_scale ;
if ( pass_list - > src_rendering_target )
{
ITexture * src_rt = materials - > FindTexture ( pass_list - > src_rendering_target ,
TEXTURE_GROUP_RENDER_TARGET ) ;
int src_width = src_rt - > GetActualWidth ( ) ;
int src_height = src_rt - > GetActualHeight ( ) ;
int ssrc_width = ( src_width - 1 ) / pass_list - > xsrc_scale ;
int ssrc_height = ( src_height - 1 ) / pass_list - > ysrc_scale ;
DrawClippedScreenSpaceRectangle (
src_mat , 0 , 0 , dest_width , dest_height ,
0 , 0 , ssrc_width , ssrc_height , src_width , src_height , cb ) ;
if ( ( pass_list - > dest_rendering_target ) & & ( debug_postproc . GetInt ( ) = = 1 ) )
{
pRenderContext - > SetRenderTarget ( NULL ) ;
int row = pcount / 2 ;
int col = pcount % 2 ;
int vdest_width , vdest_height ;
pRenderContext - > GetRenderTargetDimensions ( vdest_width , vdest_height ) ;
pRenderContext - > Viewport ( 0 , 0 , vdest_width , vdest_height ) ;
pRenderContext - > DrawScreenSpaceRectangle (
src_mat , col * 400 , 200 + row * 300 , dest_width , dest_height ,
0 , 0 , ssrc_width , ssrc_height , src_width , src_height ) ;
}
}
else
{
// just draw the whole source
if ( ( pass_list - > dest_rendering_target = = 0 ) & & split_postproc . GetInt ( ) )
{
DrawClippedScreenSpaceRectangle ( src_mat , 0 , 0 , dest_width / 2 , dest_height ,
0 , 0 , .5 , 1 , 1 , 1 , cb ) ;
}
else
{
DrawClippedScreenSpaceRectangle ( src_mat , 0 , 0 , dest_width , dest_height ,
0 , 0 , 1 , 1 , 1 , 1 , cb ) ;
}
if ( ( pass_list - > dest_rendering_target ) & & ( debug_postproc . GetInt ( ) = = 1 ) )
{
pRenderContext - > SetRenderTarget ( NULL ) ;
int row = pcount / 4 ;
int col = pcount % 4 ;
int dest_width , dest_height ;
pRenderContext - > GetRenderTargetDimensions ( dest_width , dest_height ) ;
pRenderContext - > Viewport ( 0 , 0 , dest_width , dest_height ) ;
DrawClippedScreenSpaceRectangle ( src_mat , 10 + col * 220 , 10 + row * 220 ,
200 , 200 ,
0 , 0 , 1 , 1 , 1 , 1 , cb ) ;
}
}
if ( dest_coords_out )
{
dest_coords_out - > m_minx = 0 ;
dest_coords_out - > m_maxx = dest_width - 1 ;
dest_coords_out - > m_miny = 0 ;
dest_coords_out - > m_maxy = dest_height - 1 ;
}
}
pass_list + + ;
pcount + + ;
}
pRenderContext - > SetRenderTarget ( pSaveRenderTarget ) ;
}
PostProcessingPass HDRFinal_Float [ ] =
{
PPP_PROCESS_SRCTEXTURE ( " dev/downsample " , " _rt_FullFrameFB " , " _rt_SmallFB0 " ) ,
PPP_PROCESS_SRCTEXTURE ( " dev/blurfilterx " , " _rt_SmallFB0 " , " _rt_SmallFB1 " ) ,
PPP_PROCESS_SRCTEXTURE ( " dev/blurfiltery " , " _rt_SmallFB1 " , " _rt_SmallFB0 " ) ,
PPP_PROCESS_SRCTEXTURE ( " dev/floattoscreen_combine " , " _rt_FullFrameFB " , NULL ) ,
PPP_END
} ;
PostProcessingPass HDRFinal_Float_NoBloom [ ] =
{
PPP_PROCESS_SRCTEXTURE ( " dev/copyfullframefb " , " _rt_FullFrameFB " , NULL ) ,
PPP_END
} ;
static void SetRenderTargetAndViewPort ( ITexture * rt )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
pRenderContext - > SetRenderTarget ( rt ) ;
pRenderContext - > Viewport ( 0 , 0 , rt - > GetActualWidth ( ) , rt - > GetActualHeight ( ) ) ;
}
# define FILTER_KERNEL_SLOP 20
// Note carefully about the downsampling: the first downsampling samples from the full rendertarget
// down to a temp. When doing this sampling, the texture source clamping will take care of the out
// of bounds sampling done because of the filter kernels's width. However, on any of the subsequent
// sampling operations, we will be sampling from a partially filled render target. So, texture
// coordinate clamping cannot help us here. So, we need to always render a few more pixels to the
// destination than we actually intend to, so as to replicate the border pixels so that garbage
// pixels do not get sucked into the sampling. To deal with this, we always add FILTER_KERNEL_SLOP
// to our widths/heights if there is room for them in the destination.
static void DrawScreenSpaceRectangleWithSlop (
ITexture * dest_rt ,
IMaterial * pMaterial ,
int destx , int desty ,
int width , int height ,
float src_texture_x0 , float src_texture_y0 , // which texel you want to appear at
// destx/y
float src_texture_x1 , float src_texture_y1 , // which texel you want to appear at
// destx+width-1, desty+height-1
int src_texture_width , int src_texture_height // needed for fixup
)
{
// add slop
2011-04-28 01:30:09 -05:00
int slopwidth = width + FILTER_KERNEL_SLOP ; //MIN(dest_rt->GetActualWidth()-destx,width+FILTER_KERNEL_SLOP);
int slopheight = height + FILTER_KERNEL_SLOP ; //MIN(dest_rt->GetActualHeight()-desty,height+FILTER_KERNEL_SLOP);
2008-09-15 01:07:45 -05:00
// adjust coordinates for slop
src_texture_x1 = FLerp ( src_texture_x0 , src_texture_x1 , destx , destx + width - 1 , destx + slopwidth - 1 ) ;
src_texture_y1 = FLerp ( src_texture_y0 , src_texture_y1 , desty , desty + height - 1 , desty + slopheight - 1 ) ;
width = slopwidth ;
height = slopheight ;
CMatRenderContextPtr pRenderContext ( materials ) ;
pRenderContext - > DrawScreenSpaceRectangle ( pMaterial , destx , desty , width , height ,
src_texture_x0 , src_texture_y0 ,
src_texture_x1 , src_texture_y1 ,
src_texture_width , src_texture_height ) ;
}
enum Histogram_entry_state_t
{
HESTATE_INITIAL = 0 ,
HESTATE_FIRST_QUERY_IN_FLIGHT ,
HESTATE_QUERY_IN_FLIGHT ,
HESTATE_QUERY_DONE ,
} ;
# define N_LUMINANCE_RANGES 31
# define N_LUMINANCE_RANGES_NEW 17
# define MAX_QUERIES_PER_FRAME 1
class CHistogram_entry_t
{
public :
Histogram_entry_state_t m_state ;
OcclusionQueryObjectHandle_t m_occ_handle ; // the occlusion query handle
int m_frame_queued ; // when this query was last queued
int m_npixels ; // # of pixels this histogram represents
int m_npixels_in_range ;
float m_min_lum , m_max_lum ; // the luminance range this entry was queried with
float m_minx , m_miny , m_maxx , m_maxy ; // range is 0..1 in fractions of the screen
bool ContainsValidData ( void )
{
return ( m_state = = HESTATE_QUERY_DONE ) | | ( m_state = = HESTATE_QUERY_IN_FLIGHT ) ;
}
void IssueQuery ( int frm_num ) ;
} ;
void CHistogram_entry_t : : IssueQuery ( int frm_num )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
if ( ! m_occ_handle )
{
m_occ_handle = pRenderContext - > CreateOcclusionQueryObject ( ) ;
}
int xl , yl , dest_width , dest_height ;
pRenderContext - > GetViewport ( xl , yl , dest_width , dest_height ) ;
// Find min and max gamma-space text range
float flTestRangeMin = m_min_lum ;
float flTestRangeMax = ( m_max_lum = = 1.0f ) ? 10000.0f : m_max_lum ; // Count all pixels >1.0 as 1.0
// First, set stencil bits where the colors match
IMaterial * test_mat = materials - > FindMaterial ( " dev/lumcompare " , TEXTURE_GROUP_OTHER , true ) ;
IMaterialVar * pMinVar = test_mat - > FindVar ( " $C0_X " , NULL ) ;
pMinVar - > SetFloatValue ( flTestRangeMin ) ;
IMaterialVar * pMaxVar = test_mat - > FindVar ( " $C0_Y " , NULL ) ;
pMaxVar - > SetFloatValue ( flTestRangeMax ) ;
int scrx_min = FLerp ( xl , ( xl + dest_width - 1 ) , 0 , 1 , m_minx ) ;
int scrx_max = FLerp ( xl , ( xl + dest_width - 1 ) , 0 , 1 , m_maxx ) ;
int scry_min = FLerp ( yl , ( yl + dest_height - 1 ) , 0 , 1 , m_miny ) ;
int scry_max = FLerp ( yl , ( yl + dest_height - 1 ) , 0 , 1 , m_maxy ) ;
float exposure_width_scale , exposure_height_scale ;
// now, shrink region of interest if the flashlight is on
if ( g_bFlashlightIsOn )
{
exposure_width_scale = ( 0.5f * ( 1.0f - mat_exposure_center_region_x_flashlight . GetFloat ( ) ) ) ;
exposure_height_scale = ( 0.5f * ( 1.0f - mat_exposure_center_region_y_flashlight . GetFloat ( ) ) ) ;
}
else
{
exposure_width_scale = ( 0.5f * ( 1.0f - mat_exposure_center_region_x . GetFloat ( ) ) ) ;
exposure_height_scale = ( 0.5f * ( 1.0f - mat_exposure_center_region_y . GetFloat ( ) ) ) ;
}
int skip_edgex = ( 1 + scrx_max - scrx_min ) * exposure_width_scale ;
int skip_edgey = ( 1 + scry_max - scry_min ) * exposure_height_scale ;
// now, do luminance compare
float tscale = 1.0 ;
if ( g_pMaterialSystemHardwareConfig - > GetHDRType ( ) = = HDR_TYPE_FLOAT )
{
tscale = pRenderContext - > GetToneMappingScaleLinear ( ) . x ;
}
IMaterialVar * use_t_scale = test_mat - > FindVar ( " $C0_Z " , NULL ) ;
use_t_scale - > SetFloatValue ( tscale ) ;
m_npixels = ( 1 + scrx_max - scrx_min ) * ( 1 + scry_max - scry_min ) ;
if ( mat_tonemapping_occlusion_use_stencil . GetInt ( ) )
{
pRenderContext - > SetStencilWriteMask ( 1 ) ;
// AV - We don't need to clear stencil here because it's already been cleared at the beginning of the frame
//pRenderContext->ClearStencilBufferRectangle( scrx_min, scry_min, scrx_max, scry_max, 0 );
pRenderContext - > SetStencilEnable ( true ) ;
pRenderContext - > SetStencilPassOperation ( STENCILOPERATION_REPLACE ) ;
pRenderContext - > SetStencilCompareFunction ( STENCILCOMPARISONFUNCTION_ALWAYS ) ;
pRenderContext - > SetStencilFailOperation ( STENCILOPERATION_KEEP ) ;
pRenderContext - > SetStencilZFailOperation ( STENCILOPERATION_KEEP ) ;
pRenderContext - > SetStencilReferenceValue ( 1 ) ;
}
else
{
pRenderContext - > BeginOcclusionQueryDrawing ( m_occ_handle ) ;
}
scrx_min + = skip_edgex ;
scry_min + = skip_edgey ;
scrx_max - = skip_edgex ;
scry_max - = skip_edgey ;
pRenderContext - > DrawScreenSpaceRectangle ( test_mat ,
scrx_min , scry_min ,
1 + scrx_max - scrx_min ,
1 + scry_max - scry_min ,
scrx_min , scry_min ,
scrx_max , scry_max ,
dest_width , dest_height ) ;
if ( mat_tonemapping_occlusion_use_stencil . GetInt ( ) )
{
// now, start counting how many pixels had their stencil bit set via an occlusion query
pRenderContext - > BeginOcclusionQueryDrawing ( m_occ_handle ) ;
// now, issue an occlusion query using stencil as the mask
pRenderContext - > SetStencilEnable ( true ) ;
pRenderContext - > SetStencilTestMask ( 1 ) ;
pRenderContext - > SetStencilPassOperation ( STENCILOPERATION_KEEP ) ;
pRenderContext - > SetStencilCompareFunction ( STENCILCOMPARISONFUNCTION_EQUAL ) ;
pRenderContext - > SetStencilFailOperation ( STENCILOPERATION_KEEP ) ;
pRenderContext - > SetStencilZFailOperation ( STENCILOPERATION_KEEP ) ;
pRenderContext - > SetStencilReferenceValue ( 1 ) ;
IMaterial * stest_mat = materials - > FindMaterial ( " dev/no_pixel_write " , TEXTURE_GROUP_OTHER , true ) ;
pRenderContext - > DrawScreenSpaceRectangle ( stest_mat ,
scrx_min , scry_min ,
1 + scrx_max - scrx_min ,
1 + scry_max - scry_min ,
scrx_min , scry_min ,
scrx_max , scry_max ,
dest_width , dest_height ) ;
pRenderContext - > SetStencilEnable ( false ) ;
}
pRenderContext - > EndOcclusionQueryDrawing ( m_occ_handle ) ;
if ( m_state = = HESTATE_INITIAL )
m_state = HESTATE_FIRST_QUERY_IN_FLIGHT ;
else
m_state = HESTATE_QUERY_IN_FLIGHT ;
m_frame_queued = frm_num ;
}
# define HISTOGRAM_BAR_SIZE 200
class CLuminanceHistogramSystem
{
CHistogram_entry_t CurHistogram [ N_LUMINANCE_RANGES ] ;
int cur_query_frame ;
public :
float FindLocationOfPercentBrightPixels ( float flPercentBrightPixels , float flPercentTarget ) ;
float GetTargetTonemapScalar ( bool bGetIdealTargetForDebugMode ) ;
void Update ( void ) ;
void DisplayHistogram ( void ) ;
void UpdateLuminanceRanges ( void ) ;
CLuminanceHistogramSystem ( void )
{
UpdateLuminanceRanges ( ) ;
}
} ;
void CLuminanceHistogramSystem : : Update ( void )
{
UpdateLuminanceRanges ( ) ;
// find which histogram entries should have something done this frame
int n_queries_issued_this_frame = 0 ;
cur_query_frame + + ;
int nNumRanges = N_LUMINANCE_RANGES ;
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 )
nNumRanges = N_LUMINANCE_RANGES_NEW ;
for ( int i = 0 ; i < nNumRanges ; i + + )
{
switch ( CurHistogram [ i ] . m_state )
{
case HESTATE_INITIAL :
if ( n_queries_issued_this_frame < MAX_QUERIES_PER_FRAME )
{
CurHistogram [ i ] . IssueQuery ( cur_query_frame ) ;
n_queries_issued_this_frame + + ;
}
break ;
case HESTATE_FIRST_QUERY_IN_FLIGHT :
case HESTATE_QUERY_IN_FLIGHT :
if ( cur_query_frame > CurHistogram [ i ] . m_frame_queued + 2 )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
int np = pRenderContext - > OcclusionQuery_GetNumPixelsRendered (
CurHistogram [ i ] . m_occ_handle ) ;
if ( np ! = - 1 ) // -1=query not finished. wait until
// next time
{
CurHistogram [ i ] . m_npixels_in_range = np ;
// if (mat_debug_autoexposure.GetInt())
// Warning("min=%f max=%f np = %d\n",CurHistogram[i].m_min_lum,CurHistogram[i].m_max_lum,np);
CurHistogram [ i ] . m_state = HESTATE_QUERY_DONE ;
}
}
break ;
}
}
// now, issue queries for the oldest finished queries we have
while ( n_queries_issued_this_frame < MAX_QUERIES_PER_FRAME )
{
int nNumRanges = N_LUMINANCE_RANGES ;
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 )
nNumRanges = N_LUMINANCE_RANGES_NEW ;
int oldest_so_far = - 1 ;
for ( int i = 0 ; i < nNumRanges ; i + + )
if ( ( CurHistogram [ i ] . m_state = = HESTATE_QUERY_DONE ) & &
( ( oldest_so_far = = - 1 ) | |
( CurHistogram [ i ] . m_frame_queued <
CurHistogram [ oldest_so_far ] . m_frame_queued ) ) )
oldest_so_far = i ;
if ( oldest_so_far = = - 1 ) // nothing to do
break ;
CurHistogram [ oldest_so_far ] . IssueQuery ( cur_query_frame ) ;
n_queries_issued_this_frame + + ;
}
}
float CLuminanceHistogramSystem : : FindLocationOfPercentBrightPixels ( float flPercentBrightPixels , float flPercentTargetToSnapToIfInSameBin = - 1.0f )
{
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 ) // New algorithm
{
int nTotalValidPixels = 0 ;
for ( int i = 0 ; i < N_LUMINANCE_RANGES_NEW - 1 ; i + + )
{
if ( CurHistogram [ i ] . ContainsValidData ( ) )
{
nTotalValidPixels + = CurHistogram [ i ] . m_npixels_in_range ;
}
}
if ( nTotalValidPixels = = 0 )
{
return - 1.0f ;
}
// Find where percent range border is
float flTotalPercentRangeTested = 0.0f ;
float flTotalPercentPixelsTested = 0.0f ;
for ( int i = N_LUMINANCE_RANGES_NEW - 2 ; i > = 0 ; i - - ) // Start at the bright end
{
if ( ! CurHistogram [ i ] . ContainsValidData ( ) )
return - 1.0f ;
float flPixelPercentNeeded = ( flPercentBrightPixels / 100.0f ) - flTotalPercentPixelsTested ;
float flThisBinPercentOfTotalPixels = float ( CurHistogram [ i ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ;
float flThisBinLuminanceRange = CurHistogram [ i ] . m_max_lum - CurHistogram [ i ] . m_min_lum ;
if ( flThisBinPercentOfTotalPixels > = flPixelPercentNeeded ) // We found the bin needed
{
if ( flPercentTargetToSnapToIfInSameBin > = 0.0f )
{
if ( ( CurHistogram [ i ] . m_min_lum < = ( flPercentTargetToSnapToIfInSameBin / 100.0f ) ) & & ( CurHistogram [ i ] . m_max_lum > = ( flPercentTargetToSnapToIfInSameBin / 100.0f ) ) )
{
// Sticky bin...We're in the same bin as the target so keep the tonemap scale where it is
return ( flPercentTargetToSnapToIfInSameBin / 100.0f ) ;
}
}
float flPercentOfThesePixelsNeeded = flPixelPercentNeeded / flThisBinPercentOfTotalPixels ;
float flPercentLocationOfBorder = 1.0f - ( flTotalPercentRangeTested + ( flThisBinLuminanceRange * flPercentOfThesePixelsNeeded ) ) ;
2011-04-28 01:30:09 -05:00
flPercentLocationOfBorder = MAX ( CurHistogram [ i ] . m_min_lum , MIN ( CurHistogram [ i ] . m_max_lum , flPercentLocationOfBorder ) ) ; // Clamp to this bin just in case
2008-09-15 01:07:45 -05:00
return flPercentLocationOfBorder ;
}
flTotalPercentPixelsTested + = flThisBinPercentOfTotalPixels ;
flTotalPercentRangeTested + = flThisBinLuminanceRange ;
}
return - 1.0f ;
}
else
{
// Don't know what to do for other algorithms yet
return - 1.0f ;
}
}
float CLuminanceHistogramSystem : : GetTargetTonemapScalar ( bool bGetIdealTargetForDebugMode = false )
{
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 ) // New algorithm
{
float flPercentLocationOfTarget ;
if ( bGetIdealTargetForDebugMode = = true )
flPercentLocationOfTarget = FindLocationOfPercentBrightPixels ( mat_tonemap_percent_bright_pixels . GetFloat ( ) ) ; // Don't pass in the second arg so the scalar doesn't snap to a bin
else
flPercentLocationOfTarget = FindLocationOfPercentBrightPixels ( mat_tonemap_percent_bright_pixels . GetFloat ( ) , mat_tonemap_percent_target . GetFloat ( ) ) ;
if ( flPercentLocationOfTarget < 0.0f ) // This is the return error code
{
flPercentLocationOfTarget = mat_tonemap_percent_target . GetFloat ( ) / 100.0f ; // Pretend we're at the target
}
// Make sure this is > 0.0f
2011-04-28 01:30:09 -05:00
flPercentLocationOfTarget = MAX ( 0.0001f , flPercentLocationOfTarget ) ;
2008-09-15 01:07:45 -05:00
// Compute target scalar
float flTargetScalar = ( mat_tonemap_percent_target . GetFloat ( ) / 100.0f ) / flPercentLocationOfTarget ;
// Compute secondary target scalar
float flAverageLuminanceLocation = FindLocationOfPercentBrightPixels ( 50.0f ) ;
if ( flAverageLuminanceLocation > 0.0f )
{
float flTargetScalar2 = ( mat_tonemap_min_avglum . GetFloat ( ) / 100.0f ) / flAverageLuminanceLocation ;
// Only override it if it's trying to brighten the image more than the primary algorithm
if ( flTargetScalar2 > flTargetScalar )
{
flTargetScalar = flTargetScalar2 ;
}
}
// Apply this against last frames scalar
CMatRenderContextPtr pRenderContext ( materials ) ;
float flLastScale = pRenderContext - > GetToneMappingScaleLinear ( ) . x ;
flTargetScalar * = flLastScale ;
2011-04-28 01:30:09 -05:00
flTargetScalar = MAX ( 0.001f , flTargetScalar ) ;
2008-09-15 01:07:45 -05:00
return flTargetScalar ;
}
else // Original tonemapping
{
float average_luminance = 0.5f ;
float total = 0 ;
int total_pixels = 0 ;
float scale_value = 1.0 ;
if ( CurHistogram [ N_LUMINANCE_RANGES - 1 ] . ContainsValidData ( ) )
{
scale_value = CurHistogram [ N_LUMINANCE_RANGES - 1 ] . m_npixels * ( 1.0f / CurHistogram [ N_LUMINANCE_RANGES - 1 ] . m_npixels_in_range ) ;
if ( mat_debug_autoexposure . GetInt ( ) )
{
engine - > Con_NPrintf ( 20 , " Scale value = %f " , scale_value ) ;
//Warning( "scale value=%f\n", scale_value );
}
}
else
average_luminance = 0.5 ;
if ( ! IsFinite ( scale_value ) )
scale_value = 1.0f ;
for ( int i = 0 ; i < N_LUMINANCE_RANGES - 1 ; i + + )
{
if ( CurHistogram [ i ] . ContainsValidData ( ) )
{
total + = scale_value * CurHistogram [ i ] . m_npixels_in_range * AVG ( CurHistogram [ i ] . m_min_lum , CurHistogram [ i ] . m_max_lum ) ;
total_pixels + = CurHistogram [ i ] . m_npixels ;
}
else
average_luminance = 0.5 ; // always return 0.5 until we've queried a whole frame
}
if ( total_pixels > 0 )
average_luminance = total * ( 1.0 / total_pixels ) ;
else
average_luminance = 0.5 ;
// Make sure this is > 0.0f
2011-04-28 01:30:09 -05:00
average_luminance = MAX ( 0.0001f , average_luminance ) ;
2008-09-15 01:07:45 -05:00
// Compute target scalar
float flTargetScalar = 0.005 / average_luminance ;
return flTargetScalar ;
}
}
static float GetCurrentBloomScale ( void )
{
// Use the appropriate bloom scale settings. Mapmakers's overrides the convar settings.
float flCurrentBloomScale = 1.0f ;
if ( g_bUseCustomBloomScale )
{
flCurrentBloomScale = g_flCustomBloomScale ;
}
else
{
flCurrentBloomScale = mat_bloomscale . GetFloat ( ) ;
}
return flCurrentBloomScale ;
}
static void GetExposureRange ( float * flAutoExposureMin , float * flAutoExposureMax )
{
// Get min
if ( ( g_bUseCustomAutoExposureMin ) & & ( g_flCustomAutoExposureMin > 0.0f ) )
{
* flAutoExposureMin = g_flCustomAutoExposureMin ;
}
else
{
* flAutoExposureMin = mat_autoexposure_min . GetFloat ( ) ;
}
// Get max
if ( ( g_bUseCustomAutoExposureMax ) & & ( g_flCustomAutoExposureMax > 0.0f ) )
{
* flAutoExposureMax = g_flCustomAutoExposureMax ;
}
else
{
* flAutoExposureMax = mat_autoexposure_max . GetFloat ( ) ;
}
// Override
if ( mat_hdr_uncapexposure . GetInt ( ) )
{
* flAutoExposureMax = 20.0f ;
* flAutoExposureMin = 0.0f ;
}
// Make sure min <= max
if ( * flAutoExposureMin > * flAutoExposureMax )
{
* flAutoExposureMax = * flAutoExposureMin ;
}
}
void CLuminanceHistogramSystem : : UpdateLuminanceRanges ( void )
{
// Only update if our mode changed
static int s_nCurrentBucketAlgorithm = - 1 ;
if ( s_nCurrentBucketAlgorithm = = mat_tonemap_algorithm . GetInt ( ) )
return ;
s_nCurrentBucketAlgorithm = mat_tonemap_algorithm . GetInt ( ) ;
//==================================================================//
// Force fallback to original tone mapping algorithm for these mods //
//==================================================================//
static bool s_bFirstTime = true ;
if ( engine = = NULL )
{
// Force this code to get hit again so we can change algorithm based on the client
s_nCurrentBucketAlgorithm = - 1 ;
}
else if ( s_bFirstTime = = true )
{
s_bFirstTime = false ;
// This seems like a bad idea but it's fine for now
const char * sModsForOriginalAlgorithm [ ] = { " dod " , " cstrike " , " lostcoast " } ;
for ( int i = 0 ; i < 3 ; i + + )
{
if ( strlen ( engine - > GetGameDirectory ( ) ) > = strlen ( sModsForOriginalAlgorithm [ i ] ) )
{
if ( stricmp ( & ( engine - > GetGameDirectory ( ) [ strlen ( engine - > GetGameDirectory ( ) ) - strlen ( sModsForOriginalAlgorithm [ i ] ) ] ) , sModsForOriginalAlgorithm [ i ] ) = = 0 )
{
mat_tonemap_algorithm . SetValue ( 0 ) ; // Original algorithm
s_nCurrentBucketAlgorithm = mat_tonemap_algorithm . GetInt ( ) ;
break ;
}
}
}
}
int nNumRanges = N_LUMINANCE_RANGES ;
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 )
nNumRanges = N_LUMINANCE_RANGES_NEW ;
cur_query_frame = 0 ;
for ( int bucket = 0 ; bucket < nNumRanges ; bucket + + )
{
int idx = bucket ;
CHistogram_entry_t & e = CurHistogram [ idx ] ;
e . m_state = HESTATE_INITIAL ;
e . m_minx = 0 ;
e . m_maxx = 1 ;
e . m_miny = 0 ;
e . m_maxy = 1 ;
if ( bucket ! = nNumRanges - 1 ) // Last bucket is special
{
if ( mat_tonemap_algorithm . GetInt ( ) = = 0 ) // Original algorithm
{
// Use a logarithmic ramp for high range in the low range
e . m_min_lum = - 0.01 + exp ( FLerp ( log ( .01 ) , log ( .01 + 1 ) , 0 , nNumRanges - 1 , bucket ) ) ;
e . m_max_lum = - 0.01 + exp ( FLerp ( log ( .01 ) , log ( .01 + 1 ) , 0 , nNumRanges - 1 , bucket + 1 ) ) ;
}
else
{
// Use even distribution
e . m_min_lum = float ( bucket ) / float ( nNumRanges - 1 ) ;
e . m_max_lum = float ( bucket + 1 ) / float ( nNumRanges - 1 ) ;
// Use a distribution with slightly more bins in the low range
e . m_min_lum = e . m_min_lum > 0.0f ? powf ( e . m_min_lum , 1.5f ) : e . m_min_lum ;
e . m_max_lum = e . m_max_lum > 0.0f ? powf ( e . m_max_lum , 1.5f ) : e . m_max_lum ;
}
}
else
{
// The last bucket is used as a test to determine the return range for occlusion
// queries to use as a scale factor. some boards (nvidia) have their occlusion
// query return values larger when using AA.
e . m_min_lum = 0 ;
e . m_max_lum = 100000.0 ;
}
//Warning( "Bucket %d: min/max %f / %f ", bucket, e.m_min_lum, e.m_max_lum );
}
}
void CLuminanceHistogramSystem : : DisplayHistogram ( void )
{
bool bDrawTextThisFrame = true ;
if ( IsX360 ( ) )
{
static float s_flLastTimeUpdate = 0.0f ;
if ( int ( gpGlobals - > curtime ) - int ( s_flLastTimeUpdate ) > = 2 )
{
s_flLastTimeUpdate = gpGlobals - > curtime ;
bDrawTextThisFrame = true ;
}
else
{
bDrawTextThisFrame = false ;
}
}
CMatRenderContextPtr pRenderContext ( materials ) ;
pRenderContext - > PushRenderTargetAndViewport ( ) ;
int nNumRanges = N_LUMINANCE_RANGES - 1 ;
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 )
nNumRanges = N_LUMINANCE_RANGES_NEW - 1 ;
int nMaxValidPixels = 0 ;
int nTotalValidPixels = 0 ;
int nTotalGraphPixelsWide = 0 ;
for ( int l = 0 ; l < nNumRanges ; l + + )
{
CHistogram_entry_t & e = CurHistogram [ l ] ;
if ( e . ContainsValidData ( ) )
{
nTotalValidPixels + = e . m_npixels_in_range ;
if ( e . m_npixels_in_range > nMaxValidPixels )
{
nMaxValidPixels = e . m_npixels_in_range ;
}
}
2011-04-28 01:30:09 -05:00
int width = MAX ( 1 , 500 * ( e . m_max_lum - e . m_min_lum ) ) ;
2008-09-15 01:07:45 -05:00
nTotalGraphPixelsWide + = width + 2 ;
}
int xl , yl , dest_width , dest_height ;
pRenderContext - > GetViewport ( xl , yl , dest_width , dest_height ) ;
if ( bDrawTextThisFrame = = true )
{
engine - > Con_NPrintf ( 17 , " (All values in linear space) " ) ;
engine - > Con_NPrintf ( 21 , " AvgLum @ %4.2f%% mat_tonemap_min_avglum = %4.2f%% Using %d pixels of %d pixels on screen (%3d%%) " ,
2011-04-28 01:30:09 -05:00
MAX ( 0.0f , FindLocationOfPercentBrightPixels ( 50.0f ) ) * 100.0f , mat_tonemap_min_avglum . GetFloat ( ) ,
2008-09-15 01:07:45 -05:00
nTotalValidPixels , ( dest_width * dest_height ) , int ( float ( nTotalValidPixels ) * 100.0f / float ( dest_width * dest_height ) ) ) ;
engine - > Con_NPrintf ( 23 , " BloomScale = %4.2f mat_hdr_manual_tonemap_rate = %4.2f mat_accelerate_adjust_exposure_down = %4.2f " ,
GetCurrentBloomScale ( ) , mat_hdr_manual_tonemap_rate . GetFloat ( ) , mat_accelerate_adjust_exposure_down . GetFloat ( ) ) ;
}
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 ) // New algorithm only
{
float vTotalPixelsAndHigher [ N_LUMINANCE_RANGES ] ;
for ( int i = 0 ; i < nNumRanges ; i + + )
{
vTotalPixelsAndHigher [ i ] = CurHistogram [ nNumRanges - 1 - i ] . m_npixels_in_range ;
if ( i > 0 )
{
vTotalPixelsAndHigher [ i ] + = vTotalPixelsAndHigher [ i - 1 ] ;
}
}
/* // This code works when N_LUMINANCE_RANGES_NEW = 11
if ( bDrawTextThisFrame = = true )
{
engine - > Con_NPrintf ( 17 , " %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f " ,
100.0f * float ( vTotalPixelsAndHigher [ 9 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 8 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 7 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 6 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 5 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 4 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 3 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 2 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 1 ] ) / float ( nTotalValidPixels ) ,
100.0f * float ( vTotalPixelsAndHigher [ 0 ] ) / float ( nTotalValidPixels ) ) ;
engine - > Con_NPrintf ( 15 , " %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f %04.2f " ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 9 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 8 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 7 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 6 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 5 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 4 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 3 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 2 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 1 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ,
100.0f * float ( CurHistogram [ nNumRanges - 1 - 0 ] . m_npixels_in_range ) / float ( nTotalValidPixels ) ) ;
}
//*/
}
else
{
if ( bDrawTextThisFrame = = true )
{
engine - > Con_NPrintf ( 17 , " " ) ;
engine - > Con_NPrintf ( 15 , " " ) ;
}
}
int xpStart = dest_width - nTotalGraphPixelsWide - 10 ;
if ( IsX360 ( ) )
{
xpStart - = 50 ;
}
int xp = xpStart ;
for ( int l = 0 ; l < nNumRanges ; l + + )
{
int np = 0 ;
CHistogram_entry_t & e = CurHistogram [ l ] ;
if ( e . ContainsValidData ( ) )
np + = e . m_npixels_in_range ;
2011-04-28 01:30:09 -05:00
int width = MAX ( 1 , 500 * ( e . m_max_lum - e . m_min_lum ) ) ;
2008-09-15 01:07:45 -05:00
//Warning( "Bucket %d: min/max %f / %f. m_npixels_in_range=%d m_npixels=%d\n", l, e.m_min_lum, e.m_max_lum, e.m_npixels_in_range, e.m_npixels );
if ( np )
{
2011-04-28 01:30:09 -05:00
int height = MAX ( 1 , MIN ( HISTOGRAM_BAR_SIZE , ( ( float ) np / ( float ) nMaxValidPixels ) * HISTOGRAM_BAR_SIZE ) ) ;
2008-09-15 01:07:45 -05:00
pRenderContext - > ClearColor3ub ( 255 , 0 , 0 ) ;
pRenderContext - > Viewport ( xp , 4 + HISTOGRAM_BAR_SIZE - height , width , height ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
}
else
{
int height = 1 ;
pRenderContext - > ClearColor3ub ( 0 , 0 , 255 ) ;
pRenderContext - > Viewport ( xp , 4 + HISTOGRAM_BAR_SIZE - height , width , height ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
}
xp + = width + 2 ;
}
if ( mat_tonemap_algorithm . GetInt ( ) = = 1 ) // New algorithm only
{
float flYellowTargetPixelStart = ( xpStart + ( float ( nTotalGraphPixelsWide ) * mat_tonemap_percent_target . GetFloat ( ) / 100.0f ) ) ;
float flYellowAveragePixelStart = ( xpStart + ( float ( nTotalGraphPixelsWide ) * mat_tonemap_min_avglum . GetFloat ( ) / 100.0f ) ) ;
float flTargetPixelStart = ( xpStart + ( float ( nTotalGraphPixelsWide ) * FindLocationOfPercentBrightPixels ( mat_tonemap_percent_bright_pixels . GetFloat ( ) , mat_tonemap_percent_target . GetFloat ( ) ) ) ) ;
float flAveragePixelStart = ( xpStart + ( float ( nTotalGraphPixelsWide ) * FindLocationOfPercentBrightPixels ( 50.0f ) ) ) ;
// Draw target yellow border bar
int height = HISTOGRAM_BAR_SIZE ;
// Green is current percent target location
pRenderContext - > Viewport ( flYellowTargetPixelStart , 4 + HISTOGRAM_BAR_SIZE - height , 4 , height ) ;
pRenderContext - > ClearColor3ub ( 200 , 200 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
pRenderContext - > Viewport ( flTargetPixelStart , 4 + HISTOGRAM_BAR_SIZE - height , 4 , height ) ;
pRenderContext - > ClearColor3ub ( 0 , 255 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// Blue is average luminance location
pRenderContext - > Viewport ( flYellowAveragePixelStart , 4 + HISTOGRAM_BAR_SIZE - height , 4 , height ) ;
pRenderContext - > ClearColor3ub ( 200 , 200 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
pRenderContext - > Viewport ( flAveragePixelStart , 4 + HISTOGRAM_BAR_SIZE - height , 4 , height ) ;
pRenderContext - > ClearColor3ub ( 0 , 200 , 200 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
}
// Show actual tonemap value
if ( 1 )
{
float flAutoExposureMin ;
float flAutoExposureMax ;
GetExposureRange ( & flAutoExposureMin , & flAutoExposureMax ) ;
float flBarWidth = 600.0f ;
float flBarStart = dest_width - flBarWidth - 10.0f ;
if ( IsX360 ( ) )
{
flBarStart - = 50 ;
}
pRenderContext - > Viewport ( flBarStart , 4 + HISTOGRAM_BAR_SIZE - 4 + 75 , flBarWidth , 4 ) ;
pRenderContext - > ClearColor3ub ( 200 , 200 , 200 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
pRenderContext - > Viewport ( flBarStart , 4 + HISTOGRAM_BAR_SIZE - 4 + 75 + 1 , flBarWidth , 2 ) ;
pRenderContext - > ClearColor3ub ( 0 , 0 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
pRenderContext - > Viewport ( flBarStart + ( flBarWidth * ( ( pRenderContext - > GetToneMappingScaleLinear ( ) . x - flAutoExposureMin ) / ( flAutoExposureMax - flAutoExposureMin ) ) ) ,
4 + HISTOGRAM_BAR_SIZE - 4 + 75 - 6 , 4 , 16 ) ;
pRenderContext - > ClearColor3ub ( 255 , 0 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
if ( bDrawTextThisFrame = = true )
{
if ( IsX360 ( ) )
engine - > Con_NPrintf ( 26 , " Min: %.2f Max: %.2f " , flAutoExposureMin , flAutoExposureMax ) ;
else
engine - > Con_NPrintf ( 26 , " %.2f %.2f %.2f " , flAutoExposureMin , ( flAutoExposureMax + flAutoExposureMin ) / 2.0f , flAutoExposureMax ) ;
}
}
// Last bar doesn't clear properly so draw an extra pixel
pRenderContext - > Viewport ( 0 , 0 , 1 , 1 ) ;
pRenderContext - > ClearColor3ub ( 0 , 0 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
pRenderContext - > PopRenderTargetAndViewport ( ) ;
}
static CLuminanceHistogramSystem g_HDR_HistogramSystem ;
static float s_MovingAverageToneMapScale [ 10 ] = { 1.0f , 1.0f , 1.0f , 1.0f , 1.0f , 1.0f , 1.0f , 1.0f , 1.0f , 1.0f } ;
static int s_nInAverage = 0 ;
void ResetToneMapping ( float value )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
s_nInAverage = 0 ;
pRenderContext - > ResetToneMappingScale ( value ) ;
}
static ConVar mat_force_tonemap_scale ( " mat_force_tonemap_scale " , " 0.0 " , FCVAR_CHEAT ) ;
static void SetToneMapScale ( IMatRenderContext * pRenderContext , float newvalue , float minvalue , float maxvalue )
{
Assert ( IsFinite ( newvalue ) ) ;
if ( ! IsFinite ( newvalue ) )
return ;
float flForcedTonemapScale = mat_force_tonemap_scale . GetFloat ( ) ;
if ( mat_fullbright . GetInt ( ) = = 1 )
{
flForcedTonemapScale = 1.0f ;
}
if ( flForcedTonemapScale > 0.0f )
{
mat_hdr_tonemapscale . SetValue ( flForcedTonemapScale ) ;
pRenderContext - > ResetToneMappingScale ( flForcedTonemapScale ) ;
return ;
}
mat_hdr_tonemapscale . SetValue ( newvalue ) ;
pRenderContext - > SetGoalToneMappingScale ( newvalue ) ;
if ( s_nInAverage < ARRAYSIZE ( s_MovingAverageToneMapScale ) )
{
s_MovingAverageToneMapScale [ s_nInAverage + + ] = newvalue ;
}
else
{
// scroll, losing oldest
for ( int i = 0 ; i < ARRAYSIZE ( s_MovingAverageToneMapScale ) - 1 ; i + + )
s_MovingAverageToneMapScale [ i ] = s_MovingAverageToneMapScale [ i + 1 ] ;
s_MovingAverageToneMapScale [ ARRAYSIZE ( s_MovingAverageToneMapScale ) - 1 ] = newvalue ;
}
// now, use the average of the last tonemap calculations as our goal scale
if ( s_nInAverage = = ARRAYSIZE ( s_MovingAverageToneMapScale ) ) // got full buffer yet?
{
float avg = 0. ;
float sumweights = 0 ;
int sample_pt = ARRAYSIZE ( s_MovingAverageToneMapScale ) / 2 ;
for ( int i = 0 ; i < ARRAYSIZE ( s_MovingAverageToneMapScale ) ; i + + )
{
float weight = abs ( i - sample_pt ) * ( 1.0 / ( ARRAYSIZE ( s_MovingAverageToneMapScale ) / 2 ) ) ;
sumweights + = weight ;
avg + = weight * s_MovingAverageToneMapScale [ i ] ;
}
avg * = ( 1.0 / sumweights ) ;
2011-04-28 01:30:09 -05:00
avg = MIN ( maxvalue , MAX ( minvalue , avg ) ) ;
2008-09-15 01:07:45 -05:00
pRenderContext - > SetGoalToneMappingScale ( avg ) ;
mat_hdr_tonemapscale . SetValue ( avg ) ;
}
}
//=====================================================================================================================
// Engine_Post material proxy ============================================================================================
//=====================================================================================================================
static ConVar mat_software_aa_strength ( " mat_software_aa_strength " , " -1.0 " , FCVAR_ARCHIVE , " Software AA - perform a software anti-aliasing post-process (an alternative/supplement to MSAA) . This value sets the strength of the effect : ( 0.0 - off ) , ( 1.0 - full ) " ) ;
static ConVar mat_software_aa_quality ( " mat_software_aa_quality " , " 0 " , FCVAR_ARCHIVE , " Software AA quality mode: (0 - 5-tap filter) , ( 1 - 9 - tap filter ) " ) ;
static ConVar mat_software_aa_edge_threshold ( " mat_software_aa_edge_threshold " , " 1.0 " , FCVAR_ARCHIVE , " Software AA - adjusts the sensitivity of the software AA shader's edge detection (default 1.0 - a lower value will soften more edges, a higher value will soften fewer) " ) ;
static ConVar mat_software_aa_blur_one_pixel_lines ( " mat_software_aa_blur_one_pixel_lines " , " 0.5 " , FCVAR_ARCHIVE , " How much software AA should blur one-pixel thick lines: (0.0 - none) , ( 1.0 - lots ) " ) ;
static ConVar mat_software_aa_tap_offset ( " mat_software_aa_tap_offset " , " 1.0 " , FCVAR_ARCHIVE , " Software AA - adjusts the displacement of the taps used by the software AA shader (default 1.0 - a lower value will make the image sharper, higher will make it blurrier) " ) ;
static ConVar mat_software_aa_debug ( " mat_software_aa_debug " , " 0 " , FCVAR_NONE , " Software AA debug mode: (0 - off) , ( 1 - show number of ' unlike ' samples : 0 - > black , 1 - > red , 2 - > green , 3 - > blue ) , ( 2 - show anti - alias blend strength ) , ( 3 - show averaged ' unlike ' colour ) " ) ;
static ConVar mat_software_aa_strength_vgui ( " mat_software_aa_strength_vgui " , " -1.0 " , FCVAR_ARCHIVE , " Same as mat_software_aa_strength, but forced to this value when called by the post vgui AA pass. " ) ;
class CEnginePostMaterialProxy : public CEntityMaterialProxy
{
public :
CEnginePostMaterialProxy ( ) ;
virtual ~ CEnginePostMaterialProxy ( ) ;
virtual bool Init ( IMaterial * pMaterial , KeyValues * pKeyValues ) ;
virtual void OnBind ( C_BaseEntity * pEntity ) ;
virtual IMaterial * GetMaterial ( ) ;
private :
IMaterialVar * m_pMaterialParam_AAValues ;
IMaterialVar * m_pMaterialParam_AAValues2 ;
IMaterialVar * m_pMaterialParam_BloomEnable ;
IMaterialVar * m_pMaterialParam_BloomUVTransform ;
IMaterialVar * m_pMaterialParam_ColCorrectEnable ;
IMaterialVar * m_pMaterialParam_ColCorrectNumLookups ;
IMaterialVar * m_pMaterialParam_ColCorrectDefaultWeight ;
IMaterialVar * m_pMaterialParam_ColCorrectLookupWeights ;
public :
static IMaterial * SetupEnginePostMaterial ( const Vector4D & fullViewportBloomUVs , const Vector4D & fullViewportFBUVs , const Vector2D & destTexSize ,
bool bPerformSoftwareAA , bool bPerformBloom , bool bPerformColCorrect , float flAAStrength ) ;
static void SetupEnginePostMaterialAA ( bool bPerformSoftwareAA , float flAAStrength ) ;
static void SetupEnginePostMaterialTextureTransform ( const Vector4D & fullViewportBloomUVs , const Vector4D & fullViewportFBUVs , Vector2D destTexSize ) ;
private :
static float s_vBloomAAValues [ 4 ] ;
static float s_vBloomAAValues2 [ 4 ] ;
static float s_vBloomUVTransform [ 4 ] ;
static int s_PostBloomEnable ;
} ;
float CEnginePostMaterialProxy : : s_vBloomAAValues [ 4 ] = { 0.0f , 0.0f , 0.0f , 0.0f } ;
float CEnginePostMaterialProxy : : s_vBloomAAValues2 [ 4 ] = { 0.0f , 0.0f , 0.0f , 0.0f } ;
float CEnginePostMaterialProxy : : s_vBloomUVTransform [ 4 ] = { 0.0f , 0.0f , 0.0f , 0.0f } ;
int CEnginePostMaterialProxy : : s_PostBloomEnable = 1 ;
CEnginePostMaterialProxy : : CEnginePostMaterialProxy ( )
{
m_pMaterialParam_AAValues = NULL ;
m_pMaterialParam_AAValues2 = NULL ;
m_pMaterialParam_BloomUVTransform = NULL ;
m_pMaterialParam_BloomEnable = NULL ;
m_pMaterialParam_ColCorrectEnable = NULL ;
m_pMaterialParam_ColCorrectNumLookups = NULL ;
m_pMaterialParam_ColCorrectDefaultWeight = NULL ;
m_pMaterialParam_ColCorrectLookupWeights = NULL ;
}
CEnginePostMaterialProxy : : ~ CEnginePostMaterialProxy ( )
{
// Do nothing
}
bool CEnginePostMaterialProxy : : Init ( IMaterial * pMaterial , KeyValues * pKeyValues )
{
bool bFoundVar = false ;
m_pMaterialParam_AAValues = pMaterial - > FindVar ( " $AAInternal1 " , & bFoundVar , false ) ;
m_pMaterialParam_AAValues2 = pMaterial - > FindVar ( " $AAInternal3 " , & bFoundVar , false ) ;
m_pMaterialParam_BloomUVTransform = pMaterial - > FindVar ( " $AAInternal2 " , & bFoundVar , false ) ;
m_pMaterialParam_BloomEnable = pMaterial - > FindVar ( " $bloomEnable " , & bFoundVar , false ) ;
m_pMaterialParam_ColCorrectEnable = pMaterial - > FindVar ( " $colCorrectEnable " , & bFoundVar , false ) ;
m_pMaterialParam_ColCorrectNumLookups = pMaterial - > FindVar ( " $colCorrect_NumLookups " , & bFoundVar , false ) ;
m_pMaterialParam_ColCorrectDefaultWeight = pMaterial - > FindVar ( " $colCorrect_DefaultWeight " , & bFoundVar , false ) ;
m_pMaterialParam_ColCorrectLookupWeights = pMaterial - > FindVar ( " $colCorrect_LookupWeights " , & bFoundVar , false ) ;
return true ;
}
void CEnginePostMaterialProxy : : OnBind ( C_BaseEntity * pEnt )
{
if ( m_pMaterialParam_AAValues )
m_pMaterialParam_AAValues - > SetVecValue ( s_vBloomAAValues , 4 ) ;
if ( m_pMaterialParam_AAValues2 )
m_pMaterialParam_AAValues2 - > SetVecValue ( s_vBloomAAValues2 , 4 ) ;
if ( m_pMaterialParam_BloomUVTransform )
m_pMaterialParam_BloomUVTransform - > SetVecValue ( s_vBloomUVTransform , 4 ) ;
if ( m_pMaterialParam_BloomEnable )
m_pMaterialParam_BloomEnable - > SetIntValue ( s_PostBloomEnable ) ;
}
IMaterial * CEnginePostMaterialProxy : : GetMaterial ( )
{
if ( m_pMaterialParam_AAValues = = NULL )
return NULL ;
return m_pMaterialParam_AAValues - > GetOwningMaterial ( ) ;
}
void CEnginePostMaterialProxy : : SetupEnginePostMaterialAA ( bool bPerformSoftwareAA , float flAAStrength )
{
if ( bPerformSoftwareAA )
{
// Pass ConVars to the material by proxy
// - the strength of the AA effect (from 0 to 1)
// - how much to allow 1-pixel lines to be blurred (from 0 to 1)
// - pick one of the two quality modes (5-tap or 9-tap filter)
// - optionally enable one of several debug modes (via dynamic combos)
// NOTE: this order matches pixel shader constants in Engine_Post_ps2x.fxc
s_vBloomAAValues [ 0 ] = flAAStrength ;
s_vBloomAAValues [ 1 ] = 1.0f - mat_software_aa_blur_one_pixel_lines . GetFloat ( ) ;
s_vBloomAAValues [ 2 ] = mat_software_aa_quality . GetInt ( ) ;
s_vBloomAAValues [ 3 ] = mat_software_aa_debug . GetInt ( ) ;
s_vBloomAAValues2 [ 0 ] = mat_software_aa_edge_threshold . GetFloat ( ) ;
s_vBloomAAValues2 [ 1 ] = mat_software_aa_tap_offset . GetFloat ( ) ;
//s_vBloomAAValues2[2] = unused;
//s_vBloomAAValues2[3] = unused;
}
else
{
// Zero-strength AA is interpreted as "AA disabled"
s_vBloomAAValues [ 0 ] = 0.0f ;
}
}
void CEnginePostMaterialProxy : : SetupEnginePostMaterialTextureTransform ( const Vector4D & fullViewportBloomUVs , const Vector4D & fullViewportFBUVs , Vector2D fbSize )
{
// Engine_Post uses a UV transform (from (quarter-res) bloom texture coords ('1')
// to (full-res) framebuffer texture coords ('2')).
//
// We compute the UV transform as an offset and a scale, using the texture coordinates
// of the top-left corner of the screen to compute the offset and the coordinate
// change from the top-left to the bottom-right of the quad to compute the scale.
// Take texel coordinates (start = top-left, end = bottom-right):
Vector2D texelStart1 = Vector2D ( fullViewportBloomUVs . x , fullViewportBloomUVs . y ) ;
Vector2D texelStart2 = Vector2D ( fullViewportFBUVs . x , fullViewportFBUVs . y ) ;
Vector2D texelEnd1 = Vector2D ( fullViewportBloomUVs . z , fullViewportBloomUVs . w ) ;
Vector2D texelEnd2 = Vector2D ( fullViewportFBUVs . z , fullViewportFBUVs . w ) ;
// ...and transform to UV coordinates:
Vector2D texRes1 = fbSize / 4 ;
Vector2D texRes2 = fbSize ;
Vector2D uvStart1 = ( texelStart1 + Vector2D ( 0.5 , 0.5 ) ) / texRes1 ;
Vector2D uvStart2 = ( texelStart2 + Vector2D ( 0.5 , 0.5 ) ) / texRes2 ;
Vector2D dUV1 = ( texelEnd1 - texelStart1 ) / texRes1 ;
Vector2D dUV2 = ( texelEnd2 - texelStart2 ) / texRes2 ;
// We scale about the rect's top-left pixel centre (not the origin) in UV-space:
// uv' = ((uv - uvStart1)*uvScale + uvStart1) + uvOffset
// = uvScale*uv + uvOffset + uvStart1*(1 - uvScale)
Vector2D uvOffset = uvStart2 - uvStart1 ;
Vector2D uvScale = dUV2 / dUV1 ;
uvOffset = uvOffset + uvStart1 * ( Vector2D ( 1 , 1 ) - uvScale ) ;
s_vBloomUVTransform [ 0 ] = uvOffset . x ;
s_vBloomUVTransform [ 1 ] = uvOffset . y ;
s_vBloomUVTransform [ 2 ] = uvScale . x ;
s_vBloomUVTransform [ 3 ] = uvScale . y ;
}
IMaterial * CEnginePostMaterialProxy : : SetupEnginePostMaterial ( const Vector4D & fullViewportBloomUVs , const Vector4D & fullViewportFBUVs , const Vector2D & destTexSize ,
bool bPerformSoftwareAA , bool bPerformBloom , bool bPerformColCorrect , float flAAStrength )
{
// Shouldn't get here if none of the effects are enabled
Assert ( bPerformSoftwareAA | | bPerformBloom | | bPerformColCorrect ) ;
s_PostBloomEnable = bPerformBloom ? 1 : 0 ;
SetupEnginePostMaterialAA ( bPerformSoftwareAA , flAAStrength ) ;
if ( bPerformSoftwareAA | | bPerformColCorrect )
{
SetupEnginePostMaterialTextureTransform ( fullViewportBloomUVs , fullViewportFBUVs , destTexSize ) ;
return materials - > FindMaterial ( " dev/engine_post " , TEXTURE_GROUP_OTHER , true ) ;
}
else
{
// Just use the old bloomadd material (which uses additive blending, unlike engine_post)
// NOTE: this path is what gets used for DX8 (which cannot enable AA or col-correction)
return materials - > FindMaterial ( " dev/bloomadd " , TEXTURE_GROUP_OTHER , true ) ;
}
}
EXPOSE_INTERFACE ( CEnginePostMaterialProxy , IMaterialProxy , " engine_post " IMATERIAL_PROXY_INTERFACE_VERSION ) ;
static void DrawBloomDebugBoxes ( IMatRenderContext * pRenderContext )
{
// draw inset rects which should have a centered bloom
pRenderContext - > SetRenderTarget ( NULL ) ;
int dest_width , dest_height ;
pRenderContext - > GetRenderTargetDimensions ( dest_width , dest_height ) ;
// full screen clear
pRenderContext - > Viewport ( 0 , 0 , dest_width , dest_height ) ;
pRenderContext - > ClearColor3ub ( 0 , 0 , 0 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// inset for screensafe
int inset = 64 ;
int size = 32 ;
// centerish, translating
static int wx = 0 ;
wx = ( wx + 1 ) & 63 ;
pRenderContext - > Viewport ( dest_width / 2 + wx , dest_height / 2 , size , size ) ;
pRenderContext - > ClearColor3ub ( 255 , 255 , 255 ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// upper left
pRenderContext - > Viewport ( inset , inset , size , size ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// upper right
pRenderContext - > Viewport ( dest_width - inset - size , inset , size , size ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// lower right
pRenderContext - > Viewport ( dest_width - inset - size , dest_height - inset - size , size , size ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// lower left
pRenderContext - > Viewport ( inset , dest_height - inset - size , size , size ) ;
pRenderContext - > ClearBuffers ( true , true ) ;
// restore
pRenderContext - > Viewport ( 0 , 0 , dest_width , dest_height ) ;
}
static float GetBloomAmount ( void )
{
// return bloom amount ( 0.0 if disabled or otherwise turned off )
if ( engine - > GetDXSupportLevel ( ) < 80 )
return 0.0 ;
HDRType_t hdrType = g_pMaterialSystemHardwareConfig - > GetHDRType ( ) ;
bool bBloomEnabled = ( mat_hdr_level . GetInt ( ) > = 1 ) ;
if ( ! engine - > MapHasHDRLighting ( ) )
bBloomEnabled = false ;
if ( mat_force_bloom . GetInt ( ) )
bBloomEnabled = true ;
if ( mat_disable_bloom . GetInt ( ) )
bBloomEnabled = false ;
if ( building_cubemaps . GetBool ( ) )
bBloomEnabled = false ;
if ( mat_fullbright . GetInt ( ) = = 1 )
{
bBloomEnabled = false ;
}
float flBloomAmount = 0.0 ;
if ( bBloomEnabled )
{
static float currentBloomAmount = 1.0f ;
float rate = mat_bloomamount_rate . GetFloat ( ) ;
// Use the appropriate bloom scale settings. Mapmakers's overrides the convar settings.
currentBloomAmount = GetCurrentBloomScale ( ) * rate + ( 1.0f - rate ) * currentBloomAmount ;
flBloomAmount = currentBloomAmount ;
}
if ( hdrType = = HDR_TYPE_NONE )
{
flBloomAmount * = mat_non_hdr_bloom_scalefactor . GetFloat ( ) ;
}
flBloomAmount * = mat_bloom_scalefactor_scalar . GetFloat ( ) ;
return flBloomAmount ;
}
static bool s_bScreenEffectTextureIsUpdated = false ;
static void Generate8BitBloomTexture ( IMatRenderContext * pRenderContext , float flBloomScale ,
int x , int y , int w , int h )
{
pRenderContext - > PushRenderTargetAndViewport ( ) ;
ITexture * pSrc = materials - > FindTexture ( " _rt_FullFrameFB " , TEXTURE_GROUP_RENDER_TARGET ) ;
int nSrcWidth = pSrc - > GetActualWidth ( ) ;
int nSrcHeight = pSrc - > GetActualHeight ( ) ; //,dest_height;
IMaterial * downsample_mat = materials - > FindMaterial ( " dev/downsample_non_hdr " , TEXTURE_GROUP_OTHER , true ) ;
IMaterial * xblur_mat = materials - > FindMaterial ( " dev/blurfilterx_nohdr " , TEXTURE_GROUP_OTHER , true ) ;
IMaterial * yblur_mat = materials - > FindMaterial ( " dev/blurfiltery_nohdr " , TEXTURE_GROUP_OTHER , true ) ;
ITexture * dest_rt0 = materials - > FindTexture ( " _rt_SmallFB0 " , TEXTURE_GROUP_RENDER_TARGET ) ;
ITexture * dest_rt1 = materials - > FindTexture ( " _rt_SmallFB1 " , TEXTURE_GROUP_RENDER_TARGET ) ;
// *Everything* in here relies on the small RTs being exactly 1/4 the full FB res
Assert ( dest_rt0 - > GetActualWidth ( ) = = pSrc - > GetActualWidth ( ) / 4 ) ;
Assert ( dest_rt0 - > GetActualHeight ( ) = = pSrc - > GetActualHeight ( ) / 4 ) ;
Assert ( dest_rt1 - > GetActualWidth ( ) = = pSrc - > GetActualWidth ( ) / 4 ) ;
Assert ( dest_rt1 - > GetActualHeight ( ) = = pSrc - > GetActualHeight ( ) / 4 ) ;
// downsample fb to rt0
SetRenderTargetAndViewPort ( dest_rt0 ) ;
// note the -2's below. Thats because we are downsampling on each axis and the shader
// accesses pixels on both sides of the source coord
pRenderContext - > DrawScreenSpaceRectangle ( downsample_mat , 0 , 0 , nSrcWidth / 4 , nSrcHeight / 4 ,
0 , 0 , nSrcWidth - 2 , nSrcHeight - 2 ,
nSrcWidth , nSrcHeight ) ;
if ( IsX360 ( ) )
{
pRenderContext - > CopyRenderTargetToTextureEx ( dest_rt0 , 0 , NULL , NULL ) ;
}
// guassian blur x rt0 to rt1
SetRenderTargetAndViewPort ( dest_rt1 ) ;
pRenderContext - > DrawScreenSpaceRectangle ( xblur_mat , 0 , 0 , nSrcWidth / 4 , nSrcHeight / 4 ,
0 , 0 , nSrcWidth / 4 - 1 , nSrcHeight / 4 - 1 ,
nSrcWidth / 4 , nSrcHeight / 4 ) ;
if ( IsX360 ( ) )
{
pRenderContext - > CopyRenderTargetToTextureEx ( dest_rt1 , 0 , NULL , NULL ) ;
}
// GR - gaussian blur y rt1 to rt0
SetRenderTargetAndViewPort ( dest_rt0 ) ;
IMaterialVar * pBloomAmountVar = yblur_mat - > FindVar ( " $bloomamount " , NULL ) ;
pBloomAmountVar - > SetFloatValue ( flBloomScale ) ;
pRenderContext - > DrawScreenSpaceRectangle ( yblur_mat , 0 , 0 , nSrcWidth / 4 , nSrcHeight / 4 ,
0 , 0 , nSrcWidth / 4 - 1 , nSrcHeight / 4 - 1 ,
nSrcWidth / 4 , nSrcHeight / 4 ) ;
if ( IsX360 ( ) )
{
pRenderContext - > CopyRenderTargetToTextureEx ( dest_rt0 , 0 , NULL , NULL ) ;
}
pRenderContext - > PopRenderTargetAndViewport ( ) ;
}
static void DoPreBloomTonemapping ( IMatRenderContext * pRenderContext , int nX , int nY , int nWidth , int nHeight , float flAutoExposureMin , float flAutoExposureMax )
{
// Update HDR histogram before bloom
if ( mat_dynamic_tonemapping . GetInt ( ) | | mat_show_histogram . GetInt ( ) )
{
if ( s_bScreenEffectTextureIsUpdated = = false )
{
// FIXME: nX/nY/nWidth/nHeight are used here, but the equivalent parameters are ignored in Generate8BitBloomTexture
UpdateScreenEffectTexture ( 0 , nX , nY , nWidth , nHeight , true ) ;
s_bScreenEffectTextureIsUpdated = true ;
}
g_HDR_HistogramSystem . Update ( ) ;
if ( mat_dynamic_tonemapping . GetInt ( ) | | mat_show_histogram . GetInt ( ) )
{
float flTargetScalar = g_HDR_HistogramSystem . GetTargetTonemapScalar ( ) ;
2011-04-28 01:30:09 -05:00
float flTargetScalarClamped = MAX ( flAutoExposureMin , MIN ( flAutoExposureMax , flTargetScalar ) ) ;
flTargetScalarClamped = MAX ( 0.001f , flTargetScalarClamped ) ; // Don't let this go to 0!
2008-09-15 01:07:45 -05:00
if ( mat_dynamic_tonemapping . GetInt ( ) )
{
SetToneMapScale ( pRenderContext , flTargetScalarClamped , flAutoExposureMin , flAutoExposureMax ) ;
}
if ( mat_debug_autoexposure . GetInt ( ) | | mat_show_histogram . GetInt ( ) )
{
bool bDrawTextThisFrame = true ;
if ( IsX360 ( ) )
{
static float s_flLastTimeUpdate = 0.0f ;
if ( int ( gpGlobals - > curtime ) - int ( s_flLastTimeUpdate ) > = 2 )
{
s_flLastTimeUpdate = gpGlobals - > curtime ;
bDrawTextThisFrame = true ;
}
else
{
bDrawTextThisFrame = false ;
}
}
if ( bDrawTextThisFrame = = true )
{
if ( mat_tonemap_algorithm . GetInt ( ) = = 0 )
{
engine - > Con_NPrintf ( 19 , " (Original algorithm) Target Scalar = %4.2f Min/Max( %4.2f, %4.2f ) Final Scalar: %4.2f Actual: %4.2f " ,
flTargetScalar , flAutoExposureMin , flAutoExposureMax , mat_hdr_tonemapscale . GetFloat ( ) , pRenderContext - > GetToneMappingScaleLinear ( ) . x ) ;
}
else
{
engine - > Con_NPrintf ( 19 , " %.2f%% of pixels above %d%% target @ %4.2f%% Target Scalar = %4.2f Min/Max( %4.2f, %4.2f ) Final Scalar: %4.2f Actual: %4.2f " ,
mat_tonemap_percent_bright_pixels . GetFloat ( ) , mat_tonemap_percent_target . GetInt ( ) ,
( g_HDR_HistogramSystem . FindLocationOfPercentBrightPixels ( mat_tonemap_percent_bright_pixels . GetFloat ( ) , mat_tonemap_percent_target . GetFloat ( ) ) * 100.0f ) ,
g_HDR_HistogramSystem . GetTargetTonemapScalar ( true ) , flAutoExposureMin , flAutoExposureMax , mat_hdr_tonemapscale . GetFloat ( ) , pRenderContext - > GetToneMappingScaleLinear ( ) . x ) ;
}
}
}
}
}
}
static void DoPostBloomTonemapping ( IMatRenderContext * pRenderContext , int nX , int nY , int nWidth , int nHeight , float flAutoExposureMin , float flAutoExposureMax )
{
if ( mat_show_histogram . GetInt ( ) & & ( engine - > GetDXSupportLevel ( ) > = 90 ) )
{
g_HDR_HistogramSystem . DisplayHistogram ( ) ;
}
}
static void CenterScaleQuadUVs ( Vector4D & quadUVs , const Vector2D & uvScale )
{
Vector2D uvMid = 0.5f * Vector2D ( ( quadUVs . z + quadUVs . x ) , ( quadUVs . w + quadUVs . y ) ) ;
Vector2D uvRange = 0.5f * Vector2D ( ( quadUVs . z - quadUVs . x ) , ( quadUVs . w - quadUVs . y ) ) ;
quadUVs . x = uvMid . x - uvScale . x * uvRange . x ;
quadUVs . y = uvMid . y - uvScale . y * uvRange . y ;
quadUVs . z = uvMid . x + uvScale . x * uvRange . x ;
quadUVs . w = uvMid . y + uvScale . y * uvRange . y ;
}
static ConVar r_queued_post_processing ( " r_queued_post_processing " , " 0 " ) ;
// How much to dice up the screen during post-processing on 360
// This has really marginal effects, but 4x1 does seem vaguely better for post-processing
static ConVar mat_postprocess_x ( " mat_postprocess_x " , " 4 " ) ;
static ConVar mat_postprocess_y ( " mat_postprocess_y " , " 1 " ) ;
void DoEnginePostProcessing ( int x , int y , int w , int h , bool bFlashlightIsOn , bool bPostVGui )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
# if defined( _X360 )
pRenderContext - > PushVertexShaderGPRAllocation ( 16 ) ; //max out pixel shader threads
# endif
if ( r_queued_post_processing . GetInt ( ) )
{
ICallQueue * pCallQueue = pRenderContext - > GetCallQueue ( ) ;
if ( pCallQueue )
{
pCallQueue - > QueueCall ( DoEnginePostProcessing , x , y , w , h , bFlashlightIsOn , bPostVGui ) ;
return ;
}
}
float flBloomScale = GetBloomAmount ( ) ;
HDRType_t hdrType = g_pMaterialSystemHardwareConfig - > GetHDRType ( ) ;
g_bFlashlightIsOn = bFlashlightIsOn ;
// Use the appropriate autoexposure min / max settings.
// Mapmaker's overrides the convar settings.
float flAutoExposureMin ;
float flAutoExposureMax ;
GetExposureRange ( & flAutoExposureMin , & flAutoExposureMax ) ;
if ( mat_debug_bloom . GetInt ( ) = = 1 )
{
DrawBloomDebugBoxes ( pRenderContext ) ;
}
switch ( hdrType )
{
case HDR_TYPE_NONE :
case HDR_TYPE_INTEGER :
{
s_bScreenEffectTextureIsUpdated = false ;
if ( hdrType ! = HDR_TYPE_NONE )
{
DoPreBloomTonemapping ( pRenderContext , x , y , w , h , flAutoExposureMin , flAutoExposureMax ) ;
}
// Set software-AA on by default for 360
if ( mat_software_aa_strength . GetFloat ( ) = = - 1.0f )
{
if ( IsX360 ( ) )
{
mat_software_aa_strength . SetValue ( 1.0f ) ;
if ( g_pMaterialSystem - > GetCurrentConfigForVideoCard ( ) . m_VideoMode . m_Height > 480 )
{
mat_software_aa_quality . SetValue ( 0 ) ;
}
else
{
// For standard-def, we have fewer pixels so we can afford 'high quality' mode (5->9 taps/pixel)
mat_software_aa_quality . SetValue ( 1 ) ;
}
}
else
{
mat_software_aa_strength . SetValue ( 0.0f ) ;
}
}
// Same trick for setting up the vgui aa strength
if ( mat_software_aa_strength_vgui . GetFloat ( ) = = - 1.0f )
{
if ( IsX360 ( ) & & ( g_pMaterialSystem - > GetCurrentConfigForVideoCard ( ) . m_VideoMode . m_Height = = 720 ) )
{
mat_software_aa_strength_vgui . SetValue ( 2.0f ) ;
}
else
{
mat_software_aa_strength_vgui . SetValue ( 1.0f ) ;
}
}
float flAAStrength ;
// We do a second AA blur pass over the TF intro menus. use mat_software_aa_strength_vgui there instead
if ( IsX360 ( ) & & bPostVGui )
{
flAAStrength = mat_software_aa_strength_vgui . GetFloat ( ) ;
}
else
{
flAAStrength = mat_software_aa_strength . GetFloat ( ) ;
}
// bloom, software-AA and colour-correction (applied in 1 pass, after generation of the bloom texture)
bool bPerformSoftwareAA = ( engine - > GetDXSupportLevel ( ) > = 90 ) & & ( flAAStrength ! = 0.0f ) ;
bool bPerformBloom = ! bPostVGui & & ( flBloomScale > 0.0f ) ;
bool bPerformColCorrect = ! bPostVGui & &
( g_pMaterialSystemHardwareConfig - > GetDXSupportLevel ( ) > = 90 ) & &
( g_pMaterialSystemHardwareConfig - > GetHDRType ( ) ! = HDR_TYPE_FLOAT ) & &
g_pColorCorrectionMgr - > HasNonZeroColorCorrectionWeights ( ) & &
mat_colorcorrection . GetInt ( ) ;
pRenderContext - > EnableColorCorrection ( bPerformColCorrect ) ;
if ( bPerformBloom | | bPerformSoftwareAA | | bPerformColCorrect )
{
ITexture * pSrc = materials - > FindTexture ( " _rt_FullFrameFB " , TEXTURE_GROUP_RENDER_TARGET ) ;
int nSrcWidth = pSrc - > GetActualWidth ( ) ;
int nSrcHeight = pSrc - > GetActualHeight ( ) ;
ITexture * dest_rt1 = materials - > FindTexture ( " _rt_SmallFB1 " , TEXTURE_GROUP_RENDER_TARGET ) ;
if ( ! s_bScreenEffectTextureIsUpdated )
{
// NOTE: UpdateScreenEffectTexture() uses StretchRect, so _rt_FullFrameFB is always 100%
// filled, even when the viewport is not fullscreen (e.g. with 'mat_viewportscale 0.5')
UpdateScreenEffectTexture ( 0 , x , y , w , h , true ) ;
s_bScreenEffectTextureIsUpdated = true ;
}
if ( bPerformBloom )
{
Generate8BitBloomTexture ( pRenderContext , flBloomScale , x , y , w , h ) ;
}
// Now add bloom (dest_rt0) to the framebuffer and perform software anti-aliasing and
// colour correction, all in one pass (improves performance, reduces quantization errors)
//
// First, set up texel coords (in the bloom and fb textures) at the centres of the outer pixel of the viewport:
Vector4D fullViewportPostSrcCorners ( 0.0f , - 0.5f , nSrcWidth / 4 - 1 , nSrcHeight / 4 - 1 ) ;
Vector4D fullViewportPostDestCorners ( 0.0f , 0.0f , nSrcWidth - 1 , nSrcHeight - 1 ) ;
Rect_t fullViewportPostDestRect = { x , y , w , h } ;
Vector2D destTexSize ( nSrcWidth , nSrcHeight ) ;
// When the viewport is not fullscreen, the UV-space size of a pixel changes
// (due to a stretchrect blit being used in UpdateScreenEffectTexture()), so
// we need to adjust the corner-pixel UVs sent to our drawrect call:
Vector2D uvScale ( ( nSrcWidth - ( nSrcWidth / ( float ) w ) ) / ( nSrcWidth - 1 ) ,
( nSrcHeight - ( nSrcHeight / ( float ) h ) ) / ( nSrcHeight - 1 ) ) ;
CenterScaleQuadUVs ( fullViewportPostSrcCorners , uvScale ) ;
CenterScaleQuadUVs ( fullViewportPostDestCorners , uvScale ) ;
Rect_t partialViewportPostDestRect = fullViewportPostDestRect ;
Vector4D partialViewportPostSrcCorners = fullViewportPostSrcCorners ;
if ( debug_postproc . GetInt ( ) = = 2 )
{
// Restrict the post effects to the centre quarter of the screen
// (we only use a portion of the bloom texture, so this *does* affect bloom texture UVs)
partialViewportPostDestRect . x + = 0.25f * fullViewportPostDestRect . width ;
partialViewportPostDestRect . y + = 0.25f * fullViewportPostDestRect . height ;
partialViewportPostDestRect . width - = 0.50f * fullViewportPostDestRect . width ;
partialViewportPostDestRect . height - = 0.50f * fullViewportPostDestRect . height ;
// This math interprets texel coords as being at corner pixel centers (*not* at corner vertices):
Vector2D uvScale ( 1.0f - ( ( w / 2 ) / ( float ) ( w - 1 ) ) ,
1.0f - ( ( h / 2 ) / ( float ) ( h - 1 ) ) ) ;
CenterScaleQuadUVs ( partialViewportPostSrcCorners , uvScale ) ;
}
// Temporary hack... Color correction was crashing on the first frame
// when run outside the debugger for some mods (DoD). This forces it to skip
// a frame, ensuring we don't get the weird texture crash we otherwise would.
// FIXME: This will be removed when the true cause is found [added: Main CL 144694]
static bool bFirstFrame = ! IsX360 ( ) ;
if ( ! bFirstFrame | | ! bPerformColCorrect )
{
if ( mat_postprocessing_combine . GetInt ( ) )
{
// Perform post-processing in one combined pass
IMaterial * post_mat = CEnginePostMaterialProxy : : SetupEnginePostMaterial ( fullViewportPostSrcCorners , fullViewportPostDestCorners , destTexSize , bPerformSoftwareAA , bPerformBloom , bPerformColCorrect , flAAStrength ) ;
pRenderContext - > DrawScreenSpaceRectangle ( post_mat ,
partialViewportPostDestRect . x , partialViewportPostDestRect . y ,
partialViewportPostDestRect . width , partialViewportPostDestRect . height ,
partialViewportPostSrcCorners . x , partialViewportPostSrcCorners . y ,
partialViewportPostSrcCorners . z , partialViewportPostSrcCorners . w ,
dest_rt1 - > GetActualWidth ( ) , dest_rt1 - > GetActualHeight ( ) ,
GetClientWorldEntity ( ) - > GetClientRenderable ( ) ,
mat_postprocess_x . GetInt ( ) , mat_postprocess_y . GetInt ( ) ) ;
}
else
{
// Perform post-processing in three separate passes
bool bFBUpdated = false ;
if ( bPerformSoftwareAA )
{
IMaterial * aa_mat = CEnginePostMaterialProxy : : SetupEnginePostMaterial ( fullViewportPostSrcCorners , fullViewportPostDestCorners , destTexSize , bPerformSoftwareAA , false , false , flAAStrength ) ;
pRenderContext - > DrawScreenSpaceRectangle ( aa_mat ,
partialViewportPostDestRect . x , partialViewportPostDestRect . y ,
partialViewportPostDestRect . width , partialViewportPostDestRect . height ,
partialViewportPostSrcCorners . x , partialViewportPostSrcCorners . y ,
partialViewportPostSrcCorners . z , partialViewportPostSrcCorners . w ,
dest_rt1 - > GetActualWidth ( ) , dest_rt1 - > GetActualHeight ( ) ,
GetClientWorldEntity ( ) - > GetClientRenderable ( ) ) ;
bFBUpdated = true ;
}
if ( bPerformBloom )
{
IMaterial * bloom_mat = CEnginePostMaterialProxy : : SetupEnginePostMaterial ( fullViewportPostSrcCorners , fullViewportPostDestCorners , destTexSize , false , bPerformBloom , false , flAAStrength ) ;
pRenderContext - > DrawScreenSpaceRectangle ( bloom_mat ,
partialViewportPostDestRect . x , partialViewportPostDestRect . y ,
partialViewportPostDestRect . width , partialViewportPostDestRect . height ,
partialViewportPostSrcCorners . x , partialViewportPostSrcCorners . y ,
partialViewportPostSrcCorners . z , partialViewportPostSrcCorners . w ,
dest_rt1 - > GetActualWidth ( ) , dest_rt1 - > GetActualHeight ( ) ,
GetClientWorldEntity ( ) - > GetClientRenderable ( ) ) ;
bFBUpdated = true ;
}
if ( bPerformColCorrect )
{
if ( bFBUpdated )
{
Rect_t actualRect ;
UpdateScreenEffectTexture ( 0 , x , y , w , h , false , & actualRect ) ;
}
IMaterial * colcorrect_mat = CEnginePostMaterialProxy : : SetupEnginePostMaterial ( fullViewportPostSrcCorners , fullViewportPostDestCorners , destTexSize , false , false , bPerformColCorrect , flAAStrength ) ;
pRenderContext - > DrawScreenSpaceRectangle ( colcorrect_mat ,
partialViewportPostDestRect . x , partialViewportPostDestRect . y ,
partialViewportPostDestRect . width , partialViewportPostDestRect . height ,
partialViewportPostSrcCorners . x , partialViewportPostSrcCorners . y ,
partialViewportPostSrcCorners . z , partialViewportPostSrcCorners . w ,
dest_rt1 - > GetActualWidth ( ) , dest_rt1 - > GetActualHeight ( ) ,
GetClientWorldEntity ( ) - > GetClientRenderable ( ) ) ;
bFBUpdated = true ;
}
}
}
bFirstFrame = false ;
}
if ( hdrType ! = HDR_TYPE_NONE )
{
DoPostBloomTonemapping ( pRenderContext , x , y , w , h , flAutoExposureMin , flAutoExposureMax ) ;
}
}
break ;
case HDR_TYPE_FLOAT :
{
int dest_width , dest_height ;
pRenderContext - > GetRenderTargetDimensions ( dest_width , dest_height ) ;
if ( mat_dynamic_tonemapping . GetInt ( ) | | mat_show_histogram . GetInt ( ) )
{
g_HDR_HistogramSystem . Update ( ) ;
// Warning("avg_lum=%f\n",g_HDR_HistogramSystem.GetTargetTonemapScalar());
if ( mat_dynamic_tonemapping . GetInt ( ) )
{
2011-04-28 01:30:09 -05:00
float avg_lum = MAX ( 0.0001 , g_HDR_HistogramSystem . GetTargetTonemapScalar ( ) ) ;
float scalevalue = MAX ( flAutoExposureMin ,
MIN ( flAutoExposureMax , 0.18 / avg_lum ) ) ;
2008-09-15 01:07:45 -05:00
pRenderContext - > SetGoalToneMappingScale ( scalevalue ) ;
mat_hdr_tonemapscale . SetValue ( scalevalue ) ;
}
}
IMaterial * pBloomMaterial ;
pBloomMaterial = materials - > FindMaterial ( " dev/floattoscreen_combine " , " " ) ;
IMaterialVar * pBloomAmountVar = pBloomMaterial - > FindVar ( " $bloomamount " , NULL ) ;
pBloomAmountVar - > SetFloatValue ( flBloomScale ) ;
if ( flBloomScale > 0.0 )
{
ApplyPostProcessingPasses ( HDRFinal_Float ) ;
}
else
{
ApplyPostProcessingPasses ( HDRFinal_Float_NoBloom ) ;
}
pRenderContext - > SetRenderTarget ( NULL ) ;
if ( mat_show_histogram . GetInt ( ) & & ( engine - > GetDXSupportLevel ( ) > = 90 ) )
g_HDR_HistogramSystem . DisplayHistogram ( ) ;
if ( mat_dynamic_tonemapping . GetInt ( ) )
{
2011-04-28 01:30:09 -05:00
float avg_lum = MAX ( 0.0001 , g_HDR_HistogramSystem . GetTargetTonemapScalar ( ) ) ;
float scalevalue = MAX ( flAutoExposureMin ,
MIN ( flAutoExposureMax , 0.023 / avg_lum ) ) ;
2008-09-15 01:07:45 -05:00
SetToneMapScale ( pRenderContext , scalevalue , flAutoExposureMin , flAutoExposureMax ) ;
}
pRenderContext - > SetRenderTarget ( NULL ) ;
break ;
}
}
# if defined( _X360 )
pRenderContext - > PopVertexShaderGPRAllocation ( ) ;
# endif
}
// Motion Blur Material Proxy =========================================================================================
static float g_vMotionBlurValues [ 4 ] = { 0.0f , 0.0f , 0.0f , 0.0f } ;
class CMotionBlurMaterialProxy : public CEntityMaterialProxy
{
public :
CMotionBlurMaterialProxy ( ) ;
virtual ~ CMotionBlurMaterialProxy ( ) ;
virtual bool Init ( IMaterial * pMaterial , KeyValues * pKeyValues ) ;
virtual void OnBind ( C_BaseEntity * pEntity ) ;
virtual IMaterial * GetMaterial ( ) ;
private :
IMaterialVar * m_pMaterialParam ;
} ;
CMotionBlurMaterialProxy : : CMotionBlurMaterialProxy ( )
{
m_pMaterialParam = NULL ;
}
CMotionBlurMaterialProxy : : ~ CMotionBlurMaterialProxy ( )
{
// Do nothing
}
bool CMotionBlurMaterialProxy : : Init ( IMaterial * pMaterial , KeyValues * pKeyValues )
{
bool bFoundVar = false ;
m_pMaterialParam = pMaterial - > FindVar ( " $MotionBlurInternal " , & bFoundVar , false ) ;
if ( bFoundVar = = false )
return false ;
return true ;
}
void CMotionBlurMaterialProxy : : OnBind ( C_BaseEntity * pEnt )
{
if ( m_pMaterialParam ! = NULL )
{
m_pMaterialParam - > SetVecValue ( g_vMotionBlurValues , 4 ) ;
}
}
IMaterial * CMotionBlurMaterialProxy : : GetMaterial ( )
{
if ( m_pMaterialParam = = NULL )
return NULL ;
return m_pMaterialParam - > GetOwningMaterial ( ) ;
}
EXPOSE_INTERFACE ( CMotionBlurMaterialProxy , IMaterialProxy , " MotionBlur " IMATERIAL_PROXY_INTERFACE_VERSION ) ;
//=====================================================================================================================
// Image-space Motion Blur ============================================================================================
//=====================================================================================================================
ConVar mat_motion_blur_enabled ( " mat_motion_blur_enabled " , " 1 " , FCVAR_ARCHIVE ) ;
ConVar mat_motion_blur_forward_enabled ( " mat_motion_blur_forward_enabled " , " 0 " ) ;
ConVar mat_motion_blur_falling_min ( " mat_motion_blur_falling_min " , " 10.0 " ) ;
ConVar mat_motion_blur_falling_max ( " mat_motion_blur_falling_max " , " 20.0 " ) ;
ConVar mat_motion_blur_falling_intensity ( " mat_motion_blur_falling_intensity " , " 1.0 " ) ;
//ConVar mat_motion_blur_roll_intensity( "mat_motion_blur_roll_intensity", "1.0" );
ConVar mat_motion_blur_rotation_intensity ( " mat_motion_blur_rotation_intensity " , " 1.0 " ) ;
ConVar mat_motion_blur_strength ( " mat_motion_blur_strength " , " 1.0 " ) ;
void DoImageSpaceMotionBlur ( const CViewSetup & view , int x , int y , int w , int h )
{
if ( ( ! mat_motion_blur_enabled . GetInt ( ) ) | | ( g_pMaterialSystemHardwareConfig - > GetDXSupportLevel ( ) < 90 ) )
{
return ;
}
//======================================================================================================//
// Get these convars here to make it easier to remove them later and to default each client differently //
//======================================================================================================//
float flMotionBlurRotationIntensity = mat_motion_blur_rotation_intensity . GetFloat ( ) * 0.15f ; // The default is to not blur past 15% of the range
float flMotionBlurRollIntensity = 0.3f ; // * mat_motion_blur_roll_intensity.GetFloat(); // The default is to not blur past 30% of the range
float flMotionBlurFallingIntensity = mat_motion_blur_falling_intensity . GetFloat ( ) ;
float flMotionBlurFallingMin = mat_motion_blur_falling_min . GetFloat ( ) ;
float flMotionBlurFallingMax = mat_motion_blur_falling_max . GetFloat ( ) ;
float flMotionBlurGlobalStrength = mat_motion_blur_strength . GetFloat ( ) ;
//===============================================================================//
// Set global g_vMotionBlurValues[4] values so material proxy can get the values //
//===============================================================================//
if ( true )
{
//=====================//
// Previous frame data //
//=====================//
static float s_flLastTimeUpdate = 0.0f ;
static float s_flPreviousPitch = 0.0f ;
static float s_flPreviousYaw = 0.0f ;
static float s_vPreviousPositon [ 3 ] = { 0.0f , 0.0f , 0.0f } ;
static matrix3x4_t s_mPreviousFrameBasisVectors ;
static float s_flNoRotationalMotionBlurUntil = 0.0f ;
//float vPreviousSideVec[3] = { s_mPreviousFrameBasisVectors[0][1], s_mPreviousFrameBasisVectors[1][1], s_mPreviousFrameBasisVectors[2][1] };
//float vPreviousForwardVec[3] = { s_mPreviousFrameBasisVectors[0][0], s_mPreviousFrameBasisVectors[1][0], s_mPreviousFrameBasisVectors[2][0] };
//float vPreviousUpVec[3] = { s_mPreviousFrameBasisVectors[0][2], s_mPreviousFrameBasisVectors[1][2], s_mPreviousFrameBasisVectors[2][2] };
float flTimeElapsed = gpGlobals - > realtime - s_flLastTimeUpdate ;
//===================================//
// Get current pitch & wrap to +-180 //
//===================================//
float flCurrentPitch = view . angles [ PITCH ] ;
while ( flCurrentPitch > 180.0f )
flCurrentPitch - = 360.0f ;
while ( flCurrentPitch < - 180.0f )
flCurrentPitch + = 360.0f ;
//=================================//
// Get current yaw & wrap to +-180 //
//=================================//
float flCurrentYaw = view . angles [ YAW ] ;
while ( flCurrentYaw > 180.0f )
flCurrentYaw - = 360.0f ;
while ( flCurrentYaw < - 180.0f )
flCurrentYaw + = 360.0f ;
//engine->Con_NPrintf( 0, "Blur Pitch: %6.2f Yaw: %6.2f", flCurrentPitch, flCurrentYaw );
//engine->Con_NPrintf( 1, "Blur FOV: %6.2f Aspect: %6.2f Ortho: %s", view.fov, view.m_flAspectRatio, view.m_bOrtho ? "Yes" : "No" );
//===========================//
// Get current basis vectors //
//===========================//
matrix3x4_t mCurrentBasisVectors ;
AngleMatrix ( view . angles , mCurrentBasisVectors ) ;
float vCurrentSideVec [ 3 ] = { mCurrentBasisVectors [ 0 ] [ 1 ] , mCurrentBasisVectors [ 1 ] [ 1 ] , mCurrentBasisVectors [ 2 ] [ 1 ] } ;
float vCurrentForwardVec [ 3 ] = { mCurrentBasisVectors [ 0 ] [ 0 ] , mCurrentBasisVectors [ 1 ] [ 0 ] , mCurrentBasisVectors [ 2 ] [ 0 ] } ;
//float vCurrentUpVec[3] = { mCurrentBasisVectors[0][2], mCurrentBasisVectors[1][2], mCurrentBasisVectors[2][2] };
//======================//
// Get current position //
//======================//
float vCurrentPosition [ 3 ] = { view . origin . x , view . origin . y , view . origin . z } ;
//===============================================================//
// Evaluate change in position to determine if we need to update //
//===============================================================//
float vPositionChange [ 3 ] = { 0.0f , 0.0f , 0.0f } ;
VectorSubtract ( s_vPreviousPositon , vCurrentPosition , vPositionChange ) ;
if ( ( VectorLength ( vPositionChange ) > 30.0f ) & & ( flTimeElapsed > = 0.5f ) )
{
//=======================================================//
// If we moved a far distance in one frame and more than //
// half a second elapsed, disable motion blur this frame //
//=======================================================//
//engine->Con_NPrintf( 8, " Pos change && time > 0.5 seconds %f ", gpGlobals->realtime );
g_vMotionBlurValues [ 0 ] = 0.0f ;
g_vMotionBlurValues [ 1 ] = 0.0f ;
g_vMotionBlurValues [ 2 ] = 0.0f ;
g_vMotionBlurValues [ 3 ] = 0.0f ;
}
else if ( flTimeElapsed > ( 1.0f / 15.0f ) )
{
//==========================================//
// If slower than 15 fps, don't motion blur //
//==========================================//
g_vMotionBlurValues [ 0 ] = 0.0f ;
g_vMotionBlurValues [ 1 ] = 0.0f ;
g_vMotionBlurValues [ 2 ] = 0.0f ;
g_vMotionBlurValues [ 3 ] = 0.0f ;
}
else if ( VectorLength ( vPositionChange ) > 50.0f )
{
//================================================================================//
// We moved a far distance in a frame, use the same motion blur as last frame //
// because I think we just went through a portal (should we ifdef this behavior?) //
//================================================================================//
//engine->Con_NPrintf( 8, " Position changed %f units @ %.2f time ", VectorLength( vPositionChange ), gpGlobals->realtime );
s_flNoRotationalMotionBlurUntil = gpGlobals - > realtime + 1.0f ; // Wait a second until the portal craziness calms down
}
else
{
//====================//
// Normal update path //
//====================//
// Compute horizontal and vertical fov
float flHorizontalFov = view . fov ;
float flVerticalFov = ( view . m_flAspectRatio < = 0.0f ) ? ( view . fov ) : ( view . fov / view . m_flAspectRatio ) ;
//engine->Con_NPrintf( 2, "Horizontal Fov: %6.2f Vertical Fov: %6.2f", flHorizontalFov, flVerticalFov );
//=====================//
// Forward motion blur //
//=====================//
float flViewDotMotion = DotProduct ( vCurrentForwardVec , vPositionChange ) ;
if ( mat_motion_blur_forward_enabled . GetBool ( ) ) // Want forward and falling
g_vMotionBlurValues [ 2 ] = flViewDotMotion ;
else // Falling only
g_vMotionBlurValues [ 2 ] = flViewDotMotion * fabs ( vCurrentForwardVec [ 2 ] ) ; // Only want this if we're looking up or down;
//====================================//
// Yaw (Compensate for circle strafe) //
//====================================//
float flSideDotMotion = DotProduct ( vCurrentSideVec , vPositionChange ) ;
float flYawDiffOriginal = s_flPreviousYaw - flCurrentYaw ;
if ( ( ( s_flPreviousYaw - flCurrentYaw > 180.0f ) | | ( s_flPreviousYaw - flCurrentYaw < - 180.0f ) ) & &
( ( s_flPreviousYaw + flCurrentYaw > - 180.0f ) & & ( s_flPreviousYaw + flCurrentYaw < 180.0f ) ) )
flYawDiffOriginal = s_flPreviousYaw + flCurrentYaw ;
float flYawDiffAdjusted = flYawDiffOriginal + ( flSideDotMotion / 3.0f ) ; // Yes, 3.0 is a magic number, sue me
// Make sure the adjustment only lessens the effect, not magnify it or reverse it
if ( flYawDiffOriginal < 0.0f )
flYawDiffAdjusted = clamp ( flYawDiffAdjusted , flYawDiffOriginal , 0.0f ) ;
else
flYawDiffAdjusted = clamp ( flYawDiffAdjusted , 0.0f , flYawDiffOriginal ) ;
// Use pitch to dampen yaw
float flUndampenedYaw = flYawDiffAdjusted / flHorizontalFov ;
g_vMotionBlurValues [ 0 ] = flUndampenedYaw * ( 1.0f - ( fabs ( flCurrentPitch ) / 90.0f ) ) ; // Dampen horizontal yaw blur based on pitch
//engine->Con_NPrintf( 4, "flSideDotMotion: %6.2f yaw diff: %6.2f ( %6.2f, %6.2f )", flSideDotMotion, ( s_flPreviousYaw - flCurrentYaw ), flYawDiffOriginal, flYawDiffAdjusted );
//=======================================//
// Pitch (Compensate for forward motion) //
//=======================================//
float flPitchCompensateMask = 1.0f - ( ( 1.0f - fabs ( vCurrentForwardVec [ 2 ] ) ) * ( 1.0f - fabs ( vCurrentForwardVec [ 2 ] ) ) ) ;
float flPitchDiffOriginal = s_flPreviousPitch - flCurrentPitch ;
float flPitchDiffAdjusted = flPitchDiffOriginal ;
if ( flCurrentPitch > 0.0f )
flPitchDiffAdjusted = flPitchDiffOriginal - ( ( flViewDotMotion / 2.0f ) * flPitchCompensateMask ) ; // Yes, 2.0 is a magic number, sue me
else
flPitchDiffAdjusted = flPitchDiffOriginal + ( ( flViewDotMotion / 2.0f ) * flPitchCompensateMask ) ; // Yes, 2.0 is a magic number, sue me
// Make sure the adjustment only lessens the effect, not magnify it or reverse it
if ( flPitchDiffOriginal < 0.0f )
flPitchDiffAdjusted = clamp ( flPitchDiffAdjusted , flPitchDiffOriginal , 0.0f ) ;
else
flPitchDiffAdjusted = clamp ( flPitchDiffAdjusted , 0.0f , flPitchDiffOriginal ) ;
g_vMotionBlurValues [ 1 ] = flPitchDiffAdjusted / flVerticalFov ;
//engine->Con_NPrintf( 5, "flViewDotMotion %6.2f, flPitchCompensateMask %6.2f, flPitchDiffOriginal %6.2f, flPitchDiffAdjusted %6.2f, g_vMotionBlurValues[1] %6.2f", flViewDotMotion, flPitchCompensateMask, flPitchDiffOriginal, flPitchDiffAdjusted, g_vMotionBlurValues[1]);
//========================================================//
// Roll (Enabled when we're looking down and yaw changes) //
//========================================================//
g_vMotionBlurValues [ 3 ] = flUndampenedYaw ; // Roll starts out as undampened yaw intensity and is then scaled by pitch
g_vMotionBlurValues [ 3 ] * = ( fabs ( flCurrentPitch ) / 90.0f ) * ( fabs ( flCurrentPitch ) / 90.0f ) * ( fabs ( flCurrentPitch ) / 90.0f ) ; // Dampen roll based on pitch^3
//engine->Con_NPrintf( 4, "[2] before scale and bias: %6.2f", g_vMotionBlurValues[2] );
//engine->Con_NPrintf( 5, "[3] before scale and bias: %6.2f", g_vMotionBlurValues[3] );
//==============================================================//
// Time-adjust falling effect until we can do something smarter //
//==============================================================//
if ( flTimeElapsed > 0.0f )
g_vMotionBlurValues [ 2 ] / = flTimeElapsed * 30.0f ; // 1/30th of a second?
else
g_vMotionBlurValues [ 2 ] = 0.0f ;
// Scale and bias values after time adjustment
g_vMotionBlurValues [ 2 ] = clamp ( ( fabs ( g_vMotionBlurValues [ 2 ] ) - flMotionBlurFallingMin ) / ( flMotionBlurFallingMax - flMotionBlurFallingMin ) , 0.0f , 1.0f ) * ( g_vMotionBlurValues [ 2 ] > = 0.0f ? 1.0f : - 1.0f ) ;
g_vMotionBlurValues [ 2 ] / = 30.0f ; // To counter-adjust for time adjustment above
//=================//
// Apply intensity //
//=================//
g_vMotionBlurValues [ 0 ] * = flMotionBlurRotationIntensity * flMotionBlurGlobalStrength ;
g_vMotionBlurValues [ 1 ] * = flMotionBlurRotationIntensity * flMotionBlurGlobalStrength ;
g_vMotionBlurValues [ 2 ] * = flMotionBlurFallingIntensity * flMotionBlurGlobalStrength ;
g_vMotionBlurValues [ 3 ] * = flMotionBlurRollIntensity * flMotionBlurGlobalStrength ;
//===============================================================//
// Dampen motion blur from 100%-0% as fps drops from 50fps-30fps //
//===============================================================//
if ( ! IsX360 ( ) ) // I'm not doing this on the 360 yet since I can't test it
{
float flSlowFps = 30.0f ;
float flFastFps = 50.0f ;
float flCurrentFps = ( flTimeElapsed > 0.0f ) ? ( 1.0f / flTimeElapsed ) : 0.0f ;
float flDampenFactor = clamp ( ( ( flCurrentFps - flSlowFps ) / ( flFastFps - flSlowFps ) ) , 0.0f , 1.0f ) ;
//engine->Con_NPrintf( 4, "gpGlobals->realtime %.2f gpGlobals->curtime %.2f", gpGlobals->realtime, gpGlobals->curtime );
//engine->Con_NPrintf( 5, "flCurrentFps %.2f", flCurrentFps );
//engine->Con_NPrintf( 7, "flTimeElapsed %.2f", flTimeElapsed );
g_vMotionBlurValues [ 0 ] * = flDampenFactor ;
g_vMotionBlurValues [ 1 ] * = flDampenFactor ;
g_vMotionBlurValues [ 2 ] * = flDampenFactor ;
g_vMotionBlurValues [ 3 ] * = flDampenFactor ;
//engine->Con_NPrintf( 6, "Dampen: %.2f", flDampenFactor );
}
//engine->Con_NPrintf( 6, "Final values: { %6.2f%%, %6.2f%%, %6.2f%%, %6.2f%% }", g_vMotionBlurValues[0]*100.0f, g_vMotionBlurValues[1]*100.0f, g_vMotionBlurValues[2]*100.0f, g_vMotionBlurValues[3]*100.0f );
}
//============================================//
// Zero out blur if still in that time window //
//============================================//
if ( gpGlobals - > realtime < s_flNoRotationalMotionBlurUntil )
{
//engine->Con_NPrintf( 9, " No Rotation @ %f ", gpGlobals->realtime );
// Zero out rotational blur but leave forward/falling blur alone
g_vMotionBlurValues [ 0 ] = 0.0f ; // X
g_vMotionBlurValues [ 1 ] = 0.0f ; // Y
g_vMotionBlurValues [ 3 ] = 0.0f ; // Roll
}
else
{
s_flNoRotationalMotionBlurUntil = 0.0f ;
}
//====================================//
// Store current frame for next frame //
//====================================//
VectorCopy ( vCurrentPosition , s_vPreviousPositon ) ;
s_mPreviousFrameBasisVectors = mCurrentBasisVectors ;
s_flPreviousPitch = flCurrentPitch ;
s_flPreviousYaw = flCurrentYaw ;
s_flLastTimeUpdate = gpGlobals - > realtime ;
}
//=============================================================================================//
// Render quad and let material proxy pick up the g_vMotionBlurValues[4] values just set above //
//=============================================================================================//
if ( true )
{
CMatRenderContextPtr pRenderContext ( materials ) ;
//pRenderContext->PushRenderTargetAndViewport();
ITexture * pSrc = materials - > FindTexture ( " _rt_FullFrameFB " , TEXTURE_GROUP_RENDER_TARGET ) ;
int nSrcWidth = pSrc - > GetActualWidth ( ) ;
int nSrcHeight = pSrc - > GetActualHeight ( ) ;
int dest_width , dest_height , nDummy ;
pRenderContext - > GetViewport ( nDummy , nDummy , dest_width , dest_height ) ;
if ( g_pMaterialSystemHardwareConfig - > GetHDRType ( ) ! = HDR_TYPE_FLOAT )
{
UpdateScreenEffectTexture ( 0 , x , y , w , h , true ) ; // Do we need to check if we already did this?
}
// Get material pointer
IMaterial * pMatMotionBlur = materials - > FindMaterial ( " dev/motion_blur " , TEXTURE_GROUP_OTHER , true ) ;
//SetRenderTargetAndViewPort( dest_rt0 );
//pRenderContext->PopRenderTargetAndViewport();
if ( pMatMotionBlur ! = NULL )
{
pRenderContext - > DrawScreenSpaceRectangle (
pMatMotionBlur ,
0 , 0 , dest_width , dest_height ,
0 , 0 , nSrcWidth - 1 , nSrcHeight - 1 ,
nSrcWidth , nSrcHeight , GetClientWorldEntity ( ) - > GetClientRenderable ( ) ) ;
}
}
}