在DirectX11下用Stencil Buffer绘制可视化Depth Complexity
这是一道在《Introduction to 3D Game Programming with DirectX 11》上的练习题。
要求把某个像素点上的Depth Complexity(深度复杂度?)可视化输出。Depth Complexity即某个点上有多少次depth test。
根据题目中的提示,我们可以用stencil buffer完成。
简要步骤如下:
绘制场景,设置DepthStencilState为stencil test永远通过,通过后stencil buffer中的值+1。这样每个点绘制后在stencil buffer中的点就会+1。
根据stencil buffer中的值绘制深度信息。
要做到这一点,我们只需要置StencilFunc为D3D11_COMPARISON_EQUAL,此时只有在stencil buffer中的值和stencilRef中的值相同的像素点会通过测试。接着对所有的可能的深度值(比如0~5),把整个屏幕用设定的颜色重新绘制一遍即可。这里的绘制不需要具体的顶点信息,只需要覆盖整个屏幕即可(我们会绘制一个覆盖屏幕的四边形,在接下来的shader中可以看到我们是如何做到这一点的)
综上所述,我们需要创建两个depth stencil state,第一个用来绘制场景,同时给stencil buffer中对应的点+1。注意,这个depth stencil state应该设置为在不通过或者通过depth test时都为stencil buffer +1。
建立一个名为DepthCounterDSS的depth stencil state
D3D11_DEPTH_STENCIL_DESC dsdesc = { 0 };
dsdesc.DepthEnable = true; //深度测试无所谓
dsdesc.DepthFunc = D3D11_COMPARISON_LESS;
dsdesc.DepthWriteMask = D3D11_DEPTH_WRITE_MASK_ALL;
dsdesc.StencilEnable = true;
dsdesc.StencilReadMask = 0xff;
dsdesc.StencilWriteMask = 0xff;
dsdesc.FrontFace.StencilFunc = D3D11_COMPARISON_ALWAYS; //Stencil test永远通过
dsdesc.FrontFace.StencilDepthFailOp = D3D11_STENCIL_OP_INCR; //depth fail失败了也+1
dsdesc.FrontFace.StencilPassOp = D3D11_STENCIL_OP_INCR;
dsdesc.FrontFace.StencilFailOp = D3D11_STENCIL_OP_KEEP; //不会失败,这项无意义。
//背面被剔除了,下面这些设置无意义。
dsdesc.BackFace.StencilDepthFailOp = D3D11_STENCIL_OP_INCR;
dsdesc.BackFace.StencilFailOp = D3D11_STENCIL_OP_KEEP;
dsdesc.BackFace.StencilFunc = D3D11_COMPARISON_ALWAYS;
dsdesc.BackFace.StencilPassOp = D3D11_STENCIL_OP_INCR;
HR(device->CreateDepthStencilState(&dsdesc, &DepthCounterDSS));
第二个depth stencil state用于绘制深度信息。需要做的工作就是让stencil buffer中的值和stencilRef相等时通过stencil test,否则不通过即可。
注意,此时我们要绘制的图形只有一个覆盖整个屏幕的四边形(两个三角形组成)。
建立一个名为DepthVisualDSS的depth stencil state
D3D11_DEPTH_STENCIL_DESC dvdesc = { 0 };
dvdesc.DepthEnable = false; //深度测试无所谓
dvdesc.DepthFunc = D3D11_COMPARISON_LESS;
dvdesc.DepthWriteMask = D3D11_DEPTH_WRITE_MASK_ALL;
dvdesc.StencilEnable = true;
dvdesc.StencilReadMask = 0xff;
dvdesc.StencilWriteMask = 0xff;
dvdesc.FrontFace.StencilFunc = D3D11_COMPARISON_EQUAL; //相等时才通过
dvdesc.FrontFace.StencilDepthFailOp = D3D11_STENCIL_OP_KEEP; //不管成功失败都不修改stencil buffer
dvdesc.FrontFace.StencilPassOp = D3D11_STENCIL_OP_KEEP;
dvdesc.FrontFace.StencilFailOp = D3D11_STENCIL_OP_KEEP;
//背面无所谓。
dvdesc.BackFace.StencilDepthFailOp = D3D11_STENCIL_OP_KEEP;
dvdesc.BackFace.StencilFailOp = D3D11_STENCIL_OP_KEEP;
dvdesc.BackFace.StencilFunc = D3D11_COMPARISON_EQUAL;
dvdesc.BackFace.StencilPassOp = D3D11_STENCIL_OP_KEEP;
HR(device->CreateDepthStencilState(&dvdesc, &DepthVisualDSS));
接下来我们修改DrawScene()函数,使其在绘制场景前设置stencil depth state为我们的DepthCounterDSS,然后照常绘制即可。(注意在绘制中途stencil depth state有无改变)
void BlendApp::DrawScene()
{
md3dImmediateContext->ClearRenderTargetView(mRenderTargetView, reinterpret_cast<const float*>(&Colors::Silver));
md3dImmediateContext->ClearDepthStencilView(mDepthStencilView, D3D11_CLEAR_DEPTH|D3D11_CLEAR_STENCIL, 1.0f, 0);
md3dImmediateContext->OMSetDepthStencilState(RenderStates::DepthCounterDSS, 0); //StencilRef没有用
//普通绘制
//……
//绘制结束,此时stencil buffer中存的就是每个像素的深度复杂度
DrawComplexity();
}
再实现DrawComplexity函数
void BlendApp::DrawComplexity()
{
ID3DX11EffectTechnique* quadTech = Effects::BasicFX->DepthVisual;
D3DX11_TECHNIQUE_DESC techDesc;
quadTech->GetDesc(&techDesc);
for (UINT p = 0; p<techDesc.Passes; ++p)
{
for (int i = 0; i<10; ++i) //假设最多有10层深度
{
md3dImmediateContext->OMSetDepthStencilState(RenderStates::DepthVisualDSS, i); //设置stencilref=i
Effects::BasicFX->SetDVColor(Colors::White/10*i); //设置绘制颜色,i越大白色越浓
quadTech->GetPassByIndex(p)->Apply(0, md3dImmediateContext);
md3dImmediateContext->Draw(3, 0); //绘制一个四边形,具体的顶点信息由vertex shader自己给出
}
}
}
以上代码中的DepthVisual是一个简单的对全屏幕进行一次指定颜色绘制的technique,shader代码如下。
cbuffer quadColor
{
float4 gDVColor;
}
VertexOut VSfullscreenQuad(uint id : SV_VERTEXID) //SV_VERTEXID由gpu给出,指定当前顶点序号
{
VertexOut output;
// 计算齐次坐标
output.PosH.x = (float) (id / 2) * 4.0 - 1.0f;
output.PosH.y = (float) (id % 2) * 4.0 - 1.0f;
output.PosH.z = 0.0f;
output.PosH.w = 1.0f;
return output;
}
float4 PScolorOnly(VertexOut pin) : SV_Target
{
return gDVColor;
}
technique11 DepthVisual
{
pass P0
{
SetVertexShader(CompileShader(vs_5_0, VSfullscreenQuad()));
SetGeometryShader(NULL);
SetPixelShader(CompileShader(ps_5_0, PScolorOnly()));
}
}
完成效果如下

在DirectX11下用Stencil Buffer绘制可视化Depth Complexity的更多相关文章
- windows API下的模板缓冲(stencil buffer)
在windows API搭建的OpenGL窗口中使用模板缓冲,需要在像素格式描述表中设置stencil buffer位宽为8,这样窗口会自动生成stencil buffer,然后可以在opengl环境 ...
- Directx11教程(50) 输出depth/stencil buffer的内容
原文:Directx11教程(50) 输出depth/stencil buffer的内容 有时候,我们需要查看depth/stencil buffer的内容,比如上一章中,我们要查看sten ...
- Directx11教程(48) depth/stencil buffer的作用
原文:Directx11教程(48) depth/stencil buffer的作用 在D3D11中,有depth/stencil buffer,它们和framebuffer相对应,如下图所 ...
- Stencil Buffer
刚在草稿箱里发现了这篇充满特色的好日志.发表之. ------------------吃货的分割线---------------------------------------- Stencil Bu ...
- UnityShader实例09:Stencil Buffer&Stencil Test
http://blog.csdn.net/u011047171/article/details/46928463 Stencil Buffer&Stencil Test 在开始前先吐槽下uni ...
- pytorch下可采用visidom作为可视化工具
2018/9/18更新 感觉tensorboardX插件更好用,已转用https://github.com/lanpa/tensorboardX 更新:新版visdom0.1.7安装方式为:cond ...
- Directx11教程(49) stencil的应用-镜面反射
原文:Directx11教程(49) stencil的应用-镜面反射 本教程中,我们利用stencil来实现一个镜面反射效果. 1.首先我们要在D3DClass中增加几个成员变量及函数. I ...
- UnityShader学习笔记- Stencil Buffer
模板测试(Stencil Test)是现代渲染流水线的一环,其中涉及到的就是模板缓冲(Stencil Buffer),模板缓冲可以用来制作物体的遮罩.轮廓描边.阴影.遮挡显示等等效果 目录 Stenc ...
- depth/stencil buffer的作用 ----------理解模板缓存 opengl
在D3D11中,有depth/stencil buffer,它们和framebuffer相对应,如下图所示,framebuffer中一个像素,有相对应的depth buffer和stencil buf ...
随机推荐
- RevealTrans图片切换效果
RevealTrans 更新时间:2013-06-01 17:11:59 | RevealTrans兼容性:IE5.5+ 语法: filter : progid:DXImageTransform.Mi ...
- 理解JAVA常量池
下面是一些String相关的常见问题: String中的final用法和理解final StringBuffer a = new StringBuffer("111");final ...
- 虚拟现实-VR-UE4-编译UE4源码
通过Git将UE4源代码获取到本地计算机 切记路径不要有中文 这里面我已经在进行编译了,有部分文件是多余出来的, 第一步就是点击 setup.bat批处理,这个过程回取决与你的网速的快慢,我等了一下午 ...
- BZOJ1222[HNOI 2001]产品加工
题面描述 某加工厂有A.B两台机器,来加工的产品可以由其中任何一台机器完成,或者两台机器共同完成.由于受到机器性能和产品特性的限制,不同的机器加工同一产品所需的时间会不同,若同时由两台机器共同进行加工 ...
- 孤荷凌寒自学python第七十二天开始写Python的第一个爬虫2
孤荷凌寒自学python第七十二天开始写Python的第一个爬虫2 (完整学习过程屏幕记录视频地址在文末) 今天在上一天的基础上继续完成对我的第一个代码程序的书写. 直接上代码.详细过程见文末屏幕录像 ...
- LeetCode - 70. Climbing Stairs(0ms)
You are climbing a stair case. It takes n steps to reach to the top. Each time you can either climb ...
- 使用Scrapy自带的ImagesPipeline下载图片,并对其进行分类。
ImagesPipeline是scrapy自带的类,用来处理图片(爬取时将图片下载到本地)用的. 优势: 将下载图片转换成通用的JPG和RGB格式 避免重复下载 缩略图生成 图片大小过滤 异步下载 . ...
- 今日头条 2018 AI Camp 6 月 2 日在线笔试编程题第二道——两数差的和
题目 给 n 个实数 a_1, a_2 ... a_n, 要求计算这 n 个数两两之间差的绝对值下取整后的和是多少. 输入描述 第一行为一个正整数 n 和一个整数 m.接下来 n 行,第 i 行代表一 ...
- Drools 7.4.1.Final参考手册(十四)集成Spring
集成Spring Drools 6.0重要变更 Drools Spring集成经历了与Drools 6.0的变化完全一致的改造. 以下是一些主要的变化: T*推荐的Drools Spring的前缀已经 ...
- Elasticsearch 监控和部署
Elasticsearch: ! [ https://elasticsearch.cn/book/elasticsearch_definitive_guide_2.x/_cluster_health. ...