성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] How can I tell whether two programs...
[정성태] The case of the fail-fast crashes c...
[정성태] Creating Docker multi-arch images f...
[정성태] BinaryFormatter removed from .NET 9...
[정성태] Extending the Windows Shell Progres...
[우광현] 와..... 범위를 잡았으니 클라이언트가 해당 범위를 확인해본다...
[정성태] 딱히, 그것 이상으로 더 설명할 내용이 없습니다. 동적 포...
[정성태] If Windows 3.11 required a 32-bit p...
[정성태] What is a hard error, and what make...
[괴물신인] 질문작성자인데 이 글을 이제봤네요 ㄷㄷ 이 글처럼 타입별로 인...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>Unity로 실습하는 Shader (8) - 다중 패스(Multi-Pass Shader)</h1> <p> Unity shader에서 다중 패스를 이용한 방법이 은근히 까다롭군요. 아래의 글을 읽고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 유니티에서 다른 타입의 쉐이더를 멀티 패스로 통합하기 / Unity merging different type shaders using multi pass ; <a target='tab' href='http://rapapa.net/?p=2723'>http://rapapa.net/?p=2723</a> ; <a target='tab' href='https://github.com/inbgche/ShaderMixingSample/blob/master/Assets/Shader/FV_FV.shader'>https://github.com/inbgche/ShaderMixingSample/blob/master/Assets/Shader/FV_FV.shader</a> </pre> <br /> 다음과 같이 그대로 베껴서 구현해 봤는데,<br /> <br /> <pre style='height: 400px; margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > //Multi-Pass Shader Test. Rapapa.net Shader "Test/Multi-Pass" { Properties { } SubShader { Tags {"Queue" = "Geometry" "RenderType" = "Opaque" } //////////////////////////////////////////////////////// //Vertex-Fragment Functionality shader - RED // //////////////////////////////////////////////////////// Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag struct vertexInput { float4 vertex : POSITION; float4 color : COLOR; }; struct vertexOutput { float4 pos : POSITION; float4 color : COLOR; }; vertexOutput vert(vertexInput v) { vertexOutput o; o.pos = mul(UNITY_MATRIX_MVP, v.vertex); o.color = v.color; return o; } fixed4 frag( vertexOutput i) : COLOR { return fixed4(1, 0, 0, 1); } ENDCG } //////////////////////////////////////////////////////// //Vertex-Fragment Functionality shader - GREEN // //////////////////////////////////////////////////////// Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag struct vertexInput { float4 vertex : POSITION; float4 color : COLOR; }; struct vertexOutput { float4 pos : POSITION; float4 color : COLOR; }; vertexOutput vert(vertexInput v) { vertexOutput o; o.pos = mul(UNITY_MATRIX_MVP, v.vertex); o.color = v.color; return o; } fixed4 frag( vertexOutput i) : COLOR { return fixed4(i.color.r, 1, 0, 1); } ENDCG } } Fallback "Diffuse" } </pre> <br /> 첫 번째 pass에서 붉은색을 반환하고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > fixed4 frag( vertexOutput i) : COLOR { return fixed4(1, 0, 0, 1); } </pre> <br /> 두 번째 pass에서 이전 값의 Red 값과 새롭게 Green 값을 합성한다고 하는데,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > fixed4 frag( vertexOutput i) : COLOR { return fixed4(<span style='color: blue; font-weight: bold'>i.color.r, 1,</span> 0, 1); } </pre> <br /> 실제로 해보면 진짜 노란색이 나옵니다. 문제는, i.color의 rgb 값이 전부 1이기 때문에 그런 식으로 잘 동작하는 것처럼 보인 것입니다. 일례로 다음과 같이 해도 실행해 보면 노란색이 나옵니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > fixed4 frag( vertexOutput i) : COLOR { // 이렇게 반환해도 노란색, return fixed4(<span style='color: blue; font-weight: bold'>i.color.g</span>, 1, 0, 1); // 이렇게 반환해도 노란색, return fixed4(<span style='color: blue; font-weight: bold'>i.color.b</span>, 1, 0, 1); // 이렇게 반환하면 하얀색으로 출력 return <span style='color: blue; font-weight: bold'>i.color</span>; } </pre> <br /> "<a target='tab' href='http://rapapa.net/?p=2723'>유니티에서 다른 타입의 쉐이더를 멀티 패스로 통합하기 / Unity merging different type shaders using multi pass</a>" 원 글에서 참조 링크로 걸은 글에 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Rules for Multi-pass Shaders in Unity ; <a target='tab' href='http://albertshih.blogspot.com/2014/11/rules-for-multi-pass-shaders-in-unity.html'>http://albertshih.blogspot.com/2014/11/rules-for-multi-pass-shaders-in-unity.html</a> </pre> <br /> 다음과 같이 지적하고 있습니다.<br /> <br /> <div style='BACKGROUND-COLOR: #ccffcc; padding: 10px 10px 5px 10px; MARGIN: 0px 10px 10px 10px; FONT-FAMILY: Malgun Gothic, Consolas, Verdana; COLOR: #005555'> (Each pass has its own properties <span style='color: blue; font-weight: bold'>that will not be "passed" on to other passes</span>. For a list of those properties, check here) </div><br /> <br /> QA에 답변과 종합해 보면,<br /> <br /> <div style='BACKGROUND-COLOR: #ccffcc; padding: 10px 10px 5px 10px; MARGIN: 0px 10px 10px 10px; FONT-FAMILY: Malgun Gothic, Consolas, Verdana; COLOR: #005555'> Q: Can you put two vertex/fragment shaders together?<br /> A: Yes, just put two passes with vertex/fragment shaders right next to each other. <span style='color: blue; font-weight: bold'>The second one will be drawn over the first one.</span> (Note that some properties will be passed on from one pass to the other. For example, if you use a vertex shader to change the mesh geometry, the changes will still be there in the next pass.) </div><br /> <br /> 그러니까, 각각의 vertex shader에 들어온 vertexInput.color는 이전 패스에서 넘어온 값이 아닙니다. 이것은 초기화 값으로 Unity의 render target은 기본적으로 "white"로 칠해져 있으므로 다중 패스의 모든 vertex shader는 색상 값이 float4(1, 1, 1, 1)로 넘어오는 것입니다.<br /> <br /> 그리고, 나중에 실행되는 패스가 이전 패스에서 칠한 값에 상관없이 자신의 값으로 덮어쓰는 결과가 됩니다. 그러니까, 위의 멀티 패스는 전혀 사용할 수 없는 예제인 것입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 실제로 멀티 패스가 덧씌우는 동작을 하고 있는지 확인을 해볼까요?<br /> <br /> 이를 위해서는 투명 렌더링을 해보면 됩니다. 따라서 Blend 및 RednerType 옵션을 설정하고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Shader "My/multipassTestShader" { Properties { } SubShader { <span style='color: blue; font-weight: bold'>Tags{ "RenderType" = "Transparent" "Queue" = "Transparent" } Blend SrcAlpha OneMinusSrcAlpha</span> // ...[생략]... } } </pre> <br /> 첫 번째 패스에서 다음과 같이 Red만 설정하고 렌더링하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Pass { CGPROGRAM // ...[생략]... vertexOutput vert(vertexInput v) { vertexOutput o; o.pos = UnityObjectToClipPos(v.vertex); o.color = v.color; return o; } fixed4 frag(vertexOutput i) : COLOR { <span style='color: blue; font-weight: bold'>return fixed4(1, 0, 0, 1);</span> } ENDCG } </pre> <br /> (모델을 Sphere로 했다고 가정했을 때) 화면에는 빨간색 구체가 그려집니다. 반면 fixed4(1,0,0,0)으로 alpha 값을 0으로 반환하면 Blend 옵션으로 인해 아무것도 안 그려지게 됩니다.<br /> <br /> 이렇게 빨간색 구체가 그려진 상태에서,<br /> <br /> <img alt='multi_pass_1.png' src='/SysWebRes/bbs/multi_pass_1.png' /><br /> <br /> 이제 두 번째 패스를 추가하겠습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Pass { CGPROGRAM // ...[생략]... vertexOutput vert(vertexInput v) { vertexOutput o; if (v.vertex.x > 0.1) // 로컬 좌표계로 x가 0.1보다 큰 경우에는 Green 색상을 그리고, { o.color = float4(0, 1, 0, <span style='color: blue; font-weight: bold'>1</span>); } else { // 그렇지 않은 경우 alpha == 0을 주어 그리지 않게 만듦. o.color = float4(0, 0, 0, <span style='color: blue; font-weight: bold'>0</span>); } o.pos = UnityObjectToClipPos(v.vertex); return o; } fixed4 frag(vertexOutput i) : COLOR { return i.color; } ENDCG } </pre> <br /> 중간의 if 문의 역할은 의미를 두고 해석하지 말고, 단순히 덮어쓸 여부를 확인하기 위한 정도로만 보시면 됩니다. 즉, 일부는 Green 색상이 alpha == 1이므로 덮어 그릴 것이고, 일부는 alpha == 0이므로 덮어 그리지 않을 것입니다. 실제로 이것을 실행해 보면 다음과 같이 나옵니다.<br /> <br /> <img alt='multi_pass_2.png' src='/SysWebRes/bbs/multi_pass_2.png' /><br /> <br /> 이 정도면 어떤 것인지 확인이 되었겠죠? ^^<br /> <br /> <hr style='width: 50%' /><br /> <br /> 참고로, shader 코드는 Pass 외부로 빼서 공통 코드를 공유하는 것이 가능합니다. 즉, 아래와 같이 각각의 pass에서 구현하는 것도 가능하지만,<br /> <br /> <pre style='height: 400px; margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Shader "My/multipassTestShader" { Properties { } SubShader { Tags{ "RenderType" = "Transparent" "Queue" = "Transparent" } Blend SrcAlpha OneMinusSrcAlpha Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag struct vertexInput { float4 vertex : POSITION; float4 color : COLOR; }; struct vertexOutput { float4 pos : POSITION; float4 color : COLOR; }; vertexOutput vert(vertexInput v) { vertexOutput o; o.pos = UnityObjectToClipPos(v.vertex); o.color = v.color; return o; } fixed4 frag(vertexOutput i) : COLOR { return fixed4(1, 0, 0, 1); } ENDCG } Pass { CGPROGRAM #pragma vertex vert #pragma fragment frag struct vertexInput { float4 vertex : POSITION; float4 color : COLOR; }; struct vertexOutput { float4 pos : POSITION; float4 color : COLOR; }; vertexOutput vert(vertexInput v) { vertexOutput o; if (v.vertex.x > 0.1) { o.color = float4(0, 1, 0, 1); } else { o.color = float4(0, 0, 0, 0); } o.pos = UnityObjectToClipPos(v.vertex); return o; } fixed4 frag(vertexOutput i) : COLOR { return i.color; } ENDCG } } } </pre> <br /> 다음과 같이 CGINCLUDE/ENDCG 쌍을 이용해 외부로 빼서 공통 코드는 재활용하는 것도 가능합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Shader "My/multipassTestShader" { Properties { } <span style='color: blue; font-weight: bold'>CGINCLUDE</span> struct vertexInput { float4 vertex : POSITION; float4 color : COLOR; }; struct vertexOutput { float4 pos : POSITION; float4 color : COLOR; }; vertexOutput vert(vertexInput v) { vertexOutput o; o.pos = UnityObjectToClipPos(v.vertex); o.color = v.color; return o; } fixed4 frag(vertexOutput i) : COLOR { return fixed4(1, 0, 0, 1); } vertexOutput vert2(vertexInput v) { vertexOutput o; if (v.vertex.x > 0.1) { o.color = float4(0, 1, 0, 1); } else { o.color = float4(0, 0, 0, 0); } o.pos = UnityObjectToClipPos(v.vertex); return o; } fixed4 frag2(vertexOutput i) : COLOR { return i.color; } <span style='color: blue; font-weight: bold'>ENDCG</span> SubShader { Tags{ "RenderType" = "Transparent" "Queue" = "Transparent" } Blend SrcAlpha OneMinusSrcAlpha Pass { <span style='color: blue; font-weight: bold'>CGPROGRAM #pragma vertex vert #pragma fragment frag ENDCG</span> } Pass { <span style='color: blue; font-weight: bold'>CGPROGRAM #pragma vertex vert2 #pragma fragment frag2 ENDCG</span> } } } </pre> <br /> <hr style='width: 50%' /><br /> <br /> 자, 그럼 우리가 원하는 멀티 패스를 어떻게 구현해야 하는 걸까요? 검색해 보면 다음의 결과가 있군요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > CommandBuffer 를 이용한 Multipass Shader 기법 ; <a target='tab' href='http://scripter.co.kr/298'>http://scripter.co.kr/298</a> </pre> <br /> 근데, 좀 복잡합니다. ^^; 예전 글에서 구현한 Gaussian Blur가,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Unity로 실습하는 Shader (7) - Blur (평균값, 가우스, 중간값) 필터 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/11620'>http://www.sysnet.pe.kr/2/0/11620</a> </pre> <br /> 사용한 2차원 커널을 1차원 커널로 x와 y에 대해 멀티 패스로 구현하면 된다고 했는데요. "<a target='tab' href='http://scripter.co.kr/298'>CommandBuffer 를 이용한 Multipass Shader 기법</a>" 글에서 설명한 것처럼 복잡하다면 차라리 2차원 커널을 사용해 단일 shader로 그리는 것이 더 효율적인 것 같습니다.<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
6716
(왼쪽의 숫자를 입력해야 합니다.)