2022-02-11 22:03:25 +08:00
[#]: subject: "How Much of a Genius-Level Move Was Using Binary Space Partitioning in Doom?"
[#]: via: "https://twobithistory.org/2019/11/06/doom-bsp.html"
[#]: author: "Two-Bit History https://twobithistory.org"
[#]: collector: "lujun9972"
2022-06-27 09:58:25 +08:00
[#]: translator: "aREversez"
2022-08-09 15:41:37 +08:00
[#]: reviewer: "wxy"
2022-08-19 16:17:46 +08:00
[#]: publisher: "wxy"
[#]: url: "https://linux.cn/article-14945-1.html"
2022-02-11 22:03:25 +08:00
2022-08-19 11:04:49 +08:00
在《毁灭战士》中应用二叉空间分割( BSP) 是何等天才之举?
2022-02-11 22:03:25 +08:00
======
2022-08-19 16:17:46 +08:00
![](https://img.linux.net.cn/data/attachment/album/202208/19/161257n99vkniexsjdehvh.jpg)
2022-08-09 15:41:37 +08:00
1993 年,游戏开发公司 id Software 发行了一款第一人称射击游戏 《< ruby > 毁灭战士< rt > DOOM< / rt > < / ruby > 》,游戏一经发行迅速爆火。在今天看来,《毁灭战士》可谓有史以来最具影响力的游戏之一。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
《毁灭战士》发行之后的第十年( 2003 年),记者 < ruby > 大卫·库什纳< rt > David Kushner< / rt > < / ruby > 出版了一本关于 id Software 的书,书名为《< ruby > Doom 启示录< rt > Masters of Doom< / rt > < / ruby > 》,后被奉为记录毁灭战士创作史的典范读物。几年前我曾读过这本书,如今内容已记得不太真切了,但是书中有一个关于 id Software 首席程序员 < ruby > 约翰·卡马克< rt > John Carmack< / rt > < / ruby > 的故事,我印象特别深刻。这里只对故事做粗略描述(具体情节请往下阅读)。实际上,早在《毁灭战士》开发前期,卡马克就发现自己为这款游戏编写的 3D 渲染器在渲染某些关卡时慢得像爬一样。对于《毁灭战士》这一对动感和速度有着相当高要求的射击游戏来说,这是一个非常严重的问题。意识到了这一问题的严重性,卡马克需要一个更加有效的渲染算法,于是他开始阅读相关论文。最后,他实现了一种叫做“< ruby > 二叉空间分割< rt > binary space partitioning< / rt > < / ruby > ( BSP) ”的技术, 极大地提升了《毁灭战士》游戏引擎的运行速度, 而这项技术此前从未用于电子游戏当中。
2022-02-11 22:03:25 +08:00
2022-08-09 15:41:37 +08:00
一直以来,我对这个故事的印象十分深刻。卡马克将学术前沿研究运用于电子游戏之中,我觉得这正是他之所以成为传奇人物的原因。无论从哪个角度来看,卡马克都应该是电子游戏行业中人尽皆知的典型的天才程序员,只不过上面这个故事是我最先能够想到的理由。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
显而易见,“二叉空间分割”这个术语听起来就是难度相当高的课题,能够自行阅读论文并将其付诸实施实属不易,所以这个故事给我留下了深刻的印象。我一直认为卡马克的做法十分具有创见性,不过由于我既不懂二叉空间分割到底是怎样的一项技术,也不晓得这项技术在当时究竟有多么革新,所以我也并不确定自己的观点是否正确。如果按照从 < ruby > 霍默·辛普森< rt > Homer Simpson< / rt > < / ruby > ( LCTT 译注:《辛普森一家人》中的那个老爹)到 < ruby > 阿尔伯特·爱因斯坦< rt > Albert Einstein< / rt > < / ruby > 的顺序为天才列出一套级别体系,那么卡马克将二叉空间分割技术运用于《毁灭战士》的做法究竟属于什么级别的天才之举呢?
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
同时, 我也在想, 二叉空间分割这个概念最初是从哪儿来的, 又是怎样吸引到卡马克的? 因此, 本篇文章不仅仅会讲述约翰·卡马克和《毁灭战士》的故事, 也会探讨二叉空间分割树( BSP 树) 数据结构的发展历史。有意思的是, BSP 树和计算机科学领域其他许多技术一样,最初都起源于军事研究领域。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
没错,《毁灭战士》的第一关卡 E1M1 就受到了美国空军的启发。
2022-02-11 22:03:25 +08:00
2022-07-30 19:09:41 +08:00
### VSD 难题
2022-02-11 22:03:25 +08:00
2022-08-12 11:39:26 +08:00
BSP 树是计算机图形领域最具挑战性难题的解决方案之一。举个例子,为了渲染出三维场景,渲染器必须能够区分在一个特定角度下的可见物体和不可见物体。如果渲染时间比较充足,这一要求也算不上大问题;但是就理想情况来说,实时游戏引擎在 1 秒内至少需要完成 30 次区分任务。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
这一问题有时被称为 < ruby > 可见面检测< rt > visible surface determination< / rt > < / ruby > ( VSD) 问题。后来与卡马克合作开发《< ruby > 雷神之锤< rt > Quake< / rt > < / ruby > 》( id Software 继《毁灭战士》之后开发的游戏)的程序员 < ruby > 迈克尔·亚伯拉什< rt > Michael Abrash< / rt > < / ruby > ,在自己的著作《< ruby > 图形程序开发人员指南< rt > Graphics Programming Black Book< / rt > < / ruby > 》 中写道:
2022-02-11 22:03:25 +08:00
2022-08-12 11:39:26 +08:00
> 我想探讨一下在我看来 3D 中最棘手的一个问题:可见面检测问题(在每个像素点上绘制合适的表面)以及与之密切相关的隐面消除问题(迅速去除不可见的多边形,用于加快可见表面检测速度)。简略起见,我将在下文采用缩写 VSD 来表示可见面检测和隐面消除。
>
> 为什么我会认为 VSD 是 3D 中最棘手的问题呢?尽管纹理映射等光栅化问题更让人感兴趣而且也更重要,但是相对而言,它们是范围相对有限的任务。随着 3D 加速器的出现,它们逐渐被转移到硬件中。同时,它们只随着屏幕分辨率的增加而增加,而分辨率的增加是相对温和的。
>
> 相反, VSD 却像是一个无底洞,目前应对方案也有很多。但实际上,在采用简单的方法处理 VSD 时, 其性能会直接受到场景复杂程度的影响, 而场景的复杂程度通常会以平方级或立方级的形式增大。所以在渲染过程中, VSD 很快就会成为制约因素。[^1]
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
亚伯拉什是在上个世纪九十年代末写的关于 VSD 问题的这些困难, 这是在《毁灭战士》之后数年, 这款游戏证明了普通人盼望着能用自家电脑玩很吃图形配置的游戏。九十年代早期, id Software 成立后发行了一些游戏。尽管当时的计算机还只是用来处理文字与表格或者执行其他任务, 未尝想过要在上面运行游戏, id Software 必须对发行的游戏进行编程,使其能在计算机上流畅运行。为了实现这一飞跃,尤其是为了能让在《毁灭战士》之前 id Software 发行的少数 3D 游戏在电脑上运行, id Software 必须做出革新。在这些游戏中,所有的关卡在设计时都受到了一定限制,以便更容易解决 VSD 问题。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
例如, 在《毁灭战士》之前, id Software 发行了《< ruby > 德军总部 3D 版< rt > Wolfenstein 3D 版< / rt > < / ruby > 》,该游戏的每一个关卡都是由与坐标轴平齐的墙壁组成。换言之,在《德军总部 3D 版》的游戏画面里,你看到的只有南北方向或者东西方向的墙壁。在游戏中,墙壁与墙壁之间有着固定的间隔,所有过道的宽度或是一个方格,或是两个方格等等,但绝不会出现 2.5 个方格。如此一来,尽管 id Software 团队只能设计出外观十分相似的关卡,但这也让卡马克为 《德军总部 3D 版》 编写渲染器的工作简单了不少。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
通过将屏幕上的光线“齐射”入虚拟游戏世界,《德军总部 3D 版》的渲染器解决了 VSD 问题。通常来说,使用光线的渲染器叫做“光线投射”渲染器。这种渲染器的速度一般较慢,因为解决内部的 VSD 问题涉及到在光线和游戏中的物体之间找到第一个交点,这通常需要进行大量的计算。但在 《德军总部 3D 版》, 由于所有的墙壁都与网格平齐, 所以光线与墙壁相交的位置只能在网格线上。如此一来, 渲染器只需逐个检查这些交点即可。如果渲染器先从离玩家视角最近的交点开始检查, 接着检查下一个最近的交点, 以此类推, 最后遇到第一面墙壁时停止检查。这样, VSD 问题便轻而易举地得到了解决。光线从每一个像素点向前投射,与画面物体接触时停止,这一方法是可行的。因为从 CPU 资源来看,投射的成本很低。事实上,由于每面墙壁高度相同,因此针对同列的像素点,投射的光线只需一条。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
尽管当时还没有专业的图形显卡,《德军总部 3D 版》凭借这一取巧之法得以在配置较低的个人电脑上正常运行起来。然而,这个办法并不适用于《毁灭战士》。因为 id Software 为这款新游戏增添了许多新元素 —— 倾斜的墙面、楼梯以及高低不一的天花板。光线投射的办法自然也就不好用了,于是卡马克编写出了一个新的渲染器。《德军总部 3D 版》的渲染器关注的是图像,将光线投射到屏幕像素表示的列上,而 《毁灭战士》 关注的则是物体。换句话说,《毁灭战士》 的渲染器会记录游戏场景中的所有物体,继而将其投射到屏幕当中;而非记录屏幕上的像素点,判断每个像素点的颜色。
2022-02-11 22:03:25 +08:00
2022-08-12 11:39:26 +08:00
对于强调物体的渲染器来说,可以使用 Z 缓冲区来解决 VSD 问题,比较简单。每次将物体投射到屏幕上时,需要对每个用于绘制的像素点进行检查。如果你想绘制出的物体的部分和已经绘制在目标像素点上的物体相比更加靠近玩家,可以将其覆盖。否则,必须保持像素不变。尽管办法很简单,但是 Z 缓冲区对内存的要求较高,而且渲染器可能仍然要花费大量的 CPU 资源来投射玩家永远不会看到的水平几何体。
2022-02-11 22:03:25 +08:00
2022-08-19 16:17:46 +08:00
在 20 世纪 90 年代,使用 Z 缓冲区的方法还存在着其他缺陷: IBM 兼容机( PC) 搭载的是一种叫 VGA 的显示适配器系统,在这类电脑上,将图像写入帧缓冲区的成本非常之高。因此,在只会以后被覆盖的像素上绘制花费的时间拖慢了渲染器的性能。
2022-02-11 22:03:25 +08:00
2022-08-12 11:39:26 +08:00
考虑到将图像写入帧缓冲区的成本非常之高,理想的渲染器需要首先绘制离玩家最近的物体,接着是比较近的物体,以此类推,直到屏幕上每个像素点都写入了信息。这时,渲染器会停止运行,大幅缩短远处不可见物体的渲染时间。这种由近及远对物体进行排序的方法也可以解决 VSD 问题。那么问题又来了:什么才是玩家可以看到的?
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
最初,卡马克打算依靠《毁灭战士》的关卡布局来解决 VSD 问题。首先用渲染器绘制出玩家目前所在房间的墙壁,之后玩家冲进隔壁房间,再绘制出隔壁房间的墙壁。由于每个房间互不遮挡,这一办法也能解决 VSD 问题。而互相遮挡的房间可以分割成若干互不遮挡的“区域”。在 YouTube 上的一个 [视频][2] 中, Bisqwit 展示了自己制作出来的使用了相同算法的渲染器。可以看到,如果以超慢的速度运行,便能一睹渲染的具体过程。这一算法同样运用到了《毁灭公爵 3D 版》当中,这款游戏在 《毁灭战士》 推出三年之后发行,当时 CPU 的性能也更加强大了。1993 年,尽管在硬件上已经可以运行游戏了,但是使用这一算法的《毁灭战士》渲染器在复杂的层级结构上依旧表现吃力,尤其是在房间分割出来的各部分相互嵌套的情况下。不巧的是,这类层级结构正是构造环形楼梯等物体的唯一办法。沿着环形楼梯走下去,直到走入已经绘制好的区域,由于这其中涉及多次循环下降运动,导致游戏引擎的运行速度大幅降低。
2022-02-11 22:03:25 +08:00
2022-08-19 16:17:46 +08:00
在 id Software 团队意识到《毁灭战士》游戏引擎的速度可能过慢时,公司还面临着其他任务:将《德军总部 3D 版》移植到超级任天堂游戏机(简称“超任”)上。那时,超任的性能比 IBM 兼容机还要差。结果表明,尽管光线投射渲染器非常简单,但是想要在超任上快速运行是不可能的。于是,卡马克着手研究更为高效的算法。事实上,也正是为了顺利将《德军总部》移植到超任,卡马克首次研究了二叉空间分割技术,并将其付诸应用。由于《德军总部 3D 版》的墙壁与坐标轴平齐,所以二叉空间分割技术应用起来也比较简单直接;但是《毁灭战士》的情况则比较复杂。不过,卡马克发现,二叉空间分割树同样可以用来解决《毁灭战士》速度过慢的问题。
2022-02-11 22:03:25 +08:00
2022-07-30 19:09:41 +08:00
### 二叉空间分割
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
< ruby > 二叉空间分割< rt > binary space partitioning< / rt > < / ruby > ( BSP) 会提前将 3D 场景分割为若干部分,使 VSD 问题易于解决。讲到这里,你需要先了解一下为什么分割场景可以奏效:如果你在场景上画条线(对应三维空间里的一个平面),你就可以指出玩家或者摄像机视角在这条线的哪一侧,在这条线另一侧的物体无法遮挡玩家所在一侧的物体。如果多次重复这一操作,该三维场景最终会被分割为多个区域,这并不是对原始场景的改进,只是现在你知道了更多关于场景的不同部分会如何相互阻挡。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
首次阐述上述三维场景分割的是美国空军的研究员, 他们曾尝试向美国空军证明计算机图形已经非常先进, 可以应用到飞行模拟器领域。1969 年,他们将研究发现发表在一份题为《计算机生成图像在图形仿真中的应用研究》的报告中。该报告的总结部分指出,计算机图形可用于训练飞行员,但也警告说,其实际应用可能会受制于 VSD 问题:
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
> 实时图像处理需要解决的一个关键问题就是优先级问题,或称隐藏线问题。在我们平时用眼睛观察外界时,大自然替我们轻易地解决了这一问题:不透明物体上的一个点,掩盖了同一视觉方向上、且距离较远的所有其它物体。但在计算机中,这项任务却非常困难。图像处理需要解决的优先级问题,随着环境复杂程度的增加,计算量会呈指数级增长,随即就会超过绘制物体透视图所需的计算负载。[^2]
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
他们在报告中提出了一项基于构造“遮挡矩阵”的方案,这一方案据说早些时候曾被应用于 NASA 的项目当中。研究员指出, 平面将场景一分为二, 可用来解决平面两侧物体之间存在的“任何优先级问题”。通常情况下, 可能需要明确将这些平面添加到场景中, 但对某些几何体, 只需借助你已经拥有的几何体的表面即可。他们举了一个例子, 如下图: p~1~、p~2~ 以及 p~3~ 是三个不同的平面, 如果摄像机视角位于其中一个平面的前方, 即“正”面, p~i~ 的值就等于 1。这种矩阵展示出基于三个不同平面和摄像机视角位置的三个物体之间的关系 —— 如果物体 a~i~ 遮挡了物体 a~j~,那么 a~ij~ 在此矩阵中的数值等于 1。
2022-02-11 22:03:25 +08:00
![][4]
2022-08-15 11:02:09 +08:00
研究人员指出,这种矩阵可以应用到硬件中,对每一帧进行重新评估。该矩阵基本上可以用作大型的开关,或者一种预置的 Z 缓冲区。在绘制给定的物体时,如果在物体所在列上得出数值 1, 并且所在行已经在绘制中, 那么物体被遮挡的部分就不会绘制出来。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
不过,该矩阵方法的主要缺点在于,为了在场景中表示出 n 个物体,你需要一个尺寸为 n^2^ 的矩阵。于是,研究人员们继续深入,探究将遮挡矩阵表示为“优先级列表”的可行性,该列表的尺寸是 n, 可确定物体绘制的顺序。他们随即发现, 诸如上图此类场景根本无法确定顺序( 因为它存在循环阻塞的现象) 。因此, 他们花了很多时间来阐明“合适”与“不合适”场景之间的数学区别。最后, 他们得出了一个结论: 至少对于“合适的”场景下, 优先级列表是可以制作出来的; 而对场景设计师来说, 避免设计出“不合适”的场景也不是一件难事。但是, 他们并没有说明如何生成该列表。可以说, 这份 1969 年的研究的首要贡献在于提出了:至少,在 _理论上_ ,可以采用平面分割的方法,对场景中的物体进行渲染排序。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
直到 1980 年,一份题为《基于优先级树结构的可见表面生成》的论文提出了解决该问题的具体算法。在这份论文中,作者 < ruby > 亨利·福克斯< rt > Henry Fuchs< / rt > < / ruby > 、< ruby > 泽维·凯德姆< rt > Zvi Kedem< / rt > < / ruby > 以及 < ruby > 布鲁斯·内勒< rt > Bruce Naylor< / rt > < / ruby > 介绍了 BSP 树。他们指出,这种新的数据结构“可以替代十年前首次使用,但由于一些问题未得到广泛发展的方案”(即前文 1969 年美国空军相关研究中的方案)。[^3] BSP 树一经生成,即可用于确定场景中物体的优先级顺序。
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
三人在论文中对 BSP 树的工作原理给出了相当可读的解释。但在本文,我将尝试使用更加通俗的语言,介绍给大家。
2022-02-11 22:03:25 +08:00
2022-07-30 19:09:41 +08:00
首先,在场景中选定一个多边形,将该多边形所在的平面作为分割平面。同时,该多边形充当树的根节点。场景中剩下的多边形会分散在分割平面的两侧。位于分割表面“前方”或者与分割平面相交后位于“前”半部分的多边形落在了根节点左侧的左子树上;位于分割表面“后方”或者与分割平面相交后位于“后”半部分的多边形落在了右子树上。接着,递归重复这一过程:在左子树和右子树上各选定一个多边形,作为各自空间新的分割平面,继而二分出来更多的子空间和子树。等到全部的多边形均选定之后,二叉空间分割也就结束了。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
假设你想由后向前将场景中的几何图形进行渲染。(这就是所谓的“< ruby > 画家算法< rt > painter's algorithm< / rt > < / ruby > ”。因为在绘制时,距离摄像机较远的多边形会被距离摄像机较近的多边形所覆盖,借此正确进行渲染任务。)如果想要实现这一算法,必须按顺序遍历 BSP 树,左右子树的渲染顺序由摄像机视角与节点所在分割平面的位置关系决定的。因此,针对树上的每个节点,首先渲染距离分割平面较“远”一侧的所有多边形,接着是位于平面上的多边形,最后是距离平面较“近”一侧的所有多边形 —— “远”与“近”相对于摄像机视角而言。根据前文,距离分割平面较远一侧的多边形无法遮挡近侧的物体,所以这种方法可以解决 VSD 问题。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
下图表示一个简单的二维场景的 BSP 树的构造与遍历过程。在二维中,分割平面变成了分割线,但就基本原理而言,与复杂的三维场景并无二致。
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
![][6]
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
第一步:根分割线落在 D 墙上,将剩下的几何图形分为两组。
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
![][7]
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
第二步:继续分割位于 D 墙两侧的空间。C 墙是其中一侧的唯一一堵墙壁, 因此无需再分。另一侧, B 墙形成新的分割平面。因为 A 墙与新的分割平面相交,所以必须将其分割为两堵墙。
2022-02-11 22:03:25 +08:00
2022-08-15 11:02:09 +08:00
![][8]
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
第三步:参照右上方视角,由后向前对墙壁进行排序,对执行画家算法很有帮助。这就是树的顺序遍历过程。
2022-08-15 11:02:09 +08:00
福克斯、凯德姆以及内勒多次强调了 BSP 树的优势:它只需构建一次。可能有些难以置信,但实际上无论摄像机视角位于何处,同一棵 BSP 树都可以用来渲染一个场景。只要场景中的多边形没有移动, BSP 树就不会失效。因此, BSP 树在实时渲染任务中非常实用 —— 构建树时的所有艰巨任务都可以在渲染工作开展之前完成。
2022-08-19 12:21:58 +08:00
同时,三人也提到了一项需要进一步深入研究的问题:究竟怎样才能构建出一棵 “高质量的” BSP 树? BSP 树的质量取决于用作分割平面的多边形的选择。我在前文跳过了这一问题,不过如果用作分割平面的多边形与其他多边形相交,那么为了让 BSP 算法发挥作用, 必须将相交的多边形一分为二, 这样两部分就可以分在不同的空间。但是如果这种现象反复出现, BSP 树的构建势必会大幅增加场景中多边形的数量。
2022-08-15 11:02:09 +08:00
2022-08-19 12:21:58 +08:00
内勒后来在其 1993 年的论文《构建高质量的分割树》中提及这一问题。卡马克的同事, id Software 的共同创始人 < ruby > 约翰·罗梅洛< rt > John Romero< / rt > < / ruby > 指出,这篇论文是卡马克在《毁灭战士》中引入 BSP 树时读到的论文之一。[^4]
2022-02-11 22:03:25 +08:00
2022-07-30 19:09:41 +08:00
### 《毁灭战士》中的 BSP 树
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
别忘了, 卡马克首次为《毁灭战士》设计渲染器时, 通过让渲染器渲染玩家所在房间之外的临近房间, 试图为关卡几何图形建立一套渲染顺序。对此, BSP 树是个不错的选择, 因为在玩家进入之前的房间( 区域) 时, BSP 树能够避免让渲染器重复劳动,从而节省 CPU 资源。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
实际上,“将 BSP 树引入《毁灭战士》”意味着将 BSP 树生成器引入《毁灭战士》的关卡编辑器中。当完成一个《毁灭战士》的关卡的制作时, BSP 树就会在关卡几何图形的基础上生成。根据程序员 < ruby > 法比安·桑格勒德< rt > Fabien Sanglard< / rt > < / ruby > 的说法,在原版《毁灭战士》中,一个关卡的 BSP 树生成时间需要 8 秒,全部关卡合计共需 11 分钟 [^5]。之所以生成时间较长,部分原因在于卡马克所用的 BSP 生成算法,该算法尝试使用各种启发式方法找出 “高质量” BSP 树。在运行时, 8 秒的延时可能让人无法接受;但是离线等 8 秒,时间并不算长,尤其是考虑到 BSP 树提升了渲染器的性能。为每个关卡生成的 BSP 树将在游戏启动时作为关卡数据载入。
2022-02-11 22:03:25 +08:00
2022-08-19 16:17:46 +08:00
卡马克对 1980 年论文中提出的 BSP 树算法进行了改造,因为在《毁灭战士》开始运行时,当前关卡的 BSP 树就会读取到内存中,渲染器通过 BSP 树由前向后绘制物体,而非由后向前进行绘制。福克斯三人在那篇论文中演示了 BSP 树可用于执行由后向前的画家算法,但是画家算法会造成许多重复的绘制任务,对于 IBM 兼容机来说负担较大。因此,《毁灭战士》的渲染器换了个方向,首先绘制距离玩家较近的图形,之后再绘制离玩家较远的。这种反向排序很容易通过 BSP 树来实现,因为你可以在树的每个节点都进行反向遍历。为了避免绘制出来的远处图形遮挡到近处的图形,《毁灭战士》的渲染器使用了一种隐式 Z 缓冲区,这种缓冲区不仅具备普通 Z 缓冲区的优势,而且对内存的要求也较低。这种 Z 缓冲区有两组数组,一组记录水平方向的遮挡关系,另两组记录自屏幕顶部和底部的垂直方向的遮挡关系。《毁灭战士》的渲染器就算不使用实际的 Z 缓冲区也无伤大雅,因为从技术上来看它并不是真正的 3D 游戏。BSP 树数据结构的成本虽然不高,但却能够起作用,其原因在于《毁灭战士》不会发生以下问题:水平方向的遮挡数组能够发挥作用,是因为该游戏中没有倾斜的墙体;垂直方向的遮挡数组能够发挥作用,是因为该游戏不存在有着一上一下两扇窗户的墙体。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
剩下比较棘手的问题是如何将《毁灭战士》中处于运动中的角色融入到借助 BSP 树绘制的静止的关卡几何图形中。该游戏中的敌人不可能纳入 BSP 树之中,因为他们会移动,而 BSP 树只对静止的几何形状起作用。所以渲染器首先绘制静止的关卡几何图形,同时与另一个内存使用效率较高的数据结构协作,记录屏幕上分割出来用于绘制的区域。之后,渲染器按照由后往前的顺序绘制敌人,并消除被屏幕上的区域遮挡住的敌人。这一过程与使用 BSP 树进行渲染相比,效果稍差一些。但是由于关卡中能看到的敌人的数量少于几何图形的数量,所以速度并不是一个严重的问题。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
将 BSP 树应用到《毁灭战士》中可谓一大成功。卡马克能够想到 BSP 树是解决 VSD 问题的最佳方案,无疑非常高明。但是这可以称得上是天才之举吗?
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
桑格勒德在其关于《毁灭战士》游戏引擎的书中引用了罗梅洛的话:内勒的论文《构建高质量的分割树》主要讲述使用 BSP 树消除 3D 模型的背面。[^6] 根据罗梅洛所言,卡马克认为这种算法对《毁灭战士》依然有效,所以他放手一试,将 BSP 技术应用到了该游戏中。不过这话说得有些奉承的意味 —— 意在暗示卡马克在别人仍然使用 BSP 树渲染静止的场景时, 发现该技术可以用于实时游戏领域。在《Doom 启示录》中也有给卡马克戴高帽的故事。该书作者库什纳认为,卡马克在阅读内勒的论文之后,问了自己,“如果使用 BSP 技术创造一整个虚拟世界,而不仅仅是一张 3D 图像,会怎么样呢?” [^7]。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
这些“片面之词”忽视了 BSP 树的发展历史。当美国空军研究人员开始意识到场景分割可能会加快渲染任务的时候,他们就对提升 _实时_ 渲染的速度产生了兴趣, 毕竟他们当时试图创建一个飞行模拟器。1980 年,同样的案例再次出现在了福克斯等人的论文中,他们探讨了 BSP 树如何应用于飞行模拟器中, 帮助飞行员进行训练: 飞行员用它来反复练习将飞机降至同一空港。由于空港的地形不会发生改变, BSP 树只需生成一次,即可一劳永逸。很明显,他们考虑的是实时模拟。在论文的引言部分,福克斯等人还谈到实时图形系统必须在至少 1/30 秒内生成一张图像,由此激励了他们的研究。
2022-02-11 22:03:25 +08:00
2022-08-19 16:17:46 +08:00
因此,卡马克不是第一个想到在实时图形模拟中应用 BSP 树的人。诚然, 设想与付诸实践是两码事。但是即使在实施的过程中, 卡马克受到的帮助与指导可比人们想象的要多得多。至少是到这篇文章写成之时, BSP 树的 [维基百科词条][13] 页面显示,卡马克参考了 1991 年 < ruby > 陈< rt > Chen< / rt > < / ruby > 和 < ruby > 戈登< rt > Gordon< / rt > < / ruby > 的一篇论文,以及 1990 年的一本教材《计算机图形学:原理及实践》。尽管该页面并未提供引用信息,但可信度没什么问题。陈和戈登的论文介绍了运用 BSP 树由前向后的渲染方法,这种方法与《毁灭战士》中用到的方法基本一致,还包括了我称之为“隐式 Z 缓冲区”的数据结构,可用于防止远处的图形在绘制时遮挡近处的图形。《计算机图形学:原理及实践》详细介绍了 BSP 树,以及一些构建并展示 BSP 树的伪代码(非常感谢我大学的图书馆,让我能够一睹这本教材 1990 年的版本)。因为这本书是计算机图形学的经典之作,所以卡马克很可能也有一本。
2022-02-11 22:03:25 +08:00
2022-08-19 12:21:58 +08:00
然而,卡马克发现自己遇到一个新问题:如何让第一人称射击游戏在一台 CPU 甚至都无法进行浮点操作的电脑上运行呢?通过调查研究,他证明了 BSP 树的数据结构非常适用于实时电子游戏渲染。尽管 BSP 树早已提出,而且到了卡马克的时代,相关理论已经非常成熟了,但我始终认为,卡马克的做法可谓惊人之壮举。也许,得到人们称誉的应该是整个《毁灭战士》的游戏引擎,它的确非常精致。我在前文也提及过,但是桑格勒德的《< ruby > 游戏引擎黑皮书:毁灭战士< rt > Game Engine Black Book: DOOM< / rt > < / ruby > 》 很好地讲解了这款游戏引擎的非凡之处, 以及这些优势相互契合之法。要明白, VSD 问题只是卡马克在编写《毁灭战士》游戏引擎时需要解决的诸多问题之一。不得不说,面对不为大多数程序员所知的复杂的数据结构,卡马克能够查阅相关文献,将其付诸实践,仅此一点就足以说明其技术之精湛、匠心之独到。
2022-02-11 22:03:25 +08:00
2022-07-30 19:09:41 +08:00
_如果你喜欢这篇文章, 欢迎关注推特 [@TwoBitHistory][14],也可通过 [RSS feed][15] 订阅, 获取最新文章( 每四周更新一篇) 。_
2022-02-11 22:03:25 +08:00
2022-08-12 11:39:26 +08:00
[^1]: Michael Abrash, “Michael Abrash’ s Graphics Programming Black Book,” James Gregory, accessed November 6, 2019, < http: / / www . jagregory . com / abrash-black-book / # chapter-64-quakes-visible-surface-determination > .
[^2]: R. Schumacher, B. Brand, M. Gilliland, W. Sharp, “Study for Applying Computer-Generated Images to Visual Simulation,” Air Force Human Resources Laboratory, December 1969, accessed on November 6, 2019, < https: / / apps . dtic . mil / dtic / tr / fulltext / u2 / 700375 . pdf > .
[^3]: Henry Fuchs, Zvi Kedem, Bruce Naylor, “On Visible Surface Generation By A Priori Tree Structures,” ACM SIGGRAPH Computer Graphics, July 1980.
[^4]: Fabien Sanglard, Game Engine Black Book: DOOM (CreateSpace Independent Publishing Platform, 2018), 200.
[^5]: Sanglard, 206.
[^6]: Sanglard, 200.
[^7]: David Kushner, Masters of Doom (Random House Trade Paperbacks, 2004), 142.
2022-02-11 22:03:25 +08:00
--------------------------------------------------------------------------------
via: https://twobithistory.org/2019/11/06/doom-bsp.html
作者:[Two-Bit History][a]
选题:[lujun9972][b]
2022-06-27 09:58:25 +08:00
译者:[aREversez](https://github.com/aREversez)
2022-08-19 11:04:49 +08:00
校对:[wxy](https://github.com/wxy)
2022-02-11 22:03:25 +08:00
本文由 [LCTT ](https://github.com/LCTT/TranslateProject ) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
[a]: https://twobithistory.org
[b]: https://github.com/lujun9972
[1]: tmp.eMwywbWYsp#fn:1
[2]: https://youtu.be/HQYsFshbkYw?t=822
[3]: tmp.eMwywbWYsp#fn:2
[4]: https://twobithistory.org/images/matrix_figure.png
[5]: tmp.eMwywbWYsp#fn:3
[6]: https://twobithistory.org/images/bsp.svg
[7]: https://twobithistory.org/images/bsp1.svg
[8]: https://twobithistory.org/images/bsp2.svg
[9]: tmp.eMwywbWYsp#fn:4
[10]: tmp.eMwywbWYsp#fn:5
[11]: tmp.eMwywbWYsp#fn:6
[12]: tmp.eMwywbWYsp#fn:7
[13]: https://en.wikipedia.org/wiki/Binary_space_partitioning
[14]: https://twitter.com/TwoBitHistory
[15]: https://twobithistory.org/feed.xml
[16]: https://twitter.com/TwoBitHistory/status/1164631020353859585?ref_src=twsrc%5Etfw
[17]: tmp.eMwywbWYsp#fnref:1
[18]: tmp.eMwywbWYsp#fnref:2
[19]: tmp.eMwywbWYsp#fnref:3
[20]: tmp.eMwywbWYsp#fnref:4
[21]: tmp.eMwywbWYsp#fnref:5
[22]: tmp.eMwywbWYsp#fnref:6
[23]: tmp.eMwywbWYsp#fnref:7