diff --git a/modules/video_output/opengl/fragment_shaders.c b/modules/video_output/opengl/fragment_shaders.c
index 340672620b9f5b781a0b6749926c062e917355dc..1d5b455416808367ab332d7cbfb76bf1124c7340 100644
--- a/modules/video_output/opengl/fragment_shaders.c
+++ b/modules/video_output/opengl/fragment_shaders.c
@@ -516,6 +516,11 @@ opengl_fragment_shader_init_impl(opengl_tex_converter_t *tc, GLenum tex_target,
     bool yuv_swap_uv = false;
     int ret;
 
+    assert(!tc->interop.fmt.p_palette);
+    tc->interop.sw_fmt = tc->interop.fmt;
+    tc->interop.sw_fmt.i_chroma = chroma;
+    tc->interop.sw_fmt.space = yuv_space;
+
     const vlc_chroma_description_t *desc = vlc_fourcc_GetChromaDescription(chroma);
     if (desc == NULL)
         return 0;
diff --git a/modules/video_output/opengl/interop.h b/modules/video_output/opengl/interop.h
index 46aee2a3f4feb8c4375e7807ba3c1a92b0f20af3..2eed0eb02f19404000973ce9d7dc0006e864f50d 100644
--- a/modules/video_output/opengl/interop.h
+++ b/modules/video_output/opengl/interop.h
@@ -108,6 +108,9 @@ struct vlc_gl_interop {
     /* Can only be changed from the module open function */
     video_format_t fmt;
 
+    /* Software format (useful if fmt only exposes opaque chroma) */
+    video_format_t sw_fmt;
+
     /* Pointer to decoder video context, set by the caller (can be NULL) */
     vlc_video_context *vctx;
 
diff --git a/modules/video_output/opengl/vout_helper.c b/modules/video_output/opengl/vout_helper.c
index 875dfa58feedd2aba1ddf9f1d4ca8319524c6c76..0bac994067da33e2a338d07dc8d925abdfed0e8c 100644
--- a/modules/video_output/opengl/vout_helper.c
+++ b/modules/video_output/opengl/vout_helper.c
@@ -556,6 +556,8 @@ opengl_init_program(vout_display_opengl_t *vgl, vlc_video_context *context,
     interop->ops = NULL;
     interop->glexts = glexts;
     interop->fmt = *fmt;
+    /* this is the only allocated field, and we don't need it */
+    interop->fmt.p_palette = NULL;
 
     interop->gl = tc->gl;
     interop->vt = tc->vt;