====== 간단한 3D 파이프라인 ====== - Vertex Shader -> Rasterizer : 정점위치 - Rasterize -> Pixel Shader : 픽셀위치 - Pixel Shader -> Screen : 색상 정점쉐이더 (Vertex Shader) * 정점수만큼 실행 된다. * 정점셰이더가 출력하는 점을 모아서 픽셀을 찍을 위치를 정한다음 * 픽셀 셰이더로 넘어간다. ====== BAsic (HLSL) ====== ====== 기본 코드 ====== ===== VertexShader ===== VS_OUTPUT vs_main( VS_INPUT input ) * VS_INPUT : 입력용 구조체 * VS_OUTPUT : 출력용 구조체 * 입,출력에 대한 구조체는 형식을 스스로 정의한다. ==== 변수 정의를 위한 타입 ==== * float4 : (x,y,z,w)의 성분을 갖는 벡터 변수 타입 * float4x4 : float4 타입의 4x4 행렬 변수 ==== 입출력 부분 구조체 선언 ==== // POSITION은 시맨틱으로 좌표정보임을 알리는 역할을 한다. struct VS_INPUT { float4 mPosition : POSITION; } struct VS_OUTPUT { float4 mPosition : POSITION; } ====화면 변환을 위한 행렬 변수 정의==== * 모델의 정점 정보를 읽어와서 - 월드좌표계로 변환 (행렬곱) 후 저장 - 위의 값에서 뷰좌표계로 변환 (행렬곱) 후 저장 - 원근/평면 투시를 위한 투시 변환 (행렬곱) 후 저장 이 과정이 끝나야 화면에 표시할 수 있는 좌표계로 변환된다. // 좌표계변환을 위한 각 행렬들. 셰이더 작성 프로그램에는 사전에 정의된 값이 있음. float4x4 gWorldMatrix; float4x4 gViewMatrix; float4x4 gProjectionMatrix; ====정점의 좌표 변환==== VS_OUTPUT vs_main(VS_INPUT Input) { VS_OUTPUT Output; Output.mPosition = mul( Input.mPosition, gWorldMatrix ); Output.mPosition = mul( Output.mPosition, gViewMatrix ); Output.mPosition = mul( Output.mPosition, gProjectionMatrix ); return Output; } ===== 픽셀셰이더 ===== * 리턴값이 float4 형식이지만 COLOR (rgba) 임을 명시 * 이해하기 좋게 쓰는 듯한.. float4 ps_main() : COLOR { return float4( 1.0f, 0.9f, 0.0f, 1.0f ); } ====== 난반사 (Diffuse) ====== * 여러 방향으로 고르게 퍼지는 직접광원의 하나 * 람베르트 모델을 사용해서 난반사광의 빛의 강도를 계산한다. 람베르트 모델 * 표면법선과 입사광의 각의 코사인을 구하면 난반사광의 양이 된다. * Cos는 0도일때 '1', 180일때 0이 되는 볼록한 모양의 그래프 * 표면법선 바로 위에 광원이 있을때, 최대치 * 표면법선과 90도 또는 그 이상이면 최소치가 된다. 난반사광 구하기 * cos을 사용하는 대신, * 내적dot product연산을 활용해서 구한다. 이런 순서로 * dot(A,B) = cos{theta} delim{|}{A}{|}delim{|}{B}{|} * theta : Vector_A, Vector_B의 각도 * delim{|}{A}{|} : Vector_A의 길이 * delim{|}{B}{|} : Vector_B의 길이 * costheta를 구하기 위해서 자리 이동 : \\ costheta = {dot(A,B)} / {delim{|}{A}{|}delim{|}{B}{|}} * 벡터길이를 1(단위벡터)로 설정 : \\ costheta = dot(A,B) ====== 정반사 (Specular) ====== 정반사Supercular Light는 입사각과 출사각이 같은 빛. * 난반사는 전체에 골고루 퍼지는 것 대비. 정반사를 구하는 것은 퐁모델을 사용. * 카메라벡터가 이루는 각도의 코사인을 구해서 여러번 거듭제곱하면 구한다고.. 구할때 사용할 벡터 * 출사각 벡터 * 카메라의 벡터 * 이 둘의 Cos 값 * 거듭제곱 : \\ 거듭제곱을 하면 cos 값이 점점 작아져서 빛이 동그라게 모이게 된다. ====== HLSL function ====== | reflect() | 벡터의 반사벡터를 (빛의 입사각의 반사각을 구하는) 구하는 함수 | | ::: | param1 : 입사각 방향벡터, param2 : 입사각이 닿는, 반사면의 법선벡터 | | mul() | 벡터 곱하기 | | normalize() | 노말벡터 구하기 | | dot() | 벡터 내적을 구한다. 두 벡터의 cos{theta}를 구하는데도 사용 | | | |