首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >OpenSceneGraph浮点图像

OpenSceneGraph浮点图像
EN

Stack Overflow用户
提问于 2012-07-31 15:42:49
回答 2查看 3.7K关注 0票数 3

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

代码语言:javascript
复制
glTexImage2D(GL_TEXTURE_2D, 0, GL_LUMINANCE32F_ARB, width, height, 0, 
             GL_LUMINANCE, GL_FLOAT, data);

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

代码在使用时运行良好

代码语言:javascript
复制
Image* image = osgDB::readImageFile("someImage.jpg");  

而不是

代码语言:javascript
复制
image = new Image;  

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

我希望有人能在这里帮助我,因为谷歌不能帮助我(谷歌搜索例如: osg浮动图像)。所以这是我的代码。

代码语言:javascript
复制
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;行之后插入了这个。

代码语言:javascript
复制
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.
EN

回答 2

Stack Overflow用户

发布于 2013-07-16 15:48:57

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

代码语言:javascript
复制
// 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。

票数 1
EN

Stack Overflow用户

发布于 2018-11-16 19:47:58

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

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

png文件不支持32位/通道数据,因此不能以这种方式将纹理写入文件。See the libpng book

PNG灰度图像支持任何图像类型中最大范围的像素深度。支持1、2、4、8和16位的深度,涵盖从简单的黑白扫描到全深度医学和原始天文图像的所有内容。63校准的天文图像数据通常存储为32位或64位浮点值,一些原始数据表示为32位整数。PNG不直接支持这两种格式,尽管原则上可以设计一个辅助块来保存适当的转换信息。但是,动态范围超过16位的数据的转换将是一种有损转换--至少,除非滥用PNG的alpha通道或RGB功能。

有关每通道32位的信息,请查看the OpenEXR format

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

代码语言:javascript
复制
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);

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

代码语言:javascript
复制
glm::detail::hdata *data = reinterpret_cast<glm::detail::hdata*>(heightImage->data());

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

代码语言:javascript
复制
data[currentRow*tex_width+ currentColumn] = glm::detail::toFloat16(3.1415f);

这并不是说,如果您将相同的数据写入bmptif文件(使用osg插件),结果将是不正确的。在我的例子中,我只是把想要的图像的左半部分拉伸到整个宽度上,而不是灰度,而是一些奇怪的颜色编码。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/11735418

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档