1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00
occt/src/OpenGl/OpenGl_RaytraceSource.cxx
dbp 265d4508bd 0024503: TKOpenGl - Porting ray-tracing component on BVH package
Updated test case v3d/raytrace/bug24130. The new version checks correctness of shadows.
2014-01-30 12:55:06 +04:00

1164 lines
50 KiB
C++
Executable File

// Created on: 2013-10-16
// Created by: Denis BOGOLEPOV
// Copyright (c) 2013 OPEN CASCADE SAS
//
// This file is part of Open CASCADE Technology software library.
//
// This library is free software; you can redistribute it and / or modify it
// under the terms of the GNU Lesser General Public version 2.1 as published
// by the Free Software Foundation, with special exception defined in the file
// OCCT_LGPL_EXCEPTION.txt. Consult the file LICENSE_LGPL_21.txt included in OCCT
// distribution for complete text of the license and disclaimer of any warranty.
//
// Alternatively, this file may be used under the terms of Open CASCADE
// commercial license or contractual agreement.
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#ifdef HAVE_OPENCL
#define EOL "\n"
extern const char THE_RAY_TRACE_OPENCL_SOURCE[] =
/////////////////////////////////////////////////////////////////////////////////////////
// Specific data types
EOL
//! Stores ray parameters.
EOL" typedef struct __SRay"
EOL" {"
EOL" float4 Origin;"
EOL" float4 Direct;"
EOL" }"
EOL" SRay;"
EOL
//! Stores parameters of intersection point.
EOL" typedef struct __SIntersect"
EOL" {"
EOL" float4 Normal;"
EOL" float Time;"
EOL" float U;"
EOL" float V;"
EOL" }"
EOL" SIntersect;"
EOL
EOL
/////////////////////////////////////////////////////////////////////////////////////////
// Some useful vector constants
EOL
EOL" #define ZERO ( float4 )( 0.f, 0.f, 0.f, 0.f )"
EOL" #define UNIT ( float4 )( 1.f, 1.f, 1.f, 0.f )"
EOL
EOL" #define AXIS_X ( float4 )( 1.f, 0.f, 0.f, 0.f )"
EOL" #define AXIS_Y ( float4 )( 0.f, 1.f, 0.f, 0.f )"
EOL" #define AXIS_Z ( float4 )( 0.f, 0.f, 1.f, 0.f )"
EOL
/////////////////////////////////////////////////////////////////////////////////////////
// Support functions
EOL
// =======================================================================
// function : GenerateRay
// purpose : Generates primary ray for current work item
// =======================================================================
EOL" void GenerateRay (SRay* theRay,"
EOL" const float theX,"
EOL" const float theY,"
EOL" const int theSizeX,"
EOL" const int theSizeY,"
EOL" const float16 theOrigins,"
EOL" const float16 theDirects)"
EOL" {"
EOL" float2 aPixel = (float2) (theX / (float)theSizeX,"
EOL" theY / (float)theSizeY);"
EOL
EOL" float4 aP0 = mix (theOrigins.lo.lo, theOrigins.lo.hi, aPixel.x);"
EOL" float4 aP1 = mix (theOrigins.hi.lo, theOrigins.hi.hi, aPixel.x);"
EOL
EOL" theRay->Origin = mix (aP0, aP1, aPixel.y);"
EOL
EOL" aP0 = mix (theDirects.lo.lo, theDirects.lo.hi, aPixel.x);"
EOL" aP1 = mix (theDirects.hi.lo, theDirects.hi.hi, aPixel.x);"
EOL
EOL" theRay->Direct = mix (aP0, aP1, aPixel.y);"
EOL" }"
EOL
EOL
/////////////////////////////////////////////////////////////////////////////////////////
// Functions for compute ray-object intersection
EOL
EOL" #define _OOEPS_ exp2 (-80.0f)"
EOL
// =======================================================================
// function : IntersectSphere
// purpose : Computes ray-sphere intersection
// =======================================================================
EOL" float IntersectSphere (const SRay* theRay, float theRadius)"
EOL" {"
EOL" float aDdotD = dot (theRay->Direct.xyz, theRay->Direct.xyz);"
EOL" float aDdotO = dot (theRay->Direct.xyz, theRay->Origin.xyz);"
EOL" float aOdotO = dot (theRay->Origin.xyz, theRay->Origin.xyz);"
EOL
EOL" float aD = aDdotO * aDdotO - aDdotD * (aOdotO - theRadius * theRadius);"
EOL
EOL" if (aD > 0.f)"
EOL" {"
EOL" float aTime = (-aDdotO + native_sqrt (aD)) * (1.f / aDdotD);"
EOL
EOL" return aTime > 0.f ? aTime : MAXFLOAT;"
EOL" }"
EOL
EOL" return MAXFLOAT;"
EOL" }"
EOL
// =======================================================================
// function : IntersectBox
// purpose : Computes ray-box intersection (slab test)
// =======================================================================
EOL" float IntersectBox (const SRay* theRay,"
EOL" float4 theMinPoint,"
EOL" float4 theMaxPoint)"
EOL" {"
EOL" const float4 aInvDirect = (float4)("
EOL" 1.f / (fabs (theRay->Direct.x) > _OOEPS_ ?"
EOL" theRay->Direct.x : copysign (_OOEPS_, theRay->Direct.x)),"
EOL" 1.f / (fabs (theRay->Direct.y) > _OOEPS_ ?"
EOL" theRay->Direct.y : copysign (_OOEPS_, theRay->Direct.y)),"
EOL" 1.f / (fabs (theRay->Direct.z) > _OOEPS_ ?"
EOL" theRay->Direct.z : copysign (_OOEPS_, theRay->Direct.z)),"
EOL" 0.f);"
EOL
EOL" const float4 aTime0 = (theMinPoint - theRay->Origin) * aInvDirect;"
EOL" const float4 aTime1 = (theMaxPoint - theRay->Origin) * aInvDirect;"
EOL
EOL" const float4 aTimeMax = max (aTime0, aTime1);"
EOL" const float4 aTimeMin = min (aTime0, aTime1);"
EOL
EOL" const float theTimeFinal = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" const float theTimeStart = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" return (theTimeStart <= theTimeFinal) && (theTimeFinal >= 0.f) ? theTimeStart : MAXFLOAT;"
EOL" }"
EOL
// =======================================================================
// function : IntersectNodes
// purpose : Computes intersection of ray with two child nodes (boxes)
// =======================================================================
EOL" void IntersectNodes (const SRay* theRay,"
EOL" float4 theMinPoint0,"
EOL" float4 theMaxPoint0,"
EOL" float4 theMinPoint1,"
EOL" float4 theMaxPoint1,"
EOL" float* theTimeStart0,"
EOL" float* theTimeStart1,"
EOL" float theMaxTime)"
EOL" {"
EOL" const float4 aInvDirect = (float4)("
EOL" 1.f / (fabs (theRay->Direct.x) > _OOEPS_ ?"
EOL" theRay->Direct.x : copysign (_OOEPS_, theRay->Direct.x)),"
EOL" 1.f / (fabs (theRay->Direct.y) > _OOEPS_ ?"
EOL" theRay->Direct.y : copysign (_OOEPS_, theRay->Direct.y)),"
EOL" 1.f / (fabs (theRay->Direct.z) > _OOEPS_ ?"
EOL" theRay->Direct.z : copysign (_OOEPS_, theRay->Direct.z)),"
EOL" 0.f);"
EOL
EOL" float4 aTime0 = (theMinPoint0 - theRay->Origin) * aInvDirect;"
EOL" float4 aTime1 = (theMaxPoint0 - theRay->Origin) * aInvDirect;"
EOL
EOL" float4 aTimeMax = max (aTime0, aTime1);"
EOL" float4 aTimeMin = min (aTime0, aTime1);"
EOL
EOL" aTime0 = (theMinPoint1 - theRay->Origin) * aInvDirect;"
EOL" aTime1 = (theMaxPoint1 - theRay->Origin) * aInvDirect;"
EOL
EOL" float aTimeFinal = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" float aTimeStart = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" aTimeMax = max (aTime0, aTime1);"
EOL" aTimeMin = min (aTime0, aTime1);"
EOL
EOL" *theTimeStart0 = (aTimeStart <= aTimeFinal) & (aTimeFinal >= 0.f) & (aTimeStart <= theMaxTime)"
EOL" ? aTimeStart : -MAXFLOAT;"
EOL
EOL" aTimeFinal = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" aTimeStart = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" *theTimeStart1 = (aTimeStart <= aTimeFinal) & (aTimeFinal >= 0.f) & (aTimeStart <= theMaxTime)"
EOL" ? aTimeStart : -MAXFLOAT;"
EOL" }"
EOL
// =======================================================================
// function : IntersectTriangle
// purpose : Computes ray-triangle intersection (branchless version)
// =======================================================================
EOL" float IntersectTriangle (const SRay* theRay,"
EOL" const float4 thePoint0,"
EOL" const float4 thePoint1,"
EOL" const float4 thePoint2,"
EOL" float4* theNormal,"
EOL" float* theU,"
EOL" float* theV)"
EOL" {"
EOL" const float4 aEdge0 = thePoint1 - thePoint0;"
EOL" const float4 aEdge1 = thePoint0 - thePoint2;"
EOL
EOL" *theNormal = cross (aEdge1, aEdge0);"
EOL
EOL" const float4 aEdge2 = (1.f / dot (*theNormal, theRay->Direct)) * (thePoint0 - theRay->Origin);"
EOL
EOL" const float aTime = dot (*theNormal, aEdge2);"
EOL
EOL" const float4 theVec = cross (theRay->Direct, aEdge2);"
EOL
EOL" *theU = dot (theVec, aEdge1);"
EOL" *theV = dot (theVec, aEdge0);"
EOL
EOL" return (aTime >= 0.f) & (*theU >= 0.f) & (*theV >= 0.f) & (*theU + *theV <= 1.f) ? aTime : MAXFLOAT;"
EOL" }"
EOL
/////////////////////////////////////////////////////////////////////////////////////////
// Support shading functions
EOL
EOL" const sampler_t EnvironmentSampler ="
EOL" CLK_NORMALIZED_COORDS_TRUE | CLK_ADDRESS_REPEAT | CLK_FILTER_LINEAR;"
EOL
// =======================================================================
// function : SmoothNormal
// purpose : Interpolates normal across the triangle
// =======================================================================
EOL" float4 SmoothNormal (__global float4* theNormals,"
EOL" const SIntersect* theHit,"
EOL" const int4 theIndices)"
EOL" {"
EOL" float4 aNormal0 = theNormals[theIndices.x],"
EOL" aNormal1 = theNormals[theIndices.y],"
EOL" aNormal2 = theNormals[theIndices.z];"
EOL
EOL" return fast_normalize (aNormal1 * theHit->U +"
EOL" aNormal2 * theHit->V +"
EOL" aNormal0 * (1.f - theHit->U - theHit->V));"
EOL" }"
EOL
// =======================================================================
// function : Shade
// purpose : Computes Phong-based illumination
// =======================================================================
EOL" float4 Shade (const float4 theMatDiff,"
EOL" const float4 theMatSpec,"
EOL" const float4 theLight,"
EOL" const float4 theView,"
EOL" const float4 theNormal,"
EOL" const float4 theIntens,"
EOL" const float theTranspr)"
EOL" {"
EOL" float aLambert = dot (theNormal, theLight);"
EOL
EOL" aLambert = theTranspr > 0.f ? fabs (aLambert) : aLambert;"
EOL
EOL" if (aLambert > 0.f)"
EOL" {"
EOL" const float4 aReflect = 2.f * dot (theLight, theNormal) * theNormal - theLight;"
EOL
EOL" const float aSpecular = pow (max (dot (aReflect.xyz, theView.xyz), 0.f), theMatSpec.w);"
EOL
EOL" return theIntens * (theMatDiff * aLambert + theMatSpec * aSpecular);"
EOL" }"
EOL
EOL" return ZERO;"
EOL" }"
EOL
// =======================================================================
// function : Latlong
// purpose : Converts world direction to environment texture coordinates
// =======================================================================
EOL" float2 Latlong (const float4 thePoint, const float theRadius)"
EOL" {"
EOL" float aPsi = acospi (-thePoint.y / theRadius);"
EOL" float aPhi = atan2pi (thePoint.z, thePoint.x);"
EOL
EOL" aPhi = (aPhi < 0.f) ? aPhi + 2.f : aPhi;"
EOL
EOL" return (float2) (aPhi * 0.5f, aPsi);"
EOL" }"
EOL
/////////////////////////////////////////////////////////////////////////////////////////
// Core ray tracing function
EOL
// =======================================================================
// function : push
// purpose : Pushes BVH node index to local stack
// =======================================================================
EOL" void push (uint* theStack, char* thePos, const uint theValue)"
EOL" {"
EOL" (*thePos)++;"
EOL" theStack[*thePos] = theValue;"
EOL" }"
EOL
// =======================================================================
// function : pop
// purpose : Pops BVH node index from local stack
// =======================================================================
EOL" void pop (uint* theStack, char* thePos, uint* theValue)"
EOL" {"
EOL" *theValue = theStack[*thePos];"
EOL" (*thePos)--;"
EOL" }"
EOL
// =======================================================================
// function : ObjectNearestHit
// purpose : Finds intersection with nearest object triangle
// =======================================================================
EOL" int4 ObjectNearestHit (const SRay* theRay,"
EOL" SIntersect* theIntersect,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer)"
EOL" {"
EOL" uint aStack [32];"
EOL
EOL" char aHead = -1;" // stack pointer
EOL" uint aNode = 0;" // node to visit
EOL
EOL" const float4 aInvDirect = (float4) ("
EOL" 1.f / (fabs (theRay->Direct.x) > _OOEPS_ ?"
EOL" theRay->Direct.x : copysign (_OOEPS_, theRay->Direct.x)),"
EOL" 1.f / (fabs (theRay->Direct.y) > _OOEPS_ ?"
EOL" theRay->Direct.y : copysign (_OOEPS_, theRay->Direct.y)),"
EOL" 1.f / (fabs (theRay->Direct.z) > _OOEPS_ ?"
EOL" theRay->Direct.z : copysign (_OOEPS_, theRay->Direct.z)),"
EOL" 0.f);"
EOL
EOL" int4 aTriangleIndex = (int4) (-1);"
EOL
EOL" float aTimeExit;"
EOL" float aTimeMin1;"
EOL" float aTimeMin2;"
EOL
EOL" while (true)"
EOL" {"
EOL" const int3 aData = theObjectNodeInfoBuffer[aNode].xyz;"
EOL
EOL" if (aData.x == 0)" // if inner node
EOL" {"
EOL" float4 aNodeMin = theObjectMinPointBuffer[aData.y];"
EOL" float4 aNodeMax = theObjectMaxPointBuffer[aData.y];"
EOL
EOL" float4 aTime0 = (aNodeMin - theRay->Origin) * aInvDirect;"
EOL" float4 aTime1 = (aNodeMax - theRay->Origin) * aInvDirect;"
EOL
EOL" float4 aTimeMax = max (aTime0, aTime1);"
EOL" float4 aTimeMin = min (aTime0, aTime1);"
EOL
EOL" aTimeExit = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" aTimeMin1 = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" const bool aHitLft = (aTimeMin1 <= aTimeExit) & (aTimeExit >= 0.f) & (aTimeMin1 <= theIntersect->Time);"
EOL
EOL" aNodeMin = theObjectMinPointBuffer[aData.z];"
EOL" aNodeMax = theObjectMaxPointBuffer[aData.z];"
EOL
EOL" aTime0 = (aNodeMin - theRay->Origin) * aInvDirect;"
EOL" aTime1 = (aNodeMax - theRay->Origin) * aInvDirect;"
EOL
EOL" aTimeMax = max (aTime0, aTime1);"
EOL" aTimeMin = min (aTime0, aTime1);"
EOL
EOL" aTimeExit = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" aTimeMin2 = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" const bool aHitRgh = (aTimeMin2 <= aTimeExit) & (aTimeExit >= 0.f) & (aTimeMin2 <= theIntersect->Time);"
EOL
EOL" if (aHitLft & aHitRgh)"
EOL" {"
EOL" aNode = (aTimeMin1 < aTimeMin2) ? aData.y : aData.z;"
EOL
EOL" push (aStack, &aHead, (aTimeMin1 < aTimeMin2) ? aData.z : aData.y);"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHitLft | aHitRgh)"
EOL" {"
EOL" aNode = aHitLft ? aData.y : aData.z;"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHead < 0)"
EOL" return aTriangleIndex;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL" }"
EOL" else " // if leaf node
EOL" {"
EOL" for (int aTriIdx = aData.y; aTriIdx <= aData.z; ++aTriIdx)"
EOL" {"
EOL" const int4 aTestTriangle = theGeometryTriangBuffer[aTriIdx];"
EOL
EOL" const float4 aPoint0 = theGeometryVertexBuffer[aTestTriangle.x];"
EOL" const float4 aPoint1 = theGeometryVertexBuffer[aTestTriangle.y];"
EOL" const float4 aPoint2 = theGeometryVertexBuffer[aTestTriangle.z];"
EOL
EOL" float4 aNormal; float aU, aV;"
EOL
EOL" float aTime = IntersectTriangle (theRay,"
EOL" aPoint0,"
EOL" aPoint1,"
EOL" aPoint2,"
EOL" &aNormal,"
EOL" &aU,"
EOL" &aV);"
EOL
EOL" if (aTime < theIntersect->Time)"
EOL" {"
EOL" aTriangleIndex = aTestTriangle;"
EOL" theIntersect->Normal = aNormal;"
EOL" theIntersect->Time = aTime;"
EOL" theIntersect->U = aU;"
EOL" theIntersect->V = aV;"
EOL" }"
EOL" }"
EOL
EOL" if (aHead < 0)"
EOL" return aTriangleIndex;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL
EOL" return aTriangleIndex;"
EOL" }"
EOL
// =======================================================================
// function : ObjectAnyHit
// purpose : Finds intersection with any object triangle
// =======================================================================
EOL" float ObjectAnyHit (const SRay* theRay,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer,"
EOL" const float theDistance)"
EOL" {"
EOL" uint aStack [32];"
EOL
EOL" char aHead = -1;" // stack pointer
EOL" uint aNode = 0;" // node to visit
EOL
EOL" const float4 aInvDirect = (float4) ("
EOL" 1.f / (fabs (theRay->Direct.x) > _OOEPS_ ?"
EOL" theRay->Direct.x : copysign (_OOEPS_, theRay->Direct.x)),"
EOL" 1.f / (fabs (theRay->Direct.y) > _OOEPS_ ?"
EOL" theRay->Direct.y : copysign (_OOEPS_, theRay->Direct.y)),"
EOL" 1.f / (fabs (theRay->Direct.z) > _OOEPS_ ?"
EOL" theRay->Direct.z : copysign (_OOEPS_, theRay->Direct.z)),"
EOL" 0.f);"
EOL
EOL" float aTimeExit;"
EOL" float aTimeMin1;"
EOL" float aTimeMin2;"
EOL
EOL" while (true)"
EOL" {"
EOL" const int3 aData = theObjectNodeInfoBuffer[aNode].xyz;"
EOL
EOL" if (aData.x == 0)" // if inner node
EOL" {"
EOL" float4 aNodeMin = theObjectMinPointBuffer[aData.y];"
EOL" float4 aNodeMax = theObjectMaxPointBuffer[aData.y];"
EOL
EOL" float4 aTime0 = (aNodeMin - theRay->Origin) * aInvDirect;"
EOL" float4 aTime1 = (aNodeMax - theRay->Origin) * aInvDirect;"
EOL
EOL" float4 aTimeMax = max (aTime0, aTime1);"
EOL" float4 aTimeMin = min (aTime0, aTime1);"
EOL
EOL" aTimeExit = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" aTimeMin1 = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" const bool aHitLft = (aTimeMin1 <= aTimeExit) & (aTimeExit >= 0.f) & (aTimeMin1 <= theDistance);"
EOL
EOL" aNodeMin = theObjectMinPointBuffer[aData.z];"
EOL" aNodeMax = theObjectMaxPointBuffer[aData.z];"
EOL
EOL" aTime0 = (aNodeMin - theRay->Origin) * aInvDirect;"
EOL" aTime1 = (aNodeMax - theRay->Origin) * aInvDirect;"
EOL
EOL" aTimeMax = max (aTime0, aTime1);"
EOL" aTimeMin = min (aTime0, aTime1);"
EOL
EOL" aTimeExit = min (aTimeMax.x, min (aTimeMax.y, aTimeMax.z));"
EOL" aTimeMin2 = max (aTimeMin.x, max (aTimeMin.y, aTimeMin.z));"
EOL
EOL" const bool aHitRgh = (aTimeMin2 <= aTimeExit) & (aTimeExit >= 0.f) & (aTimeMin2 <= theDistance);"
EOL
EOL" if (aHitLft & aHitRgh)"
EOL" {"
EOL" aNode = (aTimeMin1 < aTimeMin2) ? aData.y : aData.z;"
EOL
EOL" push (aStack, &aHead, (aTimeMin1 < aTimeMin2) ? aData.z : aData.y);"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHitLft | aHitRgh)"
EOL" {"
EOL" aNode = aHitLft ? aData.y : aData.z;"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHead < 0)"
EOL" return 1.f;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL" }"
EOL" else " // if leaf node
EOL" {"
EOL" for (int aTriIdx = aData.y; aTriIdx <= aData.z; ++aTriIdx)"
EOL" {"
EOL" const int4 aTestTriangle = theGeometryTriangBuffer[aTriIdx];"
EOL
EOL" const float4 aPoint0 = theGeometryVertexBuffer[aTestTriangle.x];"
EOL" const float4 aPoint1 = theGeometryVertexBuffer[aTestTriangle.y];"
EOL" const float4 aPoint2 = theGeometryVertexBuffer[aTestTriangle.z];"
EOL
EOL" float4 aNormal; float aU, aV;"
EOL
EOL" float aTime = IntersectTriangle (theRay,"
EOL" aPoint0,"
EOL" aPoint1,"
EOL" aPoint2,"
EOL" &aNormal,"
EOL" &aU,"
EOL" &aV);"
EOL
EOL" if (aTime < theDistance)"
EOL" {"
EOL" return 0.f;"
EOL" }"
EOL" }"
EOL
EOL" if (aHead < 0)"
EOL" return 1.f;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL
EOL" return 1.f;"
EOL" }"
EOL
// =======================================================================
// function : NearestHit
// purpose : Finds intersection with nearest scene triangle
// =======================================================================
EOL" int4 NearestHit (const SRay* theRay,"
EOL" SIntersect* theIntersect,"
EOL" __global int4* theSceneNodeInfoBuffer,"
EOL" __global float4* theSceneMinPointBuffer,"
EOL" __global float4* theSceneMaxPointBuffer,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer)"
EOL" {"
EOL" theIntersect->Time = MAXFLOAT;"
EOL
EOL" uint aStack [16];"
EOL
EOL" char aHead = -1;" // stack pointer
EOL" uint aNode = 0;" // node to visit
EOL
EOL" int4 aNearestTriangle = (int4) (-1);"
EOL
EOL" while (true)"
EOL" {"
EOL" const int4 aData = theSceneNodeInfoBuffer[aNode];"
EOL
EOL" if (aData.x != 0)" // if leaf node
EOL" {"
EOL" const float4 aNodeMin = theSceneMinPointBuffer[aNode];"
EOL" const float4 aNodeMax = theSceneMaxPointBuffer[aNode];"
EOL
EOL" if (IntersectBox (theRay, aNodeMin, aNodeMax) <= theIntersect->Time)"
EOL" {"
EOL" int4 anIndex = ObjectNearestHit (theRay,"
EOL" theIntersect,"
EOL" theObjectNodeInfoBuffer + aData.y,"
EOL" theObjectMinPointBuffer + aData.y,"
EOL" theObjectMaxPointBuffer + aData.y,"
EOL" theGeometryTriangBuffer + aData.w,"
EOL" theGeometryVertexBuffer + aData.z);"
EOL
EOL" if (anIndex.x != -1)"
EOL" aNearestTriangle = (int4) (anIndex.x + aData.z,"
EOL" anIndex.y + aData.z,"
EOL" anIndex.z + aData.z,"
EOL" anIndex.w);"
EOL" }"
EOL
EOL" if (aHead < 0)"
EOL" return aNearestTriangle;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" else " // if inner node
EOL" {"
EOL" float4 aNodeMinLft = theSceneMinPointBuffer[aData.y];"
EOL" float4 aNodeMinRgh = theSceneMinPointBuffer[aData.z];"
EOL" float4 aNodeMaxLft = theSceneMaxPointBuffer[aData.y];"
EOL" float4 aNodeMaxRgh = theSceneMaxPointBuffer[aData.z];"
EOL
EOL" float aTimeMin1;"
EOL" float aTimeMin2;"
EOL
EOL" IntersectNodes (theRay,"
EOL" aNodeMinLft,"
EOL" aNodeMaxLft,"
EOL" aNodeMinRgh,"
EOL" aNodeMaxRgh,"
EOL" &aTimeMin1,"
EOL" &aTimeMin2,"
EOL" theIntersect->Time);"
EOL
EOL" const bool aHitLft = (aTimeMin1 != -MAXFLOAT);"
EOL" const bool aHitRgh = (aTimeMin2 != -MAXFLOAT);"
EOL
EOL" if (aHitLft & aHitRgh)"
EOL" {"
EOL" aNode = (aTimeMin1 < aTimeMin2) ? aData.y : aData.z;"
EOL
EOL" push (aStack, &aHead, (aTimeMin1 < aTimeMin2) ? aData.z : aData.y);"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHitLft | aHitRgh)"
EOL" {"
EOL" aNode = aHitLft ? aData.y : aData.z;"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHead < 0)"
EOL" return aNearestTriangle;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL" }"
EOL" }"
EOL
EOL" return aNearestTriangle;"
EOL" }"
EOL
// =======================================================================
// function : AnyHit
// purpose : Finds intersection with any scene triangle
// =======================================================================
EOL" float AnyHit (const SRay* theRay,"
EOL" __global int4* theSceneNodeInfoBuffer,"
EOL" __global float4* theSceneMinPointBuffer,"
EOL" __global float4* theSceneMaxPointBuffer,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer,"
EOL" const float theDistance)"
EOL" {"
EOL" uint aStack [16];"
EOL
EOL" char aHead = -1;" // stack pointer
EOL" uint aNode = 0;" // node to visit
EOL
EOL" while (true)"
EOL" {"
EOL" const int4 aData = theSceneNodeInfoBuffer[aNode];"
EOL
EOL" if (aData.x != 0)" // if leaf node
EOL" {"
EOL" const float4 aNodeMin = theSceneMinPointBuffer[aNode];"
EOL" const float4 aNodeMax = theSceneMaxPointBuffer[aNode];"
EOL
EOL" if (IntersectBox (theRay, aNodeMin, aNodeMax) <= theDistance)"
EOL" {"
EOL" if (0.f == ObjectAnyHit (theRay,"
EOL" theObjectNodeInfoBuffer + aData.y,"
EOL" theObjectMinPointBuffer + aData.y,"
EOL" theObjectMaxPointBuffer + aData.y,"
EOL" theGeometryTriangBuffer + aData.w,"
EOL" theGeometryVertexBuffer + aData.z,"
EOL" theDistance))"
EOL" {"
EOL" return 0.f;"
EOL" }"
EOL" }"
EOL
EOL" if (aHead < 0)"
EOL" return 1.f;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" else" // if inner node
EOL" {"
EOL" float4 aNodeMinLft = theSceneMinPointBuffer[aData.y];"
EOL" float4 aNodeMinRgh = theSceneMinPointBuffer[aData.z];"
EOL" float4 aNodeMaxLft = theSceneMaxPointBuffer[aData.y];"
EOL" float4 aNodeMaxRgh = theSceneMaxPointBuffer[aData.z];"
EOL
EOL" float aTimeMin1;"
EOL" float aTimeMin2;"
EOL
EOL" IntersectNodes (theRay,"
EOL" aNodeMinLft,"
EOL" aNodeMaxLft,"
EOL" aNodeMinRgh,"
EOL" aNodeMaxRgh,"
EOL" &aTimeMin1,"
EOL" &aTimeMin2,"
EOL" theDistance);"
EOL
EOL" const bool aHitLft = (aTimeMin1 != -MAXFLOAT);"
EOL" const bool aHitRgh = (aTimeMin2 != -MAXFLOAT);"
EOL
EOL" if (aHitLft & aHitRgh)"
EOL" {"
EOL" aNode = (aTimeMin1 < aTimeMin2) ? aData.y : aData.z;"
EOL
EOL" push (aStack, &aHead, (aTimeMin1 < aTimeMin2) ? aData.z : aData.y);"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHitLft | aHitRgh)"
EOL" {"
EOL" aNode = aHitLft ? aData.y : aData.z;"
EOL" }"
EOL" else"
EOL" {"
EOL" if (aHead < 0)"
EOL" return 1.f;"
EOL
EOL" pop (aStack, &aHead, &aNode);"
EOL" }"
EOL" }"
EOL" }"
EOL" }"
EOL" }"
EOL
EOL" #define _MAX_DEPTH_ 5"
EOL
EOL" #define THRESHOLD (float4) (0.1f, 0.1f, 0.1f, 1.f)"
EOL
EOL" #define LIGHT_POS(Buffer, LightID) Buffer[2 * LightID + 1]"
EOL" #define LIGHT_RAD(Buffer, LightID) Buffer[2 * LightID + 0]"
EOL
EOL" #define MATERIAL_AMBN(Buffer, TriangleID) Buffer[7 * TriangleID.w + 0]"
EOL" #define MATERIAL_DIFF(Buffer, TriangleID) Buffer[7 * TriangleID.w + 1]"
EOL" #define MATERIAL_SPEC(Buffer, TriangleID) Buffer[7 * TriangleID.w + 2]"
EOL" #define MATERIAL_EMIS(Buffer, TriangleID) Buffer[7 * TriangleID.w + 3]"
EOL" #define MATERIAL_REFL(Buffer, TriangleID) Buffer[7 * TriangleID.w + 4]"
EOL" #define MATERIAL_REFR(Buffer, TriangleID) Buffer[7 * TriangleID.w + 5]"
EOL" #define MATERIAL_TRAN(Buffer, TriangleID) Buffer[7 * TriangleID.w + 6]"
EOL
// =======================================================================
// function : Radiance
// purpose : Computes color of specified ray
// =======================================================================
EOL" float4 Radiance (SRay* theRay,"
EOL" __read_only image2d_t theEnvMap,"
EOL" __global int4* theSceneNodeInfoBuffer,"
EOL" __global float4* theSceneMinPointBuffer,"
EOL" __global float4* theSceneMaxPointBuffer,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer,"
EOL" __global float4* theGeometryNormalBuffer,"
EOL" __global float4* theLightSourceBuffer,"
EOL" __global float4* theMaterialBuffer,"
EOL" const float4 theGlobalAmbient,"
EOL" const int theLightBufferSize,"
EOL" const int theShadowsEnabled,"
EOL" const int theReflectEnabled,"
EOL" const float theSceneEpsilon,"
EOL" const float theSceneRadius)"
EOL" {"
EOL" float4 aResult = (float4) (0.f, 0.f, 0.f, 0.f);"
EOL" float4 aWeight = (float4) (1.f, 1.f, 1.f, 1.f);"
EOL
EOL" SIntersect aHit;"
EOL
EOL" for (int aDepth = 0; aDepth < _MAX_DEPTH_; ++aDepth)"
EOL" {"
EOL" int4 aTriangle = NearestHit (theRay,"
EOL" &aHit,"
EOL" theSceneNodeInfoBuffer,"
EOL" theSceneMinPointBuffer,"
EOL" theSceneMaxPointBuffer,"
EOL" theObjectNodeInfoBuffer,"
EOL" theObjectMinPointBuffer,"
EOL" theObjectMaxPointBuffer,"
EOL" theGeometryTriangBuffer,"
EOL" theGeometryVertexBuffer);"
EOL
EOL" if (aTriangle.x < 0.f)"
EOL" {"
EOL" if (aWeight.w != 0.f)"
EOL" break;"
EOL
EOL" float aTime = IntersectSphere (theRay, theSceneRadius);"
EOL
EOL" if (aTime != MAXFLOAT)"
EOL" {"
EOL" aResult += aWeight * read_imagef (theEnvMap, EnvironmentSampler,"
EOL" Latlong (theRay->Origin + theRay->Direct * aTime, theSceneRadius));"
EOL" }"
EOL
EOL" return (float4) (aResult.x,"
EOL" aResult.y,"
EOL" aResult.z,"
EOL" aWeight.w);"
EOL" }"
EOL
EOL // Compute geometric normal
EOL" float4 aGeomNormal = aHit.Normal; aGeomNormal = fast_normalize (aGeomNormal);"
EOL
EOL // Compute interpolated normal
EOL" float4 aNormal = SmoothNormal (theGeometryNormalBuffer, &aHit, aTriangle);"
EOL
EOL // Compute intersection point
EOL" float4 aPoint = theRay->Direct * aHit.Time + theRay->Origin;"
EOL
EOL" float4 aMaterAmb = MATERIAL_AMBN (theMaterialBuffer, aTriangle);"
EOL" float4 aMaterTrn = MATERIAL_TRAN (theMaterialBuffer, aTriangle);"
EOL
EOL" aResult += aWeight * theGlobalAmbient * aMaterAmb *"
EOL" (aMaterTrn.x * max (fabs (dot (theRay->Direct, aNormal)), 0.5f));"
EOL
EOL" for (int nLight = 0; nLight < theLightBufferSize; ++nLight)"
EOL" {"
EOL" float4 aLightPosition = LIGHT_POS (theLightSourceBuffer, nLight);"
EOL
EOL" SRay aShadow;"
EOL" aShadow.Direct = aLightPosition;"
EOL
EOL" float aLightDistance = MAXFLOAT;"
EOL" if (aLightPosition.w != 0.f)"
EOL" {"
EOL" aLightDistance = length (aLightPosition - aPoint);"
EOL" aShadow.Direct = (aLightPosition - aPoint) * (1.f / aLightDistance);"
EOL" }"
EOL
EOL" aShadow.Origin = aPoint + aShadow.Direct * theSceneEpsilon +"
EOL" aGeomNormal * copysign (theSceneEpsilon, dot (aGeomNormal, aShadow.Direct));"
EOL
EOL" float aVisibility = 1.f;"
EOL
EOL" if (theShadowsEnabled)"
EOL" {"
EOL" aVisibility = AnyHit (&aShadow,"
EOL" theSceneNodeInfoBuffer,"
EOL" theSceneMinPointBuffer,"
EOL" theSceneMaxPointBuffer,"
EOL" theObjectNodeInfoBuffer,"
EOL" theObjectMinPointBuffer,"
EOL" theObjectMaxPointBuffer,"
EOL" theGeometryTriangBuffer,"
EOL" theGeometryVertexBuffer,"
EOL" aLightDistance);"
EOL" }"
EOL
EOL" if (aVisibility > 0.f)"
EOL" {"
EOL" aResult += aMaterTrn.x * aWeight * Shade (MATERIAL_DIFF (theMaterialBuffer, aTriangle),"
EOL" MATERIAL_SPEC (theMaterialBuffer, aTriangle),"
EOL" aShadow.Direct,"
EOL" -theRay->Direct,"
EOL" aNormal,"
EOL" LIGHT_RAD (theLightSourceBuffer, nLight),"
EOL" aMaterTrn.y);"
EOL" }"
EOL" }"
EOL
EOL" if (aMaterTrn.y > 0.f)"
EOL" {"
EOL" aWeight *= aMaterTrn.y;"
EOL" }"
EOL" else"
EOL" {"
EOL" aWeight *= theReflectEnabled ? MATERIAL_REFL (theMaterialBuffer, aTriangle) : ZERO;"
EOL
EOL" float4 aDirect = theRay->Direct - 2.f * dot (theRay->Direct, aNormal) * aNormal;"
EOL
EOL" float aDdotN = dot (aDirect, aGeomNormal);"
EOL" if (aDdotN < 0.f)"
EOL" theRay->Direct -= 2.f * dot (theRay->Direct, aGeomNormal) * aGeomNormal;"
EOL" else"
EOL" theRay->Direct = aDirect;"
EOL" }"
EOL
EOL" if (all (islessequal (aWeight, THRESHOLD)))"
EOL" {"
EOL" return (float4) (aResult.x,"
EOL" aResult.y,"
EOL" aResult.z,"
EOL" aWeight.w);"
EOL" }"
EOL
EOL" theRay->Origin = theRay->Direct * theSceneEpsilon + aPoint;"
EOL" }"
EOL
EOL" return (float4) (aResult.x,"
EOL" aResult.y,"
EOL" aResult.z,"
EOL" aWeight.w);"
EOL" }"
EOL
///////////////////////////////////////////////////////////////////////////////
// Ray tracing kernel functions
EOL
// =======================================================================
// function : RaytraceRender
// purpose : Computes pixel color using ray-tracing
// =======================================================================
EOL" __kernel void RaytraceRender (const int theSizeX,"
EOL" const int theSizeY,"
EOL" const float16 theOrigins,"
EOL" const float16 theDirects,"
EOL" __read_only image2d_t theEnvMap,"
EOL" __write_only image2d_t theOutput,"
EOL" __global int4* theSceneNodeInfoBuffer,"
EOL" __global float4* theSceneMinPointBuffer,"
EOL" __global float4* theSceneMaxPointBuffer,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer,"
EOL" __global float4* theGeometryNormalBuffer,"
EOL" __global float4* theLightSourceBuffer,"
EOL" __global float4* theMaterialBuffer,"
EOL" const float4 theGlobalAmbient,"
EOL" const int theLightBufferSize,"
EOL" const int theShadowsEnabled,"
EOL" const int theReflectEnabled,"
EOL" const float theSceneEpsilon,"
EOL" const float theSceneRadius)"
EOL" {"
EOL" const int aPixelX = get_global_id (0);"
EOL" const int aPixelY = get_global_id (1);"
EOL
EOL" if (aPixelX >= theSizeX || aPixelY >= theSizeY)"
EOL" return;"
EOL
EOL" private SRay aRay;"
EOL
EOL" GenerateRay (&aRay,"
EOL" aPixelX,"
EOL" aPixelY,"
EOL" theSizeX,"
EOL" theSizeY,"
EOL" theOrigins,"
EOL" theDirects);"
EOL
EOL" float4 aColor = (float4) (0.f, 0.f, 0.f, 1.f);"
EOL
EOL" float aTimeStart = IntersectBox (&aRay, theSceneMinPointBuffer[0], theSceneMaxPointBuffer[0]);"
EOL
EOL" if (aTimeStart != MAXFLOAT)"
EOL" {"
EOL" aRay.Origin += aRay.Direct * max (aTimeStart - theSceneEpsilon, 0.f);"
EOL
EOL" aColor = clamp (Radiance (&aRay,"
EOL" theEnvMap,"
EOL" theSceneNodeInfoBuffer,"
EOL" theSceneMinPointBuffer,"
EOL" theSceneMaxPointBuffer,"
EOL" theObjectNodeInfoBuffer,"
EOL" theObjectMinPointBuffer,"
EOL" theObjectMaxPointBuffer,"
EOL" theGeometryTriangBuffer,"
EOL" theGeometryVertexBuffer,"
EOL" theGeometryNormalBuffer,"
EOL" theLightSourceBuffer,"
EOL" theMaterialBuffer,"
EOL" theGlobalAmbient,"
EOL" theLightBufferSize,"
EOL" theShadowsEnabled,"
EOL" theReflectEnabled,"
EOL" theSceneEpsilon,"
EOL" theSceneRadius), 0.f, 1.f);"
EOL" }"
EOL
EOL" write_imagef (theOutput, (int2) (aPixelX, aPixelY), aColor);"
EOL" }"
EOL
EOL" const sampler_t OutputSampler ="
EOL" CLK_NORMALIZED_COORDS_FALSE | CLK_ADDRESS_CLAMP | CLK_FILTER_NEAREST;"
EOL
EOL" #define _LUM_DELTA_ 0.085f"
EOL
EOL" #define AA_MAX 0.559017f"
EOL" #define AA_MIN 0.186339f"
EOL
// =======================================================================
// function : RaytraceSmooth
// purpose : Performs adaptive sub-pixel rendering
// =======================================================================
EOL" __kernel void RaytraceSmooth (const int theSizeX,"
EOL" const int theSizeY,"
EOL" const float16 theOrigins,"
EOL" const float16 theDirects,"
EOL" __read_only image2d_t theInput,"
EOL" __read_only image2d_t theEnvMap,"
EOL" __write_only image2d_t theOutput,"
EOL" __global int4* theSceneNodeInfoBuffer,"
EOL" __global float4* theSceneMinPointBuffer,"
EOL" __global float4* theSceneMaxPointBuffer,"
EOL" __global int4* theObjectNodeInfoBuffer,"
EOL" __global float4* theObjectMinPointBuffer,"
EOL" __global float4* theObjectMaxPointBuffer,"
EOL" __global int4* theGeometryTriangBuffer,"
EOL" __global float4* theGeometryVertexBuffer,"
EOL" __global float4* theGeometryNormalBuffer,"
EOL" __global float4* theLightSourceBuffer,"
EOL" __global float4* theMaterialBuffer,"
EOL" const float4 theGlobalAmbient,"
EOL" const int theLightBufferSize,"
EOL" const int theShadowsEnabled,"
EOL" const int theReflectEnabled,"
EOL" const float theSceneEpsilon,"
EOL" const float theSceneRadius)"
EOL" {"
EOL" const int aPixelX = get_global_id (0);"
EOL" const int aPixelY = get_global_id (1);"
EOL
EOL" if (aPixelX >= theSizeX || aPixelY >= theSizeY)"
EOL" return;"
EOL
EOL" float4 aClr0 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 0, aPixelY + 0));"
EOL" float4 aClr1 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 0, aPixelY - 1));"
EOL" float4 aClr2 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 0, aPixelY + 1));"
EOL
EOL" float4 aClr3 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 1, aPixelY + 0));"
EOL" float4 aClr4 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 1, aPixelY - 1));"
EOL" float4 aClr5 = read_imagef (theInput, OutputSampler, (float2) (aPixelX + 1, aPixelY + 1));"
EOL
EOL" float4 aClr6 = read_imagef (theInput, OutputSampler, (float2) (aPixelX - 1, aPixelY + 0));"
EOL" float4 aClr7 = read_imagef (theInput, OutputSampler, (float2) (aPixelX - 1, aPixelY - 1));"
EOL" float4 aClr8 = read_imagef (theInput, OutputSampler, (float2) (aPixelX - 1, aPixelY + 1));"
EOL
EOL" bool render = fabs (aClr1.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr2.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr3.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr4.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr5.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr6.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr7.w - aClr0.w) > _LUM_DELTA_ ||"
EOL" fabs (aClr8.w - aClr0.w) > _LUM_DELTA_;"
EOL
EOL" if (!render)"
EOL" {"
EOL" aClr1 = (aClr1.w == 1.f) ? -UNIT : aClr1;"
EOL" aClr2 = (aClr2.w == 1.f) ? -UNIT : aClr2;"
EOL" aClr3 = (aClr3.w == 1.f) ? -UNIT : aClr3;"
EOL" aClr4 = (aClr4.w == 1.f) ? -UNIT : aClr4;"
EOL" aClr5 = (aClr5.w == 1.f) ? -UNIT : aClr5;"
EOL" aClr6 = (aClr6.w == 1.f) ? -UNIT : aClr6;"
EOL" aClr7 = (aClr7.w == 1.f) ? -UNIT : aClr7;"
EOL" aClr8 = (aClr8.w == 1.f) ? -UNIT : aClr8;"
EOL
EOL" float aLum = (aClr0.w == 1.f) ? -1.f : (0.2126f * aClr0.x + 0.7152f * aClr0.y + 0.0722f * aClr0.z);"
EOL
EOL" render = fabs (0.2126f * aClr1.x + 0.7152f * aClr1.y + 0.0722f * aClr1.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr2.x + 0.7152f * aClr2.y + 0.0722f * aClr2.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr3.x + 0.7152f * aClr3.y + 0.0722f * aClr3.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr4.x + 0.7152f * aClr4.y + 0.0722f * aClr4.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr5.x + 0.7152f * aClr5.y + 0.0722f * aClr5.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr6.x + 0.7152f * aClr6.y + 0.0722f * aClr6.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr7.x + 0.7152f * aClr7.y + 0.0722f * aClr7.z - aLum) > _LUM_DELTA_ ||"
EOL" fabs (0.2126f * aClr8.x + 0.7152f * aClr8.y + 0.0722f * aClr8.z - aLum) > _LUM_DELTA_;"
EOL" }"
EOL
EOL" float4 aColor = clamp (aClr0, 0.f, 1.f);"
EOL
EOL" private SRay aRay;"
EOL
EOL" const float4 aBoxMin = theSceneMinPointBuffer[0];"
EOL" const float4 aBoxMax = theSceneMaxPointBuffer[0];"
EOL
EOL" if (render)"
EOL" {"
EOL" for (int aSample = 0; aSample <= 3; ++aSample)"
EOL" {"
EOL" float fX = aPixelX, fY = aPixelY;"
EOL
EOL" if (aSample == 0)"
EOL" {"
EOL" fX -= AA_MIN; fY -= AA_MAX;"
EOL" }"
EOL" else if (aSample == 1)"
EOL" {"
EOL" fX -= AA_MAX; fY += AA_MIN;"
EOL" }"
EOL" else if (aSample == 2)"
EOL" {"
EOL" fX += AA_MIN; fY += AA_MAX;"
EOL" }"
EOL" else"
EOL" {"
EOL" fX += AA_MAX; fY -= AA_MIN;"
EOL" }"
EOL
EOL" GenerateRay (&aRay,"
EOL" fX,"
EOL" fY,"
EOL" theSizeX,"
EOL" theSizeY,"
EOL" theOrigins,"
EOL" theDirects);"
EOL
EOL" float aTimeStart = IntersectBox (&aRay, aBoxMin, aBoxMax);"
EOL
EOL" if (aTimeStart != MAXFLOAT)"
EOL" {"
EOL" aRay.Origin += aRay.Direct * max (aTimeStart - theSceneEpsilon, 0.f);"
EOL
EOL" aColor += clamp (Radiance (&aRay,"
EOL" theEnvMap,"
EOL" theSceneNodeInfoBuffer,"
EOL" theSceneMinPointBuffer,"
EOL" theSceneMaxPointBuffer,"
EOL" theObjectNodeInfoBuffer,"
EOL" theObjectMinPointBuffer,"
EOL" theObjectMaxPointBuffer,"
EOL" theGeometryTriangBuffer,"
EOL" theGeometryVertexBuffer,"
EOL" theGeometryNormalBuffer,"
EOL" theLightSourceBuffer,"
EOL" theMaterialBuffer,"
EOL" theGlobalAmbient,"
EOL" theLightBufferSize,"
EOL" theShadowsEnabled,"
EOL" theReflectEnabled,"
EOL" theSceneEpsilon,"
EOL" theSceneRadius), 0.f, 1.f);"
EOL" }"
EOL" else"
EOL" aColor += (float4) (0.f, 0.f, 0.f, 1.f);"
EOL" }"
EOL
EOL" aColor *= 1.f / 5.f;"
EOL" }"
EOL
EOL" write_imagef (theOutput, (int2) (aPixelX, aPixelY), aColor);"
EOL" }";
#endif