LibGDX - 在桌面��工作的着色器在Android上无法工作

4
我写了一个简单的程序,可以在3D环境中渲染一个球体,并根据球体周围的四个光源对其进行着色。当我在桌面上运行程序时,它可以正常工作,但在Android设备上,球体只是单色的。
下面是图片来说明我的问题: enter image description here -> 桌面 enter image description here -> Android
这是着色器代码:
#ifdef GL_ES
    precision mediump float;
#endif

uniform mat4 u_projectionMatrix;
uniform mat3 u_normalMatrix;
uniform mat4 u_modelViewMatrix;

const int MAX_LIGHTS = 8;
uniform int u_lights_active;
uniform vec3 u_ambient;
uniform vec3 u_position[ MAX_LIGHTS ];
uniform vec3 u_diffuse[ MAX_LIGHTS ];
uniform vec3 u_att_coeffs[ MAX_LIGHTS ];

// since builtins aren't used, we use attributes as substitute
attribute vec2 a_texCoord0;
attribute vec4 a_position;
attribute vec3 a_normal;

// outputs to fragment shader
varying vec2 v_tex_coord;
varying vec4 v_color;

void main()
{
    vec3 tempColor  = u_ambient;
    vec3 ecPosition = vec3( u_modelViewMatrix * a_position );
    vec3 viewVec    = normalize( -ecPosition );
    vec3 tnorm      = normalize( u_normalMatrix * a_normal );

    for ( int i = 0; i < u_lights_active; ++i )
    {
        float dist   = length( ecPosition - u_position[i] ); // distance from light to fragment
        float att    = 1.0 / ( u_att_coeffs[i].x + u_att_coeffs[i].y*dist + u_att_coeffs[i].z*dist*dist );

        vec3 lightVec  = normalize( u_position[i] - ecPosition );
        float diffuse  = max( dot( lightVec, tnorm ), 0.0 );
        tempColor      += att * u_diffuse[i] * diffuse;
    }

    tempColor    = clamp( tempColor, 0.0, 1.0 );
    v_color      = vec4( tempColor, 0.0 );

    gl_Position  = u_projectionMatrix * vec4( ecPosition, 1.0 );
    v_tex_coord  = a_texCoord0.xy;
}

sphere.frag

#ifdef GL_ES
    precision mediump float;
#endif

uniform sampler2D u_texture;

varying vec2 v_tex_coord;
varying vec4 v_color;

void main()
{
    vec4 texColor = texture2D( u_texture, v_tex_coord );
    gl_FragColor  = texColor * v_color;
}

我希望你能帮我解释一下,我做错了什么。
版本号:
LibGDX:0.9.8
ADT:Build v22.0.1-685705
Android设备:Sony Xperia S,Android 4.1.2
项目构建目标:Android 4.3,API 18
清单包含内容:
<uses-feature android:glEsVersion="0x00020000" android:required="true" />

着色器的创建方式:

shaderProgram = new ShaderProgram( Gdx.files.internal( "shaders/sphere.vert" ), Gdx.files.internal( "shaders/sphere.frag" ) );
if ( !shaderProgram.isCompiled() )
{
    Gdx.app.error( TAG, shaderProgram.getLog() );
}

这个球是一个静态模型:

创建:

final ModelLoaderHints hint = new ModelLoaderHints( true );
model = ModelLoaderRegistry.loadStillModel( Gdx.files.internal( "data/sphere.obj" ), hint );
texture = new Texture( Gdx.files.internal( "data/sphere_tex.png" ), Format.RGB888, false );
material = new Material( "mat", new TextureAttribute( texture, 0, "u_texture" ) );

渲染:

shaderProgram.begin();
texture.bind( 0 );

shaderProgram.setUniformMatrix( C.U_PROJECTION_MATRIX, cam.projection );
// light values
shaderProgram.setUniformi( C.U_LIGHTS_ACTIVE, lightsActive );
shaderProgram.setUniform3fv( C.U_LIGHT_AMBIENT, lightAmbient, 0, 3 );
shaderProgram.setUniform3fv( C.U_LIGHT_POSITION, lightPosition, 0, 3 * lightsActive );
shaderProgram.setUniform3fv( C.U_LIGHT_DIFFUSE, lightDiffuse, 0, 3 * lightsActive );
shaderProgram.setUniform3fv( C.U_LIGHT_ATT_COEFFS, lightAttCoeffs, 0, 3 * lightsActive );

modelMatrix.setToTranslation( positionWrap );
modelMatrix.rotate( rotationAxis, rotation );
modelMatrix.scale( scaleX, scaleY, scaleZ );

modelViewMatrix.set( cam.view ).mul( modelMatrix );
normalMatrix.set( modelViewMatrix ).inv().tra();

shaderProgram.setUniformMatrix( C.U_NORMAL_MATRIX, normalMatrix3x3.set( normalMatrix ) );
shaderProgram.setUniformMatrix( C.U_MODEL_VIEW_MATRIX, modelViewMatrix );

stillModel.render( shaderProgram );

shaderProgram.end();

希望这些信息足够了解IT技术相关内容。谢谢您的帮助!

你的顶点着色器和片段着色器颠倒了,说实话我很惊讶它们在任何一个系统上都能工作。 - Andon M. Coleman
1
你正在测试哪个安卓设备,使用的SDK是什么? - Methnani Bilel
@Andon M. Coleman:你说得对,但这只是一个错别字 :) @Methnani Bilel:我在问题中添加了版本。 - grAPPfruit
你确定 Android 启动器已配置为使用 OpenGL ES 2 吗?你是如何创建着色器的?你用什么来渲染立方体(SpriteBatchMesh……)? - Mickäel A.
1
尝试在您的for循环中使用固定边界(即,将其硬编码为2,看看是否会改善情况)。 - P.T.
P.T. 你真是个天才!这确实解决了问题。现在我的问题是:为什么?这对我来说毫无意义!你有解释吗?我不认为这只是一个幸运的猜测 :) - grAPPfruit
1个回答

1
将我的评论转换为答案:
GLSL规范仅要求支持非常简单的循环(非常“常量”的循环边界等)。请参见我的答案{{link1:在着色器代码中使用硬编码数字工作但使用统一变量不起作用的for循环}}
在您的情况下,您可能可以通过始终循环MAX_LIGHTS来解决此问题,并仅将非活动光线的影响乘以0(或确保非活动灯光的设置全部为零或全部为1.0或任何使它们无效的内容)。
请注意,在循环内添加“跳过”不必要工作的分支可能会更慢,因为像素着色器的运行方式(SIMD并行执行)如何。尝试使所有像素始终运行相同的指令序列。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接