16.4关键字、语句和强制转换
16.4.1 关键字
为便于参考,这里给出一个HLSL定义的关键字列表:
asm bool compile const decl do
double else extern false float for
half if in inline inout int
matrix out pass pixelshader return sampler
shared static string struct technique texture
true typedef uniform vector vertexshader void
volatile while
下面的集合显示了被保留并且未使用但是将来可能成为关键字的标识符:
auto break case catch char class
const_cast continue default delete dynamic cast enum
explicit friend goto long mutable namespace
new operator private protected public register
reinterpret_cast short signed sizeof static_cast switch
template this throw try typename union
unsigned using virtual
16.4.2 基本程序流程
HLSL支持很多与C++相似的选择、重复、和一般程序流程语句。这些语句的语法和C++极为相似。
return语句:
return (expression);
if和if…else语句:
if( condition )
{
statement(s);
}
if( condition )
{
statement(s);
}
else
{
statement(s);
}
for语句:
for(initial; condition; increment)
{
statement(s);
}
while语句:
while( condition )
{
statement(s);
}
do…while语句:
do
{
statement(s);
}while( condition );
16.4.3 强制转换(casting)
HLSL支持一种非常自由的强制转换设计。HLSL中强制转换的语法和C程序语言中的一样。例如要把float转换到matrix,我们写:
float f = 5.0f;
matrix m = (matrix)f;
16.5 操作符
HLSL支持很多类似C++的操作符。除了很少一些底下注释的例外以外,他们的用法和C++里的完全一样。下表列出了HLSL的操作符:
[ ] |
|
> |
< |
< = |
> = |
! = |
= = |
! |
&& |
|
?: |
+ |
+ = |
- |
- = |
* |
*= |
/ |
/= |
% |
%= |
+ + |
-- |
= |
() |
' |
|
|
|
虽然操作符的行为和C++很相似,但是也有一些差异。第一,求模%运算符对整型和浮点型都起作用。为了使用求模操作符,左边的值和右边的值都必须有相同的正负号(如:左边和右边必须都是正或者负)。
第二,要注意HLSL操作是以每个分量为基础的。这是由于实际上向量和矩阵是语言内建的,并且这些类型是由若干个分量组成。通过将这些操作施加在分量级别之上,我们可以像使用数值类型一样完成诸如向量/矩阵的加法,减法和相等测试这些操作(),见下例:
注意:操作符的行为正如对数值操作一样(也就是说,按一般C++的方式)。
vector u = {1.0f, 0.0f, -3.0f, 1.0f};
vector v = {-4.0f, 2.0f, 1.0f, 0.0f};
// adds corresponding components
vector sum = u + v; // sum = (-3.0f, 2.0f, -2.0f, 1.0f)
增量一个向量就是增量其每个分量:
// before increment: sum = (-3.0f, 2.0f, -2.0f, 1.0f)
sum++; // after increment: sum = (-2.0f, 3.0f, -1.0f, 2.0f)
向量相乘也是按分量的:
vector u = {1.0f, 0.0f, -3.0f, 1.0f};
vector v = {-4.0f, 2.0f, 1.0f, 0.0f};
// multiply corresponding components
vector sum = u * v; // product = (-4.0f, 0.0f, -3.0f, 0.0f)
比较操作也是按分量进行的,并且返回一个每个分量都为bool类型的向量或者数组。作为结果的“bool”向量包含了每个分量比较的结果。例如:
vector u = { 1.0f, 0.0f, -3.0f, 1.0f};
vector v = {-4.0f, 0.0f, 1.0f, 1.0f};
vector b = (u == v); // b = (false, true, false, true)
最后,我们以讨论二元操作的变量提升(promotion)作为结束:
对于二元操作,如果(操作符的)左边和右边维数不同,则维数较少的一边提升(强制转换)到具有和维数较大的一边相同的维数。例如,如果x的类型为float,而y的类型为float3,在表达式(x + y)中变量x被提升到float3,并且计算出来的表达式的值的类型也为float3。提升使用已定义的转换完成。注意,若转换未定义则提升也是未定义的。例如,我们不能转换float2到float3,因为没有定义这个转换。
对于二元操作,如果左边和右边类型不同,那么较低精度的类型(the lower type resolution)被提升(强制转换)到具有同类型的较高精度的类型(the higher type resolution)。例如,如果x类型为int,y类型为half,则表达式(x + y)中的变量x被提升到half,并且计算出来的表达式的值的类型也为half。
16.6 用户定义函数
HLSL中的函数有下例属性:
函数使用类似C++的语法
参数总是按值传递
递归不被支持
函数总是inline的
此外,函数还加上了一些用于其上的额外的关键字。例如,考虑一个写在HLSL中的下面这个函数:
bool foo(in const bool b, // input bool out int r1, // output int inout float r2) // input/output float { if( b ) // test input value { r1 = 5; // output a value through r1 } else { r1 = 1; // output a value through r1 }
// since r2 is inout we can use it as an input // value and also output a value through it r2 = r2 * r2 * r2;
return true; } |
函数几乎和C++函数是一样的,除了in,out和inout关键字:
in——指定型参(argument,特指传递给实参的变量)应该在函数开始前被拷贝给实参。传入参数不必强制指定,因为实参默认是in的。例如,下面两段是等价的:
float square(in float x) { return x * x; } |
也可以不强制指定in:
float square(float x) { return x * x; } |
out——指定实参应该在函数返回时被拷贝给型参。这样可以通过参数返回值。out关键字是必须的,因为HLSL不允许传递一个引用或一个指针。我们要注意:如果实参标记为out,在函数开始前,型参就不拷贝给实参。换句话说,out实参仅可以被用于输出数据——它不能用于输入。
void square(in float x, out float y) { y = x * x; } |
这里,我们输入了要被乘方的数x,并且通过参数y返回了x的乘方。
inout——这是一个指示实参既用于输入又用于输出的快捷方法。如果要使用实参同时用作输入和输出,就指定inout。
void square(inout float x) { x = x * x; } |
这里,我们输入了要被乘方的数x,同时又通过x返回了的x的乘方。