This discussion has been locked.
You can no longer post new replies to this discussion. If you have a question you can start a new discussion

How to do complex image processing in compute shaders?

Hi there,

I'm trying to implement different image processing algorithms like grayscaling, filtering, quantizing etc. on Android smartphones. I've got the camera preview rendered to a quad
as an external texture, it works fine with grayscaling in fragment shader. But when I try to do the same in OpenGL ES 3.1 with compute shaders, I get invalid operation error,
when calling glBindImageTexture(...) before dispatch compute. According to https://www.khronos.org/registry/gles/extensions/OES/OES_EGL_image_external_essl3.txt

if I bind an external texture like GL_TEXTURE_EXTERNAL_OES and bind it with glBindImageTexture, I should be able to access the image via image2D sampler in GLSL.
What am I doing wrong?

Parents
  • Hi David,

    For external surfaces you can't (and should not need to) allocate new storage. The texture memory has already been allocated externally by the system, and should be pre-populated with the data you need from the external media device.

    The wording in the extension spec related to immutability is an "neither ..., nor ..."

    "An INVALID_OPERATION error is generated if <texture> is neither the name of an immutable texture object, nor the name of an external texture object."

    ... so the fact that it is an external texture object should be sufficient to allow the glBindImageTexture to succeed, even if the texture is not marked as immutable.

    Cheers,
    Pete

Reply
  • Hi David,

    For external surfaces you can't (and should not need to) allocate new storage. The texture memory has already been allocated externally by the system, and should be pre-populated with the data you need from the external media device.

    The wording in the extension spec related to immutability is an "neither ..., nor ..."

    "An INVALID_OPERATION error is generated if <texture> is neither the name of an immutable texture object, nor the name of an external texture object."

    ... so the fact that it is an external texture object should be sufficient to allow the glBindImageTexture to succeed, even if the texture is not marked as immutable.

    Cheers,
    Pete

Children
  • Hi Pete,

    Thanks for your help so far, I've made the changes, so now this is how I generate the texture:

    textures = new int[2];
    GLES31.glGenTextures(2, textures, 0);
    GLES31.glBindTexture(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, textures[0]);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_WRAP_S, GLES31.GL_CLAMP_TO_EDGE);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_WRAP_T, GLES31.GL_CLAMP_TO_EDGE);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_MIN_FILTER, GLES31.GL_NEAREST);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_MAG_FILTER, GLES31.GL_NEAREST);

    GLES31.glBindTexture(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, textures[1]);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_WRAP_S, GLES31.GL_CLAMP_TO_EDGE);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_WRAP_T, GLES31.GL_CLAMP_TO_EDGE);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_MIN_FILTER, GLES31.GL_NEAREST);
    GLES31.glTexParameteri(GLES11Ext.GL_TEXTURE_EXTERNAL_OES, GLES31.GL_TEXTURE_MAG_FILTER, GLES31.GL_NEAREST);

    The compute shader code:

    #version 310 es

    #extension GL_OES_EGL_image_external_essl3 : enable

    layout(rgba8, binding = 0) uniform readonly image2D inTexture;

    layout(rgba8, binding = 1) uniform writeonly image2D outTexture;

    layout (local_size_x = 8, local_size_y = 8, local_size_z = 1) in;

    void main()

    {

       ivec2 storePos = ivec2(gl_GlobalInvocationID.xy);

       vec4 texColor = imageLoad(inTexture, storePos).rgba;

       float newPixel = .299f * texColor.r + .587f * texColor.g + .114f * texColor.b;

       imageStore(outTexture, storePos, vec4(newPixel, newPixel, newPixel, texColor.a));
    }

    And this is when I try to bind the texture to the shader:

    GLES31.glBindImageTexture(0, textures[0], 0, false, 0, GLES31.GL_READ_ONLY, GLES31.GL_RGBA8);

    GLES31.glBindImageTexture(1, textures[1], 0, false, 0, GLES31.GL_WRITE_ONLY, GLES31.GL_RGBA8);

    I am receiving invalid operation (1282) error on that call, so can you help me what am I doing wrong here?

    Also rendering the external texture (surfacetexture from the camera) with just a fragment shader works fine.

    Thanks,

    David

  • Eventually I managed to figure it out. First render the external texture to a normal GL_TEXTURE_2D using a frame buffer (to set up a frame buffer, check this out: http://stackoverflow.com/questions/29003414/render-camera-preview-on-a-texture-with-target-gl-texture-2d). Then you can bind the normal (immutable) texture to the compute shader, just be aware if the bindings (because GL_TEXTURE0 is occupied by the external, the input and the output texture bindings of the compute shader is 1 and 2 respectively). Then do the image processing by reading from GL_TEXTURE1 to GL_TEXTURE2 (both are created as immutable, and the first one is which was bound to the frame buffer to render the external texture into it), then finally use the 3rd texture (GL_TEXTURE2) to display the results on the screen.

  • Glad you got it working - it's a shame you needed a copy the external surface to an internal surface though - half defeats the point of zero copy external imports. I'll keep digging to see if I can find a way of avoiding that ...

    Cheers,
    Pete

  • Thanks for the advice aborges, well I did know that, but you can call it a bad habit, as I'm mainly developing in C++ and got used to
    write the namespaces everytime, in order to avoid misunderstandings.

  • Hi, fireblade

    this is just a code tip, not an address to your already solved issue.

    You can make your code less verbose by using static imports. This avoid that you have to type GLES31 in every single static method call.

    just add it to your imports:

    import static android.opengl.GLES31.*;

    Example for Math:

    import static java.lang.Math.*;

    public class HelloWorld{  

         public static void main(String []args){

           // no need to type Math.sin() and Math.cos() - just sin() and cos()

           System.out.println("Hello static import: "  + sin(0.2312f) * cos(0.2312f) );

        }

    }