Initial commit.
This commit is contained in:
commit
d3bb49b3f5
1073 changed files with 484757 additions and 0 deletions
203
Framework/external/embree/tutorials/common/texture/texture2d.cpp
vendored
Normal file
203
Framework/external/embree/tutorials/common/texture/texture2d.cpp
vendored
Normal file
|
|
@ -0,0 +1,203 @@
|
|||
// Copyright 2009-2021 Intel Corporation
|
||||
// SPDX-License-Identifier: Apache-2.0
|
||||
|
||||
#include "texture2d.h"
|
||||
|
||||
namespace embree {
|
||||
|
||||
|
||||
// Low-level texel accessors
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
// TODO blocking
|
||||
|
||||
inline Vec4f getTexel_RGBA8(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
const uint32_t c = ((const uint32_t *)self->data)[i.y*self->size.x + i.x];
|
||||
const uint32_t r = c & 0xff;
|
||||
const uint32_t g = (c >> 8) & 0xff;
|
||||
const uint32_t b = (c >> 16) & 0xff;
|
||||
const uint32_t a = c >> 24;
|
||||
return Vec4f((float)r, (float)g, (float)b, (float)a)*(1.f/255.f);
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_RGB8(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
const uint8_t *texel = (const uint8_t *)self->data;
|
||||
const uint32_t texelOfs = 3*(i.y*self->size.x + i.x);
|
||||
const uint32_t r = texel[texelOfs];
|
||||
const uint32_t g = texel[texelOfs+1];
|
||||
const uint32_t b = texel[texelOfs+2];
|
||||
return Vec4f(Vec3fa((float)r, (float)g, (float)b)*(1.f/255.f), 1.f);
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_R8(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
const uint8_t c = ((const uint8_t *)self->data)[i.y*self->size.x + i.x];
|
||||
return Vec4f(c*(1.f/255.f), 0.0f, 0.0f, 1.f);
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_SRGBA(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
return srgba_to_linear(getTexel_RGBA8(self, i));
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_SRGB(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
return srgba_to_linear(getTexel_RGB8(self, i));
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_RGBA32F(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
return ((const Vec4f *)self->data)[i.y*self->size.x + i.x];
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_RGB32F(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
Vec3fa v = ((const Vec3fa*)self->data)[i.y*self->size.x + i.x];
|
||||
return Vec4f(v, 1.f);
|
||||
}
|
||||
|
||||
inline Vec4f getTexel_R32F(const Texture2D *self, const Vec2i i)
|
||||
{
|
||||
assert(self);
|
||||
float v = ((const float*)self->data)[i.y*self->size.x + i.x];
|
||||
return Vec4f(v, 0.f, 0.f, 1.f);
|
||||
}
|
||||
|
||||
|
||||
// Texture coordinate utilities
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
inline Vec2i nearest_coords(const Texture2D *self, const Vec2f p)
|
||||
{
|
||||
// repeat: get remainder within [0..1] parameter space
|
||||
Vec2f tc = frac(p);
|
||||
tc = max(tc, Vec2f(0.0f)); // filter out inf/NaN
|
||||
|
||||
// scale by texture size
|
||||
tc = tc * self->sizef;
|
||||
|
||||
// nearest
|
||||
return Vec2i(tc);
|
||||
}
|
||||
|
||||
struct BilinCoords {
|
||||
Vec2i st0;
|
||||
Vec2i st1;
|
||||
Vec2f frac;
|
||||
};
|
||||
|
||||
inline BilinCoords bilinear_coords(const Texture2D *self, const Vec2f p)
|
||||
{
|
||||
BilinCoords coords;
|
||||
|
||||
// repeat: get remainder within [0..1] parameter space
|
||||
// lower sample shifted by half a texel
|
||||
Vec2f tc = frac(p - self->halfTexel);
|
||||
tc = max(tc, Vec2f(0.0f)); // filter out inf/NaN
|
||||
|
||||
// scale by texture size
|
||||
tc = tc * self->sizef;
|
||||
coords.frac = frac(tc);
|
||||
|
||||
coords.st0 = Vec2i(tc);
|
||||
coords.st1 = coords.st0 + 1;
|
||||
// handle border cases
|
||||
if (coords.st1.x >= self->size.x)
|
||||
coords.st1.x = 0;
|
||||
if (coords.st1.y >= self->size.y)
|
||||
coords.st1.y = 0;
|
||||
|
||||
return coords;
|
||||
}
|
||||
|
||||
inline Vec4f bilerp(const Vec2f frac, const Vec4f c00, const Vec4f c01, const Vec4f c10, const Vec4f c11)
|
||||
{
|
||||
return lerpr(frac.y,
|
||||
lerpr(frac.x, c00, c01),
|
||||
lerpr(frac.x, c10, c11));
|
||||
}
|
||||
|
||||
|
||||
// Implementations of Texture2D_get for different formats and filter modi
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#define __define_tex_get(FMT) \
|
||||
\
|
||||
static Vec4f Texture2D_nearest_##FMT(const Texture2D *self, \
|
||||
const Vec2f &p) \
|
||||
{ \
|
||||
return getTexel_##FMT(self, nearest_coords(self, p)); \
|
||||
} \
|
||||
\
|
||||
static Vec4f Texture2D_bilinear_##FMT(const Texture2D *self, \
|
||||
const Vec2f &p) \
|
||||
{ \
|
||||
BilinCoords cs = bilinear_coords(self, p); \
|
||||
\
|
||||
const Vec4f c00 = getTexel_##FMT(self, Vec2i(cs.st0.x, cs.st0.y)); \
|
||||
const Vec4f c01 = getTexel_##FMT(self, Vec2i(cs.st1.x, cs.st0.y)); \
|
||||
const Vec4f c10 = getTexel_##FMT(self, Vec2i(cs.st0.x, cs.st1.y)); \
|
||||
const Vec4f c11 = getTexel_##FMT(self, Vec2i(cs.st1.x, cs.st1.y)); \
|
||||
\
|
||||
return bilerp(cs.frac, c00, c01, c10, c11); \
|
||||
}
|
||||
|
||||
#define __define_tex_get_case(FMT) \
|
||||
case TEXTURE_##FMT: return filter_nearest ? &Texture2D_nearest_##FMT : \
|
||||
&Texture2D_bilinear_##FMT;
|
||||
|
||||
#define __foreach_fetcher(FCT) \
|
||||
FCT(RGBA8) \
|
||||
FCT(SRGBA) \
|
||||
FCT(RGBA32F) \
|
||||
FCT(RGB8) \
|
||||
FCT(SRGB) \
|
||||
FCT(RGB32F) \
|
||||
FCT(R8) \
|
||||
FCT(R32F)
|
||||
|
||||
__foreach_fetcher(__define_tex_get)
|
||||
|
||||
static Texture2D_get Texture2D_get_addr(const uint32_t type,
|
||||
const bool filter_nearest)
|
||||
{
|
||||
switch (type) {
|
||||
__foreach_fetcher(__define_tex_get_case)
|
||||
}
|
||||
return 0;
|
||||
};
|
||||
|
||||
#undef __define_tex_get
|
||||
#undef __define_tex_get_addr
|
||||
#undef __foreach_fetcher
|
||||
|
||||
|
||||
// Exports (called from C++)
|
||||
//////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
extern "C" void *Texture2D_create(Vec2i &size, void *data,
|
||||
uint32_t type, uint32_t flags)
|
||||
{
|
||||
Texture2D *self = (Texture2D*) alignedUSMMalloc(sizeof(Texture2D),16);
|
||||
self->size = size;
|
||||
|
||||
// Due to float rounding frac(x) can be exactly 1.0f (e.g. for very small
|
||||
// negative x), although it should be strictly smaller than 1.0f. We handle
|
||||
// this case by having sizef slightly smaller than size, such that
|
||||
// frac(x)*sizef is always < size.
|
||||
self->sizef = Vec2f(nextafter((float)size.x, -1.0f), nextafter((float)size.y, -1.0f));
|
||||
self->halfTexel = Vec2f(0.5f/size.x, 0.5f/size.y);
|
||||
self->data = data;
|
||||
self->get = Texture2D_get_addr(type, flags & TEXTURE_FILTER_NEAREST);
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
} // namespace embree
|
||||
Loading…
Add table
Add a link
Reference in a new issue