2014-07-08 03:11:14 -04:00
# include "EmberCLPch.h"
# include "FinalAccumOpenCLKernelCreator.h"
namespace EmberCLns
{
/// <summary>
/// Constructor that creates all kernel strings.
/// The caller will access these strings through the accessor functions.
/// </summary>
2015-08-10 23:10:23 -04:00
FinalAccumOpenCLKernelCreator : : FinalAccumOpenCLKernelCreator ( bool doublePrecision )
2014-07-08 03:11:14 -04:00
{
2015-08-10 23:10:23 -04:00
m_DoublePrecision = doublePrecision ;
2014-07-08 03:11:14 -04:00
m_GammaCorrectionWithAlphaCalcEntryPoint = " GammaCorrectionWithAlphaCalcKernel " ;
m_GammaCorrectionWithoutAlphaCalcEntryPoint = " GammaCorrectionWithoutAlphaCalcKernel " ;
m_GammaCorrectionWithAlphaCalcKernel = CreateGammaCorrectionKernelString ( true ) ;
m_GammaCorrectionWithoutAlphaCalcKernel = CreateGammaCorrectionKernelString ( false ) ;
m_FinalAccumEarlyClipEntryPoint = " FinalAccumEarlyClipKernel " ;
m_FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumEntryPoint = " FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumKernel " ;
m_FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumEntryPoint = " FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumKernel " ;
m_FinalAccumEarlyClipKernel = CreateFinalAccumKernelString ( true , false , false ) ;
m_FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumKernel = CreateFinalAccumKernelString ( true , true , true ) ;
m_FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumKernel = CreateFinalAccumKernelString ( true , false , true ) ;
m_FinalAccumLateClipEntryPoint = " FinalAccumLateClipKernel " ;
m_FinalAccumLateClipWithAlphaCalcWithAlphaAccumEntryPoint = " FinalAccumLateClipWithAlphaCalcWithAlphaAccumKernel " ;
m_FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumEntryPoint = " FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumKernel " ;
m_FinalAccumLateClipKernel = CreateFinalAccumKernelString ( false , false , false ) ;
m_FinalAccumLateClipWithAlphaCalcWithAlphaAccumKernel = CreateFinalAccumKernelString ( false , true , true ) ;
m_FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumKernel = CreateFinalAccumKernelString ( false , false , true ) ;
}
/// <summary>
/// Kernel source and entry point properties, getters only.
/// </summary>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : GammaCorrectionWithAlphaCalcKernel ( ) { return m_GammaCorrectionWithAlphaCalcKernel ; }
string FinalAccumOpenCLKernelCreator : : GammaCorrectionWithAlphaCalcEntryPoint ( ) { return m_GammaCorrectionWithAlphaCalcEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : GammaCorrectionWithoutAlphaCalcKernel ( ) { return m_GammaCorrectionWithoutAlphaCalcKernel ; }
string FinalAccumOpenCLKernelCreator : : GammaCorrectionWithoutAlphaCalcEntryPoint ( ) { return m_GammaCorrectionWithoutAlphaCalcEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipKernel ( ) { return m_FinalAccumEarlyClipKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipEntryPoint ( ) { return m_FinalAccumEarlyClipEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumKernel ( ) { return m_FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumEntryPoint ( ) { return m_FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumKernel ( ) { return m_FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumEntryPoint ( ) { return m_FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipKernel ( ) { return m_FinalAccumLateClipKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipEntryPoint ( ) { return m_FinalAccumLateClipEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipWithAlphaCalcWithAlphaAccumKernel ( ) { return m_FinalAccumLateClipWithAlphaCalcWithAlphaAccumKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipWithAlphaCalcWithAlphaAccumEntryPoint ( ) { return m_FinalAccumLateClipWithAlphaCalcWithAlphaAccumEntryPoint ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumKernel ( ) { return m_FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumKernel ; }
string FinalAccumOpenCLKernelCreator : : FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumEntryPoint ( ) { return m_FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumEntryPoint ; }
2014-07-08 03:11:14 -04:00
/// <summary>
/// Get the gamma correction entry point.
/// </summary>
/// <param name="channels">The number of channels used, 3 or 4.</param>
/// <param name="transparency">True if channels equals 4 and using transparency, else false.</param>
/// <returns>The name of the gamma correction entry point kernel function</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : GammaCorrectionEntryPoint ( size_t channels , bool transparency )
2014-07-08 03:11:14 -04:00
{
2014-07-26 15:03:51 -04:00
bool alphaCalc = ( ( channels > 3 ) & & transparency ) ;
2014-07-08 03:11:14 -04:00
return alphaCalc ? m_GammaCorrectionWithAlphaCalcEntryPoint : m_GammaCorrectionWithoutAlphaCalcEntryPoint ;
}
/// <summary>
/// Get the gamma correction kernel string.
/// </summary>
/// <param name="channels">The number of channels used, 3 or 4.</param>
/// <param name="transparency">True if channels equals 4 and using transparency, else false.</param>
/// <returns>The gamma correction kernel string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : GammaCorrectionKernel ( size_t channels , bool transparency )
2014-07-08 03:11:14 -04:00
{
2014-07-26 15:03:51 -04:00
bool alphaCalc = ( ( channels > 3 ) & & transparency ) ;
2014-07-08 03:11:14 -04:00
return alphaCalc ? m_GammaCorrectionWithAlphaCalcKernel : m_GammaCorrectionWithoutAlphaCalcKernel ;
}
/// <summary>
/// Get the final accumulation entry point.
/// </summary>
/// <param name="earlyClip">True if early clip is desired, else false.</param>
/// <param name="channels">The number of channels used, 3 or 4.</param>
/// <param name="transparency">True if channels equals 4 and using transparency, else false.</param>
/// <param name="alphaBase">Storage for the alpha base value used in the kernel. 0 if transparency is true, else 255.</param>
/// <param name="alphaScale">Storage for the alpha scale value used in the kernel. 255 if transparency is true, else 0.</param>
/// <returns>The name of the final accumulation entry point kernel function</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : FinalAccumEntryPoint ( bool earlyClip , size_t channels , bool transparency , double & alphaBase , double & alphaScale )
2014-07-08 03:11:14 -04:00
{
2014-07-26 15:03:51 -04:00
bool alphaCalc = ( ( channels > 3 ) & & transparency ) ;
2014-07-08 03:11:14 -04:00
bool alphaAccum = channels > 3 ;
if ( alphaAccum )
{
2015-08-10 23:10:23 -04:00
alphaBase = transparency ? 0 : 255 ; //See the table below.
alphaScale = transparency ? 255 : 0 ;
2014-07-08 03:11:14 -04:00
}
if ( earlyClip )
{
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
return FinalAccumEarlyClipEntryPoint ( ) ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
return FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumEntryPoint ( ) ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
return FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumEntryPoint ( ) ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
else
{
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
return FinalAccumLateClipEntryPoint ( ) ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
return FinalAccumLateClipWithAlphaCalcWithAlphaAccumEntryPoint ( ) ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
return FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumEntryPoint ( ) ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
}
/// <summary>
/// Get the final accumulation kernel string.
/// </summary>
/// <param name="earlyClip">True if early clip is desired, else false.</param>
/// <param name="channels">The number of channels used, 3 or 4.</param>
/// <param name="transparency">True if channels equals 4 and using transparency, else false.</param>
/// <returns>The final accumulation kernel string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : FinalAccumKernel ( bool earlyClip , size_t channels , bool transparency )
2014-07-08 03:11:14 -04:00
{
bool alphaCalc = ( channels > 3 & & transparency ) ;
bool alphaAccum = channels > 3 ;
if ( earlyClip )
{
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
return FinalAccumEarlyClipKernel ( ) ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
return FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumKernel ( ) ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
return FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumKernel ( ) ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
else
{
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
return FinalAccumLateClipKernel ( ) ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
return FinalAccumLateClipWithAlphaCalcWithAlphaAccumKernel ( ) ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
return FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumKernel ( ) ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
}
/// <summary>
/// Wrapper around CreateFinalAccumKernelString().
/// </summary>
/// <param name="earlyClip">True if early clip is desired, else false.</param>
/// <param name="channels">The number of channels used, 3 or 4.</param>
/// <param name="transparency">True if channels equals 4 and using transparency, else false.</param>
/// <returns>The final accumulation kernel string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : CreateFinalAccumKernelString ( bool earlyClip , size_t channels , bool transparency )
2014-07-08 03:11:14 -04:00
{
return CreateFinalAccumKernelString ( earlyClip , ( channels > 3 & & transparency ) , channels > 3 ) ;
}
/// <summary>
/// Create the final accumulation kernel string
/// </summary>
/// <param name="earlyClip">True if early clip is desired, else false.</param>
/// <param name="alphaCalc">True if channels equals 4 and transparency is desired, else false.</param>
/// <param name="alphaAccum">True if channels equals 4</param>
/// <returns>The final accumulation kernel string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : CreateFinalAccumKernelString ( bool earlyClip , bool alphaCalc , bool alphaAccum )
2014-07-08 03:11:14 -04:00
{
ostringstream os ;
string channels = alphaAccum ? " 4 " : " 3 " ;
os < <
2015-08-10 23:10:23 -04:00
ConstantDefinesString ( m_DoublePrecision ) < <
2014-07-08 03:11:14 -04:00
ClampRealFunctionString < <
UnionCLStructString < <
RgbToHsvFunctionString < <
HsvToRgbFunctionString < <
CalcAlphaFunctionString < <
2015-03-21 18:27:37 -04:00
CurveAdjustFunctionString < <
2014-07-08 03:11:14 -04:00
SpatialFilterCLStructString ;
if ( earlyClip )
{
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
os < < " __kernel void " < < m_FinalAccumEarlyClipEntryPoint < < " ( \n " ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
os < < " __kernel void " < < m_FinalAccumEarlyClipWithAlphaCalcWithAlphaAccumEntryPoint < < " ( \n " ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
os < < " __kernel void " < < m_FinalAccumEarlyClipWithoutAlphaCalcWithAlphaAccumEntryPoint < < " ( \n " ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
else
{
os < <
CreateCalcNewRgbFunctionString ( false ) < <
CreateGammaCorrectionFunctionString ( false , alphaCalc , alphaAccum , true ) ;
if ( ! alphaCalc & & ! alphaAccum ) //Rgb output, the most common case.
os < < " __kernel void " < < m_FinalAccumLateClipEntryPoint < < " ( \n " ;
else if ( alphaCalc & & alphaAccum ) //Rgba output and Transparency.
os < < " __kernel void " < < m_FinalAccumLateClipWithAlphaCalcWithAlphaAccumEntryPoint < < " ( \n " ;
else if ( ! alphaCalc & & alphaAccum ) //Rgba output and !Transparency.
os < < " __kernel void " < < m_FinalAccumLateClipWithoutAlphaCalcWithAlphaAccumEntryPoint < < " ( \n " ;
else
return " " ; //Cannot have alphaCalc and !alphaAccum, it makes no sense.
}
os < <
2015-08-10 23:10:23 -04:00
" const __global real4reals_bucket* accumulator, \n "
2014-07-08 03:11:14 -04:00
" __write_only image2d_t pixels, \n "
" __constant SpatialFilterCL* spatialFilter, \n "
2015-08-10 23:10:23 -04:00
" __constant real_bucket_t* filterCoefs, \n "
" __constant real4reals_bucket* csa, \n "
2015-03-21 18:27:37 -04:00
" const uint doCurves, \n "
2015-08-10 23:10:23 -04:00
" const real_bucket_t alphaBase, \n "
" const real_bucket_t alphaScale \n "
2014-07-08 03:11:14 -04:00
" \t ) \n "
" { \n "
" \n "
" if ((GLOBAL_ID_Y >= spatialFilter->m_FinalRasH) || (GLOBAL_ID_X >= spatialFilter->m_FinalRasW)) \n "
" return; \n "
" \n "
2014-12-06 00:05:09 -05:00
" uint accumX = spatialFilter->m_DensityFilterOffset + (GLOBAL_ID_X * spatialFilter->m_Supersample); \n "
" uint accumY = spatialFilter->m_DensityFilterOffset + (GLOBAL_ID_Y * spatialFilter->m_Supersample); \n "
2014-07-08 03:11:14 -04:00
" int2 finalCoord; \n "
" finalCoord.x = GLOBAL_ID_X; \n "
2014-07-26 15:03:51 -04:00
" finalCoord.y = (int)((spatialFilter->m_YAxisUp == 1) ? ((spatialFilter->m_FinalRasH - GLOBAL_ID_Y) - 1) : GLOBAL_ID_Y); \n "
2014-07-08 03:11:14 -04:00
" float4floats finalColor; \n "
" int ii, jj; \n "
2014-12-06 00:05:09 -05:00
" uint filterKRowIndex; \n "
2015-08-10 23:10:23 -04:00
" const __global real4reals_bucket* accumBucket; \n "
" real4reals_bucket newBucket; \n "
2014-07-08 03:11:14 -04:00
" newBucket.m_Real4 = 0; \n "
" \n "
" for (jj = 0; jj < spatialFilter->m_FilterWidth; jj++) \n "
" { \n "
" filterKRowIndex = jj * spatialFilter->m_FilterWidth; \n "
" \n "
" for (ii = 0; ii < spatialFilter->m_FilterWidth; ii++) \n "
" { \n "
2015-08-10 23:10:23 -04:00
" real_bucket_t k = filterCoefs[ii + filterKRowIndex]; \n "
2014-07-08 03:11:14 -04:00
" \n "
" accumBucket = accumulator + (accumX + ii) + ((accumY + jj) * spatialFilter->m_SuperRasW); \n "
" newBucket.m_Real4 += (k * accumBucket->m_Real4); \n "
" } \n "
" } \n "
" \n " ;
//Not supporting 2 bytes per channel on the GPU. If the user wants it, run on the CPU.
2015-03-21 18:27:37 -04:00
if ( earlyClip ) //If early clip, simply assign values directly to the temp float4 since they've been gamma corrected already, then write it straight to the output image below.
2014-07-08 03:11:14 -04:00
{
os < <
" finalColor.m_Float4.x = (float)newBucket.m_Real4.x; \n " //CPU side clamps, skip here because write_imagef() does the clamping for us.
" finalColor.m_Float4.y = (float)newBucket.m_Real4.y; \n "
" finalColor.m_Float4.z = (float)newBucket.m_Real4.z; \n " ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
if ( alphaAccum )
{
if ( alphaCalc )
os < < " finalColor.m_Float4.w = (float)newBucket.m_Real4.w * 255.0f; \n " ;
else
os < < " finalColor.m_Float4.w = 255; \n " ;
}
}
else
{
//Late clip, so must gamma correct from the temp new bucket to temp float4.
2015-08-10 23:10:23 -04:00
if ( m_DoublePrecision )
2014-07-08 03:11:14 -04:00
{
os < <
2015-08-10 23:10:23 -04:00
" real4reals_bucket realFinal; \n "
2014-07-08 03:11:14 -04:00
" \n "
" GammaCorrectionFloats(&newBucket, &(spatialFilter->m_Background[0]), spatialFilter->m_Gamma, spatialFilter->m_LinRange, spatialFilter->m_Vibrancy, spatialFilter->m_HighlightPower, alphaBase, alphaScale, &(realFinal.m_Reals[0])); \n "
" finalColor.m_Float4.x = (float)realFinal.m_Real4.x; \n "
" finalColor.m_Float4.y = (float)realFinal.m_Real4.y; \n "
" finalColor.m_Float4.z = (float)realFinal.m_Real4.z; \n "
" finalColor.m_Float4.w = (float)realFinal.m_Real4.w; \n "
;
}
else
{
os < <
" GammaCorrectionFloats(&newBucket, &(spatialFilter->m_Background[0]), spatialFilter->m_Gamma, spatialFilter->m_LinRange, spatialFilter->m_Vibrancy, spatialFilter->m_HighlightPower, alphaBase, alphaScale, &(finalColor.m_Floats[0])); \n " ;
}
}
os < <
2015-03-21 18:27:37 -04:00
" \n "
" if (doCurves) \n "
" { \n "
" CurveAdjust(csa, &(finalColor.m_Floats[0]), 1); \n "
" CurveAdjust(csa, &(finalColor.m_Floats[1]), 2); \n "
" CurveAdjust(csa, &(finalColor.m_Floats[2]), 3); \n "
" } \n "
" \n "
2014-07-08 03:11:14 -04:00
" finalColor.m_Float4 /= 255.0f; \n "
" write_imagef(pixels, finalCoord, finalColor.m_Float4); \n " //Use write_imagef instead of write_imageui because only the former works when sharing with an OpenGL texture.
" barrier(CLK_GLOBAL_MEM_FENCE); \n " //Required, or else page tearing will occur during interactive rendering.
" } \n "
;
return os . str ( ) ;
}
/// <summary>
/// Creates the gamma correction function string.
/// This is not a full kernel, just a function that is used in the kernels.
/// </summary>
/// <param name="globalBucket">True if writing to a global buffer (early clip), else false (late clip).</param>
/// <param name="alphaCalc">True if channels equals 4 and transparency is desired, else false.</param>
/// <param name="alphaAccum">True if channels equals 4</param>
/// <param name="finalOut">True if writing to global buffer (late clip), else false (early clip).</param>
/// <returns>The gamma correction function string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : CreateGammaCorrectionFunctionString ( bool globalBucket , bool alphaCalc , bool alphaAccum , bool finalOut )
2014-07-08 03:11:14 -04:00
{
ostringstream os ;
string dataType ;
string unionMember ;
2015-08-10 23:10:23 -04:00
dataType = " real_bucket_t " ;
2014-07-08 03:11:14 -04:00
//Use real_t for all cases, early clip and final accum.
2015-08-10 23:10:23 -04:00
os < < " void GammaCorrectionFloats( " < < ( globalBucket ? " __global " : " " ) < < " real4reals_bucket* bucket, __constant real_bucket_t* background, real_bucket_t g, real_bucket_t linRange, real_bucket_t vibrancy, real_bucket_t highlightPower, real_bucket_t alphaBase, real_bucket_t alphaScale, " < < ( finalOut ? " " : " __global " ) < < " real_bucket_t* correctedChannels) \n " ;
2014-07-08 03:11:14 -04:00
os
< < " { \n "
2015-08-10 23:10:23 -04:00
< < " real_bucket_t alpha, ls, tmp, a; \n "
< < " real4reals_bucket newRgb; \n "
2014-07-08 03:11:14 -04:00
< < " \n "
< < " if (bucket->m_Reals[3] <= 0) \n "
< < " { \n "
< < " alpha = 0; \n "
< < " ls = 0; \n "
< < " } \n "
< < " else \n "
< < " { \n "
< < " tmp = bucket->m_Reals[3]; \n "
< < " alpha = CalcAlpha(tmp, g, linRange); \n "
< < " ls = vibrancy * 256.0 * alpha / tmp; \n "
2015-08-10 23:10:23 -04:00
< < " alpha = clamp(alpha, (real_bucket_t)0.0, (real_bucket_t)1.0); \n "
2014-07-08 03:11:14 -04:00
< < " } \n "
< < " \n "
< < " CalcNewRgb(bucket, ls, highlightPower, &newRgb); \n "
< < " \n "
2014-12-06 00:05:09 -05:00
< < " for (uint rgbi = 0; rgbi < 3; rgbi++) \n "
2014-07-08 03:11:14 -04:00
< < " { \n "
< < " a = newRgb.m_Reals[rgbi] + ((1.0 - vibrancy) * 256.0 * pow(bucket->m_Reals[rgbi], g)); \n "
< < " \n " ;
if ( ! alphaCalc )
{
os < <
" a += ((1.0 - alpha) * background[rgbi]); \n " ;
}
else
{
os
< < " if (alpha > 0) \n "
< < " a /= alpha; \n "
< < " else \n "
< < " a = 0; \n " ;
}
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
os < <
" \n "
2015-08-10 23:10:23 -04:00
" correctedChannels[rgbi] = ( " < < dataType < < " )clamp(a, (real_bucket_t)0.0, (real_bucket_t)255.0); \n "
2014-07-08 03:11:14 -04:00
" } \n "
" \n " ;
2014-12-05 21:30:46 -05:00
2014-07-08 03:11:14 -04:00
//The CPU code has 3 cases for assigning alpha:
//[3] = alpha.//Early clip.
//[3] = alpha * 255.//Final Rgba with transparency.
//[3] = 255.//Final Rgba without transparency.
//Putting conditionals in GPU code is to be avoided. So do base + alpha * scale which will
//work for all 3 cases without using a conditional, which should be faster on a GPU. This gives:
//Base = 0, scale = 1. [3] = (0 + (alpha * 1)). [3] = alpha.
//Base = 0, scale = 255. [3] = (0 + (alpha * 255)). [3] = alpha * 255.
//Base = 255, scale = 0. [3] = (255 + (alpha * 0)). [3] = 255.
if ( alphaAccum )
{
os
< < " correctedChannels[3] = ( " < < dataType < < " )(alphaBase + (alpha * alphaScale)); \n " ;
}
os < <
" } \n "
" \n " ;
return os . str ( ) ;
}
/// <summary>
/// OpenCL equivalent of Palette::CalcNewRgb().
/// </summary>
/// <param name="globalBucket">True if writing the corrected value to a global buffer (early clip), else false (late clip).</param>
/// <returns>The CalcNewRgb function string</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : CreateCalcNewRgbFunctionString ( bool globalBucket )
2014-07-08 03:11:14 -04:00
{
ostringstream os ;
os < <
2015-08-10 23:10:23 -04:00
" static void CalcNewRgb( " < < ( globalBucket ? " __global " : " " ) < < " real4reals_bucket* oldRgb, real_bucket_t ls, real_bucket_t highPow, real4reals_bucket* newRgb) \n "
2014-07-08 03:11:14 -04:00
" { \n "
" int rgbi; \n "
2015-08-10 23:10:23 -04:00
" real_bucket_t newls, lsratio; \n "
" real4reals_bucket newHsv; \n "
" real_bucket_t maxa, maxc; \n "
" real_bucket_t adjhlp; \n "
2014-07-08 03:11:14 -04:00
" \n "
" if (ls == 0 || (oldRgb->m_Real4.x == 0 && oldRgb->m_Real4.y == 0 && oldRgb->m_Real4.z == 0)) \n " //Can't do a vector compare to zero.
" { \n "
" newRgb->m_Real4 = 0; \n "
" return; \n "
" } \n "
" \n "
//Identify the most saturated channel.
" maxc = max(max(oldRgb->m_Reals[0], oldRgb->m_Reals[1]), oldRgb->m_Reals[2]); \n "
" maxa = ls * maxc; \n "
" \n "
//If a channel is saturated and highlight power is non-negative
//modify the color to prevent hue shift.
" if (maxa > 255 && highPow >= 0) \n "
" { \n "
" newls = 255.0 / maxc; \n "
" lsratio = pow(newls / ls, highPow); \n "
" \n "
//Calculate the max-value color (ranged 0 - 1).
" for (rgbi = 0; rgbi < 3; rgbi++) \n "
" newRgb->m_Reals[rgbi] = newls * oldRgb->m_Reals[rgbi] / 255.0; \n "
" \n "
//Reduce saturation by the lsratio.
" RgbToHsv(&(newRgb->m_Real4), &(newHsv.m_Real4)); \n "
" newHsv.m_Real4.y *= lsratio; \n "
" HsvToRgb(&(newHsv.m_Real4), &(newRgb->m_Real4)); \n "
" \n "
" for (rgbi = 0; rgbi < 3; rgbi++) \n " //Unrolling and vectorizing makes no difference.
" newRgb->m_Reals[rgbi] *= 255.0; \n "
" } \n "
" else \n "
" { \n "
" newls = 255.0 / maxc; \n "
" adjhlp = -highPow; \n "
" \n "
" if (adjhlp > 1) \n "
" adjhlp = 1; \n "
" \n "
" if (maxa <= 255) \n "
" adjhlp = 1; \n "
" \n "
//Calculate the max-value color (ranged 0 - 1) interpolated with the old behavior.
" for (rgbi = 0; rgbi < 3; rgbi++) \n " //Unrolling, caching and vectorizing makes no difference.
" newRgb->m_Reals[rgbi] = ((1.0 - adjhlp) * newls + adjhlp * ls) * oldRgb->m_Reals[rgbi]; \n "
" } \n "
" } \n "
" \n " ;
return os . str ( ) ;
}
/// <summary>
/// Create the gamma correction kernel string used for early clipping.
/// </summary>
/// <param name="alphaCalc">True if channels equals 4 and transparency is desired, else false.</param>
/// <returns>The gamma correction kernel string used for early clipping</returns>
2015-08-10 23:10:23 -04:00
string FinalAccumOpenCLKernelCreator : : CreateGammaCorrectionKernelString ( bool alphaCalc )
2014-07-08 03:11:14 -04:00
{
ostringstream os ;
string dataType ;
os < <
2015-08-10 23:10:23 -04:00
ConstantDefinesString ( m_DoublePrecision ) < <
2014-07-08 03:11:14 -04:00
ClampRealFunctionString < <
UnionCLStructString < <
RgbToHsvFunctionString < <
HsvToRgbFunctionString < <
CalcAlphaFunctionString < <
CreateCalcNewRgbFunctionString ( true ) < <
SpatialFilterCLStructString < <
CreateGammaCorrectionFunctionString ( true , alphaCalc , true , false ) ; //Will only be used with float in this case, early clip. Will always alpha accum.
os < < " __kernel void " < < ( alphaCalc ? m_GammaCorrectionWithAlphaCalcEntryPoint : m_GammaCorrectionWithoutAlphaCalcEntryPoint ) < < " ( \n " < <
2015-08-10 23:10:23 -04:00
" __global real4reals_bucket* accumulator, \n "
2014-07-08 03:11:14 -04:00
" __constant SpatialFilterCL* spatialFilter \n "
" ) \n "
" { \n "
" int testGutter = 0; \n "
" \n "
" if (GLOBAL_ID_Y >= (spatialFilter->m_SuperRasH - testGutter) || GLOBAL_ID_X >= (spatialFilter->m_SuperRasW - testGutter)) \n "
" return; \n "
" \n "
2014-12-06 00:05:09 -05:00
" uint superIndex = (GLOBAL_ID_Y * spatialFilter->m_SuperRasW) + GLOBAL_ID_X; \n "
2015-08-10 23:10:23 -04:00
" __global real4reals_bucket* bucket = accumulator + superIndex; \n "
2014-07-08 03:11:14 -04:00
//Pass in an alphaBase and alphaScale of 0, 1 which means to just directly assign the computed alpha value.
" GammaCorrectionFloats(bucket, &(spatialFilter->m_Background[0]), spatialFilter->m_Gamma, spatialFilter->m_LinRange, spatialFilter->m_Vibrancy, spatialFilter->m_HighlightPower, 0.0, 1.0, &(bucket->m_Reals[0])); \n "
" } \n "
;
return os . str ( ) ;
}
2014-12-05 21:30:46 -05:00
}