OpenGL(八)使用 subroutine 切换可编程管线

Subroutine 功能是在OpenGL 4.0 版本里才增加的,因此对于各种Android手机,这个功能基本跪了。如果你发现你的程序报错:ARB_shader_subroutine,那就说明当前显卡不支持。不过大体思路可以了解一下,因为思路类似的功能有其他的实现方式。

原理

在shader中声明一个函数变量,然后定义它的指针,并将其作为一个uniform变量公开出去。最后定义很多复写函数即可。

实现

由于版本限制,使用 subroutine 要注意在shader中加入版本的编译宏:

#version 400 core

在shader中编写:

subroutine vec4 SurfaceColor();
subroutine uniform SurfaceColor U_SurfaceColor;

subroutine (SurfaceColor) vec4 Ambient()
{
   //...
}

subroutine (SurfaceColor) vec4 Diffuse()
{
    //...
}

subroutine (SurfaceColor) vec4 Specular()
{
    //...
}

void main()
{
    gl_FragColor = U_SurfaceColor();
}

在shader中,每一个函数中的代码段代表一种处理函数。另一方面在GL指令中,绑定这个函数指针,并为其指定实现函数的索引值,即可实现效果的控制。

surfaceColorLocation = glGetSubroutineUniformLocation(program,GL_FRAGMENT_SHADER,"U_SurfaceColor");

GLuint ambientLightIndex = glGetSubroutineIndex(program,GL_FRAGMENT_SHADER,"Ambient");
GLuint diffuseLightIndex = glGetSubroutineIndex(program,GL_FRAGMENT_SHADER,"Diffuse");
GLuint specularLightIndex = glGetSubroutineIndex(program,GL_FRAGMENT_SHADER,"Specular");

//draw
glUniformMatrix4fv(MLocation, 1, GL_FALSE, glm::value_ptr(model));
glUniformSubroutinesuiv(GL_FRAGMENT_SHADER,1,&ambientLightIndex);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER,ibo);
glDrawElements(GL_TRIANGLES,indexCount,GL_UNSIGNED_INT,0);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);

总结

本文介绍了OpenGL中的 subroutine 机制,通过它可以切换shader的内容。在Unity3d中使用Shader.maximumLOD技术可以达到类似的效果,相关内容可以参考官方文档


松阳论道

关注我的微信公众号,获取更多优质内容

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Fortran编程中,使用PARDISO库求解大型稀疏复数矩阵方程是一个常见的需求。PARDISO是一种并行直接解法器,它能够高效地求解稀疏矩阵方程。 以下是一个使用PARDISO库求解大型稀疏复数矩阵方程的简单示例代码: ```fortran program sparse_solver implicit none ! PARDISO库的接口声明 interface subroutine pardisoinit(pt, mtype, iparm) integer, intent(inout) :: pt(:), iparm(:) integer, intent(in) :: mtype end subroutine pardisoinit subroutine pardiso(pt, maxfct, mnum, mtype, phase, n, a, ia, ja, perm, nrhs, iparm, msglvl, b, x, error) integer, intent(inout) :: pt(:), iparm(:), perm(:), ia(:), ja(:) integer, intent(in) :: maxfct, mnum, mtype, phase, n, nrhs, msglvl complex, intent(inout) :: a(:) complex, intent(inout) :: b(:), x(:) real(kind=8), intent(out) :: error end subroutine pardiso subroutine pardisofree(pt, mtype) integer, intent(inout) :: pt(:) integer, intent(in) :: mtype end subroutine pardisofree end interface ! 定义PARDISO相关参数 integer :: pt(64), iparm(64) integer :: maxfct, mnum, mtype, phase, n, nrhs integer :: ia(n+1), ja(:), perm(n) complex :: a(:), b(n), x(n) real(kind=8) :: error character(len=64) :: msg ! 初始化PARDISO库 maxfct = 1 mnum = 1 mtype = -4 ! 复数矩阵 phase = 11 ! 初始化阶段 n = ! 矩阵的维度 nrhs = 1 ! 方程右侧的列数 call pardisoinit(pt, mtype, iparm) ! 设置PARDISO的参数 iparm(1) = 1 ! 使用默认配置 iparm(3) = 0 ! 不打印统计信息 iparm(4) = 0 ! 不打印错误信息 ! 填充稀疏矩阵A的数据 ! ... ! 填充向量b的数据 ! ... ! 调用PARDISO求解方程 call pardiso(pt, maxfct, mnum, mtype, phase, n, a, ia, ja, perm, nrhs, iparm, 0, b, x, error) ! 检查求解状态 if (error /= 0.0) then write(*, *) "PARDISO solver failed with error code: ", error stop end if ! 输出解向量x的结果 ! ... ! 释放PARDISO库占用的内存 call pardisofree(pt, mtype) end program sparse_solver ``` 请注意,上述示例中的部分代码需要根据您的具体问题进行填充,包括稀疏矩阵A和向量b的数据填充以及解向量x的结果输出。此外,您需要确保已经正确安装并链接了PARDISO库。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值