VR游戏制作中“延迟”的优化方法
VR中的”延迟”, 特指”Motion-To-Photon Latency”, 指的是从用户运动开始到相应画面显示到屏幕上所花的时间。
这中间经过了大概这么几个步骤:
- 传感器采集运动输入数据
- 采集到的数据进行过滤并通过线缆传输到主机
- 游戏引擎根据获取的输入数据更新逻辑和渲染视口
- 提交到驱动并由驱动发送到显卡进行渲染
- 把渲染的结果提交到屏幕, 像素进行颜色的切换
- 用户在屏幕上看到相应的画面
当然, 实际上还有很多细节问题, 比如屏幕上的像素并不是同一时间切换的, 可能面上面的那行先切换, 再一行行更新到最下面的, 在这里就不纠结这些细节了。
这其中的每一个步骤都会产生一定的延迟, 而目前公认的大众能接受的延迟是20ms以下, 这基本上可以做为衡量一个VR头显是不是合格的一个标准. 虽然20ms是非常短的时间, 但通过努力还是可以达到的, 主要有这么几个思路:
硬件层面的优化
- 提升传感器的采样频率, 减少刷新率与传感器频率的同步等待时间消耗
- 提升传感器的精度, 减少对采样数据进行稳定性过滤产生的延迟
- 采用有线传输也有一部分原因是出于延迟的考虑
- 屏幕使用OLED替代LCD, 减少像素颜色切换的时间
- 提升屏幕刷新率, 主流的屏幕是60Hz, 那每帧就是16.67ms; 如果提升到90Hz, 那每帧就是11.11ms
大部分的手机VR产品在延迟上都是不合格的, 最明显的表现就是转头时的画面不连续/抖动/残影等:
- 市面上的手机采用OLED屏的还是少数, 比如iPhone配个VR壳子那延迟就很感人
- 如果依赖手机的陀螺仪进行转向模拟, 其精度和频率远远达不到要求
- 手机屏幕目前都是60Hz的刷新率, 在延迟上本身就受限
刷新率的提升
假设刷新率为60Hz, 并不是代表每帧就有16.67ms的延迟, 而是说屏幕图像每16.67ms才更新一次, 渲染选项中的”垂直同步”的概念就是来源于此. 这就对我们提交渲染画面的时机要求非常高, 如下图:
为了方便计算, 这里先假设传感器, 传输, 屏幕像素切换的延迟都为0
- 假设我们在每帧开始的时候(上一次垂直同步结束)采样一次传感器数据, 在垂直同步之前完成提交, 那延迟就是16.67ms
- 如果当前帧无法在16.67ms内完成渲染, 比如花了17ms, 那么就会拖到下一帧进行提交, 屏幕上显示的画面就还是上一次的图像, 这时候的延迟就变成了16.67*2=33.33ms
这就对VR的渲染提出了非常高的要求:
- FPS必须达到刷新率的要求, 90Hz就是90Hz, 80FPS是不行的, 会被垂直同步拖累成45FPS
- FPS必须保证稳定, 偶尔掉一两帧在VR中的感觉非常明显, 可能某个物体的位置已经差了几十个像素了
以Oculus Rift(消费版)为例, 1080x1200x2的屏幕分辨率, 90Hz的刷新率, 再加上因为变形所需要的UpSampling, 实际的渲染画面就是3024x1680@90Hz, 这性能压力几乎与4k@60Hz相当. 所以, 单纯的提升刷新率和分辨率, 目前来说渲染能力还是跟不上. 不过既然有了性能需求, 硬件厂商才有前进动力, 对整个行业生态来说, 是件好事.
引擎层面的优化
除了拼命优化降低每帧画面的渲染时间外, 引擎层面还可以通过一些策略进行优化, 关键的思路就是: 能不能把采样传感器数据的时间点尽量延后, 让它与垂直同步的时间点尽量靠近?
这里我们仍然假设60Hz, 每帧时间16.67ms(约17ms), 忽略硬件延迟
如果在游戏逻辑过程中(1ms时)采样传感器数据, 那延迟大约就是16ms
如果在渲染线程进行绘制之前(5ms时), 重新再采样一下传感器数据, 修正一下视口信息(不会对游戏逻辑产生影响), 那延迟就缩短到了约12ms
做过渲染优化的人都知道, 提交D3D Command后, 需要等待GPU执行完毕, 这部分时间在整帧时间中的占比还是相当高的. 那有没有办法在渲染完成之后, 提交到屏幕之前再次采样一次传感器数据呢? 如果像下图那样的话, 延迟可以缩短到3ms!!!
这就是Timewarp的主要思想, 我们来看看它是怎么实现的
Timewarp
了解过延迟渲染的人应该都知道, 我们可以利用ZBuffer的深度数据, 逆向推导出屏幕上每个像素的世界坐标
这就意味着, 我们可以把所有像素变换到世界空间, 再根据新的摄像机位置, 重新计算每个像素的屏幕坐标, 生成一幅新的图像:
可以看到之前被遮挡区域的像素是缺失的, 因为我们的摄像机位置变化了. 那如果摄像机位置不变, 仅仅是朝向变了呢? 这样就不存在像素可见性的变化了:
Timewarp正是利用了这个特性, 在保证位置不变的情况下, 手机游戏转让平台把渲染完的画面根据最新获取的传感器朝向信息计算出一帧新的画面, 再提交到显示屏. 由于角度变化非常小, 所以边缘不会出大面积的像素缺失情况。
Oculus的Demo中可以停止渲染新的画面, 完全由单帧图像计算出各个朝向的新画面:
也就是说, 只要角度变化不是非常大(上图为了演示效果偏转的角度很大了), 可以通过这项技术”凭空渲染”出下一帧的图像, SONY的PSVR正是利用这一点, 把60FPS的画面Reproject成了120FPS。
Timewarp只能处理头部转向, 不能处理头部移动的情况, 而且一旦错过了垂直同步的时机, 一样需要等待下一次垂直同步才能显示出来. 那能不能在每次垂直同步之前, 强制进行一次Timewarp呢? 那就需要驱动来开个后门了…
驱动层面的优化
假设垂直同步时, 当前帧还没有渲染完毕, 这时如果要进行Timewarp的话, 就需要驱动提供一种高优先级的异步调用, 这就是异步Timewarp的由来: Timewarp操作与场景渲染并行执行, 如果没有新的渲染画面, 就继续使用上一帧的画面进行Timewarp。
这可以在一定程度上补偿FPS不达标造成的延迟问题, GearVR中正是应用了这项技术, 保证了手机VR的体验。
当然, PC上使用项技术还是有一些限制:
- 必须是Fermi, Kepler, Maxwell(或更新)核心的GPU
- GPU是以DrawCall为单位调度的, 所以耗时太长的DrawCall是插入不了Timewarp绘制操作的
- 需要最新的Oculus和NVIDIA驱动支持
异步Timewarp并不是说FPS低于标准还能流畅跑, 这只是一种补救措施, 所以优化仍然要好好做-_-
驱动方面还有一些其它的优化空间, 比如强制提交渲染队列:
如果驱动中缓存了3帧, 那延迟优化就白做了…
另外就是大家耳熟能详的Back Buffer(Double Buffer Rendering), 其实也会增加一点延迟, 不如省掉这一步, 即Front Buffer Rendering, 或者叫Direct Mode:
VR游戏制作中“延迟”的优化方法相关推荐
- 3D/VR游戏制作软件GameStudio简介
3D/VR游戏制作软件GameStudio简介 2004-4-30 作者:虚怀若谷 转载自:5D多媒 网友评论 2 条 点击进入论坛 前言:GameStudio是一个可以制作3D/VR游戏的软件,通过 ...
- ubuntu下使用python将ppt转成图片_Ubuntu下使用Python实现游戏制作中的切分图片功能...
本文实例讲述了Ubuntu下使用Python实现游戏制作中的切分图片功能.分享给大家供大家参考,具体如下: why 拿到一个人物行走的素材,要用TexturePacker打包.TexturePacke ...
- 西安网站制作手机端的优化方法有哪些?
西安网站制作手机端的优化方法,实际上是很简单的,无论是做移动手机端还是PC端的优化,我们在应该搞清楚的就是搜索引擎的排名规则,否则你将会是在毫无章法的运营. 移动端可以根据代码结构分为跳转适 ...
- 制作移动端手机网站过程中的SEO优化方法技巧
据国内三大运营商数据来看,中国的手机用户数已达10亿,超过2/5的移动用户每个月都会从手机终端访问网页,如今的移动端手机网站比例肯定有提升,但是对于这些存在的移动版本网站来说,马海祥查看了很大一部分手 ...
- [转]游戏中各种性能优化方法(不断更新)
http://www.cppblog.com/liangairan/archive/2013/03/23/198749.aspx 谈到游戏中的性能优化,说白了就是如何提高帧率和降低内存. 提高帧率的基 ...
- Unity VR游戏开发干货教程:优化VR体验
简介 对于VR应用来说,如果想要让用户获得好的用户体验,特别是免除恶心眩晕的困扰,在VR开发中进行优化是必不可少的,惟其如此才能达到我们期望的游戏运行帧速.和其它平台上的开发不同,对VR应用的优化应该 ...
- 游戏制作中的大宝剑---常用的数据结构与算法
前言 时间流逝,物是人非,就好像涌动的河流,永无终焉,幼稚的心智将变得高尚,青年的爱慕将变得深刻,清澈之水折射着成长. ----------<塞尔塔传说> PS:为了方便大家阅读,个人认为 ...
- 冯·诺伊曼奖得主Jorge Nocedal:增强学习中零阶优化方法及其应用(附演讲视频和PPT)
2020 北京智源大会 本文属于2020北京智源大会嘉宾演讲的整理报道系列.北京智源大会是北京智源人工智能研究院主办的年度国际性人工智能高端学术交流活动,以国际性.权威性.专业性和前瞻性的" ...
- 机器学习中常用的优化方法
写在前面 在看斯坦福的次cs231n课程,里面提到一些机器学习的基础知识,比如损失函数,优化算法,正则化形式等等.然后有一些知识也都记不起来了,索性就在博客上再回顾一遍顺便记录下来日后方便查阅.今天就 ...
最新文章
- IntelliJ IDEA(五) :酷炫插件系列
- DetachedCriteria 分页Projection取消聚合条件
- element-ui中用el-dialog+el-table+el-pagination实现文件默认选中且在分页的条件下有记忆功能...
- android开发学习之路——连连看之游戏逻辑(五)
- java executor_Java并发编程(08):Executor线程池框架
- Test of etco
- oracle-11g-R2监听文件配置
- cv2.imread读取图像结果none_PyTorch 42.图像操作
- android drawable资源调用使用心得
- matlab矩阵的LU分解
- linux jdk路径
- Opencv3.4.2调用yolov2进行物体检测源代码
- IDL实现遥感影像融合(批量)TASK(三)
- yum离线安装rpm包
- Win11系统Microsoft Store应用商店加载空白,不能下载软件.
- 计算机网络在电力系统继电保护的应用毕业论文,电力系统继电保护技术 毕业论文.doc...
- 十年老撕鸡分享,五分钟搭建个人轻论坛
- 邻接矩阵/图/DFS/BFS
- 智慧水利三维可视化平台功能介绍
- 单片机的c语言应用程序设计答案第六版,单片机C语言应用程序设计马忠梅课后习题答案...
热门文章
- java to对象_java中同类对象之间的compareTo()和compare()方法对比分析
- 机器学习基础-集成学习-13
- core webapi缩略图_.Net Core WebApi上传图片的两种方式
- P2286 [HNOI2004]宠物收养场
- IdentityServer4(7)- 使用客户端认证控制API访问(客户端授权模式)
- Weblogic服务端请求伪造漏洞(SSRF)和反射型跨站请求伪造漏洞(CSS)修复教程...
- 【Unity3D】Tags和Layers
- HttpClient 使用
- 默认开机启动;通过Broadcastreceiver广播监听开机启动,实现没有activity的自启服务或者自启应用程序。...
- 插件式程序开发及其应用(C#)