GLSL基础

OpenGL Shading Language GLSL作为一种着色语言是纯粹的和GPU打交道的计算机语言。
因为GPU是多线程并行处理器,所以GLSL直接面向SIMD模型的多线程计算。
GLSL编写的着色器函数是对每个数据同时执行的。
每个顶点都会由顶点着色器中的算法处理,每个像素也都会由片段着色器中的算法处理。
因此,初学者在编写自己的着色器时,需要考虑到SIMD的并发特性,并用并行计算的思路来思考问题。

最常见用法是在顶点着色器里生成所需要的值,然后传给片断着色器用。  
GLSL能做什么?

  • 日以逼真的材质 -- 金属,岩石,木头,油漆等
  • 日益逼真的光照效果 -- 区域光和软阴影
  • 非现实材质 -- 美术效果,钢笔画,水墨画和对插画技术的模拟
  • 针对纹理内存的新用途
  • 更少的纹理访问
  • 图形处理 -- 选择,边缘钝化遮蔽和复杂混合
  • 动画效果 -- 关键帧插值,粒子系统
  • 用户可编程的反走样方法

GLSL注意

  • GLSL支持函数重载
  • GLSL不存在数据类型的自动提升,类型必须严格保持一致。
  • GLSL不支持指针,字符串,字符,它基本上是一种处理数字数据的语言
  • GLSL不支持联合、枚举类型、结构体位字段及按位运算符

数据类型


GLSL有三种基本数据类型:float,int和bool,以及由这些数据类型组成的数组和结构体。
需要注意的是,GLSL并不支持指针。与C/C++不同的是,GLSL将向量和矩阵作为基本数据类型。
注意:GLSL不存在数据类型的自动提升,类型必须严格保持一致。

标量

  • float
  • int
  • bool
   // 十进制
// 八进制
0x2A // 十六进制

GLSL不存在数据类型的自动提升,类型必须严格保持一致

矢量

矢量可以和标量甚至矩阵做加减乘除(必须符合规则)

 vec2,  vec3,  vec4 // 包含2/3/4个浮点数的矢量
ivec2, ivec3, ivec4 // 包含2/3/4个整数的矢量
bvec2, bvec3, bvec4 // 包含2/3/4个布尔值的矢量

声明:

vec3 v;             //声明三维浮点型向量v
v[]=3.0; //给向量v的第二个元素赋值 // 下面两种等价
vec3 v = vec3(0.6);
vec3 v = vec3(0.6, 0.6, 0.6);

注意: 除了用索引的方式外,还可以用选择运算符的方式来使用向量。选择运算符是对于向量的各个元素(最多为4个)约定俗成的名称,用一个小写拉丁字母来表示。根据向量表示对象的意义不同,可以使用以下选择运算符:
表示顶点可以用 (x、y、z、w)
表示颜色可以用 (r、g、b、a)
表示纹理坐标用 (s、t、p、q)
用户可以选择其中任意一种选择运算符,它们的作用是等效的。
也就是说,如果v是一个向量,那么 v[0]v.rv.x 和 v.s 都指的是向量v的第一个元素。  

例如:

vec4 v1=vec4(1.0, 2.0, 3.0, 4.0);   //用构造函数的方式声明并初始化四维浮点型
vec4 v2;
v2.xy=v1.yz; //将v1的第二个和第三个元素复制到v2的第一个和第二个元素
v2.z=2.0; //给v2的第三个元素赋值
v2.xy=v1.yx; //将v1的头两个元素互换,再复制到v2的头两个元素中

矩阵

mat2, mat3, mat4 -- 2x2/3x3/4x4/的矩阵
矩阵是按列顺序组织的,先列后行

例如:

mat4 m;             //声明四维浮点型方阵m
m[][]=2.0; //给方阵的第三列、第四行元素赋值 // 下面两种等价,初始化矩阵对角
mat2 m = mat2(1.0)
mat2 m = mat2(1.0, 0.0, 0.0, 1.0);

取样器(Sampler)

纹理查找需要制定哪个纹理或者纹理单元将制定查找。

sampler1D           // 访问一个一维纹理
sampler2D // 访问一个二维纹理
sampler3D // 访问一个三维纹理
samplerCube // 访问一个立方贴图纹理
sampler1DShadow // 访问一个带对比的一维深度纹理
sampler2DShadow // 访问一个带对比的二维深度纹理
uniform sampler2D grass;

vcc2 coord = vec2(, );
vec4 color = texture2D(grass, coord);

如果一个着色器要在程序里结合多个纹理,可以使用取样器数组

const int tex_nums = ;
uniform sampler2D textures[tex_nums]; for(int i = ; i < tex_nums; ++i) {
sampler2D tex = textures[i];
// todo ...
}

结构体:

这是唯一的用户定义类型

struct light
{
vec3 position;
vec3 color;
}; light ceiling_light;

数组

数组索引是从0开始的,而且没有指针概念

// 创建一个10个元素的数组
vec4 points[]; // 创建一个不指定大小的数组
vec4 points[];
points[] = vec4(1.0); // points现在大小为3
points[] = vec4(2.0); // points现在大小为8

void

只能用于声明函数返回值

类型转换

必须明确地进行类型转换,不会自动类型提升

float f = 2.3;
bool b = bool(f); // b is true

限定符


GLSL中有4个限定符(variable qualifiers)可供使用,它们限定了被标记的变量不能被更改的"范围"。

  • const
  • attribute
  • uniform
  • varying

const

const和C++里差不多,定义不可变常量
表示限定的变量在编译时不可被修改

attribute

attribute是应用程序传给顶点着色器用的
不允许声明时初始化
attribute限定符标记的是一种全局变量,该变量在顶点着色器中是只读(read-only)的,该变量被用作从OpenGL应用程序向顶点着色器中传递参数,因此该限定符仅能用于顶点着色器。

uniform

unifrom一般是应用程序用于设定顶点着色器和片断着色器相关初始化值。
不允许声明时初始化
uniform限定符标记的是一种全局变量,该变量对于一个图元(primitive)来说是不可更改的 它可以从OpenGL应用程序中接收传递来的参数。

varying

varying用于传递顶点着色器的值给片断着色器
不允许声明时初始化
它提供了从顶点着色器向片段着色器传递数据的方法,varying限定符可以在顶点着色器中定义变量,然后再传递给光栅化器,光栅化器对数据插值后,再将每个片段的值交给片段着色器。

限制


  • 不能在if-else中声明变量
  • 用于判断的条件必须是bool类型(if,while,for...)
  • (?:)操作符后两个参数必须类型相同
  • 不支持switch语句
vec4 toonify(in float intensify)
{
vec4 color;
color = vec4(0.8,0.8,0.8,0.8)
return color;
}

discard


discard关键字可以避免片段更新帧缓冲区,当流控制遇到这个关键字时,正在处理的片段就会被标记为丢

函数


  • 函数名可以通过参数类型重载,但是和返回值类型无关
  • 所有参数必须完全匹配,参数不会自动
  • 函数不能被递归调用
  • 函数返回值不能是数组

函数参数标示符

in: 进复制到函数中,但不返回的参数(默认)
out: 不将参数复制到函数中,但返回参数
inout: 复制到函数中并返回

混合操作


通过在选择器(.)后列出各分量名,就可以选择这些分量

vec4 v4;
v4.rgba; // 得到vec4
v4.rgb; // 得到vec3
v4.b; // 得到float
v4.xy; // 得到vec2
v4.xgba; // 错误!分量名不是同一类 v4.wxyz; // 打乱原有分量顺序
v4.xxyy; // 重复分量

最后推荐一个GLSL编辑调试工具 OpenGL Shader Builder (Mac OS X) 在苹果的开发者中心的下载Graphics Tools.dmp

GLSL基础的更多相关文章

  1. 在Android中使用OpenGL ES开发第(五)节:GLSL基础语法

    一.前期基础储备笔者之前的四篇文综述了Android中使用OpenGL ES绘制基本图形和实现了简单的相机预览,初次接触OpenGL ES开发的读者可能对其中新的概念比较迷惑,尤其是其中的顶点着色器( ...

  2. GLSL 基础量定义

    GLSL语法跟C语言非常相似: 1.数据类型: GLSL包含下面几种简单的数据类型 float bool :false or ture int 向量: vec   {2,3,4}     长度为2, ...

  3. OpenGL 4.0 GLSL 基础教程概览——VAO和VBO常用操作接口

    (一) OpenGL  4.3 最新渲染管线图 从OpenGL 2.0 到 OpenGL 3.0变化非常大,但从OpenGL 3.0 到OpenGL 4.0 变化不是太大. 着色器程序直接运行在GPU ...

  4. GLSL 基础量定义 【转】

    转载:http://blog.csdn.net/misol/article/details/7658949   GLSL语法跟C语言非常相似: 1.数据类型: GLSL包含下面几种简单的数据类型 fl ...

  5. Android OpenGL ES 开发(八): OpenGL ES 着色器语言GLSL

    前面的文章主要是整理的Android 官方文档对OpenGL ES支持的介绍.通过之前的文章,我们基本上可以完成的基本的形状的绘制. 这是本人做的整理笔记: https://github.com/re ...

  6. GLSL语言基础

    from http://www.kankanews.com/ICkengine/archives/120870.shtml 变量 GLSL的变量命名方式与C语言类似.变量的名称可以使用字母,数字以及下 ...

  7. 【转载】OpenGL超级宝典笔记——GLSL语言基础

    变量 GLSL的变量命名方式与C语言类似.变量的名称可以使用字母,数字以及下划线,但变量名不能以数字开头,还有变量名不能以gl_作为前缀,这个是GLSL保留的前缀,用于GLSL的内部变量.当然还有一些 ...

  8. 基于虎书实现LALR(1)分析并生成GLSL编译器前端代码(C#)

    基于虎书实现LALR(1)分析并生成GLSL编译器前端代码(C#) 为了完美解析GLSL源码,获取其中的信息(都有哪些in/out/uniform等),我决定做个GLSL编译器的前端(以后简称编译器或 ...

  9. CSharpGL(15)用GLSL渲染2种类型的文字

    CSharpGL(15)用GLSL渲染2种类型的文字 2016-08-13 由于CSharpGL一直在更新,现在这个教程已经不适用最新的代码了.CSharpGL源码中包含10多个独立的Demo,更适合 ...

随机推荐

  1. Eclipse中的常用快捷键

    快捷修复 Command+1 //int a=100L; //int a=(int) 100L; 快捷删除行 Command+D 快速起新行 Shift+Enter (当本行代码很长时,将光标定在本行 ...

  2. CPU大小端判断

    两种方式:1.通过指针         2.通过联合体,联合体里面的数据都是按顺序存储的,而且不论联合体里面有多少数据类型,联合体长度是最长的数据类型的长度.不论初始化多少联合体里面的数据,有效的是最 ...

  3. fwrite ,fprintf的作用与区别

    1.概念和作用 fwrite是C语言函数,指向文件写入一个数据块,写入的是 fprintf是C/C++中的一个格式化写-库函数,其作用是格式输出到一个流/文件中:原型是int fprintf( FIL ...

  4. 【BZOJ】【2178】圆的面积并

    自适应辛普森积分 Orz Hzwer 辛普森真是个强大的东西……很多东西都能积= = 这题的正解看上去很鬼畜,至少我这种不会计算几何的渣渣是写不出来……(对圆的交点求图包,ans=凸包的面积+一堆弓形 ...

  5. 【BZOJ】【3282】Tree

    LCT 喜闻乐见的Link-Cut-Tree…… srO zyf http://www.cnblogs.com/zyfzyf/p/4149109.html 目测我是第222个?………………不要在意这些 ...

  6. Leetcode#73 Set Matrix Zeroes

    原题地址 用矩形的第一行和第一列充当mask 代码: void setZeroes(vector<vector<int> > &matrix) { ].empty()) ...

  7. display:inline-block引发的间隙问题解决办法

    在网页布局中我们经常会用到display:inline-block;好处是:能够将块状元素按照内联元素的方式布局,同时能设置宽高.个人感觉很好用,可是用多了慢慢的问题就来了? 1.display:in ...

  8. Best Practices for Web Apps

    Mobile Web Best Practices Exceptional Performance Let's make the web faster

  9. Chapter 4

    1.Python中有四种函数:全局函数,局部函数,lambda函数,方法.其中全局函数与局部函数对比理解,局部函数就是定义在某函数之内的函数,否则就是全局函数,lambda函数就是表达式,方法就跟对象 ...

  10. ECMAScript5下Array的方法

    声明:ECMAScript不会兼容IE8及以下版本IE浏览器. 一.迭代方法 注:这些迭代方法不会影响数组的值. 每个方法都有两个参数: array.方法(执行函数体,当前作用域(比如this,这个可 ...