214 lines
6.3 KiB
C++
214 lines
6.3 KiB
C++
// Copyright 2009-2021 Intel Corporation
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
#include "scene_grid_mesh.h"
|
|
#include "scene.h"
|
|
|
|
namespace embree
|
|
{
|
|
#if defined(EMBREE_LOWEST_ISA)
|
|
|
|
GridMesh::GridMesh (Device* device)
|
|
: Geometry(device,GTY_GRID_MESH,0,1)
|
|
{
|
|
vertices.resize(numTimeSteps);
|
|
}
|
|
|
|
void GridMesh::setMask (unsigned mask)
|
|
{
|
|
this->mask = mask;
|
|
Geometry::update();
|
|
}
|
|
|
|
void GridMesh::setNumTimeSteps (unsigned int numTimeSteps)
|
|
{
|
|
vertices.resize(numTimeSteps);
|
|
Geometry::setNumTimeSteps(numTimeSteps);
|
|
}
|
|
|
|
void GridMesh::setVertexAttributeCount (unsigned int N)
|
|
{
|
|
vertexAttribs.resize(N);
|
|
Geometry::update();
|
|
}
|
|
|
|
void GridMesh::setBuffer(RTCBufferType type, unsigned int slot, RTCFormat format, const Ref<Buffer>& buffer, size_t offset, size_t stride, unsigned int num)
|
|
{
|
|
/* verify that all accesses are 4 bytes aligned */
|
|
if (((size_t(buffer->getPtr()) + offset) & 0x3) || (stride & 0x3))
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "data must be 4 bytes aligned");
|
|
|
|
if (type == RTC_BUFFER_TYPE_VERTEX)
|
|
{
|
|
if (format != RTC_FORMAT_FLOAT3)
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "invalid vertex buffer format");
|
|
|
|
/* if buffer is larger than 16GB the premultiplied index optimization does not work */
|
|
if (stride*num > 16ll*1024ll*1024ll*1024ll)
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "vertex buffer can be at most 16GB large");
|
|
|
|
if (slot >= vertices.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid vertex buffer slot");
|
|
|
|
vertices[slot].set(buffer, offset, stride, num, format);
|
|
vertices[slot].checkPadding16();
|
|
vertices0 = vertices[0];
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_VERTEX_ATTRIBUTE)
|
|
{
|
|
if (format < RTC_FORMAT_FLOAT || format > RTC_FORMAT_FLOAT16)
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "invalid vertex attribute buffer format");
|
|
|
|
if (slot >= vertexAttribs.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "invalid vertex attribute buffer slot");
|
|
|
|
vertexAttribs[slot].set(buffer, offset, stride, num, format);
|
|
vertexAttribs[slot].checkPadding16();
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_GRID)
|
|
{
|
|
if (slot != 0)
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
if (format != RTC_FORMAT_GRID)
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION, "invalid index buffer format");
|
|
|
|
grids.set(buffer, offset, stride, num, format);
|
|
setNumPrimitives(num);
|
|
}
|
|
else
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "unknown buffer type");
|
|
}
|
|
|
|
void* GridMesh::getBuffer(RTCBufferType type, unsigned int slot)
|
|
{
|
|
if (type == RTC_BUFFER_TYPE_GRID)
|
|
{
|
|
if (slot != 0)
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
return grids.getPtr();
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_VERTEX)
|
|
{
|
|
if (slot >= vertices.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
return vertices[slot].getPtr();
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_VERTEX_ATTRIBUTE)
|
|
{
|
|
if (slot >= vertexAttribs.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
return vertexAttribs[slot].getPtr();
|
|
}
|
|
else
|
|
{
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "unknown buffer type");
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
void GridMesh::updateBuffer(RTCBufferType type, unsigned int slot)
|
|
{
|
|
if (type == RTC_BUFFER_TYPE_GRID)
|
|
{
|
|
if (slot != 0)
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
grids.setModified();
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_VERTEX)
|
|
{
|
|
if (slot >= vertices.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
vertices[slot].setModified();
|
|
}
|
|
else if (type == RTC_BUFFER_TYPE_VERTEX_ATTRIBUTE)
|
|
{
|
|
if (slot >= vertexAttribs.size())
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "invalid buffer slot");
|
|
vertexAttribs[slot].setModified();
|
|
}
|
|
else
|
|
{
|
|
throw_RTCError(RTC_ERROR_INVALID_ARGUMENT, "unknown buffer type");
|
|
}
|
|
|
|
Geometry::update();
|
|
}
|
|
|
|
void GridMesh::commit()
|
|
{
|
|
/* verify that stride of all time steps are identical */
|
|
for (unsigned int t=0; t<numTimeSteps; t++)
|
|
if (vertices[t].getStride() != vertices[0].getStride())
|
|
throw_RTCError(RTC_ERROR_INVALID_OPERATION,"stride of vertex buffers have to be identical for each time step");
|
|
|
|
#if defined(EMBREE_SYCL_SUPPORT)
|
|
|
|
/* build quadID_to_primID_xy mapping when hardware ray tracing is supported */
|
|
if (DeviceGPU* gpu_device = dynamic_cast<DeviceGPU*>(device))
|
|
{
|
|
const size_t numQuads = getNumTotalQuads();
|
|
quadID_to_primID_xy.resize(numQuads);
|
|
|
|
for (uint32_t primID=0, quadID=0; primID<size(); primID++)
|
|
{
|
|
const Grid& g = grid(primID);
|
|
for (ssize_t y=0; y<ssize_t(g.resY)-1; y++)
|
|
for (ssize_t x=0; x<ssize_t(g.resX)-1; x++)
|
|
quadID_to_primID_xy[quadID++] = { primID, (uint16_t) x, (uint16_t) y };
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
Geometry::commit();
|
|
}
|
|
|
|
void GridMesh::addElementsToCount (GeometryCounts& counts) const
|
|
{
|
|
if (numTimeSteps == 1) {
|
|
counts.numGrids += numPrimitives;
|
|
for (size_t primID=0; primID<numPrimitives; primID++)
|
|
counts.numSubGrids += getNumSubGrids(primID);
|
|
}
|
|
else {
|
|
counts.numMBGrids += numPrimitives;
|
|
for (size_t primID=0; primID<numPrimitives; primID++)
|
|
counts.numMBSubGrids += getNumSubGrids(primID);
|
|
}
|
|
}
|
|
|
|
bool GridMesh::verify()
|
|
{
|
|
/*! verify size of vertex arrays */
|
|
if (vertices.size() == 0) return false;
|
|
for (const auto& buffer : vertices)
|
|
if (buffer.size() != numVertices())
|
|
return false;
|
|
|
|
/*! verify size of user vertex arrays */
|
|
for (const auto& buffer : vertexAttribs)
|
|
if (buffer.size() != numVertices())
|
|
return false;
|
|
|
|
/*! verify vertices */
|
|
for (const auto& buffer : vertices)
|
|
for (size_t i=0; i<buffer.size(); i++)
|
|
if (!isvalid(buffer[i]))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
void GridMesh::interpolate(const RTCInterpolateArguments* const args) {
|
|
interpolate_impl<4>(args);
|
|
}
|
|
|
|
#endif
|
|
|
|
namespace isa
|
|
{
|
|
GridMesh* createGridMesh(Device* device) {
|
|
return new GridMeshISA(device);
|
|
}
|
|
}
|
|
}
|