【Unity大气渲染】Unity Shader中实现大气散射(半成品)

news2024/11/13 15:23:51

写在前面

这是之前在做天空盒的时候同步写的分析博客,结果后面写到一半就忘了继续了,这里先贴出当时写的半成品,有小伙伴问我怎么做的,这里只能尽力把之前的半成品先放出来了(写得很乱,勿怪orz),,后面有机会会完善好的!希望能帮到大家~

前置知识学习

【Unity大气渲染】关于单次大气散射的理论知识

关于卡通渲染的大气散射

因为我想实现的是跟Unity 卡通渲染 程序化天空盒 昼夜变化,而作者实现过程的文章Unity 卡通渲染 程序化天空盒里说了只是想实现一个日出日落的效果,大概做了如下两点简化:

  • 只计算了Mie散射
  • 光学距离部分只计算了D(\overline{PA}),省去了计算D(\overline{CP})

既然不是基于物理的,那【Unity大气散射】GAMES104:3A中如何实现大气散射里整理的LUT优化方式等等内容就派不上用场了,但也无妨!多了一次学习~

那么接下来计算会尽量参考着这两点去做~ 很大部分的过程参考【实战】从零实现一套完整单次大气散射_四 - 知乎

分析原神的日出

截图来自【原神】好美,原神一天中天空的变化_哔哩哔哩_bilibili

早晨,是地平线的颜色Bloom超突出,伴随着缓缓升起的Mie散射的日出效果: 

随着太阳升起,地平线Bloom褪去,留下明显的Mie散射: 

日出结束,进入白天时间,太阳始终伴随着一个炫光(猜测是直接叠加了一个炫光图案),加上有随太阳高度角变化的镜头光晕效果: 

关于地平线Bloom和太阳炫光、镜头光晕部分这里不提,先实现如此的Mie散射。

1 整体思路

我们需要模拟出人站在地面上看向天空的过程,其实就是人眼发出射线与天空相交的过程,这个过程还须判断是否被物体挡住,挡住了返回黑色;没挡住就调用一个单次散射函数,返回散射的值。

1.2 构建坐标系

  • 首先要有一个明确的坐标系:这里将地球顶部作为世界原点建立坐标系,换句话来说,就按照Unity的世界坐标就行!
  • 人眼此时就是A了,那么在Unity中相机的位置就是A的世界坐标,从A点出发发出射线与天空相交于点B,这时射线AB就是整个模拟过程中的“视线”

下图是解释图,我觉得画的不错hhh

1.2 函数1:射线与球体求交

数学原理没什么好说的,随便找了个解释原理的文章:

射线与球体/三角面片求交、重心坐标、插值_csu_xiji的博客-CSDN博客_射线求交

函数返回值

返回的是±t,rayOrigin+t*rayDir就能得到两个交点啦。

//-------------------------------------
// RaySphereIntersection
//-------------------------------------
float2 RaySphereIntersection(float3 rayOrigin, float3 rayDir, float3 sphereCenter, float sphereRadius)
{
    rayOrigin -= sphereCenter;
    float a = dot(rayDir, rayDir);
    float b = 2.0 * dot(rayOrigin, rayDir);
    float c = dot(rayOrigin, rayOrigin) - (sphereRadius * sphereRadius);
    float d = b * b - 4 * a * c;
    if (d < 0)
    {
        return -1;
    }
    else
    {
        d = sqrt(d);
        return float2(-b - d, -b + d) / (2 * a);
    }
}

1.3 计算的部分

就是主要的单次散射函数,主要有两个关键:

  • 计算

2 计算T(PA)

参考开头的简化方法,我们不计算D(\overline{CP}),那么整体就是按照下式计算了!其中,海平面的散射系数是已知的,相位函数和光学距离(累加相对大气密度)都需要计算。

2.1 函数2:计算大气密度

//---------------------------------------------------------------------------
// GetAtmosphereDensity
//---------------------------------------------------------------------------
void GetAtmosphereDensity(float3 position, float3 planetCenter, float3 lightDir, out float dpa, out float dpc)
{
    // ρ(h)= exp(-h/H)
    float height = length(position - planetCenter) - _PlanetRadius;
    dpa = exp(-height / _DensityScaleHeight);

    // we don't computer the D(PC)
    dpc = 0;
}

其中,_DensityScaleHeight为大气厚度。

2.2 函数3:相位函数

//---------------------------------------------------------------------------
// ApplyPhaseFunction
//---------------------------------------------------------------------------
void ApplyPhaseFunction(inout float scatterMie, float cosAngle)
{
    // only Mie
    float g = _MieG;
    float g2 = g * g;
    float phase = (1.0 / (4.0 * PI)) * ((3.0 * (1.0 - g2)) / (2.0 * (2.0 + g2))) * ((1 + cosAngle * cosAngle) / (pow((1 + g2 - 2 * g * cosAngle), 3.0 / 2.0)));
    scatterMie *= phase;
}

2.3 函数3:计算总散射

//---------------------------------------------------------------------------
// IntegrateInscattering
//---------------------------------------------------------------------------
float4 IntegrateInscattering(float3 rayStart, float3 rayDir, float rayLength, float3 planetCenter, float3 lightDir, float sampleCount)
{
    float3 stepVector = rayDir * (rayLength / sampleCount);
    float stepSize = length(stepVector);

    float2 prevDPA = 0;
    float prevTransmittance = 0;

    float densityPA = 0;
    float densityCP = 0;
    float localDPA = 0;

    float scatterMie = 0;

    GetAtmosphereDensity(rayStart, planetCenter, lightDir, localDPA, densityCP);

    densityPA += localDPA * stepSize;
    prevDPA = localDPA;

    // local Inscattering, densityCP=0
    float Transmittance = exp(-(densityCP + densityPA) * _ExtinctionM) * localDPA;
    prevTransmittance = Transmittance;

    [loop]
    for (float i = 1.0; i < sampleCount; i += 1.0)
    {
        float3 p = rayStart + stepVector * i;
        GetAtmosphereDensity(p, planetCenter, lightDir, localDPA, densityCP);
        densityPA += (prevDPA + localDPA) * (stepSize / 2.0);        
        
        Transmittance = exp(-(densityCP + densityPA) * _ExtinctionM) * localDPA;

        scatterMie += (prevTransmittance + Transmittance) * stepSize / 2;

        prevTransmittance = Transmittance;
        prevDPA = densityPA;
    }
    // phase function
    ApplyPhaseFunction(scatterMie, dot(rayDir, -lightDir.xyz));

    float3 lightInscatter = _ScatteringM * scatterMie;

    return float4(lightInscatter, 1);
}

其中,为了方便用户调试,cpp里对于Mie散射的S项和T项分别提供可控参数MieScatterCoef和MieExtinctionCoef:

Vector4 MieSct = new Vector4(2.0f, 2.0f, 2.0f, 0.0f) * 0.00001f;

_ExtinctionM = MieSct * MieExtinctionCoef;
_ScatteringM = MieSct * MieScatterCoef;

也就是_ExtinctionM是计算所取的S项海平面处的散射系数,_ScatteringM是计算所取的M项海平面处的散射系数,都是由cpp直接传入,MieSct取值参考:

1.1

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

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

相关文章

用C语言实现一个任意类型的队列

下面是一个简单的无类型队列的实现&#xff1a; #include <stdio.h> #include <stdlib.h>typedef struct Node {void *data;struct Node *next; } Node;typedef struct Queue {Node *front;Node *rear;int size; } Queue;void enqueue(Queue *queue, void *data);…

[蓝桥杯] 枚举、模拟和排列问题

文章目录 一、连号区间数 1、1 题目描述 1、2 题解关键思路与解答 二、递增三元组 2、1 题目描述 2、2 题解关键思路与解答 三、错误票据 3、1 题目描述 3、2 题解关键思路与解答 四、回文日期 4、1 题目描述 4、2 题解关键思路与解答 五、归并排序 标题&#xff1a;蓝桥杯——…

windows和linux出现timewait过多的解决方法

一、timewait出现在客户端还是服务端以及什么情况下出现 我是做性能测试的。在压测过程中遇到了timewait过多的情况&#xff0c;下面来看一下timewait产生的原因及解决办法&#xff0c;我自己在服务器起了一个很简单的springboot应用来验证自己的猜想及解决办法。 说到产生原…

基于docker部署prometheus

1、prometheus架构 Prometheus Server: 收集指标和存储时间序列数据&#xff0c;并提供查询接口 ClientLibrary:客户端库 Push Gateway: 短期存储指标数据。主要用于临时性的任务 Exporters:采集已有的第三方服务监控指标并暴露 metrics Alertmanager:告警 Web UI :简单的…

VS插件CodeRush全新发布v22.2.4——改进对VS 17.5的支持

CodeRush是一个强大的Visual Studio .NET 插件&#xff0c;它利用整合技术&#xff0c;通过促进开发者和团队效率来提升开发者体验。CodeRush能帮助你以极高的效率创建和维护源代码。Consume-first 申明&#xff0c;强大的模板&#xff0c;智能的选择工具&#xff0c;智能代码分…

【C++升级之路】第九篇:vector

&#x1f31f;hello&#xff0c;各位读者大大们你们好呀&#x1f31f; &#x1f36d;&#x1f36d;系列专栏&#xff1a;【C学习与应用】 ✒️✒️本篇内容&#xff1a;vector的基本概念、vector的使用&#xff08;构造&#xff0c;迭代器&#xff0c;空间增长&#xff0c;增删…

小樽C++ 单章③ 一维数组

目录 一、一维数组认识与使用 1.4 数组的输入与输出 1.5 最大值与最小值的求解 二、一维数组的操作用法 2.1 数组的插入与删除 2.2 数组某个值的查找 2.3多个数组的合并 2.4多个数组的有序合并 三、一维数组的应用 3.1 斐波那契数列 3.2 淘淘摘苹果 3.3 翻纸牌游戏…

RTOS中互斥量的原理以及应用

互斥量的原理 RTOS中的互斥量是一种同步机制&#xff0c;用于保护共享资源&#xff0c;防止多个任务同时访问该资源&#xff0c;从而避免数据竞争和不一致性。 互斥量的原理是通过对共享资源进行加锁和解锁操作来实现的。 在RTOS中&#xff0c;互斥量通常是一个数据结构&…

为什么说要慎用BeanUtils,因为性能真的拉跨

1 背景之前在专栏中讲过“不推荐使用属性拷贝工具”&#xff0c;推荐直接定义转换类和方法使用 IDEA 插件自动填充 get / set 函数。不推荐的主要理由是&#xff1a;有些属性拷贝工具性能有点差有些属性拷贝工具有“BUG”使用属性拷贝工具容易存在一些隐患&#xff08;后面例子…

九龙证券|磷酸亚铁锂是什么?磷酸亚铁锂的特点和性能介绍

磷酸亚铁锂是一种新式锂离子电池电极资料&#xff0c;化学式&#xff1a;LiFePO4&#xff0c;磷酸亚铁锂为近来新开发的锂离子电池电极资料&#xff0c;首要用于动力锂离子电池&#xff0c;作为正极活性物质运用&#xff0c;人们习气也称其为磷酸铁锂。 磷酸亚铁锂的特色和功能…

【计算机网络】随机访问介质访问控制中的ALOHA协议、CSMA协议、CSMA/CD协议和CSMA/CA协议

二、随机访问介质访问控制 信道上的所有用户可随机发送信息 1. 简单ALOHA协议 解决信息发送时的冲突问题。其特点是不监听信道&#xff0c;不按时间槽发送&#xff0c;随机重发。 思想&#xff1a;在超时后&#xff0c;等随机一段时间后再重传 2. 时隙ALOHA协议 时隙ALOH…

送给她最最浪漫的表白(Python代码实现)

目录I 最美的浪漫 II 昨日浪漫的表白III 关于“美”的哲思 IV Python代码实现 I 最美的浪漫 II 昨日浪漫的表白 她问:有一个问题&#xff0c;我只问一遍&#xff0c;为什么是我?我微微一笑答道:笞案很长&#xff0c;我要用一生来回答。我&#xff1a;图书馆旁边的星空真美&…

Linux面试总结

一.常用命令1.目录切换cd / 切换到根目录cd ../ 切换到上级目录cd ~ 切换到home目录2.查看目录ls 列出当前目录下所有的文件ls [路径]ls / 查看根目录 ls -l 相当于 ll 最常用的命令,用了表的方式列出当前目录的内容3.查看当前目录pwd-4.创建一组空文件touch5.显示文件内容cat6…

Maven打包操作

对于企业级项目&#xff0c;无论是进行本地测试&#xff0c;还是测试环境测试以及最终的项目上线&#xff0c;都会涉及项目的打包操作。对于每个环境下的项目打包&#xff0c;对应的项目所需要的配置资源都会有所区别&#xff0c;实现打包的方式有很多种&#xff0c;可以通过an…

【Linux】进程理解与学习(Ⅰ)

环境&#xff1a;centos7.6&#xff0c;腾讯云服务器Linux文章都放在了专栏&#xff1a;【Linux】欢迎支持订阅&#x1f339;相关文章推荐&#xff1a;【Linux】冯.诺依曼体系结构与操作系统进程概念什么是进程&#xff1f;进程是什么&#xff1f;我们打开任务管理器可以看到有…

Flex弹性布局一文通【最全Flex教学】

文章目录一.Flex布局1.1 传统布局和flex布局1.1.1 传统布局1.1.2 flex弹性布局1.2 flex初步体验1.3 布局原理二.常见Flex属性2.1 常见父项属性2.2 flex-direction主轴的方向2.3 justify-content设置主轴上的子元素排列方式2.4 设置子元素是否flex-wrap换行2.5 align-itmes设置侧…

数列的极限

数列的极限的定义&#xff1a; 当正多边形的边数越来越多时&#xff0c;正多边形的面积越来越接近于圆的面积。 这个过程就叫做极限。 数列的形式&#xff1a; 数列的每一个下标对应一个数列元素值。 认识数列 这就是4个数列的表现形式。 当数列的下标无限增长时&#xff0c…

记录--uni-app中安卓包检查更新、新版本下载、下载进度条显示功能实现

这里给大家分享我在网上总结出来的一些知识&#xff0c;希望对大家有所帮助 需求描述 如果想要做一个app的话&#xff0c;可以有很多种选择方案&#xff0c;uni-app是其中的一个性价比高一些(坑多一些)的方案。本文记录一下&#xff0c;uni-app打安卓包以后&#xff0c;需要检查…

【操作系统】如何避免预读失效和缓存污染的问题?

【操作系统】如何避免预读失效和缓存污染的问题&#xff1f; 文章目录【操作系统】如何避免预读失效和缓存污染的问题&#xff1f;Linux 和 MySQL 的缓存Linux 操作系统的缓存MySQL 的缓存传统 LRU 是如何管理内存数据的&#xff1f;预读失效&#xff0c;怎么办&#xff1f;什么…

bash if 判断及特殊字符

bash if 判断 1)字符串判断 str1 str2     当两个串有相同内容、长度时为真 str1 ! str2     当串str1和str2不等时为真 -n str1       当串的长度大于0时为真(串非空) -z str1       当串的长度为0时为真(空串) str1        当串str1为非…