Path: blob/master/thirdparty/embree/kernels/subdiv/bilinear_patch.h
9913 views
// Copyright 2009-2021 Intel Corporation1// SPDX-License-Identifier: Apache-2.023#pragma once45#include "catmullclark_patch.h"6#include "bezier_curve.h"78namespace embree9{10template<typename Vertex, typename Vertex_t = Vertex>11class __aligned(64) BilinearPatchT12{13typedef CatmullClark1RingT<Vertex,Vertex_t> CatmullClarkRing;14typedef CatmullClarkPatchT<Vertex,Vertex_t> CatmullClarkPatch;1516public:17Vertex v[4];1819public:2021__forceinline BilinearPatchT () {}2223__forceinline BilinearPatchT (const HalfEdge* edge, const BufferView<Vertex>& vertices) {24init(edge,vertices.getPtr(),vertices.getStride());25}2627__forceinline BilinearPatchT (const HalfEdge* edge, const char* vertices, size_t stride) {28init(edge,vertices,stride);29}3031__forceinline void init (const HalfEdge* edge, const char* vertices, size_t stride)32{33v[0] = Vertex::loadu(vertices+edge->getStartVertexIndex()*stride); edge = edge->next();34v[1] = Vertex::loadu(vertices+edge->getStartVertexIndex()*stride); edge = edge->next();35v[2] = Vertex::loadu(vertices+edge->getStartVertexIndex()*stride); edge = edge->next();36v[3] = Vertex::loadu(vertices+edge->getStartVertexIndex()*stride); edge = edge->next();37}3839__forceinline BilinearPatchT (const CatmullClarkPatch& patch)40{41v[0] = patch.ring[0].getLimitVertex();42v[1] = patch.ring[1].getLimitVertex();43v[2] = patch.ring[2].getLimitVertex();44v[3] = patch.ring[3].getLimitVertex();45}4647__forceinline BBox<Vertex> bounds() const48{4950BBox<Vertex> bounds (v[0]);51bounds.extend(v[1]);52bounds.extend(v[2]);53bounds.extend(v[3]);54return bounds;55}5657__forceinline Vertex eval(const float uu, const float vv) const {58return lerp(lerp(v[0],v[1],uu),lerp(v[3],v[2],uu),vv);59}6061__forceinline Vertex eval_du(const float uu, const float vv) const {62return lerp(v[1]-v[0],v[2]-v[3],vv);63}6465__forceinline Vertex eval_dv(const float uu, const float vv) const {66return lerp(v[3]-v[0],v[2]-v[1],uu);67}6869__forceinline Vertex eval_dudu(const float uu, const float vv) const {70return Vertex(zero);71}7273__forceinline Vertex eval_dvdv(const float uu, const float vv) const {74return Vertex(zero);75}7677__forceinline Vertex eval_dudv(const float uu, const float vv) const {78return (v[2]-v[3]) - (v[1]-v[0]);79}8081__forceinline Vertex normal(const float uu, const float vv) const {82return cross(eval_du(uu,vv),eval_dv(uu,vv));83}8485__forceinline void eval(const float u, const float v,86Vertex* P, Vertex* dPdu, Vertex* dPdv, Vertex* ddPdudu, Vertex* ddPdvdv, Vertex* ddPdudv,87const float dscale = 1.0f) const88{89if (P) {90*P = eval(u,v);91}92if (dPdu) {93assert(dPdu); *dPdu = eval_du(u,v)*dscale;94assert(dPdv); *dPdv = eval_dv(u,v)*dscale;95}96if (ddPdudu) {97assert(ddPdudu); *ddPdudu = eval_dudu(u,v)*sqr(dscale);98assert(ddPdvdv); *ddPdvdv = eval_dvdv(u,v)*sqr(dscale);99assert(ddPdudv); *ddPdudv = eval_dudv(u,v)*sqr(dscale);100}101}102103template<class vfloat>104__forceinline Vec3<vfloat> eval(const vfloat& uu, const vfloat& vv) const105{106const vfloat x = lerp(lerp(v[0].x,v[1].x,uu),lerp(v[3].x,v[2].x,uu),vv);107const vfloat y = lerp(lerp(v[0].y,v[1].y,uu),lerp(v[3].y,v[2].y,uu),vv);108const vfloat z = lerp(lerp(v[0].z,v[1].z,uu),lerp(v[3].z,v[2].z,uu),vv);109return Vec3<vfloat>(x,y,z);110}111112template<class vfloat>113__forceinline Vec3<vfloat> eval_du(const vfloat& uu, const vfloat& vv) const114{115const vfloat x = lerp(v[1].x-v[0].x,v[2].x-v[3].x,vv);116const vfloat y = lerp(v[1].y-v[0].y,v[2].y-v[3].y,vv);117const vfloat z = lerp(v[1].z-v[0].z,v[2].z-v[3].z,vv);118return Vec3<vfloat>(x,y,z);119}120121template<class vfloat>122__forceinline Vec3<vfloat> eval_dv(const vfloat& uu, const vfloat& vv) const123{124const vfloat x = lerp(v[3].x-v[0].x,v[2].x-v[1].x,uu);125const vfloat y = lerp(v[3].y-v[0].y,v[2].y-v[1].y,uu);126const vfloat z = lerp(v[3].z-v[0].z,v[2].z-v[1].z,uu);127return Vec3<vfloat>(x,y,z);128}129130template<typename vfloat>131__forceinline Vec3<vfloat> normal(const vfloat& uu, const vfloat& vv) const {132return cross(eval_du(uu,vv),eval_dv(uu,vv));133}134135template<class vfloat>136__forceinline vfloat eval(const size_t i, const vfloat& uu, const vfloat& vv) const {137return lerp(lerp(v[0][i],v[1][i],uu),lerp(v[3][i],v[2][i],uu),vv);138}139140template<class vfloat>141__forceinline vfloat eval_du(const size_t i, const vfloat& uu, const vfloat& vv) const {142return lerp(v[1][i]-v[0][i],v[2][i]-v[3][i],vv);143}144145template<class vfloat>146__forceinline vfloat eval_dv(const size_t i, const vfloat& uu, const vfloat& vv) const {147return lerp(v[3][i]-v[0][i],v[2][i]-v[1][i],uu);148}149150template<class vfloat>151__forceinline vfloat eval_dudu(const size_t i, const vfloat& uu, const vfloat& vv) const {152return vfloat(zero);153}154155template<class vfloat>156__forceinline vfloat eval_dvdv(const size_t i, const vfloat& uu, const vfloat& vv) const {157return vfloat(zero);158}159160template<class vfloat>161__forceinline vfloat eval_dudv(const size_t i, const vfloat& uu, const vfloat& vv) const {162return (v[2][i]-v[3][i]) - (v[1][i]-v[0][i]);163}164165template<typename vbool, typename vfloat>166__forceinline void eval(const vbool& valid, const vfloat& uu, const vfloat& vv,167float* P, float* dPdu, float* dPdv, float* ddPdudu, float* ddPdvdv, float* ddPdudv,168const float dscale, const size_t dstride, const size_t N) const169{170if (P) {171for (size_t i=0; i<N; i++) vfloat::store(valid,P+i*dstride,eval(i,uu,vv));172}173if (dPdu) {174for (size_t i=0; i<N; i++) {175assert(dPdu); vfloat::store(valid,dPdu+i*dstride,eval_du(i,uu,vv)*dscale);176assert(dPdv); vfloat::store(valid,dPdv+i*dstride,eval_dv(i,uu,vv)*dscale);177}178}179if (ddPdudu) {180for (size_t i=0; i<N; i++) {181assert(ddPdudu); vfloat::store(valid,ddPdudu+i*dstride,eval_dudu(i,uu,vv)*sqr(dscale));182assert(ddPdvdv); vfloat::store(valid,ddPdvdv+i*dstride,eval_dvdv(i,uu,vv)*sqr(dscale));183assert(ddPdudv); vfloat::store(valid,ddPdudv+i*dstride,eval_dudv(i,uu,vv)*sqr(dscale));184}185}186}187};188189typedef BilinearPatchT<Vec3fa,Vec3fa_t> BilinearPatch3fa;190}191192193