/* GStreamer * Copyright (C) 2019 Seungha Yang * Copyright (C) 2020 Seungha Yang * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "gstmfconfig.h" #include "gstmfutils.h" #include using namespace Microsoft::WRL; G_BEGIN_DECLS GST_DEBUG_CATEGORY_EXTERN (gst_mf_utils_debug); #define GST_CAT_DEFAULT gst_mf_utils_debug G_END_DECLS #define MAKE_RAW_FORMAT_CAPS(format) \ "video/x-raw, format = (string) " format /* No GUID is defined for "Y16 " in mfapi.h, but it's used by several devices */ DEFINE_MEDIATYPE_GUID (MFVideoFormat_Y16, FCC ('Y16 ')); static struct { const GUID &mf_format; const gchar *caps_string; GstVideoFormat format; } raw_video_format_map[] = { {MFVideoFormat_RGB32, MAKE_RAW_FORMAT_CAPS ("BGRx"), GST_VIDEO_FORMAT_BGRx}, {MFVideoFormat_ARGB32, MAKE_RAW_FORMAT_CAPS ("BGRA"), GST_VIDEO_FORMAT_BGRA}, {MFVideoFormat_RGB24, MAKE_RAW_FORMAT_CAPS ("BGR"), GST_VIDEO_FORMAT_BGR}, {MFVideoFormat_RGB555, MAKE_RAW_FORMAT_CAPS ("RGB15"), GST_VIDEO_FORMAT_RGB15}, {MFVideoFormat_RGB565, MAKE_RAW_FORMAT_CAPS ("RGB16"), GST_VIDEO_FORMAT_RGB16}, {MFVideoFormat_AYUV, MAKE_RAW_FORMAT_CAPS ("VUYA"), GST_VIDEO_FORMAT_VUYA}, {MFVideoFormat_YUY2, MAKE_RAW_FORMAT_CAPS ("YUY2"), GST_VIDEO_FORMAT_YUY2}, {MFVideoFormat_YVYU, MAKE_RAW_FORMAT_CAPS ("YVYU"), GST_VIDEO_FORMAT_YVYU}, {MFVideoFormat_UYVY, MAKE_RAW_FORMAT_CAPS ("UYVY"), GST_VIDEO_FORMAT_UYVY}, {MFVideoFormat_NV12, MAKE_RAW_FORMAT_CAPS ("NV12"), GST_VIDEO_FORMAT_NV12}, {MFVideoFormat_YV12, MAKE_RAW_FORMAT_CAPS ("YV12"), GST_VIDEO_FORMAT_YV12}, {MFVideoFormat_I420, MAKE_RAW_FORMAT_CAPS ("I420"), GST_VIDEO_FORMAT_I420}, {MFVideoFormat_IYUV, MAKE_RAW_FORMAT_CAPS ("I420"), GST_VIDEO_FORMAT_I420}, {MFVideoFormat_P010, MAKE_RAW_FORMAT_CAPS ("P010"), GST_VIDEO_FORMAT_P010_10LE}, {MFVideoFormat_P016, MAKE_RAW_FORMAT_CAPS ("P016"), GST_VIDEO_FORMAT_P016_LE}, {MFVideoFormat_v210, MAKE_RAW_FORMAT_CAPS ("v210"), GST_VIDEO_FORMAT_v210}, {MFVideoFormat_v216, MAKE_RAW_FORMAT_CAPS ("v216"), GST_VIDEO_FORMAT_v216}, {MFVideoFormat_Y16, MAKE_RAW_FORMAT_CAPS ("GRAY16_LE"), GST_VIDEO_FORMAT_GRAY16_LE}, }; static struct { const GUID &mf_format; const gchar *caps_string; } encoded_video_format_map[] = { {MFVideoFormat_H264, "video/x-h264"}, {MFVideoFormat_HEVC, "video/x-h265"}, {MFVideoFormat_H265, "video/x-h265"}, {MFVideoFormat_VP80, "video/x-vp8"}, {MFVideoFormat_VP90, "video/x-vp9"}, {MFVideoFormat_MJPG, "image/jpeg"}, }; GstVideoFormat gst_mf_video_subtype_to_video_format (const GUID * subtype) { gint i; for (i = 0; i < G_N_ELEMENTS (raw_video_format_map); i++) { if (IsEqualGUID (raw_video_format_map[i].mf_format, *subtype)) return raw_video_format_map[i].format; } return GST_VIDEO_FORMAT_UNKNOWN; } const GUID * gst_mf_video_subtype_from_video_format (GstVideoFormat format) { gint i; for (i = 0; i < G_N_ELEMENTS (raw_video_format_map); i++) { if (raw_video_format_map[i].format == format) return &raw_video_format_map[i].mf_format; } return NULL; } static GstCaps * gst_mf_media_type_to_video_caps (IMFMediaType * media_type) { HRESULT hr; GstCaps *caps = NULL; gint i; guint32 width = 0; guint32 height = 0; guint32 num, den; guint32 val; gchar *str; GUID subtype; GstVideoChromaSite chroma_site; GstVideoColorimetry colorimetry; gboolean raw_format = TRUE; hr = media_type->GetGUID (MF_MT_SUBTYPE, &subtype); if (FAILED (hr)) { GST_WARNING ("Failed to get subtype, hr: 0x%x", (guint) hr); return NULL; } for (i = 0; i < G_N_ELEMENTS (raw_video_format_map); i++) { if (IsEqualGUID (raw_video_format_map[i].mf_format, subtype)) { caps = gst_caps_from_string (raw_video_format_map[i].caps_string); break; } } if (!caps) { for (i = 0; i < G_N_ELEMENTS (encoded_video_format_map); i++) { if (IsEqualGUID (encoded_video_format_map[i].mf_format, subtype)) { caps = gst_caps_from_string (encoded_video_format_map[i].caps_string); raw_format = FALSE; break; } } } if (!caps) { GST_WARNING ("Unknown format %" GST_FOURCC_FORMAT, GST_FOURCC_ARGS (subtype.Data1)); return NULL; } hr = MFGetAttributeSize (media_type, MF_MT_FRAME_SIZE, &width, &height); if (FAILED (hr) || !width || !height) { GST_WARNING ("Couldn't get frame size, hr: 0x%x", (guint) hr); if (raw_format) { gst_caps_unref (caps); return NULL; } } if (width > 0 && height > 0) { gst_caps_set_simple (caps, "width", G_TYPE_INT, width, "height", G_TYPE_INT, height, NULL); } hr = MFGetAttributeRatio (media_type, MF_MT_FRAME_RATE, &num, &den); if (SUCCEEDED (hr) && num > 0 && den > 0) gst_caps_set_simple (caps, "framerate", GST_TYPE_FRACTION, num, den, NULL); hr = MFGetAttributeRatio (media_type, MF_MT_PIXEL_ASPECT_RATIO, &num, &den); if (SUCCEEDED (hr) && num > 0 && den > 0) gst_caps_set_simple (caps, "pixel-aspect-ratio", GST_TYPE_FRACTION, num, den, NULL); colorimetry.range = GST_VIDEO_COLOR_RANGE_UNKNOWN; colorimetry.matrix = GST_VIDEO_COLOR_MATRIX_UNKNOWN; colorimetry.transfer = GST_VIDEO_TRANSFER_UNKNOWN; colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_UNKNOWN; hr = media_type->GetUINT32 (MF_MT_VIDEO_NOMINAL_RANGE, &val); if (SUCCEEDED (hr)) { switch (val) { case MFNominalRange_0_255: colorimetry.range = GST_VIDEO_COLOR_RANGE_0_255; break; case MFNominalRange_16_235: colorimetry.range = GST_VIDEO_COLOR_RANGE_16_235; break; default: break; } } hr = media_type->GetUINT32 (MF_MT_VIDEO_PRIMARIES, &val); if (SUCCEEDED (hr)) { switch (val) { case MFVideoPrimaries_BT709: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_BT709; break; case MFVideoPrimaries_BT470_2_SysM: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_BT470M; break; case MFVideoPrimaries_BT470_2_SysBG: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_BT470BG; break; case MFVideoPrimaries_SMPTE170M: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_SMPTE170M; break; case MFVideoPrimaries_SMPTE240M: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_SMPTE240M; break; case MFVideoPrimaries_EBU3213: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_EBU3213; break; case MFVideoPrimaries_BT2020: colorimetry.primaries = GST_VIDEO_COLOR_PRIMARIES_BT2020; break; default: GST_FIXME ("unhandled color primaries %d", val); break; } } hr = media_type->GetUINT32 (MF_MT_YUV_MATRIX, &val); if (SUCCEEDED (hr)) { switch (val) { case MFVideoTransferMatrix_BT709: colorimetry.matrix = GST_VIDEO_COLOR_MATRIX_BT709; break; case MFVideoTransferMatrix_BT601: colorimetry.matrix = GST_VIDEO_COLOR_MATRIX_BT601; break; case MFVideoTransferMatrix_SMPTE240M: colorimetry.matrix = GST_VIDEO_COLOR_MATRIX_SMPTE240M; break; case MFVideoTransferMatrix_BT2020_10: case MFVideoTransferMatrix_BT2020_12: colorimetry.matrix = GST_VIDEO_COLOR_MATRIX_BT2020; break; default: GST_FIXME ("unhandled color matrix %d", val); break; } } hr = media_type->GetUINT32 (MF_MT_TRANSFER_FUNCTION, &val); if (SUCCEEDED (hr)) { switch (val) { case MFVideoTransFunc_10: colorimetry.transfer = GST_VIDEO_TRANSFER_GAMMA10; break; case MFVideoTransFunc_18: colorimetry.transfer = GST_VIDEO_TRANSFER_GAMMA18; break; case MFVideoTransFunc_20: colorimetry.transfer = GST_VIDEO_TRANSFER_GAMMA20; break; case MFVideoTransFunc_22: colorimetry.transfer = GST_VIDEO_TRANSFER_GAMMA22; break; case MFVideoTransFunc_709: case MFVideoTransFunc_709_sym: colorimetry.transfer = GST_VIDEO_TRANSFER_BT709; break; case MFVideoTransFunc_240M: colorimetry.transfer = GST_VIDEO_TRANSFER_SMPTE240M; break; case MFVideoTransFunc_sRGB: colorimetry.transfer = GST_VIDEO_TRANSFER_SRGB; break; case MFVideoTransFunc_28: colorimetry.transfer = GST_VIDEO_TRANSFER_GAMMA28; break; case MFVideoTransFunc_Log_100: colorimetry.transfer = GST_VIDEO_TRANSFER_LOG100; break; case MFVideoTransFunc_Log_316: colorimetry.transfer = GST_VIDEO_TRANSFER_LOG316; break; case MFVideoTransFunc_2020_const: case MFVideoTransFunc_2020: colorimetry.transfer = GST_VIDEO_TRANSFER_BT2020_10; break; case MFVideoTransFunc_2084: colorimetry.transfer = GST_VIDEO_TRANSFER_SMPTE2084; break; case MFVideoTransFunc_HLG: colorimetry.transfer = GST_VIDEO_TRANSFER_ARIB_STD_B67; break; default: GST_FIXME ("unhandled color transfer %d", val); break; } } str = gst_video_colorimetry_to_string (&colorimetry); if (str) { gst_caps_set_simple (caps, "colorimetry", G_TYPE_STRING, str, NULL); g_free (str); str = NULL; } chroma_site = GST_VIDEO_CHROMA_SITE_UNKNOWN; hr = media_type->GetUINT32 (MF_MT_VIDEO_CHROMA_SITING, &val); if (SUCCEEDED (hr)) { gboolean known_value = TRUE; if ((val & MFVideoChromaSubsampling_MPEG2) == MFVideoChromaSubsampling_MPEG2) { chroma_site = GST_VIDEO_CHROMA_SITE_MPEG2; } else if ((val & MFVideoChromaSubsampling_DV_PAL) == MFVideoChromaSubsampling_DV_PAL) { chroma_site = GST_VIDEO_CHROMA_SITE_DV; } else if ((val & MFVideoChromaSubsampling_Cosited) == MFVideoChromaSubsampling_Cosited) { chroma_site = GST_VIDEO_CHROMA_SITE_COSITED; } else { known_value = FALSE; } GST_LOG ("have %s chroma site value 0x%x", known_value ? "known" : "unknown", val); } if (chroma_site != GST_VIDEO_CHROMA_SITE_UNKNOWN) gst_caps_set_simple (caps, "chroma-site", G_TYPE_STRING, gst_video_chroma_to_string (chroma_site), NULL); return caps; } GstCaps * gst_mf_media_type_to_caps (IMFMediaType * media_type) { GUID major_type; HRESULT hr; g_return_val_if_fail (media_type != NULL, NULL); hr = media_type->GetMajorType (&major_type); if (FAILED (hr)) { GST_WARNING ("failed to get major type, hr: 0x%x", (guint) hr); return NULL; } if (IsEqualGUID (major_type, MFMediaType_Video)) return gst_mf_media_type_to_video_caps (media_type); return NULL; } void gst_mf_media_type_release (IMFMediaType * media_type) { if (media_type) media_type->Release (); } gboolean _gst_mf_result (HRESULT hr, GstDebugCategory * cat, const gchar * file, const gchar * function, gint line) { #ifndef GST_DISABLE_GST_DEBUG gboolean ret = TRUE; if (FAILED (hr)) { gchar *error_text = NULL; error_text = g_win32_error_message ((gint) hr); /* g_win32_error_message() doesn't cover all HERESULT return code, * so it could be empty string, or null if there was an error * in g_utf16_to_utf8() */ gst_debug_log (cat, GST_LEVEL_WARNING, file, function, line, NULL, "MediaFoundation call failed: 0x%x, %s", (guint) hr, GST_STR_NULL (error_text)); g_free (error_text); ret = FALSE; } return ret; #else return SUCCEEDED (hr); #endif } /* Reference: * https://docs.microsoft.com/en-us/windows/win32/medfound/media-type-debugging-code */ #define GST_MF_IF_EQUAL_RETURN(guid,val) G_STMT_START { \ if (IsEqualGUID (guid, val)) \ return G_STRINGIFY (val); \ } G_STMT_END static const gchar * gst_mf_guid_to_static_string (const GUID& guid) { GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MAJOR_TYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MAJOR_TYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_SUBTYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_ALL_SAMPLES_INDEPENDENT); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_FIXED_SIZE_SAMPLES); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_COMPRESSED); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_SAMPLE_SIZE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_WRAPPED_TYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_NUM_CHANNELS); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_SAMPLES_PER_SECOND); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_FLOAT_SAMPLES_PER_SECOND); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_AVG_BYTES_PER_SECOND); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_BLOCK_ALIGNMENT); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_BITS_PER_SAMPLE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_VALID_BITS_PER_SAMPLE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_SAMPLES_PER_BLOCK); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_CHANNEL_MASK); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_FOLDDOWN_MATRIX); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_WMADRC_PEAKREF); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_WMADRC_PEAKTARGET); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_WMADRC_AVGREF); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_WMADRC_AVGTARGET); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AUDIO_PREFER_WAVEFORMATEX); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AAC_PAYLOAD_TYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AAC_AUDIO_PROFILE_LEVEL_INDICATION); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_FRAME_SIZE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_FRAME_RATE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_FRAME_RATE_RANGE_MAX); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_FRAME_RATE_RANGE_MIN); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_PIXEL_ASPECT_RATIO); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DRM_FLAGS); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_PAD_CONTROL_FLAGS); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_SOURCE_CONTENT_HINT); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_VIDEO_CHROMA_SITING); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_INTERLACE_MODE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_TRANSFER_FUNCTION); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_VIDEO_PRIMARIES); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_YUV_MATRIX); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_VIDEO_LIGHTING); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_VIDEO_NOMINAL_RANGE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_GEOMETRIC_APERTURE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MINIMUM_DISPLAY_APERTURE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_PAN_SCAN_APERTURE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_PAN_SCAN_ENABLED); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AVG_BITRATE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AVG_BIT_ERROR_RATE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MAX_KEYFRAME_SPACING); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DEFAULT_STRIDE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_PALETTE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_USER_DATA); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG_START_TIME_CODE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG2_PROFILE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG2_LEVEL); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG2_FLAGS); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG_SEQUENCE_HEADER); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_AAUX_SRC_PACK_0); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_AAUX_CTRL_PACK_0); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_AAUX_SRC_PACK_1); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_AAUX_CTRL_PACK_1); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_VAUX_SRC_PACK); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_DV_VAUX_CTRL_PACK); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_IMAGE_LOSS_TOLERANT); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG4_SAMPLE_DESCRIPTION); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_MPEG4_CURRENT_SAMPLE_ENTRY); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Audio); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Video); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Protected); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_SAMI); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Script); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Image); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_HTML); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_Binary); GST_MF_IF_EQUAL_RETURN(guid, MFMediaType_FileTransfer); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_AI44); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_ARGB32); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_AYUV); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_DV25); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_DV50); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_DVH1); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_DVSD); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_DVSL); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_H264); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_H265); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_HEVC); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_HEVC_ES); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_I420); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_IYUV); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_M4S2); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MJPG); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MP43); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MP4S); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MP4V); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MPG1); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MSS1); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_MSS2); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_NV11); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_NV12); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_P010); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_P016); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_P210); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_P216); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_RGB24); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_RGB32); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_RGB555); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_RGB565); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_RGB8); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_UYVY); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_v210); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_v410); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_VP80); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_VP90); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_WMV1); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_WMV2); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_WMV3); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_WVC1); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y210); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y216); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y410); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y416); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y41P); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_Y41T); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_YUY2); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_YV12); GST_MF_IF_EQUAL_RETURN(guid, MFVideoFormat_YVYU); /* WAVE_FORMAT_PCM */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_PCM); /* WAVE_FORMAT_IEEE_FLOAT */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_Float); /* WAVE_FORMAT_DTS */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_DTS); /* WAVE_FORMAT_DOLBY_AC3_SPDIF */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_Dolby_AC3_SPDIF); /* WAVE_FORMAT_DRM */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_DRM); /* WAVE_FORMAT_WMAUDIO2 */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_WMAudioV8); /* WAVE_FORMAT_WMAUDIO3 */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_WMAudioV9); /* WAVE_FORMAT_WMAUDIO_LOSSLESS */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_WMAudio_Lossless); /* WAVE_FORMAT_WMASPDIF */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_WMASPDIF); /* WAVE_FORMAT_WMAVOICE9 */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_MSP1); /* WAVE_FORMAT_MPEGLAYER3 */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_MP3); /* WAVE_FORMAT_MPEG */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_MPEG); /* WAVE_FORMAT_MPEG_HEAAC */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_AAC); /* WAVE_FORMAT_MPEG_ADTS_AAC */ GST_MF_IF_EQUAL_RETURN(guid, MFAudioFormat_ADTS); #if GST_MF_WINAPI_DESKTOP GST_MF_IF_EQUAL_RETURN(guid, MF_MT_CUSTOM_VIDEO_PRIMARIES); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_AM_FORMAT_TYPE); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_ARBITRARY_HEADER); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_ARBITRARY_FORMAT); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_ORIGINAL_4CC); GST_MF_IF_EQUAL_RETURN(guid, MF_MT_ORIGINAL_WAVE_FORMAT_TAG); #endif return NULL; } static gchar * gst_mf_guid_to_string (const GUID& guid) { const gchar *str = NULL; HRESULT hr; WCHAR *name = NULL; gchar *ret = NULL; str = gst_mf_guid_to_static_string (guid); if (str) return g_strdup (str); hr = StringFromCLSID (guid, &name); if (gst_mf_result (hr) && name) { ret = g_utf16_to_utf8 ((const gunichar2 *) name, -1, NULL, NULL, NULL); CoTaskMemFree (name); if (ret) return ret; } ret = g_strdup_printf ("%8.8x-%4.4x-%4.4x-%2.2x%2.2x-%2.2x%2.2x%2.2x%2.2x%2.2x%2.2x", (guint) guid.Data1, (guint) guid.Data2, (guint) guid.Data3, guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3], guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]); return ret; } static gchar * gst_mf_attribute_value_to_string (const GUID& guid, const PROPVARIANT& var) { if (IsEqualGUID (guid, MF_MT_FRAME_RATE) || IsEqualGUID (guid, MF_MT_FRAME_RATE_RANGE_MAX) || IsEqualGUID (guid, MF_MT_FRAME_RATE_RANGE_MIN) || IsEqualGUID (guid, MF_MT_FRAME_SIZE) || IsEqualGUID (guid, MF_MT_PIXEL_ASPECT_RATIO)) { UINT32 high = 0, low = 0; Unpack2UINT32AsUINT64(var.uhVal.QuadPart, &high, &low); return g_strdup_printf ("%dx%d", high, low); } if (IsEqualGUID (guid, MF_MT_GEOMETRIC_APERTURE) || IsEqualGUID (guid, MF_MT_MINIMUM_DISPLAY_APERTURE) || IsEqualGUID (guid, MF_MT_PAN_SCAN_APERTURE)) { /* FIXME: Not our usecase for now */ return g_strup ("Not parsed"); } switch (var.vt) { case VT_UI4: return g_strdup_printf ("%d", var.ulVal); case VT_UI8: return g_strdup_printf ("%" G_GUINT64_FORMAT, var.uhVal); case VT_R8: return g_strdup_printf ("%f", var.dblVal); case VT_CLSID: return gst_mf_guid_to_string (*var.puuid); case VT_LPWSTR: return g_utf16_to_utf8 ((const gunichar2 *) var.pwszVal, -1, NULL, NULL, NULL); case VT_UNKNOWN: return g_strdup ("IUnknown"); default: return g_strdup_printf ("Unhandled type (vt = %d)", var.vt); } return NULL; } static void gst_mf_dump_attribute_value_by_index (IMFAttributes * attr, const gchar * msg, guint index, GstDebugLevel level, GstDebugCategory * cat, const gchar * file, const gchar * function, gint line) { gchar *guid_name = NULL; gchar *value = NULL; GUID guid = GUID_NULL; HRESULT hr; PROPVARIANT var; PropVariantInit(&var); hr = attr->GetItemByIndex(index, &guid, &var); if (!gst_mf_result (hr)) goto done; guid_name = gst_mf_guid_to_string (guid); if (!guid_name) goto done; value = gst_mf_attribute_value_to_string (guid, var); if (!value) goto done; gst_debug_log (cat, level, file, function, line, NULL, "%s attribute %d, %s: %s", msg ? msg : "", index, guid_name, value); done: PropVariantClear(&var); g_free (guid_name); g_free (value); } void _gst_mf_dump_attributes (IMFAttributes * attr, const gchar * msg, GstDebugLevel level, GstDebugCategory * cat, const gchar * file, const gchar * function, gint line) { #ifndef GST_DISABLE_GST_DEBUG HRESULT hr; UINT32 count = 0, i; if (!attr) return; hr = attr->GetCount (&count); if (!gst_mf_result (hr) || count == 0) return; for (i = 0; i < count; i++) { gst_mf_dump_attribute_value_by_index (attr, msg, i, level, cat, file, function, line); } #endif }