Android OpenGL ES 3.0 PBO像素缓冲区对象
创始人
2024-04-12 04:37:50

1.什么是PBO

OpenGL PBO(Pixel Buffer Object),被称为像素缓冲区对象,主要被用于异步像素传输操作。PBO 仅用于执行像素传输,不连接到纹理,且与 FBO (帧缓冲区对象)无关。

OpenGL PBO(像素缓冲区对象) 类似于 VBO(顶点缓冲区对象),PBO 开辟的也是 GPU 缓存,而存储的是图像数据。

PBO 绑定相关的 Target 标签有 2 个:

  • GL_PIXEL_UNPACK_BUFFER:将PBO绑定到这个上边,glTexImage2D和glTexSubImage2D() 表示从PBO中解包像素数据并恢复到帧缓冲区

  • GL_PIXEL_PACK_BUFFER:将PBO绑定到这个上边时,glReadPixels()表示从帧缓冲区中读取数据并打包进PBO

2.为什么要使用PBO

处理高分辨率的图像时,图像数据在内存和显存之前拷贝往往会造成性能瓶颈,而利用 PBO 可以在一定程度上解决这个问题。

使用 PBO 可以在 GPU 的缓存间快速传递像素数据,不影响 CPU 时钟周期,除此之外,PBO 还支持异步传输。

传统的方式:从文件中加载纹理,图像数据首先被加载到 CPU 内存中,然后通过 glTexImage2D 函数将图像数据从 CPU 内存复制到 OpenGL 纹理对象中 (GPU 内存),两次数据传输(加载和复制)完全由 CPU 执行和控制。

使用PBO的方式:可以通过 glMapBufferRange 获取 PBO 对应 GPU 缓冲区的内存地址。将图像数据加载到 PBO 后,再将图像数据从 PBO 传输到纹理对象中完全是由 GPU 控制,不会占用 CPU 时钟周期。所以,绑定 PBO 后,执行 glTexImage2D (将图像数据从 PBO 传输到纹理对象) 操作,CPU 无需等待,可以立即返回。

通过对比这两种(将图像数据传送到纹理对象中)方式,可以看出,利用 PBO 传输图像数据,省掉了一步 CPU 耗时操作(将图像数据从 CPU 内存复制到 纹理对象中)

3.PBO的使用

3.1PBO的创建
/*** 创建PBO 用于读取纹理上的像素数据* StreamRead 流的方式读取*/
m_PBOfst = new MSOpenGLBuffer(MSOpenGLBuffer::PixelPackBuffer,MSOpenGLBuffer::StreamRead);
m_PBOsnd = new MSOpenGLBuffer(MSOpenGLBuffer::PixelPackBuffer,MSOpenGLBuffer::StreamRead);
//构造函数
MSOpenGLBuffer::MSOpenGLBuffer(MSOpenGLBuffer::Type type, MSOpenGLBuffer::UsagePattern usage):m_bufferType(type),m_usage(usage),m_buffSize(0)
{glGenBuffers(1, &m_buffID);}

创建方式是GL_PIXEL_PACK_BUFFER,用于从纹理上读取像素数据,用于流的方式进行读取。

3.2PBO设置bufferData
m_PBOfst->Bind();
m_PBOfst->SetBufferData(nullptr,m_nResolution.x*m_nResolution.y*4);
m_PBOfst->Release();m_PBOsnd->Bind();
m_PBOsnd->SetBufferData(NULL,m_nResolution.x * m_nResolution.y * 4); 
m_PBOsnd->Release();
//Bind 函数
void MSOpenGLBuffer::Bind()
{glBindBuffer(m_bufferType, m_buffID);
}
//release函数
void MSOpenGLBuffer::Release()
{glBindBuffer(m_bufferType, 0);
}void MSOpenGLBuffer::SetBufferData(const GLvoid *data, GLsizeiptr size)
{if (size > m_buffSize) {m_buffSize = size;glBufferData(m_bufferType, size, data, m_usage);} else {glBufferSubData(m_bufferType, 0, size, data);}
}

给连个PBO设置BufferData

3.3通过PBO进行数据读取
/*** 通过m_currIdx 来控制 两个PBO交替控制*/
void MSGLScene::processPBOReadPixels() {glReadBuffer(GL_FRONT);if(m_currIdx == 0){readPixelsFromPBO(m_PBOfst,m_PBOsnd);m_currIdx =1;}else{readPixelsFromPBO(m_PBOsnd,m_PBOfst);m_currIdx =0;}
}void MSGLScene::readPixelsFromPBO(MSOpenGLBuffer *fstBuff, MSOpenGLBuffer *sndBuff) {int w = m_nResolution.x;int h = m_nResolution.y;fstBuff->Bind();glReadPixels(0,0,w,h,GL_RGBA,GL_UNSIGNED_BYTE,0);//READsndBuff->Bind();void* data= glMapBufferRange(GL_PIXEL_PACK_BUFFER,0,w*h*4,GL_MAP_READ_BIT);if (data){MSVideoWriter::GetInstance()->WriteVideoFrameWithRgbData((const unsigned char*)data);}glUnmapBuffer(GL_PIXEL_PACK_BUFFER);glBindBuffer(GL_PIXEL_PACK_BUFFER,0);}

通过两个PBO交替读取数据,glMapBufferRange读取到结果就是:从纹理上读取到的RBG数据,此时数据就可以交给FFmpeg或者MediaCodec进行编码,并进行视频合成了。

相关内容

热门资讯

华尔街、贸易商齐声警告:霍尔木... 就在美国和伊朗就停战问题胶着拉扯之际,全球石油库存正在以创纪录的速度下降。华尔街分析师和能源贸易商警...
景林最新美股持仓曝光 加仓英特... 格隆汇5月10日|景林资产向美国证券交易委员会提交的季度持仓13-F表显示,该机构在去年底持有259...
全民追“光”!热门股抱团飙涨 ...   当前A股最耀眼的板块就是“光”,光模块、光纤、光器件、光芯片等概念股均是“热门行业里的热门股”,...
普京:坚决制止篡改二战历史图谋 △俄罗斯总统普京(资料图)俄罗斯总统普京9日在莫斯科克里姆林宫表示,必须坚决制止任何篡改二战历史和美...
禁飞通告 | 5月17日,长春... 长春市人民政府关于在马拉松赛事期间对低空慢速小目标航空器实施临时管控的通告2026一汽红旗长春马拉松...