GAMES202 Real-time Environment Mapping

news2025/1/20 7:06:55

文章目录

  • 前言
  • Lecture5 Real-time Environment Mapping (prefiltering,split sum)
    • 补充:反射波瓣(Lobe)
  • Lecture6 Real-time Environment Mapping (precomputed radiance transfer)
    • Spherical Harmonics
    • precomputed radiance transfer
      • DIffuse 的情况
      • Glossy 的情况
      • 总结
        • 存在的问题:
        • 更进一步的研究:

前言

课程录播:
https://www.bilibili.com/video/BV1YK4y1T7yY?p=5

课程主页:
https://sites.cs.ucsb.edu/~lingqi/teaching/games202.html

课程BBS:
http://games-cn.org/forums/forum/games202/

Lecture5 Real-time Environment Mapping (prefiltering,split sum)

环境光照:用一张图来记录在场景中往任何一个方向看,可以看到的方向。

隐含条件:环境光照来自无穷远处。

给了一个环境贴图,如何求一个点的shading呢?自然是回归我们的渲染方程:
在这里插入图片描述
我们注意到,如果 BRDF 是glossy的(BRDF是 出射光的 Ridiance 的微分 比上 入射光的 Irradiance 的微分),此时集中在一小块区域,积分域 support 是很小的;如果 BRDF 是 diffuse 的,那他是比较平滑smooth的。
在这里插入图片描述
这就联想到之前讲阴影的时候,我们说那个近似公式的成立条件了:
在这里插入图片描述
于是我们可以先把 BRDF 从渲染方程里抽出来:
在这里插入图片描述
这样拆除来的好处是:light 和 brdf 一起作用我们不好求,我们这样可以把 brdf 表示的某个区域给积分起来再平均化,其实就是把 IBL 所表示的图给模糊了:
在这里插入图片描述
也就是我们可以去滤波我们的环境光,这个滤波核取多大取决于我们的 BRDF 占多大。

补充:反射波瓣(Lobe)

参考文章:
https://zhuanlan.zhihu.com/p/144437629
发现闫老师常用 lobe ,比如:
在这里插入图片描述
其实就是用反射波瓣去表示 BRDF,也即反射的分布。波瓣小,表示反射后的分布较集中(可以理解成分布在这个波瓣lobe上?),这个shading point比较glossy;反之,波瓣大,则diffuse

继续说,如上图左侧,我们查询反射的值,只能在球面上查对应反射方向的那一个值,这样就无法体现我们的 lobe;但没关系,我们提前做好了 filter,那么查询就是已经滤波后的值,也就相当于查询了周围一片方向的值(即这个 lobe 范围的值),如上图右侧。

举个例子,比如环境贴图是2k的,然后做了一个 filter 卷积模糊到了一个新的图;我们想查询一个 lobe 打过去的值,需要在原图找到这个 lobe 打中的那一块区域;但这里我们直接在原图上做了一个卷积模糊,新的图的一个像素就包含了周围像素的信息,因此只考虑 lobe 中心那个方向打中的地方也包含了之前整个 lobe 的信息(这时候涉及到 lobe 的大小我们就需要用插值了,双线性、三线性…)

回归原来的方程,我们还剩下对剩下的部分的积分:
在这里插入图片描述
如果要做预计算,角度一维、rgb三维、Roughness一维,一共五维的查找表,要存一个五维纹理,存储势必会爆炸(一般存三维纹理都很难接受了),因此不能硬存。

回顾:Microfacet BRDF
在这里插入图片描述
Fresnel 项可以用 Schlick 近似,NDF(法线分布函数,Normal Distribution Function) 也可以用一些分布去近似:
在这里插入图片描述
回顾旧笔记:https://zhuanlan.zhihu.com/p/427854383
在这里插入图片描述
这里我们把 F0 提出来(在202里面叫做R0),可以把原来的BRDF变成这样(除以一个菲涅尔项再乘以一个,乘以的那一个用Schlick近似):
在这里插入图片描述
这样做的好处是可以把 R0 拆到积分外面去,而 R0 是事先给好的,可以看作一个常数。

注:见我前面的笔记链接:
在这里插入图片描述
R0 就是上面的 F0 即 基础反射率,这是三维的,我们这样拆其实就是把这三维给剥离出积分项,而可以事先就给定这三维的指(其实也就和这个物体表面的颜色值相关了,PBR流程中查一个albedo贴图即可),因此从原先的五维降到了二维,从而可以预先计算出纹理来查找。

而剩下的几项又其实只和 Roughness 以及 Cos 项相关,于是可以预积分成一张纹理:
在这里插入图片描述
因此对于上面右边的那个式子,拆出来的两个积分项都只和 theta 以及 Roughness 相关,我们可以预先计算出来,并分别存入纹理的 R 和 G 通道即可,也因此上图的那个纹理是红绿色的(有这两个通道的值)。

Lecture6 Real-time Environment Mapping (precomputed radiance transfer)

实时渲染从环境光计算出阴影,非常非常难。high level 下来讲就是做不到。

但是也有一个工业界的解决方案,就是只生成一个最亮的光源下的阴影(比如太阳光)。另外一个就是PRT(precomputed radiance transfer)。

Spherical Harmonics

他是一系列的二维基函数,并且定义在球面上的(三维空间中的一个方向是二维的,因为可以拿 theta 和 phi 来描述)。有点像在一维情况下的傅里叶函数。
在这里插入图片描述
如上图,球谐函数每一行所描述的频率都是一样的,只是旋转或者形状有所不同。(颜色表示值大小,蓝色中越白的表示值越大,黄色则为负的,越白则绝对值越大。这里仅仅是可视化)

因此 Spherical Harmonics 仅仅是一系列二维的函数,并且他们有着不同的频率。并且每一行函数的个数就是 2 l + 1 个。l = k 叫做第 k 阶的SH,编号从 -k 到 k。(每一阶描述一个频率,阶越高频率越高)。每一个阶函数在数学上可以用勒让德多项式去描述。

给一个球面函数,用任意一组基底(记为Bi),前面的系数(记为Ci)可以很快求出来:
在这里插入图片描述
求这个系数的过程就叫做投影。

可以额外参考:
https://zhuanlan.zhihu.com/p/359856625

不过我们要是想恢复这个函数,我们其实不需要用到那么多阶(即不需要那么精确)。比如可以用前四阶的SH来恢复原来的函数,那么恢复出来的函数保留的最高的频率就是 l = 3 的频率,更高的被我们给扔了。这意味着我们只是损失了一些细节(高频),大体还是准的。(一般用前多少阶,则这些阶的基函数通常我们都用,比如 l = 2 第二阶,这五个基函数我们都用)

于是SH可以用来近似光照,比如之前说的某一点求shading的时候求他的球面的 irradiance 值。并且在 Diffuse 的 BRDF 下,本身的 shading 就主要是低频的信息,只需要用到前三阶(l = 012)的 SH 就可以描述了。

于是只剩下两个问题了:shadow 以及 非diffuse的BRDF

precomputed radiance transfer

在这里插入图片描述
这里BRDF是一个四维的,代表入射角和观察角,球面坐标一个方向是二维。这里是对某一个特定的shading point、特定的观察点来求BRDF值,那么观察方向已知,我们就可以对不同的入射方向存成一个贴图了,就像图中的BRDF那个黑白图一样。

注:
个人理解:
出射角的radiance微分比上入射角的irradiance微分,因为 radiance 的测量和光源大小相关,所以入射用irradiance,直接弄一个面积就可以方便测量了,单位是 sr-1,也就是球面度分之1,因为radiance比irradiance多除以了一个单位立体角(单位即球面度)。
而之所以微分的比可以作为一个值存在贴图中,是因为我们默认应该都是单位大小的。就像功率是在这里插入图片描述一样,Q是焦耳,t是秒,功率是瓦特,也就是一个单位时间内通过表面或者空间区域的能量的总量,因此这个积分应该默认是0到1的。

但是这只是一个shading point的计算,每次算三个值的乘积,一张图那么多分辨率,要全算一遍积分,这也太费了。于是搞出个PRT

DIffuse 的情况

在这里插入图片描述
我们做出假设:渲染的时候只有光照可以发生变化(例如换一种光照或者光照旋转),场景其他东西不变。那么我们就可以把渲染方程看成 lighting 和 light transport 两部分。

这样假设之后,light transport 其实只是针对一个点shading point的性质,于是可以提前预计算出来。并且在其他场景都不变的情况下 light transport 其实也就是一个球面函数(观察方向为两维自变量)。
在这里插入图片描述
我们把Light球谐展开,并且把BRDF提出来(diffuse情况下),做可以交换的假设,如上。剩下的 light transport 乘上那个球谐系数的积分 其实也就是 整个函数在这组球谐基上的投影。这部分的积分是可以预计算出来的。

但是代价是 light transport 所有东西都是固定的,这就意味着 visibility 项也是固定的,也就意味着场景所有的东西都是不能动的。另外一个代价是 Light 也做了预计算,那么只要光源被预计算过就可以替换,但是光源要是一旋转也就相当于换了一种光源,然而球谐具有旋转不变性,因此其实这种情况是可以处理的。( visibility 项是从 shading point 朝四面八方看,那个方向是否会被物体挡住,和光源没关系,我认为更确切的说是和环境光没关系)

在这里插入图片描述
因此之前说旋转光源,其实跟旋转球谐系数是一样的,我们就相当于把旋转光源变成了旋转任意一个 SH Basis。但是球谐的一个好性质是,旋转任意一个SH Basis他都能被同阶的 SH 的线性组合所得到。

因此,我们得出:要是光源旋转了,相当于对 SH 旋转,每一阶的 SH 旋转后都能被同阶的 SH 的线性组合所得到,于是又产生了新的一组 SH 系数,因此我们可以预先算出系数,得到一张表就行了。

在这里插入图片描述
在这里插入图片描述

Glossy 的情况

Glossy 和视点有关,BRDF 是四维的。因此 Light Transport 在 i 的方向的 SH 投影出来一定是一个关于 o (观察方向)的函数。
在这里插入图片描述
那么人们就想到也可以在 o 的方向投影出来一个 SH 结果:
在这里插入图片描述
因此其实就是对 diffuse 情况的一组延申:对任何一个 shading point 都需要存他的 transport matrix。

在这里插入图片描述
科研领域 glossy 的物体大家觉得十阶 SH 都还不够呢。当 glossy 非常高频的情况下,SH描述的效果并不好。(diffuse、glossy、specular,一般 glossy 介于 diffuse 和 specular 中间,不太严格的话可以认为glossy和specular差不多)

在这里插入图片描述
我们之前的 light transport 投影到 SH 上面去的这个式子(上图),还可以理解成,把每个 SH 的基函数看作一种光源,上式看成 rendering equation,然后就是这个光照来照亮了每个顶点,并且把这个照亮结果给记录下来。(上图是可视化了,红的越厉害表示越正,蓝的越厉害表示越负,所以看着有些奇怪)
在这里插入图片描述
像上图这种 Spatially Varying,即一个物体上任意一点也可能是另一种 BRDF,那么整个物体的 BRDF 就上升到六维了(物体上不同点的位置占二维)

总结

总结一下最早的这篇 PRT,它首先描述了用 SH 来把描述 lighting 和 light transport 两部分,对于 diffuse 的情况下就可以在每个 shading point 做一个点乘的操作,而如果是 glossy 的情况下每个顶点就做一个向量乘以矩阵的一个操作。
在这里插入图片描述

存在的问题:

  1. SH 基本上只适合描述低频的函数,比如下图用了 26 阶才基本恢复出高频来。
  2. 做了一个假设,一旦要预计算就需要场景是固定住的,即只适用于静态场景。场景中就不能动态去改变它的材质。
  3. 需要大量的预计算数据,带来存储以及读取的开销。
    在这里插入图片描述

更进一步的研究:

PRT 一些其他思路:
不用SH换成别的基函数、不用点积等等。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/40996.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于VggNet网络与ResNet神经网络的物体分类识别研究-附Matlab代码

⭕⭕ 目 录 ⭕⭕✳️ 一、引言✳️ 二、VGGNet网络与ResNet网络模型✳️ 2.1 VGG16 网络✳️ 2.2 ResNet-18网络✳️ 三、训练与实验结果✳️ 3.1 CIFAR-10 数据集✳️ 3.2 ResNet-18训练与识别结果✳️ 3.3 VGG16 网络训练与识别结果✳️ 四、参考文献✳️ 五、Mat…

使用vue脚手架快速搭建vue 2项目

简单了解vue-cli 官网地址:https://cli.vuejs.org/zh/guide/browser-compatibility.html 前提 1.安装node(js代码的运行环境)、npm; 2.全局安装vue-cli; 命令创建项目 vue create hello-word 等待项目创建成功即可 开发工具打开刚刚创建的项目 项目结构如图…

RS232和RS485

1. 串口存在的问题 电器接口不统一。UART只是对信号的时序和格式进行定义,未定义接口的电气特性 Uart通信一般直接使用从处理器的TTL电平,但不同的处理器使用的电平存在差异,导致不通处理器一般不能使用UART相互连接 UART没有规定连接器件的连…

Css 3 动画

动画(animation)是css3中具有颠覆性的特征之一,可通过设置多个节点来精确控制一个或一组动画,常用来实现复杂的动画效果。 相比较过渡,动画可以实现更多变化,更多控制,连续自动播放等效果。 动…

【毕业设计-课程设计】-单片机电子密码锁设计

资源链接在文章最后,订阅查看获取全部内容及资料,如需可私信提供硬件。 一、 主要功能: 1.按键设置6位密码,输入密码若密码正确,则锁打开。显示open! 2.密码可以自己修改(6位密码),必须是锁打开时才能改密。为防止误操作,修改密码得输入两次。 3.若密码输入错误次数…

Metabase学习教程:仪表盘-1

BI仪表盘最佳实践 学习如何制作出色的商业智能仪表盘。 在Metabase中,仪表盘可以在网格中组织图表和文本卡片。关于基本知识,请查看我们的文档创建仪表盘。这篇文章介绍了有关什么是优秀的商业智能仪表盘的高级概念,并包括一些关于如何充分…

网络程序设计——重叠I/O模型

目录 1、重叠I/O (1)概念 (2)重叠数据结构WSAOVERLAPPED 2、重叠I/O的相关函数 (1)套接字创建 (2)发送数据函数 (3)两种获取传输数据数量的方法 3、…

17.指针的概念及其分类

内存 存储器 存储数据的器件 外部存储器 长期存放数据,掉电不丢失数据。 常见的外存数据:硬盘、ROM、U盘 内部存储器 暂时存储数据,掉电丢失 常见的内存数据:ram、DDR 物理内存 实实在在的存储设备。 虚拟内存 操作系统虚拟…

FFmpeg进阶: 截取视频生成gif动图

文章目录1.封装视频滤镜2.截取视频生成gif3.gif优化4.示例效果现在互联网上很多人都通过表情包来表达自己的情绪,常用的表情包很多都是视频文件的一部分。这里就介绍一下如何通过ffmpeg截取视频生成gif动图。其实原理很简单,首先我们seek到视频对应的位置…

Go sync.WaitGroup的学习

一.前言 二. 夯实基础 2.1 sync.WaitGroup是什么? Go语言中除了可以使用通道(channel)和互斥锁进行两个并发程序间的同步外,还可以使用等待组进行多个任务的同步,等待组可以保证在并发环境中完成指定数量的任务 在…

Spring Boot 中的Thymeleaf分页和排序示例

在上一篇文章中,我们已经知道如何构建Spring Boot Thymeleaf示例。今天,我将继续使用 Spring Data 和 Bootstrap 进行 Thymeleaf 分页和排序(按列标题)。 百里香叶分页和排序示例 假设我们在数据库中有这样的教程表:…

深入理解 Android 模块化里的资源冲突

翻译自 Understanding resource conflicts in Android ⚽ 前言 作为 Android 开发者,我们常常需要去管理非常多不同的资源文件,编译时这些资源文件会被统一地收集和整合到同一个包下面。根据官方的《Configure your build》文档介绍的构建过程可以总结这…

RFSoC应用笔记 - RF数据转换器 -22- API使用指南之配置DAC相关工作状态和中断相关函数使用

前言 本文完结后,关于RFSoC的配置的API函数部分就全部介绍完毕,后续有空将更新介绍简单的射频收发回环示例工程,不定时更新,敬请期待。 配置DAC相关工作状态 XRFdc_SetInterpolationFactor 函数原型 u32 XRFdc_SetInterpolat…

内存一致性,指令重排序,内存屏障,volatile解析

文章目录为什么会存在“内存可见性”问题重排序与内存可见性的关系as-if-serial语义单线程程序的重排序规则多线程程序的重排序规则happen-before是什么解决方案:内存屏障Volatile关键字解决内存可见性问题的实现原理为什么会存在“内存可见性”问题 下图为x86架构…

redis 的企业实战应用 (二)

前言: 如今redis的常用场景有 短信登录:使用redis共享session来实现 商户查询缓存:会理解缓存击穿,缓存穿透,缓存雪崩等问题,让小伙伴的对于这些概念的理解不仅仅是停留在概念上,更是能在代码…

【数学】仿射变换

∣降维打击NightguardSeries.∣\begin{vmatrix}\Huge{\textsf{ 降 维 打 击 }}\\\texttt{ Nightguard Series. }\end{vmatrix}∣∣∣∣∣​ 降 维 打 击 Nightguard Series. ​∣∣∣∣∣​ 注:本文讨论的仿射变换仅为y轴上的伸缩变换,且难度在高中生理…

H3CNE V7.0 视频教程

构建中小企业网络全套PPT汇总【V7版本】 第1章 计算机网络概述 第2章 OSI参考模型与TCP IP模型 第3章 局域网基本原理 第4章 广域网基本原理 第5章 IP基本原理 第6章 TCP和UDP基本原理 第7章 路由器、交换机及其操作系统介绍 第8章 命令行操作基础 第9章 网络设备文件…

mycat-3-实战篇

1 总结: 1:用的表必须在mycat的配置文件中配置。 2:mycat默认分片策略中,都是针对表的主键,默认是id,如果主键不是id的,请去rule.xml自己复制一份修改 3: 2 注意细讲解 1:schem…

Springboot启动流程分析(四):完成启动流程

目录 一 添加BeanPostProcessors到IOC容器 二 国际化支持 三 初始化监听器的多路播放器 四 刷新容器 五 注册监听器到IOC容器的多播器 六 完成bean的大规模实例化 6.1 大规模实例化bean 6.1.1 连续三层do...while循环作用 6.1.2 FactoryBean是什么?为什么要…

04 YAML kubetnetes世界里的通用语

文章目录1. 前言2. 声明式和命令式是怎么回事?3. 什么是YAML?4. 什么是API对象?4.1 k8s都有哪些资源对象4.2 列出kubectl 命令详细执行过程5. 如何描述 API 对象5.1 命令式5.2 声明式5.2.1 声明式YAML语法详解5.2.1.1 header部分详解5.2.1.2 …