HLSL学习1

来源:互联网 发布:矩阵分析作业答案 编辑:程序博客网 时间:2024/06/05 01:00
什么是着色器

DirectX使用管道技术(pipeline)进行图形渲染,其构架如下:

1.1 Direct3D Graphics Pipeline

之前我们使用管道的步骤如下:

1.      设定顶点、图元、纹理等数据信息;

2.      设定管道状态信息;

   渲染状态

通过SetRenderState方法设定渲染状态;

另外,使用以下方法设置变换、材质和光照:

            SetTransform

            SetMaterial

            SetLight

            LightEnable

            通过SetSamplerState方法设定取样器状态;

            通过SetTextureStageState设定纹理层状态;

3. 渲染;

这部分交由D3D管道按照之前的设定自行完成,这部分操作是D3D预先固定的,所以这种管道技术被称为固定功能管道(fixed function pipeline)

 

固定功能管道给我们编程提供了一定的灵活性,但是仍有很多效果难以通过这种方式实现,比如:

1.      在渲染过程中,我们要求y坐标值大于10的顶点要被绘制到坐标值(000)的地方,在之前的固定功能管道中,顶点被绘制的位置是在第1步即被设定好的,不可能在渲染过程中进行改变,所以是不可行的;

2.      谋顶点在纹理贴图1上映射为点A,在纹理贴图2上映射为点B,我们要求该顶点颜色由AB共同决定,即:

定点颜色 = A点色彩值*0.7 + B点色彩值*0.3

  这在固定管道编程中也是不可行的。

以上两个问题都可以由可编程管道(pragrammable pipeline)来解决。

      可编程管线允许用户自定义一段可以在GPU上执行的程序,代替固定管道技术中的Vertex ProcessingPixel Processing阶段(参照图1.1),从而在使我们在编程中达到更大的灵活性。其中替换Vertex Processing的部分叫做Vertex Shader(顶点着色器),替换Pixel Proccessing的部分叫做Pixel Shader(像素着色器),这就是我们所说的着色器Shader

 

什么是HLSL

Direct8.x中,着色器是通过低级着色汇编语言来编写的,这样的程序更像是汇编式的指令集合,由于其效率低、可读性差、版本限制等缺点,迫切要求出现一门更高级的着色语言。到了Direct3D9HLSLHigh Level Shading Language,高级渲染语言)应运而生了。HLSL的语法非常类似于CC++HLSL 独立的工作在 Windows 平台上,只能供微软的Direct3D使用。

 

HLSL着色器

我们可以直接把HLSL着色器代码作为一长串字符串编写进我们的应用程序源文件中,但是,更加方便和模块化的方法是把着色器的代码从应用程序代码中分离出来。因此,我们将着色器代码单独保存为文本格式,然后在应用程序中使用特定函数将其加载进来。

下面是一个完整的HLSL着色器程序代码,我们把它保存在BasicHLSL.txt中。该着色器完成顶点的世界变换、观察变换和投影变幻,并将顶点颜色设定为指定的颜色。

//

// BasicHLSL.txt

// Global variable

matrix WVPMatrix;

vector color;

struct VS_INPUT

{

      vector position : POSITION;

};

 

struct VS_OUTPUT

{

      vector position : POSITION;

      vector color : COLOR;

}; 

VS_OUTPUT SetColor(VS_INPUT input)

{

      VS_OUTPUT output = (VS_OUTPUT)0; 

      output.position = mul(input.position, WVPMatrix);     

      output.color = color;

      return output;

}

 

下面就针对上述代码讲解一下HLSL着色器程序的编写:

1.3.1全局变量

      代码中声明了两个全局变量:

matrix WVPMatrix;

vector color;

      变量WVPMatrix是一个矩阵类型,它包含了世界、观察、投影的合矩阵,用于对顶点进行坐标变换;

      变量color是一个向量类型,它用于设定顶点颜色;

      代码中并没有对全局变量进行初始化,这是因为我们对全局变量的初始化过程将在应用程序中进行,全局变量在应用程序中赋值而在着色器程序中使用,这是应用程序和着色器通信的关键所在。具体赋值过程将在后续部分讲述。

1.3.2输入输出

输入输出结构

程序中定义了两个输入输出结构VS_INPUTVS_OUTPUT

struct VS_INPUT

{

      vector position : POSITION;

};

 

struct VS_OUTPUT

{

      vector position : POSITION;

      vector color : COLOR;

};

自定义的结构可以采用任意名称,结构不过是一种组织数据的方式,并不是强制的,你也可以不使用,而将本程序的输入改为:

      vector position : POSITION;

  标志符

用于输入输出的变量采用用一种特殊的声明方式:

Type VariableName : Semantic

      这个特殊的冒号语法表示一个语义,冒号后面的标志符用来指定变量的用途,如

vector position : POSITION;

      其中,POSITION标志符表明该变量表示顶点位置,另外还有诸如COLORNORMAL等很多表示其他意义的标志符。

本节所说的输入输出其实是指着色器代码和编译器、GPU之间的通信,和应用程序是无关的,所以这些变量不需要在应用程序中进行赋值,标志符告诉编译器各个输入输出变量的用途(顶点位置、法线、颜色等),这是着色器代码和编译器、GPU之间通信的关键。

1.3.3入口函数

      程序中还定义了一个函数SetColor

OUTPUT SetColor(INPUT input)

{

      VS_OUTPUT output = (VS_OUTPUT)0;

      output.position = mul(input.position, WVPMatrix);     

      output.color = color;

      return output;

}

1.      该函数以inputoutput类型作为输入输出;

2.      使全局变量WVPMatrixinput.position相乘,以完成顶点的世界、观察、投影变换,并把结果赋值到output.position

output.position = mul(input.position, WVPMatrix);

3.      将全局变量color的值赋给output.color

output.color = color;

4.      在同一个着色器代码文件中,可以有多个用户自定义函数,因此在应用程序中需要指定一个入口函数,相当于windows程序的WinMain函数,本程序只包含SetColor一个函数而且它将被做为入口函数使用。

 

1.3.4总结

      至此,一个HLSL着色器编写完毕,渲染过程中,当一个顶点被送到着色器时:

1.      全局变量WVPMatrixcolor将在应用程序中被赋值;

2.      入口函数SetColor被调用编译器根据标志符将顶点信息填充到VS_INPUT中的各个字段;

3.      SetColor函数中,首先定义一个VS_OUTPUT信息,之后根据WVPMatrixcolor变量完成顶点的坐标变换和颜色设定操作,最后函数返回VS_OUTPUT结构;

4.      编译器将会再次根据标志符把返回的VS_OUTPUT结构中的各字段映射为顶点相应的信息。

5.      顶点被送往下一个流程接受进一步处理。

上述过程中,全局变量在应用程序中赋值而在着色器程序中使用,这是应用程序和着色器通信的关键所在;标志符告诉编译器各个输入输出变量的用途(顶点位置、法线、颜色等),这是着色器代码和编译器、GPU之间通信的关键。个人认为这是着色器中最为精义的地方

原文:http://www.cppblog.com/liangairan/articles/57952.html?opt=admin

原创粉丝点击