vtk-m/vtkm/rendering/CanvasRayTracer.cxx

201 lines
6.8 KiB
C++
Raw Normal View History

//============================================================================
// Copyright (c) Kitware, Inc.
// All rights reserved.
// See LICENSE.txt for details.
// This software is distributed WITHOUT ANY WARRANTY; without even
// the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
// PURPOSE. See the above copyright notice for more information.
//
// Copyright 2015 Sandia Corporation.
// Copyright 2015 UT-Battelle, LLC.
// Copyright 2015 Los Alamos National Security.
//
// Under the terms of Contract DE-AC04-94AL85000 with Sandia Corporation,
// the U.S. Government retains certain rights in this software.
//
// Under the terms of Contract DE-AC52-06NA25396 with Los Alamos National
// Laboratory (LANL), the U.S. Government retains certain rights in
// this software.
//============================================================================
#include <vtkm/rendering/CanvasRayTracer.h>
#include <vtkm/cont/TryExecute.h>
#include <vtkm/exec/ExecutionWholeArray.h>
#include <vtkm/rendering/Canvas.h>
#include <vtkm/rendering/Color.h>
#include <vtkm/rendering/raytracing/Ray.h>
#include <vtkm/worklet/DispatcherMapField.h>
2017-05-18 14:51:24 +00:00
#include <vtkm/worklet/WorkletMapField.h>
2017-05-18 14:29:41 +00:00
namespace vtkm
{
namespace rendering
{
namespace internal
{
class SurfaceConverter : public vtkm::worklet::WorkletMapField
{
2017-08-01 17:06:37 +00:00
vtkm::Matrix<vtkm::Float32, 4, 4> ViewProjMat;
public:
VTKM_CONT
2017-08-01 17:06:37 +00:00
SurfaceConverter(const vtkm::Matrix<vtkm::Float32, 4, 4> viewProjMat)
: ViewProjMat(viewProjMat)
{
}
2017-08-01 17:06:37 +00:00
typedef void ControlSignature(FieldIn<>,
WholeArrayInOut<>,
2017-08-01 17:06:37 +00:00
FieldIn<>,
FieldIn<>,
FieldIn<>,
ExecObject,
ExecObject);
typedef void ExecutionSignature(_1, _2, _3, _4, _5, _6, _7, WorkIndex);
template <typename Precision, typename ColorPortalType>
VTKM_EXEC void operator()(
const vtkm::Id& pixelIndex,
ColorPortalType& colorBufferIn,
const Precision& inDepth,
2017-08-01 17:06:37 +00:00
const vtkm::Vec<Precision, 3>& origin,
const vtkm::Vec<Precision, 3>& dir,
vtkm::exec::ExecutionWholeArray<vtkm::Float32>& depthBuffer,
vtkm::exec::ExecutionWholeArray<vtkm::Vec<vtkm::Float32, 4>>& colorBuffer,
const vtkm::Id& index) const
{
2017-08-01 17:06:37 +00:00
vtkm::Vec<Precision, 3> intersection = origin + inDepth * dir;
vtkm::Vec<vtkm::Float32, 4> point;
point[0] = static_cast<vtkm::Float32>(intersection[0]);
point[1] = static_cast<vtkm::Float32>(intersection[1]);
point[2] = static_cast<vtkm::Float32>(intersection[2]);
point[3] = 1.f;
vtkm::Vec<vtkm::Float32, 4> newpoint;
newpoint = vtkm::MatrixMultiply(this->ViewProjMat, point);
newpoint[0] = newpoint[0] / newpoint[3];
newpoint[1] = newpoint[1] / newpoint[3];
newpoint[2] = newpoint[2] / newpoint[3];
vtkm::Float32 depth = newpoint[2];
depth = 0.5f * (depth) + 0.5f;
vtkm::Vec<vtkm::Float32, 4> color;
color[0] = static_cast<vtkm::Float32>(colorBufferIn.Get(index * 4 + 0));
color[1] = static_cast<vtkm::Float32>(colorBufferIn.Get(index * 4 + 1));
color[2] = static_cast<vtkm::Float32>(colorBufferIn.Get(index * 4 + 2));
color[3] = static_cast<vtkm::Float32>(colorBufferIn.Get(index * 4 + 3));
// blend the mapped color with existing canvas color
vtkm::Vec<vtkm::Float32, 4> inColor = colorBuffer.Get(pixelIndex);
vtkm::Float32 alpha = inColor[3] * (1.f - color[3]);
color[0] = color[0] + inColor[0] * alpha;
color[1] = color[1] + inColor[1] * alpha;
color[2] = color[2] + inColor[2] * alpha;
color[3] = alpha + color[3];
// clamp
for (vtkm::Int32 i = 0; i < 4; ++i)
{
color[i] = vtkm::Min(1.f, vtkm::Max(color[i], 0.f));
}
// The existng depth should already been feed into thge ray mapper
// so no color contribution will exist past the existing depth.
2017-08-01 17:06:37 +00:00
depthBuffer.Set(pixelIndex, depth);
colorBuffer.Set(pixelIndex, color);
}
}; //class SurfaceConverter
template <typename Precision>
struct WriteFunctor
{
protected:
vtkm::rendering::CanvasRayTracer* Canvas;
2017-08-01 17:06:37 +00:00
const vtkm::rendering::raytracing::Ray<Precision>& Rays;
const vtkm::cont::ArrayHandle<Precision>& Colors;
const vtkm::rendering::Camera& CameraView;
2017-08-01 17:06:37 +00:00
vtkm::Matrix<vtkm::Float32, 4, 4> ViewProjMat;
public:
VTKM_CONT
WriteFunctor(vtkm::rendering::CanvasRayTracer* canvas,
2017-08-01 17:06:37 +00:00
const vtkm::rendering::raytracing::Ray<Precision>& rays,
const vtkm::cont::ArrayHandle<Precision>& colors,
const vtkm::rendering::Camera& camera)
: Canvas(canvas)
2017-08-01 17:06:37 +00:00
, Rays(rays)
, Colors(colors)
, CameraView(camera)
{
2017-08-01 17:06:37 +00:00
ViewProjMat = vtkm::MatrixMultiply(
CameraView.CreateProjectionMatrix(Canvas->GetWidth(), Canvas->GetHeight()),
CameraView.CreateViewMatrix());
}
template <typename Device>
VTKM_CONT bool operator()(Device)
{
VTKM_IS_DEVICE_ADAPTER_TAG(Device);
2017-08-01 17:06:37 +00:00
vtkm::worklet::DispatcherMapField<SurfaceConverter, Device>(SurfaceConverter(ViewProjMat))
.Invoke(
2017-08-01 17:06:37 +00:00
Rays.PixelIdx,
Colors,
2017-08-01 17:06:37 +00:00
Rays.Distance,
Rays.Origin,
Rays.Dir,
vtkm::exec::ExecutionWholeArray<vtkm::Float32>(Canvas->GetDepthBuffer()),
vtkm::exec::ExecutionWholeArray<vtkm::Vec<vtkm::Float32, 4>>(Canvas->GetColorBuffer()));
return true;
}
};
template <typename Precision>
2017-08-01 17:06:37 +00:00
VTKM_CONT void WriteToCanvas(const vtkm::rendering::raytracing::Ray<Precision>& rays,
const vtkm::cont::ArrayHandle<Precision>& colors,
const vtkm::rendering::Camera& camera,
vtkm::rendering::CanvasRayTracer* canvas)
{
2017-08-01 17:06:37 +00:00
WriteFunctor<Precision> functor(canvas, rays, colors, camera);
vtkm::cont::TryExecute(functor);
//Force the transfer so the vectors contain data from device
canvas->GetColorBuffer().GetPortalControl().Get(0);
canvas->GetDepthBuffer().GetPortalControl().Get(0);
}
} // namespace internal
CanvasRayTracer::CanvasRayTracer(vtkm::Id width, vtkm::Id height)
: Canvas(width, height)
2017-05-18 14:29:41 +00:00
{
}
CanvasRayTracer::~CanvasRayTracer()
2017-05-18 14:29:41 +00:00
{
}
2017-08-01 17:06:37 +00:00
void CanvasRayTracer::WriteToCanvas(const vtkm::rendering::raytracing::Ray<vtkm::Float32>& rays,
const vtkm::cont::ArrayHandle<vtkm::Float32>& colors,
const vtkm::rendering::Camera& camera)
{
2017-08-01 17:06:37 +00:00
internal::WriteToCanvas(rays, colors, camera, this);
}
2017-08-01 17:06:37 +00:00
void CanvasRayTracer::WriteToCanvas(const vtkm::rendering::raytracing::Ray<vtkm::Float64>& rays,
const vtkm::cont::ArrayHandle<vtkm::Float64>& colors,
const vtkm::rendering::Camera& camera)
{
2017-08-01 17:06:37 +00:00
internal::WriteToCanvas(rays, colors, camera, this);
}
2017-05-18 14:29:41 +00:00
vtkm::rendering::Canvas* CanvasRayTracer::NewCopy() const
{
return new vtkm::rendering::CanvasRayTracer(*this);
}
}
}