+
+/* */
+#define VLC_CODEC_YUV_PLANAR_410 \
+ VLC_CODEC_I410, VLC_CODEC_YV9
+
+#define VLC_CODEC_YUV_PLANAR_420 \
+ VLC_CODEC_I420, VLC_CODEC_YV12, VLC_CODEC_J420
+
+#define VLC_CODEC_YUV_PLANAR_420_16 \
+ VLC_CODEC_I420_10L, VLC_CODEC_I420_10B, VLC_CODEC_I420_9L, VLC_CODEC_I420_9B
+
+#define VLC_CODEC_YUV_PLANAR_422 \
+ VLC_CODEC_I422, VLC_CODEC_J422
+
+#define VLC_CODEC_YUV_PLANAR_422_16 \
+ VLC_CODEC_I422_10L, VLC_CODEC_I422_10B, VLC_CODEC_I422_9L, VLC_CODEC_I422_9B
+
+#define VLC_CODEC_YUV_PLANAR_440 \
+ VLC_CODEC_I440, VLC_CODEC_J440
+
+#define VLC_CODEC_YUV_PLANAR_444 \
+ VLC_CODEC_I444, VLC_CODEC_J444
+
+#define VLC_CODEC_YUV_PLANAR_444_16 \
+ VLC_CODEC_I444_10L, VLC_CODEC_I444_10B, VLC_CODEC_I444_9L, VLC_CODEC_I444_9B
+
+#define VLC_CODEC_YUV_PACKED \
+ VLC_CODEC_YUYV, VLC_CODEC_YVYU, \
+ VLC_CODEC_UYVY, VLC_CODEC_VYUY
+
+#define VLC_CODEC_FALLBACK_420 \
+ VLC_CODEC_YUV_PLANAR_422, VLC_CODEC_YUV_PACKED, \
+ VLC_CODEC_YUV_PLANAR_444, VLC_CODEC_YUV_PLANAR_440, \
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211
+
+static const vlc_fourcc_t p_I420_fallback[] = {
+ VLC_CODEC_I420, VLC_CODEC_YV12, VLC_CODEC_J420, VLC_CODEC_FALLBACK_420, 0
+};
+static const vlc_fourcc_t p_J420_fallback[] = {
+ VLC_CODEC_J420, VLC_CODEC_I420, VLC_CODEC_YV12, VLC_CODEC_FALLBACK_420, 0
+};
+static const vlc_fourcc_t p_YV12_fallback[] = {
+ VLC_CODEC_YV12, VLC_CODEC_I420, VLC_CODEC_J420, VLC_CODEC_FALLBACK_420, 0
+};
+
+#define VLC_CODEC_FALLBACK_420_16 \
+ VLC_CODEC_I420, VLC_CODEC_YV12, VLC_CODEC_J420, VLC_CODEC_FALLBACK_420
+
+static const vlc_fourcc_t p_I420_9L_fallback[] = {
+ VLC_CODEC_I420_9L, VLC_CODEC_I420_9B, VLC_CODEC_FALLBACK_420_16, 0
+};
+static const vlc_fourcc_t p_I420_9B_fallback[] = {
+ VLC_CODEC_I420_9B, VLC_CODEC_I420_9L, VLC_CODEC_FALLBACK_420_16, 0
+};
+static const vlc_fourcc_t p_I420_10L_fallback[] = {
+ VLC_CODEC_I420_10L, VLC_CODEC_I420_10B, VLC_CODEC_FALLBACK_420_16, 0
+};
+static const vlc_fourcc_t p_I420_10B_fallback[] = {
+ VLC_CODEC_I420_10B, VLC_CODEC_I420_10L, VLC_CODEC_FALLBACK_420_16, 0
+};
+
+#define VLC_CODEC_FALLBACK_422 \
+ VLC_CODEC_YUV_PACKED, VLC_CODEC_YUV_PLANAR_420, \
+ VLC_CODEC_YUV_PLANAR_444, VLC_CODEC_YUV_PLANAR_440, \
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211
+
+static const vlc_fourcc_t p_I422_fallback[] = {
+ VLC_CODEC_I422, VLC_CODEC_J422, VLC_CODEC_FALLBACK_422, 0
+};
+static const vlc_fourcc_t p_J422_fallback[] = {
+ VLC_CODEC_J422, VLC_CODEC_I422, VLC_CODEC_FALLBACK_422, 0
+};
+
+#define VLC_CODEC_FALLBACK_422_16 \
+ VLC_CODEC_I422, VLC_CODEC_J422, VLC_CODEC_FALLBACK_422
+
+static const vlc_fourcc_t p_I422_9L_fallback[] = {
+ VLC_CODEC_I422_9L, VLC_CODEC_I422_9B, VLC_CODEC_FALLBACK_422_16, 0
+};
+static const vlc_fourcc_t p_I422_9B_fallback[] = {
+ VLC_CODEC_I422_9B, VLC_CODEC_I422_9L, VLC_CODEC_FALLBACK_422_16, 0
+};
+static const vlc_fourcc_t p_I422_10L_fallback[] = {
+ VLC_CODEC_I422_10L, VLC_CODEC_I422_10B, VLC_CODEC_FALLBACK_422_16, 0
+};
+static const vlc_fourcc_t p_I422_10B_fallback[] = {
+ VLC_CODEC_I422_10B, VLC_CODEC_I422_10L, VLC_CODEC_FALLBACK_422_16, 0
+};
+
+#define VLC_CODEC_FALLBACK_444 \
+ VLC_CODEC_YUV_PLANAR_422, VLC_CODEC_YUV_PACKED, \
+ VLC_CODEC_YUV_PLANAR_420, VLC_CODEC_YUV_PLANAR_440, \
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211
+
+static const vlc_fourcc_t p_I444_fallback[] = {
+ VLC_CODEC_I444, VLC_CODEC_J444, VLC_CODEC_FALLBACK_444, 0
+};
+static const vlc_fourcc_t p_J444_fallback[] = {
+ VLC_CODEC_J444, VLC_CODEC_I444, VLC_CODEC_FALLBACK_444, 0
+};
+
+#define VLC_CODEC_FALLBACK_444_16 \
+ VLC_CODEC_I444, VLC_CODEC_J444, VLC_CODEC_FALLBACK_444
+
+static const vlc_fourcc_t p_I444_9L_fallback[] = {
+ VLC_CODEC_I444_9L, VLC_CODEC_I444_9B, VLC_CODEC_FALLBACK_444_16, 0
+};
+static const vlc_fourcc_t p_I444_9B_fallback[] = {
+ VLC_CODEC_I444_9B, VLC_CODEC_I444_9L, VLC_CODEC_FALLBACK_444_16, 0
+};
+static const vlc_fourcc_t p_I444_10L_fallback[] = {
+ VLC_CODEC_I444_10L, VLC_CODEC_I444_10B, VLC_CODEC_FALLBACK_444_16, 0
+};
+static const vlc_fourcc_t p_I444_10B_fallback[] = {
+ VLC_CODEC_I444_10B, VLC_CODEC_I444_10L, VLC_CODEC_FALLBACK_444_16, 0
+};
+
+static const vlc_fourcc_t p_I440_fallback[] = {
+ VLC_CODEC_I440,
+ VLC_CODEC_YUV_PLANAR_420,
+ VLC_CODEC_YUV_PLANAR_422,
+ VLC_CODEC_YUV_PLANAR_444,
+ VLC_CODEC_YUV_PACKED,
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211, 0
+};
+
+#define VLC_CODEC_FALLBACK_PACKED \
+ VLC_CODEC_YUV_PLANAR_422, VLC_CODEC_YUV_PLANAR_420, \
+ VLC_CODEC_YUV_PLANAR_444, VLC_CODEC_YUV_PLANAR_440, \
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211
+
+static const vlc_fourcc_t p_YUYV_fallback[] = {
+ VLC_CODEC_YUYV,
+ VLC_CODEC_YVYU,
+ VLC_CODEC_UYVY,
+ VLC_CODEC_VYUY,
+ VLC_CODEC_FALLBACK_PACKED, 0
+};
+static const vlc_fourcc_t p_YVYU_fallback[] = {
+ VLC_CODEC_YVYU,
+ VLC_CODEC_YUYV,
+ VLC_CODEC_UYVY,
+ VLC_CODEC_VYUY,
+ VLC_CODEC_FALLBACK_PACKED, 0
+};
+static const vlc_fourcc_t p_UYVY_fallback[] = {
+ VLC_CODEC_UYVY,
+ VLC_CODEC_VYUY,
+ VLC_CODEC_YUYV,
+ VLC_CODEC_YVYU,
+ VLC_CODEC_FALLBACK_PACKED, 0
+};
+static const vlc_fourcc_t p_VYUY_fallback[] = {
+ VLC_CODEC_VYUY,
+ VLC_CODEC_UYVY,
+ VLC_CODEC_YUYV,
+ VLC_CODEC_YVYU,
+ VLC_CODEC_FALLBACK_PACKED, 0
+};
+
+static const vlc_fourcc_t *pp_YUV_fallback[] = {
+ p_YV12_fallback,
+ p_I420_fallback,
+ p_I420_9L_fallback,
+ p_I420_9B_fallback,
+ p_I420_10L_fallback,
+ p_I420_10B_fallback,
+ p_J420_fallback,
+ p_I422_fallback,
+ p_I422_9L_fallback,
+ p_I422_9B_fallback,
+ p_I422_10L_fallback,
+ p_I422_10B_fallback,
+ p_J422_fallback,
+ p_I444_fallback,
+ p_J444_fallback,
+ p_I444_9L_fallback,
+ p_I444_9B_fallback,
+ p_I444_10L_fallback,
+ p_I444_10B_fallback,
+ p_I440_fallback,
+ p_YUYV_fallback,
+ p_YVYU_fallback,
+ p_UYVY_fallback,
+ p_VYUY_fallback,
+
+
+ NULL,
+};
+
+static const vlc_fourcc_t p_list_YUV[] = {
+ VLC_CODEC_YUV_PLANAR_420,
+ VLC_CODEC_YUV_PLANAR_422,
+ VLC_CODEC_YUV_PLANAR_440,
+ VLC_CODEC_YUV_PLANAR_444,
+ VLC_CODEC_YUV_PACKED,
+ VLC_CODEC_I411, VLC_CODEC_YUV_PLANAR_410, VLC_CODEC_Y211,
+ VLC_CODEC_YUV_PLANAR_420_16,
+ VLC_CODEC_YUV_PLANAR_422_16,
+ VLC_CODEC_YUV_PLANAR_444_16,
+ 0,
+};
+
+/* */
+static const vlc_fourcc_t p_RGB32_fallback[] = {
+ VLC_CODEC_RGB32,
+ VLC_CODEC_RGB24,
+ VLC_CODEC_RGB16,
+ VLC_CODEC_RGB15,
+ VLC_CODEC_RGB8,
+ 0,
+};
+static const vlc_fourcc_t p_RGB24_fallback[] = {
+ VLC_CODEC_RGB24,
+ VLC_CODEC_RGB32,
+ VLC_CODEC_RGB16,
+ VLC_CODEC_RGB15,
+ VLC_CODEC_RGB8,
+ 0,
+};
+static const vlc_fourcc_t p_RGB16_fallback[] = {
+ VLC_CODEC_RGB16,
+ VLC_CODEC_RGB24,
+ VLC_CODEC_RGB32,
+ VLC_CODEC_RGB15,
+ VLC_CODEC_RGB8,
+ 0,
+};
+static const vlc_fourcc_t p_RGB15_fallback[] = {
+ VLC_CODEC_RGB15,
+ VLC_CODEC_RGB16,
+ VLC_CODEC_RGB24,
+ VLC_CODEC_RGB32,
+ VLC_CODEC_RGB8,
+ 0,
+};
+static const vlc_fourcc_t p_RGB8_fallback[] = {
+ VLC_CODEC_RGB8,
+ VLC_CODEC_RGB15,
+ VLC_CODEC_RGB16,
+ VLC_CODEC_RGB24,
+ VLC_CODEC_RGB32,
+ 0,
+};
+static const vlc_fourcc_t *pp_RGB_fallback[] = {
+ p_RGB32_fallback,
+ p_RGB24_fallback,
+ p_RGB16_fallback,
+ p_RGB15_fallback,
+ p_RGB8_fallback,
+ NULL,
+};
+
+
+/* */
+static const vlc_fourcc_t *GetFallback( vlc_fourcc_t i_fourcc,
+ const vlc_fourcc_t *pp_fallback[],
+ const vlc_fourcc_t p_list[] )
+{
+ for( unsigned i = 0; pp_fallback[i]; i++ )
+ {
+ if( pp_fallback[i][0] == i_fourcc )
+ return pp_fallback[i];
+ }
+ return p_list;
+}
+
+const vlc_fourcc_t *vlc_fourcc_GetYUVFallback( vlc_fourcc_t i_fourcc )
+{
+ return GetFallback( i_fourcc, pp_YUV_fallback, p_list_YUV );
+}
+const vlc_fourcc_t *vlc_fourcc_GetRGBFallback( vlc_fourcc_t i_fourcc )
+{
+ return GetFallback( i_fourcc, pp_RGB_fallback, p_RGB32_fallback );
+}
+
+bool vlc_fourcc_AreUVPlanesSwapped( vlc_fourcc_t a, vlc_fourcc_t b )
+{
+ static const vlc_fourcc_t pp_swapped[][4] = {
+ { VLC_CODEC_YV12, VLC_CODEC_I420, VLC_CODEC_J420, 0 },
+ { VLC_CODEC_YV9, VLC_CODEC_I410, 0 },
+ { 0 }
+ };
+
+ for( int i = 0; pp_swapped[i][0]; i++ )
+ {
+ if( pp_swapped[i][0] == b )
+ {
+ vlc_fourcc_t t = a;
+ a = b;
+ b = t;
+ }
+ if( pp_swapped[i][0] != a )
+ continue;
+ for( int j = 1; pp_swapped[i][j]; j++ )
+ {
+ if( pp_swapped[i][j] == b )
+ return true;
+ }
+ }
+ return false;
+}
+
+bool vlc_fourcc_IsYUV(vlc_fourcc_t fcc)
+{
+ for( unsigned i = 0; p_list_YUV[i]; i++ )
+ {
+ if( p_list_YUV[i] == fcc )
+ return true;
+ }
+ return false;
+}
+
+#define PLANAR(n, w_den, h_den, size, bits) \
+ { .plane_count = n, \
+ .p = { {.w = {1, 1}, .h = {1, 1}}, \
+ {.w = {1,w_den}, .h = {1,h_den}}, \
+ {.w = {1,w_den}, .h = {1,h_den}}, \
+ {.w = {1, 1}, .h = {1, 1}} }, \
+ .pixel_size = size, \
+ .pixel_bits = bits }
+
+#define PLANAR_8(n, w_den, h_den) PLANAR(n, w_den, h_den, 1, 8)
+#define PLANAR_16(n, w_den, h_den, bits) PLANAR(n, w_den, h_den, 2, bits)
+
+#define PACKED_FMT(size, bits) \
+ { .plane_count = 1, \
+ .p = { {.w = {1,1}, .h = {1,1}} }, \
+ .pixel_size = size, \
+ .pixel_bits = bits }
+
+static const struct
+{
+ vlc_fourcc_t p_fourcc[6];
+ vlc_chroma_description_t description;
+} p_list_chroma_description[] = {
+ { { VLC_CODEC_I411, 0 }, PLANAR_8(3, 4, 1) },
+ { { VLC_CODEC_YUV_PLANAR_410, 0 }, PLANAR_8(3, 4, 4) },
+ { { VLC_CODEC_YUV_PLANAR_420, 0 }, PLANAR_8(3, 2, 2) },
+ { { VLC_CODEC_YUV_PLANAR_422, 0 }, PLANAR_8(3, 2, 1) },
+ { { VLC_CODEC_YUV_PLANAR_440, 0 }, PLANAR_8(3, 1, 2) },
+ { { VLC_CODEC_YUV_PLANAR_444, 0 }, PLANAR_8(3, 1, 1) },
+ { { VLC_CODEC_YUVA, 0 }, PLANAR_8(4, 1, 1) },
+
+ { { VLC_CODEC_I420_10L,
+ VLC_CODEC_I420_10B, 0 }, PLANAR_16(3, 2, 2, 10) },
+ { { VLC_CODEC_I420_9L,
+ VLC_CODEC_I420_9B, 0 }, PLANAR_16(3, 2, 2, 9) },
+ { { VLC_CODEC_I422_10L,
+ VLC_CODEC_I422_10B, 0 }, PLANAR_16(3, 2, 1, 10) },
+ { { VLC_CODEC_I422_9L,
+ VLC_CODEC_I422_9B, 0 }, PLANAR_16(3, 2, 1, 9) },
+ { { VLC_CODEC_I444_10L,
+ VLC_CODEC_I444_10B, 0 }, PLANAR_16(3, 1, 1, 10) },
+ { { VLC_CODEC_I444_9L,
+ VLC_CODEC_I444_9B, 0 }, PLANAR_16(3, 1, 1, 9) },
+
+ { { VLC_CODEC_YUV_PACKED, 0 }, PACKED_FMT(2, 16) },
+ { { VLC_CODEC_RGB8, VLC_CODEC_GREY,
+ VLC_CODEC_YUVP, VLC_CODEC_RGBP, 0 }, PACKED_FMT(1, 8) },
+
+ { { VLC_CODEC_RGB15, 0 }, PACKED_FMT(2, 15) },
+ { { VLC_CODEC_RGB12, 0 }, PACKED_FMT(2, 12) },
+ { { VLC_CODEC_RGB16, VLC_CODEC_RGBT,
+ VLC_CODEC_RGBA16, 0 }, PACKED_FMT(2, 16) },
+ { { VLC_CODEC_RGB24, 0 }, PACKED_FMT(3, 24) },
+ { { VLC_CODEC_RGB32, 0 }, PACKED_FMT(4, 24) },
+ { { VLC_CODEC_RGBA, 0 }, PACKED_FMT(4, 32) },
+
+ { { VLC_CODEC_Y211, 0 }, { 1, { {{1,4}, {1,1}} }, 4, 32 } },
+
+ { {0}, { 0, {}, 0, 0 } }
+};
+
+#undef PACKED_FMT
+#undef PLANAR_16
+#undef PLANAR_8
+#undef PLANAR
+
+const vlc_chroma_description_t *vlc_fourcc_GetChromaDescription( vlc_fourcc_t i_fourcc )
+{
+ for( unsigned i = 0; p_list_chroma_description[i].p_fourcc[0]; i++ )
+ {
+ const vlc_fourcc_t *p_fourcc = p_list_chroma_description[i].p_fourcc;
+ for( unsigned j = 0; p_fourcc[j]; j++ )
+ {
+ if( p_fourcc[j] == i_fourcc )
+ return &p_list_chroma_description[i].description;
+ }
+ }
+ return NULL;
+}