OpenGL ES3.1使用计算着色器(Compute Shader)_opengl es计算着色器-程序员宅基地

技术标签: OpenGL ES  Android  

OpenGL ES3.1使用计算着色器(Compute Shader)

1.基本介绍

OpenGL ES从3.1版本开始支持计算着色器
        工作模型有全局工作组和本地工作组,全局工作组包含由三维的本地工作组组成,本地工作组也由三个维度组成。本地工作组三个维度大小分别为:local_size_xlocal_size_ylocal_size_z
一个什么都不做的着色器最基本代码:

#version 310 es
layout(local_size_x=32,local_size_y=32) in;//设置本地工作组大小,local_size_z未设置默认为1
void main(void){
   
    
	//执行代码
}

2.创建、编译和链接计算着色器

创建计算着色器和顶点着色器、片元着色器过程差不多。

shader=glCreateShader(GL_COMPUTE_SHADER);//使用GL_COMPUTE
glShaderSource(shader,1,sourece,NULL);
glCompileShader(shader);
program=glCreateProgram();
glAttachShader(program,shader);
glLinkProgram(program);

3.执行计算着色器

当然要先使用glUseProgram(program)
3.1使用glDispatchCompute(GLuint num_groups_x,GLuint num_groups_y,GLuint num_groups_z)设置全局工作组大小并且执行计算着色器。
3.2使用glDispatchComputeIndirect(GLintptr indirect)使用存储在缓冲区对象参数来设置全局工作组大小并且执行计算着色器。
通过绑定GL_DISPATCH_INDIRECT_BUFFER缓冲对象来设置全局参数,缓冲对象由三个无符号整数(GLuint)来设置。indirect参数是当前绑定到 GL_DISPATCH_INDIRECT_BUFFER 目标的缓冲区的字节偏移量。

4.获取着色器属性

glGetProgramiv (GLuint program, GLenum pname, GLint *params)
pname:可为GL_MAX_COMPUTE_WORK_GROUP_SIZE

5.每个计算单元的位置

const uvec3 gl_WorkGroupSize;//本地工作组大小
in uvec3 gl_NumWorkGroups;//全局工作组大小
in uvec3 gl_LocalInvocationID;//表示当前执行单元在本地工作组中的位置
in uvec3 gl_WorkGroupID;//表示当前本地工作组在全局工作组中的位置
in uvec3 gl_GlobalInvocationID;//等于gl_WorkGroupID*gl_WorkGroupSize+gl_LocalInvocationID,所以它是当前执行单元的三维索引
in uint gl_LocalInvocationIndex;//它是gl_LocalInvocationID的一种扁平化方式,等于gl_LocalInvocationID.z*gl_WorkGroupSize.x*gl_WorkGroupSize.y+gl_LocalInvocationID.y*gl_WorkGroupSize.x+gl_LocalInvocationID.x

6.同步

同步类型有两种运行屏障(execution barrier)和内存屏障(memory barrier)

6.1运行屏障(execution barrier)

计算着色器程序中使用barrier()触发,如果计算着色器的一个请求遇到了barrier(),那么它就会停止运行,并等待同一本地工作组的所有请求到达为止,这一点在条件语句中使用尤为需要注意避免死锁。

6.2内存屏障(memory barrier)

        最基本的版本就是memoryBarrier(),它可以保证着色器的请求内存的写入操作一定提交到内存端,而不是通过缓冲区(cache)或者调度队列之类的方式。它还可以给着色器编译器做出指示,让它不要对内存操作重排序,以免因此跨越屏障函数。
        memoryBarrierAtomicCounter()会等待原子计数器更新,然后继续执行。
        memoryBarrierBuffer()和memoryBarrierImage()会等待缓存和图像变量的写入操作完成,然后继续执行。
        memoryBarrierShared()会等待带有shared限定符的变量更新,然后继续执行。

7.使用计算着色器写一个百万粒子发射器

7.1着色器代码

顶点着色器

#version 310 es
precision mediump image2D;//OpenGL ES要求显示定义image2D的精度
layout (binding = 1,rgba32f) restrict uniform image2D position_buffer;
uniform mat4 mvp;
out float intensity;//粒子年龄值
void main() {
   
    
    ivec2 size=imageSize(position_buffer);
    vec4 pos=imageLoad(position_buffer,ivec2(gl_VertexID%size.y,gl_VertexID/size.y));//将一维坐标转换为图像二维坐标
    gl_Position=mvp*vec4(pos.xyz,1.0f);
    intensity=pos.w;
}

片元着色器

#version 310 es
precision mediump float;
out vec4 color;
//这个值来自顶点着色器中读取的粒子年龄值
in float intensity;
void main(void){
   
    
    //根据粒子的年龄,在热红色到冷蓝色之间的混合结果
    color=mix(vec4(0.0f,0.2f,1.0f,1.0f),vec4(0.2f,0.05f,0.0f,1.0f),intensity);
}

计算着色器

#version 310 es
precision mediump image2D;//OpenGL ES要求显示定义image2D的精度
//uniform块中包含引力器的位置和质量
layout (std140,binding=0) uniform attractor_block{
   
    
    vec4 attractor[64];//xyz=position,w=mass
};
//每块中粒子的数量为128
layout (local_size_x=128) in;
//使用两个缓冲来包含粒子的位置和速度信息
layout (binding = 0,rgba32f) restrict uniform image2D velocity_buffer;
layout (binding = 1,rgba32f) restrict uniform image2D position_buffer;
//时间间隔
uniform float dt;
void main(void){
   
    
    //从缓存中读取当前的位置和速度
    ivec2 size=imageSize(velocity_buffer);
    ivec2 p=ivec2(int(gl_GlobalInvocationID.x)/size.y,int(gl_GlobalInvocationID.x)%size.y);
    vec4 vel=imageLoad(velocity_buffer,p);
    vec4 pos=imageLoad(position_buffer,p);
    int i;
    //使用当前速度x时间来更新位置
    pos.xyz += vel.xyz * dt;
    pos.w -= 0.0001 * dt;
    //对于每个引力器
    for (i = 0; i < 4; i++)
    {
   
    
        //计算受力并更新速度
        vec3 dist = (attractor[i].xyz - pos.xyz);
        vel.xyz += dt * dt * attractor[i].w * normalize(dist) / (dot(dist, dist) + 10.0);
    }
    //如果粒子已经过期,那么重置它
    if (pos.w <= 0.0)
    {
   
    
        pos.xyz = vec3(0.0f);
        vel.xyz *= 0.01;
        pos.w += 1.0f;
    }

    //将新的位置和速度信息重新保存在缓存中
    imageStore(position_buffer,p,pos);
    imageStore(velocity_buffer,p,vel);
}

7.2粒子系统初始化

	srand(clock());
    glEnable(GL_BLEND);
    glBlendFunc(GL_ONE, GL_ONE);
    //激活计算着色器,
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_39561000/article/details/103112147

智能推荐

13、SQL注入——SQL盲注_sql怎么盲注-程序员宅基地

文章浏览阅读137次。在SQL注入过程中,SQL语句执行后,选择的数据不能回显到前端页面,此时还需要利用一些方法进行判断或尝试,这个过程称之为盲注。在盲注中,攻击者根据其返回页面的不同来判断信息(可能时页面内容的不同,也可能时响应时间的不同)。一般情况下,盲注可分为两种:基于布尔的盲注某些情况下,页面返回的结果只要两种(正常或者错误)。通过构造SQL判断语句,查看页面的返回结果(True 或False)来判断哪些SQL判断条件成立,通过此来获取数据库中的数据。基于时间的盲注。_sql怎么盲注

unity 多个预制体的时候,随机生成某个预制物体_unity多个预制体随机生成-程序员宅基地

文章浏览阅读4.2k次,点赞4次,收藏26次。转载自:https://www.cnblogs.com/lzl0823/p/6379488.html方法1://创建一个数组,在界面那里把预制物体拖进FishPrefab里public GameObject[] FishPrefab;//计时器float timer = 0;void Update(){timer += Time.deltaTime;..._unity多个预制体随机生成

Linux操作系统入门-程序员宅基地

文章浏览阅读861次,点赞10次,收藏15次。Linux操作系统入门

二本学渣考研失败,2021年Android春招面试经历,这原因我服了_二本学渣 没有科创经历怎么-程序员宅基地

文章浏览阅读78次。Java相关无论什么级别的Android从业者,Java作为Android开发基础语言。不管是工作还是面试中,Java都是必考题。如果不懂Java的话,薪酬会非常吃亏(美团尤为重视Java基础)详细介绍了Java泛型、注解、并发编程、数据传输与序列化、高效IO、容器集合、反射与类加载以及JVM重点知识线程、内存模型、JVM运行时内存、垃圾回收与算法、Java中四种引用类型、GC 分代收集算法 VS 分区收集算法、GC 垃圾收集器、JAVA IO/NIO 、JVM 类加载机制的各大知识点。详细知识_二本学渣 没有科创经历怎么

Spring Boot 2 之 WebFlux 反应式编程解析及实战_webflux底层原理-程序员宅基地

文章浏览阅读793次。近些年来,反应式编程亦或称为响应式编程,在开发者社区中很受欢迎,特别在 Spring 5 以及 Spring Boot 2 发布之后热度再次飙升。以反应式编程为基础的 Spring WebFlux 组件作为异步非阻塞的系统解决方案,可以明显的提高系统吞吐量。首先看一下Srping官网上的一张图,对比一下SpringMvc和Spring WebFlux:Spring WebFlux 提供了两种..._webflux底层原理

利用commons-fileupload 上传图片(包含表单数据)_使用commons组件实现图片文件上传-程序员宅基地

文章浏览阅读334次。在一个表单中包含普通文本数据,另外还有需要上传的图片,那么本程序将图片保存到服务器上的一个图片目录中,文本数据则获取然后输出,查看传输是否正确,后面的处理为涉及。上传的jsp页面:最后将信息输出:(图片已经保存在特定目录中)需要用到的两个jar包,commons-fileupload-1.2.2.jar commons-io.jarjsp页面:&lt;%@ page language="java..._使用commons组件实现图片文件上传

随便推点

微信电脑版聊天图片DAT格式转为普通图片_dat转换-程序员宅基地

文章浏览阅读162次。9.2:如果你需要清理DAT图片又怕清理了重要文件,可以人工筛选上面转出来的图片,留下重要图片,删除没用的图片;然后使用下面这个工具快速清理DAT文件,不影响在微信里查看它,具体用法在清理工具里有的。5.2022年6月前后文件放的文件夹不一样,6月后的图片(包含加密的DAT图片)、视频、文件都放在【MsgAttach】里,这里以转换6月后的DAT图片成普通图片为例。但是图片不行,图片是被加密保存的,它保存在个人数据文件夹里,是dat结尾的一堆乱码dat文件。9.1:最终转换出来的图片可以在目标文件中查看。_dat转换

基于JAVA理发预约系统计算机毕业设计源码+数据库+lw文档+系统+部署-程序员宅基地

文章浏览阅读81次。基于JAVA理发预约系统计算机毕业设计源码+数据库+lw文档+系统+部署。JSP手机销售网站的设计与开发access+SQL数据库双数据库。springboot基于Bootstrap的家具商城系统设计。springboot高校体育场馆预约管理系统设计与实现。JSP企业进销存系统的设计与实现SQLServer。ssm基于JavaWeb的网上购书后台管理系统。springboot基于智能推荐的良品店铺平台。ssm基于ssm+vue的高校个性化选课推荐。ssm线上远程教学及自主学平台的设计与实现。

毕设福利129个全新、创新的Java计算机毕业设计项目,让你的毕设独树一帜-程序员宅基地

文章浏览阅读53次。Springboot基于Web的华圆小区物业管理系统设计与开发1mo07。Springboot基于spring的物业管理系统的设计与实现5z848。Springboot基于web的医院his系统-门诊模块eb6p8。Springboot社区医院住院管理系统的设计与实现 c9ooo。Springboot基于Vue框架的社区疫情监控系统ml3gs。Springboot基于人脸识别的社区防疫管理系统vdg70。Springboot基于的疫情期间高校人员管理系统304r4。

关于Python脚本开头两行的:#!/usr/bin/python和# -*- coding: utf-8 -*-的作用 – 转_#!/usr/bin/python coding:utf-8-程序员宅基地

文章浏览阅读812次。#!/usr/bin/python是用来说明脚本语言是python的是要用/usr/bin下面的程序(工具)python,这个解释器,来解释python脚本,来运行python脚本的。# -*- coding: utf-8 -*-是用来指定文件编码为utf-8的详情可以参考:PEP 0263 — Defining Python Source Code Encoding..._#!/usr/bin/python coding:utf-8

java中图形界面部件_Java-图形界面-paintComponent-程序员宅基地

文章浏览阅读586次。自己如何创建绘图组件:创建JPanel的子类并覆盖掉paintComponent()这个方法//需要引入这些包import javax.swing.*;import java.awt.*;//创建JPanel的子类public class MyDrawPanel extends JPanel {//这是非常重要的方法,你绝不能自己调用,要由系统来调用public void paintCompone..._java paintcomponent

使用jhlabs提供的类对图片进行滤镜处理_jhlabs 图片处理-程序员宅基地

文章浏览阅读1.8k次。public static void main(String[] args) throws IOException{ //调用的时候指定相关参数 AbstractBufferedImageOp bbf = new BoxBlurFilter((float)4,(float)3,3); System.out.println(System.getProperty("user.dir") +_jhlabs 图片处理