本篇讲一下相交检测的优化。有两个措施。

线段相交检测

之前的检测都是检测光线的终点是否在物体内。我们可以尝试检测光线的线段是否与物体相交。

比如说有一个非常薄的物体,光线差不多垂直于它的表面。如果用普通的方法的话,这个平面可能就会被光线跳过了。

我们将一个像素的厚度看做一维数轴上的一条线段,起点是其深度。同时将光线的起点、终点的深度值也用同样的方法看做一条线段。此时我们去检测这两条线段是否有重合。有的话则证明相交。

用这种方法可以解决薄的物体被跳过的问题。

			bool intersect(float raya, float rayb, float2 sspt) {        //raya rayb是光线两端的深度值,sspt是屏幕空间的坐标点。
float screenPCameraDepth = Linear01Depth(tex2Dlod(_CameraDepthTexture, float4(sspt / 2 + 0.5, 0, 0)));
float backZ = tex2Dlod(_BackfaceTex, float4(sspt / 2 + 0.5, 0, 0)).r; if (raya > rayb) { //因为光线方向不定(可能朝向+z或者-z)需要排序
float t = raya;
raya = rayb;
rayb = t;
}
return raya < backZ && rayb > screenPCameraDepth;
}

为了优化效果,我们在计算光线的屏幕空间的坐标时,可以取光线两个端点的中间点,投影到屏幕上作为采样点。

二分搜索优化

大部分情况下,我们的每个像素的采样次数不会很高。此时采样的质量会比较差。我们可以引入二分搜索,当检测到一个光线的相交时,我们在这段光线内部进行二分搜索,寻找精确的相交点。

这个图不好截,我找了kode80博客里的一张图进行说明。

值得注意的是,二分搜索只能对成功相交的光线进行优化,本身没有相交的是无法进行优化的。

以下代码中涉及到一些下篇的变量,但是大体意思很明了,即退回一段光线,然后在退回的区间中寻找相交。

if (intersect(screenPTrueDepth,prevDepth, screenPCurrent - dScreenPCurrent/2)){
#if 1 //二分搜索优化
float gapSize = PIXEL_STRIDE;
float2 screenPBegin = screenPCurrent - dScreenPCurrent; //回退
float oneOverZBegin = oneOverzCurrent - dOneOverZCurrent;
prevDepth = 1 / oneOverZBegin / -_ProjectionParams.z;
UNITY_LOOP
for (int j = 0; j < 10 && gapSize > 1.0; j++) {
gapSize /= 2;
dScreenPCurrent /= 2;
dOneOverZCurrent /= 2;
screenPCurrent = screenPBegin + dScreenPCurrent;
oneOverzCurrent = oneOverZBegin + dOneOverZCurrent;
screenPTrueDepth = 1 / oneOverzCurrent / -_ProjectionParams.z;
if (intersect(screenPTrueDepth, prevDepth, screenPCurrent)) { //命中了,起点不用动。(长度缩短一半即可) }
else { //没命中,将起点移动到中间。
prevDepth = screenPTrueDepth;
screenPBegin = screenPCurrent;
oneOverZBegin = oneOverzCurrent;
}
}
#endif
hitPixel = screenPCurrent / 2 + 0.5;
rayPercent = (float)i / STEP_COUNT;
return true;

在Unity中实现屏幕空间反射Screen Space Reflection(3)的更多相关文章

  1. 在Unity中实现屏幕空间反射Screen Space Reflection(4)

    第四部分讲一下如何在2D屏幕空间步进光线. http://casual-effects.blogspot.com/2014/08/screen-space-ray-tracing.html 中的代码感 ...

  2. 在Unity中实现屏幕空间反射Screen Space Reflection(1)

    本篇文章我会介绍一下我自己在Unity中实现的SSR效果 出发点是理解SSR效果的原理,因此最终效果不是非常完美的(代码都是够用就行),但是从学习的角度来说足以学习到SSR中的核心算法. 如果对核心算 ...

  3. 在Unity中实现屏幕空间反射Screen Space Reflection(2)

    traceRay函数 在上一篇中,我们有如下签名的traceRay函数 bool traceRay(float3 start, float3 direction, out float2 hitPixe ...

  4. 高级屏幕空间反射: Screen Space Reflection (SSSR)

    SSSR进一步调优,对标寒霜级技术水平,实现方式为Direct3D 11+自主实现实时渲染引擎,方法为对比测试.实现已经有段时间了,还是简要更新下吧.以下画面中的SSSR效果全部采用1:4 resol ...

  5. 高级屏幕空间反射: Screen Space Reflection (SSR)

    自从CE3首倡SSR以来,发展至今,其质量与当年早已不能同日而语.不仅强调超越性的质量,而且强调超越性的性能.乘着周末有空撸了撸,以下是增强型实时SSR结果图.与我原来的SSR原始实现相比,新的增强型 ...

  6. screen space reflection/soft alpha test/

    http://www.crytek.com/cryengine/presentations/secrets-of-cryengine-3-graphics-technology 很多宝贝里面 不止题目 ...

  7. 在Unity中实现屏幕空间阴影(1)

    接着上篇文章,我们实现了SSR效果. 其中的在屏幕空间进行光线追踪的方法是通用的.借此我们再实现一种屏幕空间的效果,即屏幕空间阴影. 文中的图片来自Catlike coding http://catl ...

  8. 在Unity中实现屏幕空间阴影(2)

    参考文章: https://www.imgtec.com/blog/implementing-fast-ray-traced-soft-shadows-in-a-game-engine/ 完成的工程: ...

  9. 关于Unity中的屏幕适配

    一.Game视图的屏幕分辨率可以先自定义添加,供以后选择,以下是手游经常用到的分辨率: 1.1136X640,iPhone5 2.1920X1080,横屏,主流游戏都是这个分辨率 3.1080X192 ...

随机推荐

  1. 敏捷开发之Scrum站立会议

    Scrum是迭代式增量软件开发过程,通常用于敏捷开发.站立会议通常指Scrun方法中的每日站立会议.顾名思义,是每天以站姿的方式召开的会议.以下从功能及要点方面对其进行解释说明: 功能:     1. ...

  2. jquery mobiscroll 滑动、滚动

    mobiscroll : 滑动选择 2.13.2版本免费,官网(mobiscroll.com)收费 先从官方下载2.13.2体验版下来,查看例子结合官方API学习( http://docs.mobis ...

  3. Mybatis 中 sql 语句的占位符 #{} 和 ${}

    #{} 表示一个占位符号,通过 #{} 可以实现 preparedStatement 向占位符中设置值,自动进行 java 类型和 jdbc 类型转换.#{} 可以有效防止   sql注入. #{}  ...

  4. 【C++】深度探索C++对象模型读书笔记--执行期语意学(Runtime Semantics)

    对象的构造和析构: 全局对象 C++程序中所有的global objects都被放置在程序的data segment中.如果显式指定给它一个值,此object将以此值为初值.否则object所配置到的 ...

  5. BZOJ 2152 聪聪可可(树形DP)

    给出一颗n个点带边权的树(n<=20000),求随机选择两个点,使得它们之间的路径边权是3的倍数的概率是多少. 首先总的对数是n*n,那么只需要统计路径边权是3的倍数的点对数量就行了. 考虑将无 ...

  6. bzoj2383[CEOI2011] ballons

    题意 在一条数轴上从左向右有一些气球,每个气球一开始位于横坐标xi的位置,是半径为0的圆.现在开始从左向右给每个气球充气.被充气的气球的半径会不断变大,直到达到这个气球的半径上限Ri或者这个气球和之前 ...

  7. 【HLSDK系列】服务端 AddToFullPack 函数

    服务端会给客户端发送一些数据,其中两大种类数据是 clientdata_t 和 entity_state_t 这里我们说说 entity_state_t 这个结构体. 你在丢在地上的枪.C4等等是服务 ...

  8. [洛谷P4503][CTSC2014]企鹅QQ

    题目大意:给你$n(n\leqslant3\times10^4)$个长度为$l(l\leqslant200)$的字符串,要你求出有多少对字符串是相似的,相似的定义是两个字符串只在一位上不同. 题解:可 ...

  9. TCP/IP协议详解---概述

        工作之后,才发现以前在学校里学的东西忘得太快太干净了,现在需要一点点地捡起来了,要不然写几行程序会闹很多笑话会出现很多bug的.从今天开始,翻一翻<TCP/IP协议详解 卷1>这本 ...

  10. NOIP2017 宝藏 题解报告【状压dp】

    题目描述 参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了 n 个深埋在地下的宝藏屋, 也给出了这 n 个宝藏屋之间可供开发的 m 条道路和它们的长度. 小明决心亲自前往挖掘所有宝藏屋中的宝藏.但是 ...