[GPU프로그래밍] 5. The Basics of GLSL Shaders
data:image/s3,"s3://crabby-images/7b6f9/7b6f973d3c77e428b0935c6919e140cea76a6209" alt="post-thumbnail"
◾ Diffuse Shading
- Diffuse and Per-vertex Shading with a Single Point Light Sources
◾ Diffuse reflection
- Light vector와 normal vector를 내적하여 밝기 결정
- Ld: 광원의 밝기
- Kd: 산란되는 입사광의 비율을 나타내는 diffuse 반사계수
data:image/s3,"s3://crabby-images/02002/02002de0a0435c322f0b86875b2b918e8194331f" alt=""
data:image/s3,"s3://crabby-images/148ca/148ca92e54116f9c55cfc4fa3c12f15eb5ce490e" alt=""
- 내적값<0인 조명의 반대편은 완전 검정이다
📃 Code
- 버텍스와 normal을 Camera coordinate로 변환
- Diffuse shading 계산 시 음수가 나오지 않도록 최소값 0으로 설정
- (보통 reflection은 vertex가 아닌 fragment에서 하긴 하는데,, 여기선 vertex에서 함 -> fixed pipeline에서는 per-vertex로 계산하긴 했었다..)
data:image/s3,"s3://crabby-images/5e8c5/5e8c593340b5296ab82d1f9674a0e1d5ab5fa49c" alt=""
- vertex shader에서 계산된 LightIntensity가 Input으로 들어오고, 그대로 내보냄
data:image/s3,"s3://crabby-images/bbc68/bbc68c0e7ddb550791a386c6e91d55391ee5a8a6" alt=""
- Position
A
, B
와 그 표면에 대한 노멀N
- Polygon
AB
가 변환M이 생겼을 때 (x축으로 2배)
- 변환된 Polygon에 맞게 normal도 변환M−1T 필요
(rotation만 있는 경우는 M=M−1T라 polygon변환과 같게 해주면 되지만, Scaling도 들어가면 M=M−1T이라 추가 계산 필요) data:image/s3,"s3://crabby-images/be146/be146cad49bea9acb5bd9c6799cd67610a807bd2" alt=""
data:image/s3,"s3://crabby-images/63c31/63c3127fdeeb9b213e9d2f5797c3e572c66d22fc" alt=""
◾ Phong reflection model
- diffuse뿐만 아니라 ambient, specular (3개의 term)으로 이루어진 reflection model
data:image/s3,"s3://crabby-images/c938f/c938faedf414fc522fbe35be21a0432ff158c723" alt=""
data:image/s3,"s3://crabby-images/397ef/397efde7a1028daea1786bb61a0205c83ce1ff61" alt=""
📃 Code
- 위와 거의 동일, 계산만 다름
- Input이 많아서 구조체 사용
data:image/s3,"s3://crabby-images/02ec2/02ec2fe2a9f62eaacffc0d6c5ddb46fdf1a8b103" alt=""
◾ Functions in shaders
- const qualifier: input 매개변수가 함수 내에서 값이 바뀌지 않도록 (not with out or inout)
- Function overloading: 같은 이름을 가진 함수가 여러개 있을 수 있음
- 배열/구조체를 함수로 보내는 경우: call by value방식(복사)이라 메모리 문제가 생길 수 있음, 가능하면 global 변수로 보내는 것이 좋다
📃 Code
- 위의 phong reflection model의 코드를 함수를 사용해서 나타내면 아래와 같다 (간결!)
data:image/s3,"s3://crabby-images/bb257/bb25713664e7d0345c5208c6393cec3acd92ed96" alt=""
◾ Two-sided Shading
- 뒷면까지 렌더링하기 위해서, 기존 노멀 방향을 뒤집어서 반대 노멀 방향까지 고려해야함
data:image/s3,"s3://crabby-images/a493c/a493c3b2176956986b1bc333e8d9e61286f96de6" alt=""
📃 Code
- normal을 반전시켜 phongModel 계산해서 뒷면의 color를 얻음
- 어떤 경우에 front/back color를 칠할지 고려해야함
-> 전/후면 구별 필요 (2가지 방법)
- gl_ForntFacing (fragment shader)
- index buffer에 저장된 정보 중 Polygon의 버텍스가 그려지는 순서(winding)을 보고 구별
- Polygon 단위 정보이므로 fragment에서만 알 수 있다
-> color 계산 후 전/후면 판별data:image/s3,"s3://crabby-images/b8e88/b8e889e48bf81ee4ea885471f2d6c5b8cebc8f7d" alt=""
- Alternative Choice (vertex shader)
- winding이 잘 안되있는 경우도 존재, 그럴때는 기존 normal과 view dir을 내적하여 확인
- vertex에서 알 수 있음
-> 전/후면 판별하면서 color 계산data:image/s3,"s3://crabby-images/31185/311851beb46aae6a280ff7d3a6f175c1beaddf89" alt=""
Debugging
- color로 확인하는 경우가 많다
- winding이 잘 돼있는지 확인하기 위해 후면인 경우 빨강으로 그려본,,~ (원래 색이랑 좀 mix한 상태, 하지 않으면 너무 평면처럼 보여서 오히려 파악이 더 어려움)
data:image/s3,"s3://crabby-images/e4b13/e4b13a3beecc13858e75f681d49f3927769043ba" alt=""
◾ Flat Shading
- 한 polygon에 같은 색
data:image/s3,"s3://crabby-images/a023c/a023c52437f1567a99519c126c50568d2417d1cb" alt=""
- fixed pipeline에서는 버텍스마다 똑같은 color를 파이프라인에 보내준다 ?
- Polygon의 vertex 중 하나만 선택하여 색을 결정하게 됨 (Provoking vertex)
(원래는 vertex에서 fragment로 넘어갈 때, interpolation해서 색 결정됨)
- vertex shader의 out, fragment shader의 in variable에
flat
qualifier 사용data:image/s3,"s3://crabby-images/cb8e5/cb8e570e848d993ca809abc892ac975ed311ae5a" alt=""
- Polygon의 몇 번째 vertex를 사용할 것인지 (default: 첫번째 vertex)
data:image/s3,"s3://crabby-images/5032d/5032df1f41709d95bab7c0848b895e5081410e67" alt=""
- flat 효과 뿐만 아니라, 한 polygon에 vertex 상관없이 동일한 값을 주고싶은 경우 등.. 활용 가능
◾ Subroutines
- select shader functionality
- 상황에 따라 렌더링을 다르게하기 위해 Uniform변수를 조건으로 활용하여 함수 mappinig
(조건문 느낌-if문은 시간이 좀 걸려서.. 요새는 아니긴 하지만)
- OpenGL Application에서 shader별로 프로그램 만들어서 if문으로 하는 방법도 있긴 있음
- Shader 코드에서는 Subroutine Type Uniform변수를 받아 그에따른 함수 실행
data:image/s3,"s3://crabby-images/dd7e5/dd7e5d7a92b30c6e1efcda2cf3957ae954c47055" alt=""
- OpenGL Application에서는 Uniform변수에 따른 함수 mapping
- 해당 shader program의 handle값을 사용하여 특정 함수의 index값을 받아온다
- 그 index값으로 subroutine에 mapping할 함수를 지정한다
(mapping할 함수는 여러개 지정할 수 있어서 주소값으로 전달)data:image/s3,"s3://crabby-images/a06a4/a06a4ac38cbe1306ea7e07a20be64d53c43918fe" alt=""
- Subroutine uniform변수를 지정하고 넘겨주는 부분이 없음 (기존 uniform변수처럼 주소값 받아서 넘겨주는..)
-> Subroutine Type이 default로 0, 1, ..로 지정되기 때문
◾ Discarding Fragments
- to create a perforated look
- fragments를 버리는 것 -> 구멍 (배경을 먼저 그리기 때문에 배경이 보임)
📃 Code
- 뚫린 부분으로 보이는 뒷부분(안쪽)을 칠해주기 위해 backColor도 계산
data:image/s3,"s3://crabby-images/98dfb/98dfb1f4341a38d32110ab4b9aa9bd858fabe02d" alt=""
- Texture coordinate를 사용해서 일정하게 뚫리도록 함
- TexCoord에 15를 곱해서 [0, 15]범위로 만든 뒤, 소수부분만 남겨서 0~0.9가 15번 반복되도록 한다
- 그 TexCoord를 vec(0.2, 0.2)와 비교하여 0.2보다 큰 경우에 fragments를 버린다 -> 격자무늬
- color를 그리는 과정은 버려지지 않은 fragments에만 하면 되므로 버린 뒤에 진행한다
data:image/s3,"s3://crabby-images/dd68d/dd68d6023039b4616e3625498d2f404c98c10ae4" alt=""