You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1053 lines
36KB

  1. /*
  2. * DXVA2 HW acceleration.
  3. *
  4. * copyright (c) 2010 Laurent Aimar
  5. *
  6. * This file is part of FFmpeg.
  7. *
  8. * FFmpeg is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU Lesser General Public
  10. * License as published by the Free Software Foundation; either
  11. * version 2.1 of the License, or (at your option) any later version.
  12. *
  13. * FFmpeg is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  16. * Lesser General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU Lesser General Public
  19. * License along with FFmpeg; if not, write to the Free Software
  20. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  21. */
  22. #include <assert.h>
  23. #include <string.h>
  24. #include <initguid.h>
  25. #include "libavutil/common.h"
  26. #include "libavutil/log.h"
  27. #include "libavutil/time.h"
  28. #include "avcodec.h"
  29. #include "decode.h"
  30. #include "dxva2_internal.h"
  31. /* define all the GUIDs used directly here,
  32. to avoid problems with inconsistent dxva2api.h versions in mingw-w64 and different MSVC version */
  33. DEFINE_GUID(ff_DXVA2_ModeMPEG2_VLD, 0xee27417f, 0x5e28,0x4e65,0xbe,0xea,0x1d,0x26,0xb5,0x08,0xad,0xc9);
  34. DEFINE_GUID(ff_DXVA2_ModeMPEG2and1_VLD, 0x86695f12, 0x340e,0x4f04,0x9f,0xd3,0x92,0x53,0xdd,0x32,0x74,0x60);
  35. DEFINE_GUID(ff_DXVA2_ModeH264_E, 0x1b81be68, 0xa0c7,0x11d3,0xb9,0x84,0x00,0xc0,0x4f,0x2e,0x73,0xc5);
  36. DEFINE_GUID(ff_DXVA2_ModeH264_F, 0x1b81be69, 0xa0c7,0x11d3,0xb9,0x84,0x00,0xc0,0x4f,0x2e,0x73,0xc5);
  37. DEFINE_GUID(ff_DXVADDI_Intel_ModeH264_E, 0x604F8E68, 0x4951,0x4C54,0x88,0xFE,0xAB,0xD2,0x5C,0x15,0xB3,0xD6);
  38. DEFINE_GUID(ff_DXVA2_ModeVC1_D, 0x1b81beA3, 0xa0c7,0x11d3,0xb9,0x84,0x00,0xc0,0x4f,0x2e,0x73,0xc5);
  39. DEFINE_GUID(ff_DXVA2_ModeVC1_D2010, 0x1b81beA4, 0xa0c7,0x11d3,0xb9,0x84,0x00,0xc0,0x4f,0x2e,0x73,0xc5);
  40. DEFINE_GUID(ff_DXVA2_ModeHEVC_VLD_Main, 0x5b11d51b, 0x2f4c,0x4452,0xbc,0xc3,0x09,0xf2,0xa1,0x16,0x0c,0xc0);
  41. DEFINE_GUID(ff_DXVA2_ModeHEVC_VLD_Main10,0x107af0e0, 0xef1a,0x4d19,0xab,0xa8,0x67,0xa1,0x63,0x07,0x3d,0x13);
  42. DEFINE_GUID(ff_DXVA2_ModeVP9_VLD_Profile0,0x463707f8,0xa1d0,0x4585,0x87,0x6d,0x83,0xaa,0x6d,0x60,0xb8,0x9e);
  43. DEFINE_GUID(ff_DXVA2_ModeVP9_VLD_10bit_Profile2,0xa4c749ef,0x6ecf,0x48aa,0x84,0x48,0x50,0xa7,0xa1,0x16,0x5f,0xf7);
  44. DEFINE_GUID(ff_DXVA2_NoEncrypt, 0x1b81beD0, 0xa0c7,0x11d3,0xb9,0x84,0x00,0xc0,0x4f,0x2e,0x73,0xc5);
  45. DEFINE_GUID(ff_GUID_NULL, 0x00000000, 0x0000,0x0000,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00);
  46. DEFINE_GUID(ff_IID_IDirectXVideoDecoderService, 0xfc51a551,0xd5e7,0x11d9,0xaf,0x55,0x00,0x05,0x4e,0x43,0xff,0x02);
  47. typedef struct dxva_mode {
  48. const GUID *guid;
  49. enum AVCodecID codec;
  50. // List of supported profiles, terminated by a FF_PROFILE_UNKNOWN entry.
  51. // If NULL, don't check profile.
  52. const int *profiles;
  53. } dxva_mode;
  54. static const int prof_mpeg2_main[] = {FF_PROFILE_MPEG2_SIMPLE,
  55. FF_PROFILE_MPEG2_MAIN,
  56. FF_PROFILE_UNKNOWN};
  57. static const int prof_h264_high[] = {FF_PROFILE_H264_CONSTRAINED_BASELINE,
  58. FF_PROFILE_H264_MAIN,
  59. FF_PROFILE_H264_HIGH,
  60. FF_PROFILE_UNKNOWN};
  61. static const int prof_hevc_main[] = {FF_PROFILE_HEVC_MAIN,
  62. FF_PROFILE_UNKNOWN};
  63. static const int prof_hevc_main10[] = {FF_PROFILE_HEVC_MAIN_10,
  64. FF_PROFILE_UNKNOWN};
  65. static const int prof_vp9_profile0[] = {FF_PROFILE_VP9_0,
  66. FF_PROFILE_UNKNOWN};
  67. static const int prof_vp9_profile2[] = {FF_PROFILE_VP9_2,
  68. FF_PROFILE_UNKNOWN};
  69. static const dxva_mode dxva_modes[] = {
  70. /* MPEG-2 */
  71. { &ff_DXVA2_ModeMPEG2_VLD, AV_CODEC_ID_MPEG2VIDEO, prof_mpeg2_main },
  72. { &ff_DXVA2_ModeMPEG2and1_VLD, AV_CODEC_ID_MPEG2VIDEO, prof_mpeg2_main },
  73. /* H.264 */
  74. { &ff_DXVA2_ModeH264_F, AV_CODEC_ID_H264, prof_h264_high },
  75. { &ff_DXVA2_ModeH264_E, AV_CODEC_ID_H264, prof_h264_high },
  76. /* Intel specific H.264 mode */
  77. { &ff_DXVADDI_Intel_ModeH264_E, AV_CODEC_ID_H264, prof_h264_high },
  78. /* VC-1 / WMV3 */
  79. { &ff_DXVA2_ModeVC1_D2010, AV_CODEC_ID_VC1 },
  80. { &ff_DXVA2_ModeVC1_D2010, AV_CODEC_ID_WMV3 },
  81. { &ff_DXVA2_ModeVC1_D, AV_CODEC_ID_VC1 },
  82. { &ff_DXVA2_ModeVC1_D, AV_CODEC_ID_WMV3 },
  83. /* HEVC/H.265 */
  84. { &ff_DXVA2_ModeHEVC_VLD_Main10, AV_CODEC_ID_HEVC, prof_hevc_main10 },
  85. { &ff_DXVA2_ModeHEVC_VLD_Main, AV_CODEC_ID_HEVC, prof_hevc_main },
  86. /* VP8/9 */
  87. { &ff_DXVA2_ModeVP9_VLD_Profile0, AV_CODEC_ID_VP9, prof_vp9_profile0 },
  88. { &ff_DXVA2_ModeVP9_VLD_10bit_Profile2, AV_CODEC_ID_VP9, prof_vp9_profile2 },
  89. { NULL, 0 },
  90. };
  91. static int dxva_get_decoder_configuration(AVCodecContext *avctx,
  92. const void *cfg_list,
  93. unsigned cfg_count)
  94. {
  95. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  96. unsigned i, best_score = 0;
  97. int best_cfg = -1;
  98. for (i = 0; i < cfg_count; i++) {
  99. unsigned score;
  100. UINT ConfigBitstreamRaw;
  101. GUID guidConfigBitstreamEncryption;
  102. #if CONFIG_D3D11VA
  103. if (sctx->pix_fmt == AV_PIX_FMT_D3D11) {
  104. D3D11_VIDEO_DECODER_CONFIG *cfg = &((D3D11_VIDEO_DECODER_CONFIG *)cfg_list)[i];
  105. ConfigBitstreamRaw = cfg->ConfigBitstreamRaw;
  106. guidConfigBitstreamEncryption = cfg->guidConfigBitstreamEncryption;
  107. }
  108. #endif
  109. #if CONFIG_DXVA2
  110. if (sctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  111. DXVA2_ConfigPictureDecode *cfg = &((DXVA2_ConfigPictureDecode *)cfg_list)[i];
  112. ConfigBitstreamRaw = cfg->ConfigBitstreamRaw;
  113. guidConfigBitstreamEncryption = cfg->guidConfigBitstreamEncryption;
  114. }
  115. #endif
  116. if (ConfigBitstreamRaw == 1)
  117. score = 1;
  118. else if (avctx->codec_id == AV_CODEC_ID_H264 && ConfigBitstreamRaw == 2)
  119. score = 2;
  120. else
  121. continue;
  122. if (IsEqualGUID(&guidConfigBitstreamEncryption, &ff_DXVA2_NoEncrypt))
  123. score += 16;
  124. if (score > best_score) {
  125. best_score = score;
  126. best_cfg = i;
  127. }
  128. }
  129. if (!best_score) {
  130. av_log(avctx, AV_LOG_VERBOSE, "No valid decoder configuration available\n");
  131. return AVERROR(EINVAL);
  132. }
  133. return best_cfg;
  134. }
  135. #if CONFIG_D3D11VA
  136. static int d3d11va_validate_output(void *service, GUID guid, const void *surface_format)
  137. {
  138. HRESULT hr;
  139. BOOL is_supported = FALSE;
  140. hr = ID3D11VideoDevice_CheckVideoDecoderFormat((ID3D11VideoDevice *)service,
  141. &guid,
  142. *(DXGI_FORMAT *)surface_format,
  143. &is_supported);
  144. return SUCCEEDED(hr) && is_supported;
  145. }
  146. #endif
  147. #if CONFIG_DXVA2
  148. static int dxva2_validate_output(void *decoder_service, GUID guid, const void *surface_format)
  149. {
  150. HRESULT hr;
  151. int ret = 0;
  152. unsigned j, target_count;
  153. D3DFORMAT *target_list;
  154. hr = IDirectXVideoDecoderService_GetDecoderRenderTargets((IDirectXVideoDecoderService *)decoder_service, &guid, &target_count, &target_list);
  155. if (SUCCEEDED(hr)) {
  156. for (j = 0; j < target_count; j++) {
  157. const D3DFORMAT format = target_list[j];
  158. if (format == *(D3DFORMAT *)surface_format) {
  159. ret = 1;
  160. break;
  161. }
  162. }
  163. CoTaskMemFree(target_list);
  164. }
  165. return ret;
  166. }
  167. #endif
  168. static int dxva_check_codec_compatibility(AVCodecContext *avctx, const dxva_mode *mode)
  169. {
  170. if (mode->codec != avctx->codec_id)
  171. return 0;
  172. if (mode->profiles && !(avctx->hwaccel_flags & AV_HWACCEL_FLAG_ALLOW_PROFILE_MISMATCH)) {
  173. int i, found = 0;
  174. for (i = 0; mode->profiles[i] != FF_PROFILE_UNKNOWN; i++) {
  175. if (avctx->profile == mode->profiles[i]) {
  176. found = 1;
  177. break;
  178. }
  179. }
  180. if (!found)
  181. return 0;
  182. }
  183. return 1;
  184. }
  185. static void dxva_list_guids_debug(AVCodecContext *avctx, void *service,
  186. unsigned guid_count, const GUID *guid_list)
  187. {
  188. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  189. int i;
  190. av_log(avctx, AV_LOG_VERBOSE, "Decoder GUIDs reported as supported:\n");
  191. for (i = 0; i < guid_count; i++) {
  192. const GUID *guid = &guid_list[i];
  193. av_log(avctx, AV_LOG_VERBOSE,
  194. "{%8.8x-%4.4x-%4.4x-%2.2x%2.2x-%2.2x%2.2x%2.2x%2.2x%2.2x%2.2x}",
  195. (unsigned) guid->Data1, guid->Data2, guid->Data3,
  196. guid->Data4[0], guid->Data4[1],
  197. guid->Data4[2], guid->Data4[3],
  198. guid->Data4[4], guid->Data4[5],
  199. guid->Data4[6], guid->Data4[7]);
  200. #if CONFIG_D3D11VA
  201. if (sctx->pix_fmt == AV_PIX_FMT_D3D11) {
  202. DXGI_FORMAT format;
  203. // We don't know the maximum valid DXGI_FORMAT, so use 200 as
  204. // arbitrary upper bound (that could become outdated).
  205. for (format = 0; format < 200; format++) {
  206. if (d3d11va_validate_output(service, *guid, &format))
  207. av_log(avctx, AV_LOG_VERBOSE, " %d", (int)format);
  208. }
  209. }
  210. #endif
  211. #if CONFIG_DXVA2
  212. if (sctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  213. const D3DFORMAT formats[] = {MKTAG('N', 'V', '1', '2'),
  214. MKTAG('P', '0', '1', '0')};
  215. int i;
  216. for (i = 0; i < FF_ARRAY_ELEMS(formats); i++) {
  217. if (dxva2_validate_output(service, *guid, &formats[i]))
  218. av_log(avctx, AV_LOG_VERBOSE, " %d", i);
  219. }
  220. }
  221. #endif
  222. av_log(avctx, AV_LOG_VERBOSE, "\n");
  223. }
  224. }
  225. static int dxva_get_decoder_guid(AVCodecContext *avctx, void *service, void *surface_format,
  226. unsigned guid_count, const GUID *guid_list, GUID *decoder_guid)
  227. {
  228. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  229. unsigned i, j;
  230. dxva_list_guids_debug(avctx, service, guid_count, guid_list);
  231. *decoder_guid = ff_GUID_NULL;
  232. for (i = 0; dxva_modes[i].guid; i++) {
  233. const dxva_mode *mode = &dxva_modes[i];
  234. int validate;
  235. if (!dxva_check_codec_compatibility(avctx, mode))
  236. continue;
  237. for (j = 0; j < guid_count; j++) {
  238. if (IsEqualGUID(mode->guid, &guid_list[j]))
  239. break;
  240. }
  241. if (j == guid_count)
  242. continue;
  243. #if CONFIG_D3D11VA
  244. if (sctx->pix_fmt == AV_PIX_FMT_D3D11)
  245. validate = d3d11va_validate_output(service, *mode->guid, surface_format);
  246. #endif
  247. #if CONFIG_DXVA2
  248. if (sctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD)
  249. validate = dxva2_validate_output(service, *mode->guid, surface_format);
  250. #endif
  251. if (validate) {
  252. *decoder_guid = *mode->guid;
  253. break;
  254. }
  255. }
  256. if (IsEqualGUID(decoder_guid, &ff_GUID_NULL)) {
  257. av_log(avctx, AV_LOG_VERBOSE, "No decoder device for codec found\n");
  258. return AVERROR(EINVAL);
  259. }
  260. if (IsEqualGUID(decoder_guid, &ff_DXVADDI_Intel_ModeH264_E))
  261. sctx->workaround |= FF_DXVA2_WORKAROUND_INTEL_CLEARVIDEO;
  262. return 0;
  263. }
  264. static void bufref_free_interface(void *opaque, uint8_t *data)
  265. {
  266. IUnknown_Release((IUnknown *)opaque);
  267. }
  268. static AVBufferRef *bufref_wrap_interface(IUnknown *iface)
  269. {
  270. return av_buffer_create((uint8_t*)iface, 1, bufref_free_interface, iface, 0);
  271. }
  272. #if CONFIG_DXVA2
  273. static int dxva2_get_decoder_configuration(AVCodecContext *avctx, const GUID *device_guid,
  274. const DXVA2_VideoDesc *desc,
  275. DXVA2_ConfigPictureDecode *config)
  276. {
  277. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  278. unsigned cfg_count;
  279. DXVA2_ConfigPictureDecode *cfg_list;
  280. HRESULT hr;
  281. int ret;
  282. hr = IDirectXVideoDecoderService_GetDecoderConfigurations(sctx->dxva2_service, device_guid, desc, NULL, &cfg_count, &cfg_list);
  283. if (FAILED(hr)) {
  284. av_log(avctx, AV_LOG_ERROR, "Unable to retrieve decoder configurations\n");
  285. return AVERROR(EINVAL);
  286. }
  287. ret = dxva_get_decoder_configuration(avctx, cfg_list, cfg_count);
  288. if (ret >= 0)
  289. *config = cfg_list[ret];
  290. CoTaskMemFree(cfg_list);
  291. return ret;
  292. }
  293. static int dxva2_create_decoder(AVCodecContext *avctx)
  294. {
  295. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  296. GUID *guid_list;
  297. unsigned guid_count;
  298. GUID device_guid;
  299. D3DFORMAT surface_format = avctx->sw_pix_fmt == AV_PIX_FMT_YUV420P10 ?
  300. MKTAG('P', '0', '1', '0') : MKTAG('N', 'V', '1', '2');
  301. DXVA2_VideoDesc desc = { 0 };
  302. DXVA2_ConfigPictureDecode config;
  303. HRESULT hr;
  304. int ret;
  305. HANDLE device_handle;
  306. AVHWFramesContext *frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data;
  307. AVDXVA2FramesContext *frames_hwctx = frames_ctx->hwctx;
  308. AVDXVA2DeviceContext *device_hwctx = frames_ctx->device_ctx->hwctx;
  309. hr = IDirect3DDeviceManager9_OpenDeviceHandle(device_hwctx->devmgr,
  310. &device_handle);
  311. if (FAILED(hr)) {
  312. av_log(avctx, AV_LOG_ERROR, "Failed to open a device handle\n");
  313. goto fail;
  314. }
  315. hr = IDirect3DDeviceManager9_GetVideoService(device_hwctx->devmgr, device_handle,
  316. &ff_IID_IDirectXVideoDecoderService,
  317. (void **)&sctx->dxva2_service);
  318. IDirect3DDeviceManager9_CloseDeviceHandle(device_hwctx->devmgr, device_handle);
  319. if (FAILED(hr)) {
  320. av_log(avctx, AV_LOG_ERROR, "Failed to create IDirectXVideoDecoderService\n");
  321. goto fail;
  322. }
  323. hr = IDirectXVideoDecoderService_GetDecoderDeviceGuids(sctx->dxva2_service, &guid_count, &guid_list);
  324. if (FAILED(hr)) {
  325. av_log(avctx, AV_LOG_ERROR, "Failed to retrieve decoder device GUIDs\n");
  326. goto fail;
  327. }
  328. ret = dxva_get_decoder_guid(avctx, sctx->dxva2_service, &surface_format,
  329. guid_count, guid_list, &device_guid);
  330. CoTaskMemFree(guid_list);
  331. if (ret < 0) {
  332. goto fail;
  333. }
  334. desc.SampleWidth = avctx->coded_width;
  335. desc.SampleHeight = avctx->coded_height;
  336. desc.Format = surface_format;
  337. ret = dxva2_get_decoder_configuration(avctx, &device_guid, &desc, &config);
  338. if (ret < 0) {
  339. goto fail;
  340. }
  341. hr = IDirectXVideoDecoderService_CreateVideoDecoder(sctx->dxva2_service, &device_guid,
  342. &desc, &config, frames_hwctx->surfaces,
  343. frames_hwctx->nb_surfaces, &sctx->dxva2_decoder);
  344. if (FAILED(hr)) {
  345. av_log(avctx, AV_LOG_ERROR, "Failed to create DXVA2 video decoder\n");
  346. goto fail;
  347. }
  348. sctx->dxva2_config = config;
  349. sctx->decoder_ref = bufref_wrap_interface((IUnknown *)sctx->dxva2_decoder);
  350. if (!sctx->decoder_ref)
  351. return AVERROR(ENOMEM);
  352. return 0;
  353. fail:
  354. return AVERROR(EINVAL);
  355. }
  356. #endif
  357. #if CONFIG_D3D11VA
  358. static int d3d11va_get_decoder_configuration(AVCodecContext *avctx,
  359. ID3D11VideoDevice *video_device,
  360. const D3D11_VIDEO_DECODER_DESC *desc,
  361. D3D11_VIDEO_DECODER_CONFIG *config)
  362. {
  363. unsigned cfg_count = 0;
  364. D3D11_VIDEO_DECODER_CONFIG *cfg_list = NULL;
  365. HRESULT hr;
  366. int i, ret;
  367. hr = ID3D11VideoDevice_GetVideoDecoderConfigCount(video_device, desc, &cfg_count);
  368. if (FAILED(hr)) {
  369. av_log(avctx, AV_LOG_ERROR, "Unable to retrieve decoder configurations\n");
  370. return AVERROR(EINVAL);
  371. }
  372. cfg_list = av_malloc_array(cfg_count, sizeof(D3D11_VIDEO_DECODER_CONFIG));
  373. if (cfg_list == NULL)
  374. return AVERROR(ENOMEM);
  375. for (i = 0; i < cfg_count; i++) {
  376. hr = ID3D11VideoDevice_GetVideoDecoderConfig(video_device, desc, i, &cfg_list[i]);
  377. if (FAILED(hr)) {
  378. av_log(avctx, AV_LOG_ERROR, "Unable to retrieve decoder configurations. (hr=0x%lX)\n", hr);
  379. av_free(cfg_list);
  380. return AVERROR(EINVAL);
  381. }
  382. }
  383. ret = dxva_get_decoder_configuration(avctx, cfg_list, cfg_count);
  384. if (ret >= 0)
  385. *config = cfg_list[ret];
  386. av_free(cfg_list);
  387. return ret;
  388. }
  389. static DXGI_FORMAT d3d11va_map_sw_to_hw_format(enum AVPixelFormat pix_fmt)
  390. {
  391. switch (pix_fmt) {
  392. case AV_PIX_FMT_NV12: return DXGI_FORMAT_NV12;
  393. case AV_PIX_FMT_P010: return DXGI_FORMAT_P010;
  394. case AV_PIX_FMT_YUV420P: return DXGI_FORMAT_420_OPAQUE;
  395. default: return DXGI_FORMAT_UNKNOWN;
  396. }
  397. }
  398. static int d3d11va_create_decoder(AVCodecContext *avctx)
  399. {
  400. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  401. GUID *guid_list;
  402. unsigned guid_count, i;
  403. GUID decoder_guid;
  404. D3D11_VIDEO_DECODER_DESC desc = { 0 };
  405. D3D11_VIDEO_DECODER_CONFIG config;
  406. AVHWFramesContext *frames_ctx = (AVHWFramesContext *)avctx->hw_frames_ctx->data;
  407. AVD3D11VADeviceContext *device_hwctx = frames_ctx->device_ctx->hwctx;
  408. AVD3D11VAFramesContext *frames_hwctx = frames_ctx->hwctx;
  409. DXGI_FORMAT surface_format = d3d11va_map_sw_to_hw_format(frames_ctx->sw_format);
  410. D3D11_TEXTURE2D_DESC texdesc;
  411. HRESULT hr;
  412. int ret;
  413. if (!frames_hwctx->texture) {
  414. av_log(avctx, AV_LOG_ERROR, "AVD3D11VAFramesContext.texture not set.\n");
  415. return AVERROR(EINVAL);
  416. }
  417. ID3D11Texture2D_GetDesc(frames_hwctx->texture, &texdesc);
  418. guid_count = ID3D11VideoDevice_GetVideoDecoderProfileCount(device_hwctx->video_device);
  419. guid_list = av_malloc_array(guid_count, sizeof(*guid_list));
  420. if (guid_list == NULL || guid_count == 0) {
  421. av_log(avctx, AV_LOG_ERROR, "Failed to get the decoder GUIDs\n");
  422. av_free(guid_list);
  423. return AVERROR(EINVAL);
  424. }
  425. for (i = 0; i < guid_count; i++) {
  426. hr = ID3D11VideoDevice_GetVideoDecoderProfile(device_hwctx->video_device, i, &guid_list[i]);
  427. if (FAILED(hr)) {
  428. av_log(avctx, AV_LOG_ERROR, "Failed to retrieve decoder GUID %d\n", i);
  429. av_free(guid_list);
  430. return AVERROR(EINVAL);
  431. }
  432. }
  433. ret = dxva_get_decoder_guid(avctx, device_hwctx->video_device, &surface_format,
  434. guid_count, guid_list, &decoder_guid);
  435. av_free(guid_list);
  436. if (ret < 0)
  437. return AVERROR(EINVAL);
  438. desc.SampleWidth = avctx->coded_width;
  439. desc.SampleHeight = avctx->coded_height;
  440. desc.OutputFormat = surface_format;
  441. desc.Guid = decoder_guid;
  442. ret = d3d11va_get_decoder_configuration(avctx, device_hwctx->video_device, &desc, &config);
  443. if (ret < 0)
  444. return AVERROR(EINVAL);
  445. sctx->d3d11_views = av_mallocz_array(texdesc.ArraySize, sizeof(sctx->d3d11_views[0]));
  446. if (!sctx->d3d11_views)
  447. return AVERROR(ENOMEM);
  448. sctx->nb_d3d11_views = texdesc.ArraySize;
  449. for (i = 0; i < sctx->nb_d3d11_views; i++) {
  450. D3D11_VIDEO_DECODER_OUTPUT_VIEW_DESC viewDesc = {
  451. .DecodeProfile = decoder_guid,
  452. .ViewDimension = D3D11_VDOV_DIMENSION_TEXTURE2D,
  453. .Texture2D = {
  454. .ArraySlice = i,
  455. }
  456. };
  457. hr = ID3D11VideoDevice_CreateVideoDecoderOutputView(device_hwctx->video_device,
  458. (ID3D11Resource*) frames_hwctx->texture,
  459. &viewDesc,
  460. (ID3D11VideoDecoderOutputView**) &sctx->d3d11_views[i]);
  461. if (FAILED(hr)) {
  462. av_log(avctx, AV_LOG_ERROR, "Could not create the decoder output view %d\n", i);
  463. return AVERROR_UNKNOWN;
  464. }
  465. }
  466. hr = ID3D11VideoDevice_CreateVideoDecoder(device_hwctx->video_device, &desc,
  467. &config, &sctx->d3d11_decoder);
  468. if (FAILED(hr)) {
  469. av_log(avctx, AV_LOG_ERROR, "Failed to create D3D11VA video decoder\n");
  470. return AVERROR(EINVAL);
  471. }
  472. sctx->d3d11_config = config;
  473. sctx->d3d11_texture = frames_hwctx->texture;
  474. sctx->decoder_ref = bufref_wrap_interface((IUnknown *)sctx->d3d11_decoder);
  475. if (!sctx->decoder_ref)
  476. return AVERROR(ENOMEM);
  477. return 0;
  478. }
  479. #endif
  480. static void ff_dxva2_lock(AVCodecContext *avctx)
  481. {
  482. #if CONFIG_D3D11VA
  483. if (ff_dxva2_is_d3d11(avctx)) {
  484. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  485. AVDXVAContext *ctx = DXVA_CONTEXT(avctx);
  486. if (D3D11VA_CONTEXT(ctx)->context_mutex != INVALID_HANDLE_VALUE)
  487. WaitForSingleObjectEx(D3D11VA_CONTEXT(ctx)->context_mutex, INFINITE, FALSE);
  488. if (sctx->device_ctx) {
  489. AVD3D11VADeviceContext *hwctx = sctx->device_ctx->hwctx;
  490. hwctx->lock(hwctx->lock_ctx);
  491. }
  492. }
  493. #endif
  494. }
  495. static void ff_dxva2_unlock(AVCodecContext *avctx)
  496. {
  497. #if CONFIG_D3D11VA
  498. if (ff_dxva2_is_d3d11(avctx)) {
  499. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  500. AVDXVAContext *ctx = DXVA_CONTEXT(avctx);
  501. if (D3D11VA_CONTEXT(ctx)->context_mutex != INVALID_HANDLE_VALUE)
  502. ReleaseMutex(D3D11VA_CONTEXT(ctx)->context_mutex);
  503. if (sctx->device_ctx) {
  504. AVD3D11VADeviceContext *hwctx = sctx->device_ctx->hwctx;
  505. hwctx->unlock(hwctx->lock_ctx);
  506. }
  507. }
  508. #endif
  509. }
  510. int ff_dxva2_common_frame_params(AVCodecContext *avctx,
  511. AVBufferRef *hw_frames_ctx)
  512. {
  513. AVHWFramesContext *frames_ctx = (AVHWFramesContext *)hw_frames_ctx->data;
  514. AVHWDeviceContext *device_ctx = frames_ctx->device_ctx;
  515. int surface_alignment, num_surfaces;
  516. if (device_ctx->type == AV_HWDEVICE_TYPE_DXVA2) {
  517. frames_ctx->format = AV_PIX_FMT_DXVA2_VLD;
  518. } else if (device_ctx->type == AV_HWDEVICE_TYPE_D3D11VA) {
  519. frames_ctx->format = AV_PIX_FMT_D3D11;
  520. } else {
  521. return AVERROR(EINVAL);
  522. }
  523. /* decoding MPEG-2 requires additional alignment on some Intel GPUs,
  524. but it causes issues for H.264 on certain AMD GPUs..... */
  525. if (avctx->codec_id == AV_CODEC_ID_MPEG2VIDEO)
  526. surface_alignment = 32;
  527. /* the HEVC DXVA2 spec asks for 128 pixel aligned surfaces to ensure
  528. all coding features have enough room to work with */
  529. else if (avctx->codec_id == AV_CODEC_ID_HEVC)
  530. surface_alignment = 128;
  531. else
  532. surface_alignment = 16;
  533. /* 1 base work surface */
  534. num_surfaces = 1;
  535. /* add surfaces based on number of possible refs */
  536. if (avctx->codec_id == AV_CODEC_ID_H264 || avctx->codec_id == AV_CODEC_ID_HEVC)
  537. num_surfaces += 16;
  538. else if (avctx->codec_id == AV_CODEC_ID_VP9)
  539. num_surfaces += 8;
  540. else
  541. num_surfaces += 2;
  542. frames_ctx->sw_format = avctx->sw_pix_fmt == AV_PIX_FMT_YUV420P10 ?
  543. AV_PIX_FMT_P010 : AV_PIX_FMT_NV12;
  544. frames_ctx->width = FFALIGN(avctx->coded_width, surface_alignment);
  545. frames_ctx->height = FFALIGN(avctx->coded_height, surface_alignment);
  546. frames_ctx->initial_pool_size = num_surfaces;
  547. #if CONFIG_DXVA2
  548. if (frames_ctx->format == AV_PIX_FMT_DXVA2_VLD) {
  549. AVDXVA2FramesContext *frames_hwctx = frames_ctx->hwctx;
  550. frames_hwctx->surface_type = DXVA2_VideoDecoderRenderTarget;
  551. }
  552. #endif
  553. #if CONFIG_D3D11VA
  554. if (frames_ctx->format == AV_PIX_FMT_D3D11) {
  555. AVD3D11VAFramesContext *frames_hwctx = frames_ctx->hwctx;
  556. frames_hwctx->BindFlags |= D3D11_BIND_DECODER;
  557. }
  558. #endif
  559. return 0;
  560. }
  561. int ff_dxva2_decode_init(AVCodecContext *avctx)
  562. {
  563. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  564. AVHWFramesContext *frames_ctx;
  565. enum AVHWDeviceType dev_type = avctx->hwaccel->pix_fmt == AV_PIX_FMT_DXVA2_VLD
  566. ? AV_HWDEVICE_TYPE_DXVA2 : AV_HWDEVICE_TYPE_D3D11VA;
  567. int ret = 0;
  568. // Old API.
  569. if (avctx->hwaccel_context)
  570. return 0;
  571. // (avctx->pix_fmt is not updated yet at this point)
  572. sctx->pix_fmt = avctx->hwaccel->pix_fmt;
  573. ret = ff_decode_get_hw_frames_ctx(avctx, dev_type);
  574. if (ret < 0)
  575. return ret;
  576. frames_ctx = (AVHWFramesContext*)avctx->hw_frames_ctx->data;
  577. sctx->device_ctx = frames_ctx->device_ctx;
  578. if (frames_ctx->format != sctx->pix_fmt) {
  579. av_log(avctx, AV_LOG_ERROR, "Invalid pixfmt for hwaccel!\n");
  580. ret = AVERROR(EINVAL);
  581. goto fail;
  582. }
  583. #if CONFIG_D3D11VA
  584. if (sctx->pix_fmt == AV_PIX_FMT_D3D11) {
  585. AVD3D11VADeviceContext *device_hwctx = frames_ctx->device_ctx->hwctx;
  586. AVD3D11VAContext *d3d11_ctx = &sctx->ctx.d3d11va;
  587. ff_dxva2_lock(avctx);
  588. ret = d3d11va_create_decoder(avctx);
  589. ff_dxva2_unlock(avctx);
  590. if (ret < 0)
  591. goto fail;
  592. d3d11_ctx->decoder = sctx->d3d11_decoder;
  593. d3d11_ctx->video_context = device_hwctx->video_context;
  594. d3d11_ctx->cfg = &sctx->d3d11_config;
  595. d3d11_ctx->surface_count = sctx->nb_d3d11_views;
  596. d3d11_ctx->surface = sctx->d3d11_views;
  597. d3d11_ctx->workaround = sctx->workaround;
  598. d3d11_ctx->context_mutex = INVALID_HANDLE_VALUE;
  599. }
  600. #endif
  601. #if CONFIG_DXVA2
  602. if (sctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  603. AVDXVA2FramesContext *frames_hwctx = frames_ctx->hwctx;
  604. struct dxva_context *dxva_ctx = &sctx->ctx.dxva2;
  605. ff_dxva2_lock(avctx);
  606. ret = dxva2_create_decoder(avctx);
  607. ff_dxva2_unlock(avctx);
  608. if (ret < 0)
  609. goto fail;
  610. dxva_ctx->decoder = sctx->dxva2_decoder;
  611. dxva_ctx->cfg = &sctx->dxva2_config;
  612. dxva_ctx->surface = frames_hwctx->surfaces;
  613. dxva_ctx->surface_count = frames_hwctx->nb_surfaces;
  614. dxva_ctx->workaround = sctx->workaround;
  615. }
  616. #endif
  617. return 0;
  618. fail:
  619. ff_dxva2_decode_uninit(avctx);
  620. return ret;
  621. }
  622. int ff_dxva2_decode_uninit(AVCodecContext *avctx)
  623. {
  624. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  625. int i;
  626. av_buffer_unref(&sctx->decoder_ref);
  627. #if CONFIG_D3D11VA
  628. for (i = 0; i < sctx->nb_d3d11_views; i++) {
  629. if (sctx->d3d11_views[i])
  630. ID3D11VideoDecoderOutputView_Release(sctx->d3d11_views[i]);
  631. }
  632. av_freep(&sctx->d3d11_views);
  633. #endif
  634. #if CONFIG_DXVA2
  635. if (sctx->dxva2_service)
  636. IDirectXVideoDecoderService_Release(sctx->dxva2_service);
  637. #endif
  638. return 0;
  639. }
  640. static void *get_surface(const AVCodecContext *avctx, const AVFrame *frame)
  641. {
  642. #if CONFIG_D3D11VA
  643. if (frame->format == AV_PIX_FMT_D3D11) {
  644. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  645. intptr_t index = (intptr_t)frame->data[1];
  646. if (index < 0 || index >= sctx->nb_d3d11_views ||
  647. sctx->d3d11_texture != (ID3D11Texture2D *)frame->data[0]) {
  648. av_log((void *)avctx, AV_LOG_ERROR, "get_buffer frame is invalid!\n");
  649. return NULL;
  650. }
  651. return sctx->d3d11_views[index];
  652. }
  653. #endif
  654. return frame->data[3];
  655. }
  656. unsigned ff_dxva2_get_surface_index(const AVCodecContext *avctx,
  657. const AVDXVAContext *ctx,
  658. const AVFrame *frame)
  659. {
  660. void *surface = get_surface(avctx, frame);
  661. unsigned i;
  662. #if CONFIG_D3D11VA
  663. if (avctx->pix_fmt == AV_PIX_FMT_D3D11)
  664. return (intptr_t)frame->data[1];
  665. if (avctx->pix_fmt == AV_PIX_FMT_D3D11VA_VLD) {
  666. D3D11_VIDEO_DECODER_OUTPUT_VIEW_DESC viewDesc;
  667. ID3D11VideoDecoderOutputView_GetDesc((ID3D11VideoDecoderOutputView*) surface, &viewDesc);
  668. return viewDesc.Texture2D.ArraySlice;
  669. }
  670. #endif
  671. #if CONFIG_DXVA2
  672. for (i = 0; i < DXVA_CONTEXT_COUNT(avctx, ctx); i++) {
  673. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD && ctx->dxva2.surface[i] == surface)
  674. return i;
  675. }
  676. #endif
  677. assert(0);
  678. return 0;
  679. }
  680. int ff_dxva2_commit_buffer(AVCodecContext *avctx,
  681. AVDXVAContext *ctx,
  682. DECODER_BUFFER_DESC *dsc,
  683. unsigned type, const void *data, unsigned size,
  684. unsigned mb_count)
  685. {
  686. void *dxva_data;
  687. unsigned dxva_size;
  688. int result;
  689. HRESULT hr = 0;
  690. #if CONFIG_D3D11VA
  691. if (ff_dxva2_is_d3d11(avctx))
  692. hr = ID3D11VideoContext_GetDecoderBuffer(D3D11VA_CONTEXT(ctx)->video_context,
  693. D3D11VA_CONTEXT(ctx)->decoder,
  694. type,
  695. &dxva_size, &dxva_data);
  696. #endif
  697. #if CONFIG_DXVA2
  698. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD)
  699. hr = IDirectXVideoDecoder_GetBuffer(DXVA2_CONTEXT(ctx)->decoder, type,
  700. &dxva_data, &dxva_size);
  701. #endif
  702. if (FAILED(hr)) {
  703. av_log(avctx, AV_LOG_ERROR, "Failed to get a buffer for %u: 0x%x\n",
  704. type, (unsigned)hr);
  705. return -1;
  706. }
  707. if (size <= dxva_size) {
  708. memcpy(dxva_data, data, size);
  709. #if CONFIG_D3D11VA
  710. if (ff_dxva2_is_d3d11(avctx)) {
  711. D3D11_VIDEO_DECODER_BUFFER_DESC *dsc11 = dsc;
  712. memset(dsc11, 0, sizeof(*dsc11));
  713. dsc11->BufferType = type;
  714. dsc11->DataSize = size;
  715. dsc11->NumMBsInBuffer = mb_count;
  716. }
  717. #endif
  718. #if CONFIG_DXVA2
  719. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  720. DXVA2_DecodeBufferDesc *dsc2 = dsc;
  721. memset(dsc2, 0, sizeof(*dsc2));
  722. dsc2->CompressedBufferType = type;
  723. dsc2->DataSize = size;
  724. dsc2->NumMBsInBuffer = mb_count;
  725. }
  726. #endif
  727. result = 0;
  728. } else {
  729. av_log(avctx, AV_LOG_ERROR, "Buffer for type %u was too small\n", type);
  730. result = -1;
  731. }
  732. #if CONFIG_D3D11VA
  733. if (ff_dxva2_is_d3d11(avctx))
  734. hr = ID3D11VideoContext_ReleaseDecoderBuffer(D3D11VA_CONTEXT(ctx)->video_context, D3D11VA_CONTEXT(ctx)->decoder, type);
  735. #endif
  736. #if CONFIG_DXVA2
  737. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD)
  738. hr = IDirectXVideoDecoder_ReleaseBuffer(DXVA2_CONTEXT(ctx)->decoder, type);
  739. #endif
  740. if (FAILED(hr)) {
  741. av_log(avctx, AV_LOG_ERROR,
  742. "Failed to release buffer type %u: 0x%x\n",
  743. type, (unsigned)hr);
  744. result = -1;
  745. }
  746. return result;
  747. }
  748. static int frame_add_buf(AVFrame *frame, AVBufferRef *ref)
  749. {
  750. int i;
  751. for (i = 0; i < AV_NUM_DATA_POINTERS; i++) {
  752. if (!frame->buf[i]) {
  753. frame->buf[i] = av_buffer_ref(ref);
  754. return frame->buf[i] ? 0 : AVERROR(ENOMEM);
  755. }
  756. }
  757. // For now we expect that the caller does not use more than
  758. // AV_NUM_DATA_POINTERS-1 buffers if the user uses a custom pool.
  759. return AVERROR(EINVAL);
  760. }
  761. int ff_dxva2_common_end_frame(AVCodecContext *avctx, AVFrame *frame,
  762. const void *pp, unsigned pp_size,
  763. const void *qm, unsigned qm_size,
  764. int (*commit_bs_si)(AVCodecContext *,
  765. DECODER_BUFFER_DESC *bs,
  766. DECODER_BUFFER_DESC *slice))
  767. {
  768. AVDXVAContext *ctx = DXVA_CONTEXT(avctx);
  769. unsigned buffer_count = 0;
  770. #if CONFIG_D3D11VA
  771. D3D11_VIDEO_DECODER_BUFFER_DESC buffer11[4];
  772. #endif
  773. #if CONFIG_DXVA2
  774. DXVA2_DecodeBufferDesc buffer2[4];
  775. #endif
  776. DECODER_BUFFER_DESC *buffer = NULL, *buffer_slice = NULL;
  777. int result, runs = 0;
  778. HRESULT hr;
  779. unsigned type;
  780. FFDXVASharedContext *sctx = DXVA_SHARED_CONTEXT(avctx);
  781. if (sctx->decoder_ref) {
  782. result = frame_add_buf(frame, sctx->decoder_ref);
  783. if (result < 0)
  784. return result;
  785. }
  786. do {
  787. ff_dxva2_lock(avctx);
  788. #if CONFIG_D3D11VA
  789. if (ff_dxva2_is_d3d11(avctx))
  790. hr = ID3D11VideoContext_DecoderBeginFrame(D3D11VA_CONTEXT(ctx)->video_context, D3D11VA_CONTEXT(ctx)->decoder,
  791. get_surface(avctx, frame),
  792. 0, NULL);
  793. #endif
  794. #if CONFIG_DXVA2
  795. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD)
  796. hr = IDirectXVideoDecoder_BeginFrame(DXVA2_CONTEXT(ctx)->decoder,
  797. get_surface(avctx, frame),
  798. NULL);
  799. #endif
  800. if (hr != E_PENDING || ++runs > 50)
  801. break;
  802. ff_dxva2_unlock(avctx);
  803. av_usleep(2000);
  804. } while(1);
  805. if (FAILED(hr)) {
  806. av_log(avctx, AV_LOG_ERROR, "Failed to begin frame: 0x%x\n", (unsigned)hr);
  807. ff_dxva2_unlock(avctx);
  808. return -1;
  809. }
  810. #if CONFIG_D3D11VA
  811. if (ff_dxva2_is_d3d11(avctx)) {
  812. buffer = &buffer11[buffer_count];
  813. type = D3D11_VIDEO_DECODER_BUFFER_PICTURE_PARAMETERS;
  814. }
  815. #endif
  816. #if CONFIG_DXVA2
  817. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  818. buffer = &buffer2[buffer_count];
  819. type = DXVA2_PictureParametersBufferType;
  820. }
  821. #endif
  822. result = ff_dxva2_commit_buffer(avctx, ctx, buffer,
  823. type,
  824. pp, pp_size, 0);
  825. if (result) {
  826. av_log(avctx, AV_LOG_ERROR,
  827. "Failed to add picture parameter buffer\n");
  828. goto end;
  829. }
  830. buffer_count++;
  831. if (qm_size > 0) {
  832. #if CONFIG_D3D11VA
  833. if (ff_dxva2_is_d3d11(avctx)) {
  834. buffer = &buffer11[buffer_count];
  835. type = D3D11_VIDEO_DECODER_BUFFER_INVERSE_QUANTIZATION_MATRIX;
  836. }
  837. #endif
  838. #if CONFIG_DXVA2
  839. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  840. buffer = &buffer2[buffer_count];
  841. type = DXVA2_InverseQuantizationMatrixBufferType;
  842. }
  843. #endif
  844. result = ff_dxva2_commit_buffer(avctx, ctx, buffer,
  845. type,
  846. qm, qm_size, 0);
  847. if (result) {
  848. av_log(avctx, AV_LOG_ERROR,
  849. "Failed to add inverse quantization matrix buffer\n");
  850. goto end;
  851. }
  852. buffer_count++;
  853. }
  854. #if CONFIG_D3D11VA
  855. if (ff_dxva2_is_d3d11(avctx)) {
  856. buffer = &buffer11[buffer_count + 0];
  857. buffer_slice = &buffer11[buffer_count + 1];
  858. }
  859. #endif
  860. #if CONFIG_DXVA2
  861. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  862. buffer = &buffer2[buffer_count + 0];
  863. buffer_slice = &buffer2[buffer_count + 1];
  864. }
  865. #endif
  866. result = commit_bs_si(avctx,
  867. buffer,
  868. buffer_slice);
  869. if (result) {
  870. av_log(avctx, AV_LOG_ERROR,
  871. "Failed to add bitstream or slice control buffer\n");
  872. goto end;
  873. }
  874. buffer_count += 2;
  875. /* TODO Film Grain when possible */
  876. assert(buffer_count == 1 + (qm_size > 0) + 2);
  877. #if CONFIG_D3D11VA
  878. if (ff_dxva2_is_d3d11(avctx))
  879. hr = ID3D11VideoContext_SubmitDecoderBuffers(D3D11VA_CONTEXT(ctx)->video_context,
  880. D3D11VA_CONTEXT(ctx)->decoder,
  881. buffer_count, buffer11);
  882. #endif
  883. #if CONFIG_DXVA2
  884. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD) {
  885. DXVA2_DecodeExecuteParams exec = {
  886. .NumCompBuffers = buffer_count,
  887. .pCompressedBuffers = buffer2,
  888. .pExtensionData = NULL,
  889. };
  890. hr = IDirectXVideoDecoder_Execute(DXVA2_CONTEXT(ctx)->decoder, &exec);
  891. }
  892. #endif
  893. if (FAILED(hr)) {
  894. av_log(avctx, AV_LOG_ERROR, "Failed to execute: 0x%x\n", (unsigned)hr);
  895. result = -1;
  896. }
  897. end:
  898. #if CONFIG_D3D11VA
  899. if (ff_dxva2_is_d3d11(avctx))
  900. hr = ID3D11VideoContext_DecoderEndFrame(D3D11VA_CONTEXT(ctx)->video_context, D3D11VA_CONTEXT(ctx)->decoder);
  901. #endif
  902. #if CONFIG_DXVA2
  903. if (avctx->pix_fmt == AV_PIX_FMT_DXVA2_VLD)
  904. hr = IDirectXVideoDecoder_EndFrame(DXVA2_CONTEXT(ctx)->decoder, NULL);
  905. #endif
  906. ff_dxva2_unlock(avctx);
  907. if (FAILED(hr)) {
  908. av_log(avctx, AV_LOG_ERROR, "Failed to end frame: 0x%x\n", (unsigned)hr);
  909. result = -1;
  910. }
  911. return result;
  912. }
  913. int ff_dxva2_is_d3d11(const AVCodecContext *avctx)
  914. {
  915. if (CONFIG_D3D11VA)
  916. return avctx->pix_fmt == AV_PIX_FMT_D3D11VA_VLD ||
  917. avctx->pix_fmt == AV_PIX_FMT_D3D11;
  918. else
  919. return 0;
  920. }