3

使用 C++ 和 OSG 我正在尝试将浮动纹理上传到我的着色器,但不知何故它似乎不起作用。最后,我发布了我的部分代码。主要问题是如何使用浮点数组中的数据创建 osg::Image 对象。在 OpenGL 中,所需的代码是

glTexImage2D(GL_TEXTURE_2D, 0, GL_LUMINANCE32F_ARB, width, height, 0, 
             GL_LUMINANCE, GL_FLOAT, data);

但在这种情况下,我必须使用 OSG。

使用时代码运行良好

Image* image = osgDB::readImageFile("someImage.jpg");  

代替

image = new Image;  

但我需要上传生成的浮点数据。也无法切换到无符号字符数组,因为我需要着色器代码中的 GL_LUMINANCE32F_ARB 数据范围。

我希望有人可以在这里帮助我,因为谷歌无法帮助我(谷歌搜索例如:osg float image)。所以这是我的代码。

using namespace std;
using namespace osg;
//...
float* data = new float[width*height];
fill_n(data, size, 1.0); // << I actually do this for testing purposes
Texture2D* texture = new Texture2D;
Image* image = new Image;
osg::State* state = new osg::State;
Uniform* uniform = new Uniform(Uniform::SAMPLER_2D, "texUniform");
texture->setInternalFormat(GL_LUMINANCE32F_ARB);
texture->setDataVariance(osg::Object::DYNAMIC);
texture->setFilter(osg::Texture2D::MIN_FILTER, osg::Texture2D::LINEAR);
texture->setFilter(osg::Texture2D::MAG_FILTER, osg::Texture2D::LINEAR);
texture->setWrap(osg::Texture2D::WRAP_T, osg::Texture2D::CLAMP_TO_EDGE);
texture->setWrap(osg::Texture2D::WRAP_S, osg::Texture2D::CLAMP_TO_EDGE);
if (data == NULL)
  cout << "texdata null" << endl; // << this is not printed
image->setImage(width, height, 1, GL_LUMINANCE32F_ARB,
                GL_LUMINANCE, GL_FLOAT,
                (unsigned char*)data, osg::Image::USE_NEW_DELETE);
if (image->getDataPointer() == NULL)
  cout << "datapointernull" << endl; // << this is printed
if (!image->valid())
  exit(1); //  << here the code exits (hard exit just for testing purposes)
osgDB::writeImageFile(*image, "blah.png");
texture->setInternalFormat(GL_LUMINANCE32F_ARB);
texture->setImage(image);
camera->getOrCreateStateSet()->setTextureAttributeAndModes(4, texture);
state->setActiveTextureUnit(4);
texture->apply(*state);
uniform->set(4);
addProgrammUniform(uniform);

我在网上找到了另一种方法,让 osg::Image 创建数据并在之后填充。但不知何故,这也行不通。我在新的 XYZ 之后插入了这个;线。

image->setInternalTextureFormat(GL_LUMINANCE32F_ARB);
image->allocateImage(width,height,1,GL_LUMINANCE,GL_FLOAT);
if (image->data() == NULL)
  cout << "null here?!" << endl; // << this is printed.
4

2 回答 2

1

我使用以下(简化的)代码来创建和设置浮点纹理:

// Create texture and image
osg::Texture* texture = new osg::Texture2D;
osg::Image* image = new osg::Image();
image->allocateImage(size, size, 1, GL_LUMINANCE, GL_FLOAT);
texture->setInternalFormat(GL_LUMINANCE32F_ARB);
texture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
texture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
texture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
texture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
texture->setImage(image);

// Set texture to node
osg::StateSet* stateSet = node->getOrCreateStateSet();
stateSet->setTextureAttributeAndModes(TEXTURE_UNIT_NUMBER, texture);

// Set data
float* data = reinterpret_cast<float*>(image->data());
/* ...data processing... */
image->dirty();

您可能想要更改一些参数,但这应该让您有一个开始。我相信在你的情况下TEXTURE_UNIT_NUMBER应该设置为 4。

于 2013-07-16T07:48:57.103 回答
0

但我需要上传生成的浮点数据。也无法切换到无符号字符数组,因为我需要着色器代码中的 GL_LUMINANCE32F_ARB 数据范围。

osgDB::writeImageFile(*image, "blah.png");

png 文件不支持每通道 32 位数据,因此您不能以这种方式将纹理写入文件。请参阅 libpng 书

PNG 灰度图像支持任何图像类型的最广泛的像素深度。支持 1、2、4、8 和 16 位的深度,涵盖从简单的黑白扫描到全深度医学和原始天文图像的所有内容。 [63]

[63] 校准后的天文图像数据通常存储为 32 位或 64 位浮点值,一些原始数据表示为 32 位整数。PNG 不直接支持这两种格式,尽管原则上可以设计一个辅助块来保存正确的转换信息。然而,转换具有超过 16 位动态范围的数据将是一种有损转换——至少,除非滥用 PNG 的 alpha 通道或 RGB 功能。

对于每通道 32 位,请查看OpenEXR 格式

但是,如果 16 位浮点数(即半个浮点数)就足够了,那么您可以这样做:

osg::ref_ptr<osg::Image> heightImage = new osg::Image;
int pixelFormat = GL_LUMINANCE;
int type = GL_HALF_FLOAT;
heightImage->allocateImage(tex_width, tex_height, 1, pixelFormat, type);

现在要实际使用和编写半浮点数,您可以使用GLM 库。您可以通过 include 获得半浮点类型<glm/detail/type_half.hpp>,然后调用hdata. 您现在需要从图像中获取数据指针并将其转换为上述格式:

glm::detail::hdata *data = reinterpret_cast<glm::detail::hdata*>(heightImage->data());

然后您可以像访问一维数组一样访问它,例如

data[currentRow*tex_width+ currentColumn] = glm::detail::toFloat16(3.1415f);

并不是说如果您将相同的数据写入bmptif文件(使用 osg 插件),结果将不正确。就我而言,我只是将预期图像的左半部分拉伸到全宽,而不是灰度,而是一些奇怪的颜色编码。

于 2018-11-16T11:47:58.317 回答