java - 仅 GLSL 统一结构数组定义第一个元素
问题描述
我最近开始编写一个新的 3D 游戏,并开始从搅拌机及其相关的 .mtl 文件中导入模型。我创建了一个系统,通过为每个基元分配索引,然后从具有所述索引的统一数组中访问材质结构,在同一网格中加载具有多种材质的模型。奇怪的是,这适用于第 0 个索引,但对于任何高于该索引的索引都失败了。
如果我切换(在这种情况下)两种材质,则只有分配给第 0 个索引的材质会被渲染(排除我的值错误的可能性)。我检查了所有统一的位置,没有一个是负面的,所以这也不应该是问题。
着色器代码:
#version 400 core
layout(triangles) in;
layout(triangle_strip, max_vertices = 3) out;
in vec3 passVertex[];
in int passMtlIndex[]; // I don't have actual per-primitive values but I just use the first index in the array
out vec4 color;
struct Material {
vec3 ambient;
vec3 diffuse;
vec3 specular;
float shininess;
float dissolve;
};
struct Light {
vec3 position;
vec3 ambient;
vec3 diffuse;
vec3 specular;
};
uniform mat4 projection;
uniform mat4 view;
const int MAX_MATERIALS = 10;
uniform Material materials[MAX_MATERIALS];
uniform Light light;
void doVertex(int index, vec4 baseColor) {
gl_Position = projection * view * vec4(passVertex[index], 1.0);
color = baseColor;
EmitVertex();
}
mat4 inverseAffine(mat4 matrix) {
return mat4(
matrix[0].x, matrix[1].x, matrix[2].x, 0,
matrix[0].y, matrix[1].y, matrix[2].y, 0,
matrix[0].z, matrix[1].z, matrix[2].z, 0,
-dot(matrix[3].xyz, matrix[0].xyz), -dot(matrix[3].xyz, matrix[1].xyz), -dot(matrix[3].xyz, matrix[2].xyz), 1);
}
void main() {
vec3 toEye = normalize(inverseAffine(view)[3].xyz - passVertex[0]);
vec3 norm = normalize(cross(passVertex[1] - passVertex[0], passVertex[2] - passVertex[0]));
vec3 lightDir = normalize(light.position - passVertex[0]);
Material material = materials[passMtlIndex[0]];
vec3 ambient = light.ambient * material.ambient;
float diff = max(dot(norm, lightDir), 0.2);
vec3 diffuse = light.diffuse * (diff * material.diffuse);
float spec = pow(max(dot(reflect(-lightDir, norm), toEye), 0.0), material.shininess);
vec3 specular = light.specular * (spec * material.specular);
vec4 baseColor = vec4(ambient + diffuse + specular, material.dissolve);
doVertex(0, baseColor);
doVertex(1, baseColor);
doVertex(2, baseColor);
EndPrimitive();
}
渲染代码:
// use program
shader.loadMaterial("materials[0]", grassMtl);
shader.loadMaterial("materials[1]", dirtMtl);
// render
着色器代码:
public void loadMaterial(String name, Material value) {
loadVector(name + ".ambient", value.getAmbient());
loadVector(name + ".diffuse", value.getDiffuse());
loadVector(name + ".specular", value.getSpecular());
loadFloat(name + ".shininess", value.getShininess());
loadFloat(name + ".dissolve", value.getDissolve());
}
public void loadVector(String name, Vector3f value) {
GL20.glUniform3f(GL20.glGetUniformLocation(programID, name), value.x, value.y, value.z);
}
public void loadFloat(String name, float value) {
GL20.glUniform1f(GL20.glGetUniformLocation(programID, name), value);
}
编辑:我实际上试图将两种单独的材质传递到着色器中,这工作得很好,但由于某种原因,阵列仍然无法正常工作。
我还尝试将数组从结构数组转换为包含结构所有成员的五个数组,但这会产生相同的结果(仅定义了第一个元素)
另外,我该如何改进这条线:Material material = materials[passMtlIndex[0]];
?我听说不推荐使用带有变量索引的索引,这会降低 GPU 的速度。
在我看来,每次提出这个问题(并且以某种形式或形式提出)时,要么被忽略,要么找到了次优的解决方案。
解决方案
事实证明,我的着色器代码很好,而我的想法却完全相反。问题是我将 int 值作为顶点数组属性加载。GPU 没有很好地处理这个问题,任何超过 0 的值都被转换为一些(看似)随机数,这导致只有索引 0 起作用,给我一种错觉,即我错误地加载了我的值。
推荐阅读
- javascript - 如何在 IndexedDB 上的 objectStore 中添加数据?
- java-compiler-api - JavaCompiler API:在运行时访问已编译程序之外的函数/变量?
- typo3 - 另一个 FLUIDTEMPLATE 在自己的扩展中嵌套 contentElement
- python - 在“功能性”python 中处理副作用的正确方法
- html - Outlook 忽略了我的电子邮件的 CSS,包括 font-family 和 background-color
- python - 如何拆分字符串中的特定单词?
- sql-server - 如何按组从 EAV 中透视列(SQL Server)
- php - 如何返回数组唯一值?在php中
- django - 使用 celery 任务中的线程池批量插入到 mongo DB
- spring - 如何捕捉 Envers 中的错误?