GAMES104课程笔记22-GPU-Driven Geometry Pipeline-Nanite
这个系列是GAMES104-现代游戏引擎:从入门到实践(GAMES 104: Modern Game Engine-Theory and Practice)的同步课程笔记。本课程会介绍现代游戏引擎所涉及的系统架构、技术点以及引擎系统相关的知识。本节课主要介绍GPU驱动的几何管线与Nanite技术。
Introduction
Traditional Rendering Pipeline
Nanite是虚幻5引擎中提出的虚拟几何系统用来实现渲染超高精度的网格。要理解Nanite首先要回顾一下经典渲染管线:当我们从CPU端发出渲染指令时会首先由CPU来准备各种渲染所需的资源,然后GPU会接收这些数据并计算实际的着色。这种模式的缺陷在于CPU可能无法跟上GPU的计算速度,而且CPU的算力会浪费在准备渲染素材这一过程中。
data:image/s3,"s3://crabby-images/1dc61/1dc61010ca897b35d9abe63d2c5fcb345ea1c609" alt=""
随着渲染场景的复杂度逐渐提升,CPU端的计算开销会成为整个渲染过程的瓶颈。
data:image/s3,"s3://crabby-images/83a47/83a4762e1abe066b19e186d81c78790bd4a272c8" alt=""
data:image/s3,"s3://crabby-images/47577/4757739bfc22f7598257017bf3c3caf33b1fbc79" alt=""
Compute Shader
为了提升渲染效率人们开发出了compute shader这样的技术,其核心在于把过去只能在CPU端执行的通用计算转移的GPU端,从而节约掉大量的CPU到GPU端的通信开销。
data:image/s3,"s3://crabby-images/4aeb5/4aeb5fe894e9886ab8bf986262289349ba8d0133" alt=""
Graphics API
在图形API层面上过去只能一次绘制一个网格,而现代图形API则支持在一次DrawCall中同时绘制多个网格。
data:image/s3,"s3://crabby-images/79e98/79e9883d5e72e2945edbdaa9b12901e835dbbf77" alt=""
GPU Driven Rendering Pipeline
总结一下,现代GPU驱动的渲染管线核心思想在于把CPU端的计算直接移动到GPU端,同时渲染所需的数据也会直接由GPU进行加载。在理想情况下CPU端只负责发出绘制指令,一切渲染数据加载和计算都在GPU端直接完成。
data:image/s3,"s3://crabby-images/b6b36/b6b3697c9606c565e1a44c9adcf25b96159fd74c" alt=""
GPU Driven Pipeline in Assassins Creed
游戏工业对GPU驱动渲染管线的大规模应用可以追溯到《刺客信条:大革命》。在游戏中我们可以看到大量的拥有真实细节的建筑和场景,如何渲染这些极其复杂的几何对象是整个渲染管线的巨大挑战。
data:image/s3,"s3://crabby-images/9aacf/9aacf133fa937509a103f3501ccc703eedf2f2be" alt=""
游戏开发团队提出了mesh cluster rendering的技术来提升渲染效率。mesh cluster rendering的思想在于对同一物体上的面片进行聚类,在渲染时首先根据cluster来判断面片的可见性
data:image/s3,"s3://crabby-images/46a07/46a07069937b397037fdd724ac2a7ce85023b38e" alt=""
整个游戏的渲染管线如下图所示。通过clustering的方法可以去除掉大量不可见的对象以及三角形,从而极大地缓解了GPU的渲染压力。
data:image/s3,"s3://crabby-images/6749d/6749db9e290e5f199d9f05510716142db915545c" alt=""
而在CPU端只负责非常少量的视锥剔除等工作,初步过滤掉不可见的物体。
data:image/s3,"s3://crabby-images/23cc5/23cc5f575499c73d7d2c7566770532deb0769433" alt=""
然后GPU端会把过滤后物体上的cluster拓展为chunk,每个instance可以属于不同的chunk而每个chunk可以包含不同的cluster。
data:image/s3,"s3://crabby-images/acba9/acba9c36ab516d72d209c36daa335de063c37286" alt=""
GPU端进行实际的可见性剔除时会先检查chunk的可见性然后计算cluster的可见性。除了利用bounding box进行剔除外,还会同时结合三角形的朝向进行过滤,最后得到所有可见的三角形编号。
data:image/s3,"s3://crabby-images/439bb/439bbb2f811e042b3687e4b42255c19ac0ffa993" alt=""
所有可见三角形的编号会存储在一个事先申请的巨大buffer中。写入过程是原子化的,因此可以利用GPU并行计算来高效处理。而在进行渲染时可以利用这个buffer来并行处理所有的三角形,从而实现对场景的渲染。
data:image/s3,"s3://crabby-images/a5768/a5768b7be5d56b64ba93bdc808893c26135aafb6" alt=""
data:image/s3,"s3://crabby-images/c50d2/c50d2916f6fb636e752d474b214b8183ca651e2f" alt=""
Occlusion Culling for Camera and Shadow
为了进一步提升渲染效率,除了剔除掉视野外的三角形外我们还希望能够把被遮挡住的三角形也同时剔除掉,这一过程称为occlusion culling。当相机在场景中的运动比较光滑时可以把前一帧的深度图投影到当前相机位置上,再结合hierarchy z-buffer就可以估计哪些cluster和三角形是可见的。
data:image/s3,"s3://crabby-images/1a044/1a044875b0d668d7fd78619da1fb93f57acaf21b" alt=""
Two-Phase Occlusion Culling
更现代的occlusion culling方法是使用上一帧和这一帧的两个z-buffer来实现。首先利用前一帧的z-buffer来快速选取可能可见的物体,然后使用这些物体来渲染新的z-buffer。显然此时的深度图会有非常多的洞等待填充,而且很多像素的深度可能是错误的。为了修正这个问题还需要再利用这一帧的深度图来测试前面过滤掉的其它物体。
data:image/s3,"s3://crabby-images/669f2/669f23e8318d5c27ef6a32a6d0121317a84e7e7c" alt=""
这种two-phase occlusion culling方法对于非常复杂的场景以及动态物体都有很好的性能。
data:image/s3,"s3://crabby-images/181e6/181e63402d869ddeac56c87166dca9e7c7e4f655" alt=""
而对于阴影的问题也可以复用前一帧阴影的深度图并结合hierarchy z-buffer来进行剔除。
data:image/s3,"s3://crabby-images/1eb44/1eb446843576a6935f40d1fc45249a7fab2403d0" alt=""
要进一步提升阴影的渲染效率还可以结合相机的可见性,把所有相机方向不可见的物体全部剔除掉。
data:image/s3,"s3://crabby-images/69f19/69f19090c37d8c39cad6d35b812e87a2383eb2ac" alt=""
data:image/s3,"s3://crabby-images/9c711/9c7113a480ef56bace8b17e7e35b78cb70176369" alt=""
Visibility Buffer
和Nanite相关的另一个技术是课程前面提到过的G-buffer和延迟渲染,我们可以把场景中的各种几何信息记录在G-buffer中从而方便渲染时的计算。
data:image/s3,"s3://crabby-images/f2941/f2941708729208291a0876092ff941c74873a8dd" alt=""
data:image/s3,"s3://crabby-images/df6fa/df6fa32ac25f7a0c50ce828008805b550da85064" alt=""
显然这样的G-buffer会占用非常多的显存,这在画面高分辨率或是复杂场景的情况下读取数据的效率会变得极其低下。
data:image/s3,"s3://crabby-images/21a06/21a0685908c427e1a8686421157fa7ddb908c8eb" alt=""
data:image/s3,"s3://crabby-images/63ff2/63ff224f0df02327a9705de96ddbbfa073a20773" alt=""
V-buffer是为了提升数据读取效率和缓存利用率而提出的一种技术。V-buffer中不会记录太多的几何信息,一般只保存像素上物体的各种编号。
data:image/s3,"s3://crabby-images/e4bb3/e4bb39a1ed88af6bbd9bac0b22b7220ec778f229" alt=""
data:image/s3,"s3://crabby-images/d257d/d257d2f7a3ea4b7c96269e8dd1c5786a40669522" alt=""
在进行着色时对每个像素需要先获取该处对应的三角形信息,然后通过插值来得到像素上相应的各种几何材质数据。这种渲染方式的优势在于计算量只与分辨率有关,而与场景的几何复杂度无关,因此拥有非常高的计算效率。
data:image/s3,"s3://crabby-images/40bbe/40bbecb2bf439aed3ca18c74f3d7dd3dfdfb0ccc" alt=""
data:image/s3,"s3://crabby-images/70513/705135d64cea0b1b7e1dba21ae731f0ae08a0b32" alt=""
V-buffer可以很容易地和延迟渲染管线进行结合。我们只需要利用V-buffer中可见物体的编号来重新写入G-buffer就可以完美融入延迟渲染管线中。
data:image/s3,"s3://crabby-images/76225/76225ddfcb61db0019e344c06e9fd0d68be12423" alt=""
当然V-buffer在实际使用时还有很多的细节要处理,比如说如何考虑纹理的梯度、如何选取合适的mip-map等。
data:image/s3,"s3://crabby-images/a45e1/a45e13eaabc527338684513e0d709a5c9105fb69" alt=""
使用V-buffer可以极大地提升具有复杂几何场景的渲染效率。
data:image/s3,"s3://crabby-images/2f233/2f233347ddc70e4860e46cd727b62488defba007" alt=""
Virtual Geometry - Nanite
Overview
Nanite的核心任务是实现实时电影级高精度几何模型的渲染,我们希望能够尽可能还原有着无限细节的真实世界。
data:image/s3,"s3://crabby-images/50ec5/50ec55d9b62cdda997df70a2a55df2030bb15edf" alt=""
data:image/s3,"s3://crabby-images/a53f3/a53f3b321fd9d518561471dc53206724da0acde0" alt=""
回忆基于virtual texture的技术我们可以为物体不同LoD的纹理烘焙在固定大小的纹理贴图上,在渲染时根据相机的位置和实际需要加载所需的纹理。这种材质表达可以提升缓存利用率以及数据加载效率。
data:image/s3,"s3://crabby-images/99f44/99f448af658716ab85844ad65381ae6ef425c1e7" alt=""
Nanite的思想与virtual texture非常相似,不过Nanite更关心的是如何建立虚拟的几何表示。当然几何数据本身要比纹理贴图要复杂得多,如何建立规范的几何表示至今仍然是一个难题。
data:image/s3,"s3://crabby-images/06005/06005237e4506a1042f8947a71e7fadb01676314" alt=""
data:image/s3,"s3://crabby-images/be7b7/be7b7739c99240355b165c1291d8cfefcbd265be" alt=""
以体素化表示为例,尽管体素本身是相对规范的但由于其巨大的数据量我们很难在游戏引擎中来直接使用。
data:image/s3,"s3://crabby-images/61070/610705c98d36bc5391b52b3f663396b58f63aff1" alt=""
另一种流行的几何表示方法是曲面细分(surface subdivision),基于这样的技术我们可以把粗略的几何表面细分为高精度包含各种细节的曲面。然而曲面细分的一个缺陷在于很难对曲面进行降采样,即从高精度曲面来获得低精度表示。
data:image/s3,"s3://crabby-images/568d1/568d1a88a180e2279eebd998f4f1d4c4f6679ebd" alt=""
其它的几何表达方式包括displacement map或是点云也都无法满足我们的需求。
data:image/s3,"s3://crabby-images/8399f/8399f8a5a4ad6613bac18a7f1eba1ef91396fa8e" alt=""
data:image/s3,"s3://crabby-images/e9f8c/e9f8c48936853ce704d44bba68977405e2eba43b" alt=""
因此在Nanite中还是选择了三角网格来表示,然后设计了一套非常复杂的算法流程来表达几何信息。
data:image/s3,"s3://crabby-images/fa763/fa763f4b64635f1a57e845e41bf59f088f775042" alt=""
Geometry Representation
Nanite的一个重要想法是利用屏幕的精度来控制渲染时所需计算三角形的数量。尽管三角形的数量可以随着模型精度的提高不断增长,但只要屏幕分辨率不变所需绘制的三角形数量应该是比较稳定的。
data:image/s3,"s3://crabby-images/b6c06/b6c06ee98666aeb6ddcdca1e49c928239bab2a83" alt=""
因此可以结合前面介绍过的mesh cluster来控制模型的细节。
data:image/s3,"s3://crabby-images/3eeb4/3eeb4719e4ff7de69644ca6981ff6a3c34a548eb" alt=""
然后根据相机与模型的相对远近关系来生成cluster在不同LoD下的几何表示。
data:image/s3,"s3://crabby-images/ed382/ed3823daf48919598e43da21dd1a284df6e1b5f5" alt=""
data:image/s3,"s3://crabby-images/091f0/091f08cf7194543be07b7d6568bd1681dcd18358" alt=""
在选择cluster的LoD时需要考虑它投影到屏幕上产生的误差。一种直观的选取方法是当误差小于1px时选择当前层的LoD,否则选取下一层的LoD。
data:image/s3,"s3://crabby-images/7a0c9/7a0c934d444d963e4c565d3609e5673e2ed5d8ef" alt=""
data:image/s3,"s3://crabby-images/53a57/53a5768a5a415decc40e92b0c2a70d9c873ec05d" alt=""
data:image/s3,"s3://crabby-images/19bc3/19bc3f187294a218c6e5a779930232bb6bb703e2" alt=""
但是在合并cluster时需要考虑不同LoD的cluster之间可能会出现缝隙。当然我们可以把cluster的边锁住,这样不管是使用哪一层的LoD都会有一致的边界。不过这样的处理并不是一个非常好的办法,可能会产生严重的artifact。
data:image/s3,"s3://crabby-images/d3f99/d3f992bfdd47c67f73ac8a12e33a446f8c8c9a9c" alt=""
data:image/s3,"s3://crabby-images/40eb4/40eb41e278917bb2f32351606195d949dff21cf0" alt=""
Nanite中提出了cluster group的概念来处理cluster之间的缝隙。cluster group之间的边界会被锁住,而内部的cluster会在生成LoD时一起进行简化。
data:image/s3,"s3://crabby-images/ad4ec/ad4ec359fa5431f2e9ecf3c8fae063b4c081b74a" alt=""
整个cluster简化的过程如下。需要注意的是简化后的cluster与原始cluster之间并不是一对多的关系,而是多对多的关系。即不同的简化后的cluster可以对应同一个原始cluster。
data:image/s3,"s3://crabby-images/65923/65923f99bf5805d32e79df97ce89482c719c855e" alt=""
data:image/s3,"s3://crabby-images/43560/435602886085ac17772d9f0b78ab1172fe3a6077" alt=""
data:image/s3,"s3://crabby-images/822cf/822cff618f0c991f9637e83c4bd6f9816de980a5" alt=""
data:image/s3,"s3://crabby-images/cfc29/cfc29bf3f7c856e59d036ab14c80b2384f4c68de" alt=""
data:image/s3,"s3://crabby-images/6326b/6326ba341c877bb74ade9241dfce84a3210e772e" alt=""
随着LoD的提高不同cluster group的边界也会发生相应的变化,这样可以避免出现高频噪声。
data:image/s3,"s3://crabby-images/8f1ff/8f1ffda5160374c86795905c925d00ad7c70176e" alt=""
data:image/s3,"s3://crabby-images/39973/39973fbdea8c59101ead6e26a9b0417c4931e540" alt=""
data:image/s3,"s3://crabby-images/5c242/5c2427fb571fee770082867c8ed72c102dc527d5" alt=""
data:image/s3,"s3://crabby-images/f72ac/f72aca7ecf0848c8431b02d7aa2a41e1a360cd25" alt=""
实际上这样的简化cluster过程可以表示为一张DAG,每个cluster在上一层LoD会有多个指向。
data:image/s3,"s3://crabby-images/e5e74/e5e7486d46f034b2005076523c77b40a271b4d9f" alt=""
data:image/s3,"s3://crabby-images/49575/49575c2a6feb34abb19478579ca3c593f7991e5c" alt=""
data:image/s3,"s3://crabby-images/fa908/fa908380885b746a13dc4417a466e9b4ec52313a" alt=""
而网格本身的简化则可以使用经典的QEM等简化算法来实现。
data:image/s3,"s3://crabby-images/86a68/86a680ed62c6eb93df2425ab3a89e344347ccac0" alt=""
data:image/s3,"s3://crabby-images/06c40/06c403563aa2a1ed1c42ef78992baf6fcef6436e" alt=""
Runtime LoD Selection
进行渲染时需要根据相机的位置来选择合适的LoD。不过对于DAG这样的数据结构进行访问时仍然是比较复杂的。
data:image/s3,"s3://crabby-images/c8983/c89833230bd849f84189a71810c36317949f42ac" alt=""
data:image/s3,"s3://crabby-images/df55a/df55a698378f48bafc5b3d9887d9399092f498f2" alt=""
Nanite还使用了并行化的技术来加速访问。
data:image/s3,"s3://crabby-images/98a9b/98a9b92e59f7df4cd07cf6f7785051d38fb8d8dd" alt=""
data:image/s3,"s3://crabby-images/7aafd/7aafdb32545f5bbe97b3cce4210c9c3f0a9d8b72" alt=""
data:image/s3,"s3://crabby-images/ae8bc/ae8bc0b3c44251860297de6a5a1bf12728e25ed8" alt=""
data:image/s3,"s3://crabby-images/f74f7/f74f7cec10cb0b4714f07e703d4d3a4ae0ea253c" alt=""
data:image/s3,"s3://crabby-images/0538a/0538a518e121b9ce3f40ac051801479ab6a72e27" alt=""
data:image/s3,"s3://crabby-images/94bc0/94bc06ea0abd4d05d29835d5d91fb8c8384a504f" alt=""
除此之外还可以使用BVH来加速LoD选择。
data:image/s3,"s3://crabby-images/7fe62/7fe624663ca3cc3081651569edf2b54a346601b3" alt=""
data:image/s3,"s3://crabby-images/714ae/714ae9318268ffecbe3653ddf42a372b5f952cd7" alt=""
data:image/s3,"s3://crabby-images/7edbc/7edbc73c91f0cdea5f9d83933aa9f1272c3fa206" alt=""
data:image/s3,"s3://crabby-images/7ef9d/7ef9d66bfceb3fe82cba5e4d0bcde63b71c0787f" alt=""
BVH的构建过程还可以使用job system来进行加速。
data:image/s3,"s3://crabby-images/49d36/49d36a94fbae35081f5b283dd83e3fc046f280f9" alt=""
data:image/s3,"s3://crabby-images/8096d/8096dff7fde11dd242b29501610bbf747729e0fc" alt=""
Rasterization
Nanite在渲染时很多三角形的大小已经接近于屏幕上的一个像素,此时需要硬件光栅化来提供支持。
data:image/s3,"s3://crabby-images/b1159/b1159c61dc985641994fc175ab7d2e04330d7ca6" alt=""
data:image/s3,"s3://crabby-images/1210d/1210da36cba100fa88a52d1eecaba1f2ea783fcd" alt=""
data:image/s3,"s3://crabby-images/ca904/ca904019cf6e6809f45769ee5ea2d96abf4c9a26" alt=""
传统光栅化对于小三角形的支持不够好,在Nanite中会结合compute shader来实现软光栅。
data:image/s3,"s3://crabby-images/d9261/d9261fb690f517b1641b1819df87209115114f73" alt=""
data:image/s3,"s3://crabby-images/6ce23/6ce23ec2ba1c56be63fe927986627de2f62cf6bd" alt=""
data:image/s3,"s3://crabby-images/b0be4/b0be4455a41d9e2e0e19d6e3bc67520f5d90cd1f" alt=""
data:image/s3,"s3://crabby-images/8279a/8279a73d0e3d35786e9ea6c919256a5e94b463ae" alt=""
在深度测试时,Nanite还利用了一些trick进行加速。实际渲染过程与V-buffer渲染过程类似。
data:image/s3,"s3://crabby-images/d471e/d471e1f2f1893fe94673f7478586f1df25d6aca2" alt=""
data:image/s3,"s3://crabby-images/bffcd/bffcda2279f5f0fdac23c6d971d7b6f649c723e2" alt=""
data:image/s3,"s3://crabby-images/c13ff/c13ff2407120699d85d564a0cec96a026904908d" alt=""
data:image/s3,"s3://crabby-images/48164/481642089b2daed36ecd703e4606bab9bf811e38" alt=""
data:image/s3,"s3://crabby-images/b775b/b775b5dcb82613edf7ea437df39350cf0d0510ac" alt=""
data:image/s3,"s3://crabby-images/4fd36/4fd36a7385fd1e956239f5f2323594676b75130a" alt=""
data:image/s3,"s3://crabby-images/ef69e/ef69e9e021f2a1fe138d29baff3c921c33fc8cc8" alt=""
Deferred Material
Nanite在绘制材质时会把材质信息转换为深度图,然后对可能出现的深度(材质)进行遍历。这样可以一次性绘制所有具有相同材质的像素。
data:image/s3,"s3://crabby-images/d93ce/d93ceb563f14dbba42ed0168be0e52af6b61ca07" alt=""
data:image/s3,"s3://crabby-images/02cd5/02cd553ca63e4529057d952b37b454992ea6fe8f" alt=""
data:image/s3,"s3://crabby-images/27987/279870cd341aae0d94159d6d144a842983052ff1" alt=""
更新的Nanite版本还会把屏幕划分为若干个tile,然后在每个tile上统计出现的材质。这样可以加速对全屏材质的遍历和绘制。
data:image/s3,"s3://crabby-images/25e3c/25e3c528c391766cf0c2c0141fbfc1ce2c0580ea" alt=""
data:image/s3,"s3://crabby-images/c7492/c7492a0b76c2f9ffa2d9ef167c2a1818bde847ef" alt=""
data:image/s3,"s3://crabby-images/35d29/35d29bb16c8dc6d91137af09e4af08d529ba6536" alt=""
data:image/s3,"s3://crabby-images/1b3bf/1b3bfdb780a2102255395675281eb7458ec1de86" alt=""
data:image/s3,"s3://crabby-images/1243f/1243fba66f4dd8fb7bdeb5dfeb84aeaea6fa0d9e" alt=""
Virtual Shadow Map
高精度几何模型还会导致阴影渲染时的困难,而且遗憾的是Nanite目前尚不支持实时光追来计算阴影。
data:image/s3,"s3://crabby-images/07920/07920755f226aa30366a30342e5a2001f0bd292a" alt=""
data:image/s3,"s3://crabby-images/d4e45/d4e45a494b0666fd7b43a1848e3519ca5c42cbd5" alt=""
不过计算阴影时也可以结合LoD,在距离相机不同远近的位置使用不同精度的模型。
data:image/s3,"s3://crabby-images/de71d/de71d0666696667f9f661cd6c5f9ec7333c6d47d" alt=""
data:image/s3,"s3://crabby-images/d7dc1/d7dc1b070189cfd1d22f27726c01fe50227d5d15" alt=""
data:image/s3,"s3://crabby-images/9c5d8/9c5d8f41a0d10249e1aa2186382b0c02461fbf17" alt=""
在这种思想下Nanite提出了virtual shadow map来表示不同精度的物体。
data:image/s3,"s3://crabby-images/07e7e/07e7e9f19231aaa748648a26c7109087c3c03c1c" alt=""
data:image/s3,"s3://crabby-images/3d37f/3d37f9810debbfdaf6d13ca0f6368585f10c33db" alt=""
对于不同类型的光源也可以定制划分virtual shadow map的方式。
data:image/s3,"s3://crabby-images/9163b/9163bc011495f01ef61522e9e6ea421124e9e5ae" alt=""
当相机和光源都不变时我们可以把shadow map相关的信息写入page中方便下一帧读取。而如果相机和光源发生变化则只需更新一部分page即可。
data:image/s3,"s3://crabby-images/bb35a/bb35ad039b1dedd778365fc0b32931b53d3ae2b5" alt=""
data:image/s3,"s3://crabby-images/12bc6/12bc690e3c5d246e33d9c3699297c43b49914fd1" alt=""
当然这种virtual shadow map在场景光源发生变化时会出现一些问题,因此比较适合主光源不变的场景。
data:image/s3,"s3://crabby-images/67c26/67c269911c1a52ccffbbc6242bbd7a7d702e947d" alt=""
data:image/s3,"s3://crabby-images/95617/95617c13691a992e72d0392b490f8f6b24eeb84f" alt=""
data:image/s3,"s3://crabby-images/59aed/59aedc0235d3acc4a9c7fa16be30c08b4d6bc5c9" alt=""
Streaming and Compression
data:image/s3,"s3://crabby-images/897b8/897b88ae96d41900880de03a7dfc71e7f95d97fc" alt=""
data:image/s3,"s3://crabby-images/20bdf/20bdff23ed8dc0f226c7eb4d5748b801b01c0d6c" alt=""
data:image/s3,"s3://crabby-images/86f39/86f39e073f6b2d52a01214339bd1d8234c9057ea" alt=""
data:image/s3,"s3://crabby-images/e2310/e2310984b2e2a4eb513eb118ad940fc20e1f567f" alt=""
data:image/s3,"s3://crabby-images/de5e9/de5e977a4d1159fb4a746177a187e6ab7283c75c" alt=""