如果你想给游戏做个截图功能,或者想把屏幕图像弄成一个纹理,你就非常需要 PBO 了

通常情况下,你想把屏幕图像的像素数据读到内存需要用 glReadPixels 然后 pixels 参数传进去一块内存地址

这样做是非常非常不好的,因为 glReadPixels 会把屏幕图像的像素数据从显卡的显存复制到内存条,这个过程就非常非常的慢,特别是数据量大的时候

然后如果你要把像素数据再用 glTexImage2D 传到纹理,数据就又要从内存条复制到显存,这个过程也是非常非常慢的,特别是数据量大的时候

那么有没有一种办法,让我们可以通过一个内存指针,直接访问显存的数据呢?当然是有的,那就是 OpenGL 的 Array Buffer

这个东西中文叫做 数组缓冲区 也可以直接省略成 缓冲区 因为它就是显存里的一块内存,所以我们下文就叫 缓冲区 吧

你可以用 glMapBuffer 得到它的内存指针,然后就可以为所欲为了,另外,OpenGL 很多用来返回数据的函数,都可以把数据写到缓冲区里,而不是复制到内存条。

就比如说 glReadPixels 原本你是要传一个内存指针进去的,但是有了缓冲区,它就可以把数据复制到缓冲区里而不是复制到内存条

因为,屏幕的像素数据是在显存里的,缓冲区也是在显存里的,所以,显存->复制数据->显存 速度就比 显存->复制数据->内存条 快非常非常的多

然后我们直接用 glMapBuffer 来获取缓冲区的内存地址,就能访问到复制好的屏幕像素数据了,接着该干嘛干嘛。

而且,OpenGL 的一些函数可以把数据写入到缓冲区里,还有些函数也可以从缓冲区里读取数据来用,比如,glTexImage2D 什么的,如果你很聪明,你已经知道接下来要干嘛了

假如我们上一步把屏幕的像素数据读取到缓冲区里了,我们就可以直接用 glTexImage2D、glTexSubImage2D 什么的函数把缓冲区里的数据传给纹理了

这样我们就把屏幕图像存储在纹理了,然后干嘛干嘛,并且这个过程完全不关内存条的事,所以速度也是非常非常的快

当然我们需要用来操作Buffer的函数。你也可以用GLEW库来偷懒

#include "glext.h"

PFNGLBINDBUFFERPROC glBindBuffer = NULL;
PFNGLBUFFERDATAPROC glBufferData = NULL;
PFNGLGENBUFFERSPROC glGenBuffers = NULL;
PFNGLMAPBUFFERPROC glMapBuffer = NULL;
PFNGLUNMAPBUFFERPROC glUnmapBuffer = NULL; void gl_init()
{
glBindBuffer = (PFNGLBINDBUFFERPROC)wglGetProcAddress("glBindBuffer");
glBufferData = (PFNGLBUFFERDATAPROC)wglGetProcAddress("glBufferData");
glGenBuffers = (PFNGLGENBUFFERSPROC)wglGetProcAddress("glGenBuffers");
glMapBuffer = (PFNGLMAPBUFFERPROC)wglGetProcAddress("glMapBuffer");
glUnmapBuffer = (PFNGLUNMAPBUFFERPROC)wglGetProcAddress("glUnmapBuffer");
}

那么上面我们已经BB了一堆,接下来就开始写代码了

GLuint Buffer;
GLuint Texture; void init()
{
// 创建1个缓冲区
glGenBuffers(, &Buffer); // 缓冲区刚创建出来的时候还没有分配内存,所以我们要初始化一下它
// 先绑定..
glBindBuffer(GL_ARRAY_BUFFER, Buffer); // 这一步十分重要,第2个参数指定了这个缓冲区的大小,单位是字节,一定要注意
// 然后第3个参数是初始化用的数据,如果你传个内存指针进去,这个函数就会把你的
// 数据复制到缓冲区里,我们这里一开始并不需要什么数据,所以传个NULL就行了
// GL内部会给缓冲区分配内存,然后什么都不干,第4个参数可以优化显存效率,指定
// 缓冲区中的数据读写频繁程度,如果缓冲区中的数据不经常读写,可以传入 GL_STATIC_****
// 这样GL会把缓冲区放在内存数据不经常变动的区域,如果要经常读写缓冲区中的数据,可以传
// 别的值,具体参考 @https://www.khronos.org/registry/OpenGL-Refpages/gl2.1/ 吧
// 注意这里的 BUFFER_SIZE 我们假设要复制整个屏幕的像素数据,格式为RGB就行了,那么
// 大小就是 屏幕宽度×屏幕高度×3,每个像素3字节
glBufferData(GL_ARRAY_BUFFER, BUFFER_SIZE, NULL, GL_STREAM_COPY); // 这样我们的缓冲区就已经初始化好了,它现在已经有一块可用的内存
// 随时可以用 glMapBuffer 来访问
// 初始化完了那么解绑吧
glBindBuffer(GL_ARRAY_BUFFER, ); // 创建1个纹理,等会把屏幕复制到这个纹理
glGenTextures(, &Texture);
// 初始化纹理,不多解释了
glBindTexture(GL_TEXTURE_2D, Texture);
// 这里data参数传NULL和上面缓冲区一样,GL仅仅给纹理分配内存而已
// ScreenWide和ScreenTall是屏幕的宽度和高度
// 格式用RGB,因为屏幕不需要透明通道,所以纹理的像素数据大小是和上面的缓冲区大小一样的
glTexImage2D(GL_TEXTURE_2D, , GL_RGB, ScreenWide, ScreenTall, , GL_RGB, GL_UNSIGNED_BYTE, NULL);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
// 初始化完了解绑
glBindTexture(GL_TEXTURE_2D, );
} void draw()
{
// 假装这里画了游戏场景 // 首先我们要把缓冲区绑定到 GL_PIXEL_PACK_BUFFER 这个地方
glBindBuffer(GL_PIXEL_PACK_BUFFER, Buffer);
// 这个函数会判断 GL_PIXEL_PACK_BUFFER 这个地方有没有绑定一个缓冲区,如果有,那就把数据写入到这个缓冲区里
// 前4个参数就是要读取的屏幕区域,不多解释
// 格式是RGB,类型是BYTE,每个像素3字节
// 如果GL_PIXEL_PACK_BUFFER有绑定缓冲区,最后一个参数就作为偏移值来使用,这里不啰嗦没用的东西了。
// 传NULL就行
glReadPixels(, , ScreenWide, ScreenTall, GL_RGB, GL_UNSIGNED_BYTE, NULL);
// 好了我们已经成功把屏幕的像素数据复制到了缓冲区里 // 这时候,你可以用 glMapBuffer 得到缓冲区的内存指针,来读取里面的像素数据,保存到图片文件
// 完成截图
/****** // 注意glMapBuffer的第1个参数不一定要是GL_PIXEL_PACK_BUFFER,你可以把缓冲区绑定到比如上面init函数的GL_ARRAY_BUFFER
// 然后这里也传GL_ARRAY_BUFFER,由于懒得再绑定一次,就接着用上面绑定的GL_PIXEL_PACK_BUFFER吧
void *data = glMapBuffer(GL_PIXEL_PACK_BUFFER, GL_READ_WRITE);
if (data)
{
WriteTGA("screenshot.tga", ScreenWide, ScreenTall, data);
glUnmapBuffer(GL_PIXEL_PACK_BUFFER); // 不要忘了解除Map
} ******/ // 完事了把GL_PIXEL_PACK_BUFFER这个地方的缓冲区解绑掉,以免别的函数误操作
qglBindBuffer(GL_PIXEL_PACK_BUFFER, ); // 接着我们演示一下把缓冲区中的像素数据传给纹理
// 首先我们把缓冲区绑定到 GL_PIXEL_UNPACK_BUFFER 这个地方。这里注意啊!GL_PIXEL_PACK_BUFFER 和 GL_PIXEL_UNPACK_BUFFER 是不同的!
glBindBuffer(GL_PIXEL_UNPACK_BUFFER, Buffer);
// 绑定纹理
glBindTexture(GL_TEXTURE_2D, Texture);
// 这个函数会判断 GL_PIXEL_UNPACK_BUFFER 这个地方有没有绑定一个缓冲区
// 如果有,就从这个缓冲区读取数据,而不是data参数指定的那个内存
// 前面参数很简单就不解释了,最后一个参数和上面glReadPixels同理,传NULL就行
// 这样glTexSubImage2D就会从我们的缓冲区中读取数据了
// 这里为什么要用glTexSubImage2D呢,因为如果用glTexImage2D,glTexImage2D会销毁纹理内存重新申请,glTexSubImage2D就仅仅只是更新纹理中的数据
// 这就提高了速度,并且优化了显存的利用率
glTexSubImage2D(GL_TEXTURE_2D, , , , ScreenWide, ScreenTall, GL_RGB, GL_UNSIGNED_BYTE, NULL);
// 完事了把GL_PIXEL_UNPACK_BUFFER这个地方的缓冲区解绑掉,以免别的函数误操作
glBindBuffer(GL_PIXEL_UNPACK_BUFFER, ); // 这时候我们已经更新了纹理,我们可以把纹理画出来看看 // 假装这里有绘制纹理的代码
}

代码就这些了。如果你想了解上面用到的函数的详细信息,一定要去看看 https://www.khronos.org/registry/OpenGL-Refpages/gl2.1 这个网页

补充:GL_PIXEL_PACK_BUFFER 和 GL_PIXEL_UNPACK_BUFFER 统称为 PBO(Pixel Buffer Object)因为这俩就是用来搞像素的,所以就叫 Pixel buffer 了呗

缓冲区什么东西都可以存,可不止像素,具体请搜索其它资料吧

这是纹理的样子:

纹理中没有看到HUD是因为我读取屏幕像素的时候HUD还没有绘制

OpenGL 使用 PBO 高速复制屏幕图像到内存或者纹理中的更多相关文章

  1. [转]OpenGL 使用 PBO 高速复制屏幕图像到内存或者纹理中

    如果你想给游戏做个截图功能,或者想把屏幕图像弄成一个纹理,你就非常需要 PBO 了 通常情况下,你想把屏幕图像的像素数据读到内存需要用 glReadPixels 然后 pixels 参数传进去一块内存 ...

  2. PowerShell定时抓取屏幕图像

         昨天的博文写了定时记录操作系统行为,其实说白了就是抓取了击键的记录和对应窗口的标题栏,而很多应用程序标题栏又包含当时记录的文件路径和文件名,用这种方式可以大致记录操作了哪些程序,打开了哪些文 ...

  3. C# 图像处理:复制屏幕到内存中,拷屏操作

    /// <summary> /// 复制屏幕到内存中 /// </summary> /// <returns>返回内存流</returns> publi ...

  4. C# 截取屏幕图像

    #region 截取屏幕图像 private static Bitmap GetScreenCapture() { Rectangle tScreenRect = , , Screen.Primary ...

  5. ##DAY3 自定义视图、视图控制器、视图控制器指定视图、loadView、 viewDidLoad、MVC、屏幕旋转、内存警告

    ##DAY3 自定义视图.视图控制器.视图控制器指定视图.loadView. viewDidLoad.MVC.屏幕旋转.内存警告 #pragma mark ———————自定义视图的步骤 —————— ...

  6. 实现把dgv里的数据完整的复制到一张内存表

    /// <summary> /// 方法实现把dgv里的数据完整的复制到一张内存表 /// </summary> /// <param name="dgv&qu ...

  7. NeHe OpenGL教程 第十三课:图像字体

    转自[翻译]NeHe OpenGL 教程 前言 声明,此 NeHe OpenGL教程系列文章由51博客yarin翻译(2010-08-19),本博客为转载并稍加整理与修改.对NeHe的OpenGL管线 ...

  8. 在屏幕上搜索图片并返回图片所在位置的坐标的AutoHotkey脚本源代码(类似大漠插件)

    ;~  在屏幕上搜索图片并返回图片所在位置的坐标的AutoHotkey脚本源代码(类似大漠插件) ; https://www.autohotkey.com/boards/viewtopic.php?t ...

  9. 【OpenGL】使用FreeType库加载字体并在GL中绘制文字

    FreeType用起来比较麻烦,这里写了一份简单的示例代码,仅供参考. 实现了FT库生成字符位图,并上传到GL纹理. 实现了字符位图缓存功能,多个字符图像保存在同一个纹理中. 实现了简单的字体管理框架 ...

随机推荐

  1. 提示文件过大无法复制到U盘怎么解决

    1.U盘作为一个便携的移动存储工具,在我们的生活中扮演重要的角色,但 是我们经常会遇到在复制文件到U盘中的时候,U盘明显有很大的空间,却 提示文件过大无法复制,今天,我教大家一步解决这个问题!! 2. ...

  2. xtrabackup 备份和恢复

    该文章接上一篇文章: 内核方面: $ cat /etc/centos-release CentOS Linux release 7.4.1708 (Core) $ uname -r 3.10.0-69 ...

  3. 【Codeforces 499D】Name That Tune

    Codeforces 499 D 题意:给\(n\)个曲子,每个曲子每一秒有\(p_i\)的几率可以被猜出来,过了\(t_i\)秒肯定能被猜出来,猜完第\(i\)首歌立即播第\(i+1\)首,问\(T ...

  4. docker知识复习

    1.镜像基于内容寻址 基于内容寻址的实现,使用了两个目录:/var/lib/docker/image和/var/lib/docker/overlay, 后面的这个根据存储驱动的名称不同,而目录名不同. ...

  5. NOIP2002-2017提高组题解

    给个人认为比较难的题目打上'*' NOIP2002(clear) //一个很吼的贪心题,将平均数减掉之后从左往右将影响消除 #include<bits/stdc++.h> using na ...

  6. CSS-Photoshop投影与CSS中box-shadow的转换

    box-shadow是给元素块添加周边阴影效果基本语法是: {box-shadow:[inset] x-offset y-offset blur-radius spread-radiuscolor} ...

  7. Linux下修改/设置环境变量JAVA_HOME

    export设置只对当前的bash登录session有效.这是存在内存里面的.你可以写入文件一般的文件.之后source它.或者放到/etc/profile 等等的位置里,不同的地方效果不同. 1. ...

  8. ABPZero中的Name和SurName处理,以及EmailAddress解决方案(完美)。

    使用ABPzero的朋友们都知道,User表中有Name和Surname两个字段,这两个字段对于国内的用户来说相当的不友好. 以及我们的一些系统中是不会涉及到EmailAddress字段.也就是说不会 ...

  9. GeForce Experience关闭自动更新

    GeForce Experience驱动更新很烦,而且有时更新后就打不开了,找到种方法关闭更新 1.安装并登陆 2.打开 C:\ProgramData\NVIDIA Corporation 3.进入D ...

  10. Docker容器学习梳理 - 容器时间跟宿主机时间同步

    在Docker容器创建好之后,可能会发现容器时间跟宿主机时间不一致,这就需要同步它们的时间,让容器时间跟宿主机时间保持一致.如下: 宿主机时间 [root@slave-1 ~]# date Fri M ...