// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/base/win/dxgi_device_manager.h"
#include <mfcaptureengine.h>
#include <mferror.h>
#include <mfreadwrite.h>
#include <utility>
#include "base/check.h"
#include "base/logging.h"
#include "media/base/win/mf_helpers.h"
#include "media/base/win/mf_initializer.h"
namespace media {
DXGIDeviceScopedHandle::DXGIDeviceScopedHandle(
IMFDXGIDeviceManager* device_manager)
: device_manager_(device_manager) {}
DXGIDeviceScopedHandle::~DXGIDeviceScopedHandle() {
if (device_handle_ == INVALID_HANDLE_VALUE) {
return;
}
HRESULT hr = device_manager_->CloseDeviceHandle(device_handle_);
LOG_IF(ERROR, FAILED(hr)) << "Failed to close device handle";
device_handle_ = INVALID_HANDLE_VALUE;
}
HRESULT DXGIDeviceScopedHandle::LockDevice(REFIID riid, void** device_out) {
HRESULT hr = S_OK;
if (device_handle_ == INVALID_HANDLE_VALUE) {
hr = device_manager_->OpenDeviceHandle(&device_handle_);
RETURN_ON_HR_FAILURE(
hr, "Failed to open device handle on MF DXGI device manager", hr);
}
// see
// https://docs.microsoft.com/en-us/windows/win32/api/mfobjects/nf-mfobjects-imfdxgidevicemanager-lockdevice
// for details of LockDevice call.
hr = device_manager_->LockDevice(device_handle_, riid, device_out,
/*block=*/FALSE);
return hr;
}
Microsoft::WRL::ComPtr<ID3D11Device> DXGIDeviceScopedHandle::GetDevice() {
HRESULT hr = S_OK;
if (device_handle_ == INVALID_HANDLE_VALUE) {
hr = device_manager_->OpenDeviceHandle(&device_handle_);
RETURN_ON_HR_FAILURE(
hr, "Failed to open device handle on MF DXGI device manager", nullptr);
}
Microsoft::WRL::ComPtr<ID3D11Device> device;
hr = device_manager_->GetVideoService(device_handle_, IID_PPV_ARGS(&device));
RETURN_ON_HR_FAILURE(hr, "Failed to get device from MF DXGI device manager",
nullptr);
return device;
}
scoped_refptr<DXGIDeviceManager> DXGIDeviceManager::Create(CHROME_LUID luid) {
if (!InitializeMediaFoundation()) {
DLOG(ERROR) << "MF DXGI Device Manager is not available";
return nullptr;
}
Microsoft::WRL::ComPtr<IMFDXGIDeviceManager> mf_dxgi_device_manager;
UINT d3d_device_reset_token = 0;
HRESULT hr = MFCreateDXGIDeviceManager(&d3d_device_reset_token,
&mf_dxgi_device_manager);
RETURN_ON_HR_FAILURE(hr, "Failed to create MF DXGI device manager", nullptr);
auto dxgi_device_manager = base::WrapRefCounted(new DXGIDeviceManager(
std::move(mf_dxgi_device_manager), d3d_device_reset_token, luid));
Microsoft::WRL::ComPtr<ID3D11Device> d3d_device;
if (dxgi_device_manager &&
FAILED(dxgi_device_manager->ResetDevice(d3d_device))) {
// If setting a device failed, ensure that an empty scoped_refptr is
// returned as the dxgi_device_manager is not usable without a device.
return nullptr;
}
return dxgi_device_manager;
}
DXGIDeviceManager::DXGIDeviceManager(
Microsoft::WRL::ComPtr<IMFDXGIDeviceManager> mf_dxgi_device_manager,
UINT d3d_device_reset_token,
CHROME_LUID luid)
: mf_dxgi_device_manager_(std::move(mf_dxgi_device_manager)),
d3d_device_reset_token_(d3d_device_reset_token),
luid_(luid) {
DETACH_FROM_SEQUENCE(sequence_checker_);
}
DXGIDeviceManager::~DXGIDeviceManager() = default;
HRESULT DXGIDeviceManager::ResetDevice(
Microsoft::WRL::ComPtr<ID3D11Device>& d3d_device) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
constexpr uint32_t kDeviceFlags =
D3D11_CREATE_DEVICE_VIDEO_SUPPORT | D3D11_CREATE_DEVICE_BGRA_SUPPORT;
const D3D_FEATURE_LEVEL kFeatureLevels[] = {
D3D_FEATURE_LEVEL_11_1, D3D_FEATURE_LEVEL_11_0, D3D_FEATURE_LEVEL_10_1,
D3D_FEATURE_LEVEL_10_0};
Microsoft::WRL::ComPtr<IDXGIAdapter> adapter;
if (luid_.HighPart || luid_.LowPart) {
Microsoft::WRL::ComPtr<IDXGIFactory1> factory;
HRESULT hr = CreateDXGIFactory1(IID_PPV_ARGS(&factory));
RETURN_ON_HR_FAILURE(hr, "Failed to create DXGIFactory1", hr);
Microsoft::WRL::ComPtr<IDXGIAdapter> temp;
for (UINT i = 0; SUCCEEDED(factory->EnumAdapters(i, &temp)); i++) {
DXGI_ADAPTER_DESC desc;
if (SUCCEEDED(temp->GetDesc(&desc)) &&
desc.AdapterLuid.HighPart == luid_.HighPart &&
desc.AdapterLuid.LowPart == luid_.LowPart) {
adapter = temp;
break;
}
}
}
// If adapter is not nullptr, the driver type must be D3D_DRIVER_TYPE_UNKNOWN
// or D3D11CreateDevice will return E_INVALIDARG.
HRESULT hr = D3D11CreateDevice(
adapter.Get(),
adapter ? D3D_DRIVER_TYPE_UNKNOWN : D3D_DRIVER_TYPE_HARDWARE, nullptr,
kDeviceFlags, kFeatureLevels, std::size(kFeatureLevels),
D3D11_SDK_VERSION, &d3d_device, nullptr, nullptr);
RETURN_ON_HR_FAILURE(hr, "D3D11 device creation failed", hr);
RETURN_ON_HR_FAILURE(
hr, media::SetDebugName(d3d_device.Get(), "Media_DXGIDeviceManager"), hr);
// Since FrameServerClient background threads in the video capture process
// call EnqueueSetEvent on Chromium's D3D11 device at the same time that
// Chromium is actively using it in a worker thread, we need to protect access
// via ID3D10Multithreaded::SetMultithreadedProtect. Unfortunately, leaving
// off the CREATE_DEVICE_SINGLETHREADED creation flag is not enough to protect
// us.
Microsoft::WRL::ComPtr<ID3D10Multithread> d3d_device_multithread;
RETURN_IF_FAILED(d3d_device.As(&d3d_device_multithread));
RETURN_IF_FAILED(d3d_device_multithread->SetMultithreadProtected(TRUE));
hr = mf_dxgi_device_manager_->ResetDevice(d3d_device.Get(),
d3d_device_reset_token_);
RETURN_ON_HR_FAILURE(hr, "Failed to reset device on MF DXGI device manager",
hr);
return S_OK;
}
HRESULT DXGIDeviceManager::CheckDeviceRemovedAndGetDevice(
Microsoft::WRL::ComPtr<ID3D11Device>* new_device) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
Microsoft::WRL::ComPtr<ID3D11Device> device = GetDevice();
HRESULT hr = device ? device->GetDeviceRemovedReason() : MF_E_UNEXPECTED;
if (FAILED(hr)) {
HRESULT reset_hr = ResetDevice(device);
if (FAILED(reset_hr)) {
LOG(ERROR) << "Failed to recreate the device: "
<< logging::SystemErrorCodeToString(reset_hr);
if (new_device) {
*new_device = nullptr;
}
return hr;
}
}
if (new_device) {
*new_device = std::move(device);
}
return hr;
}
HRESULT DXGIDeviceManager::RegisterInCaptureEngineAttributes(
IMFAttributes* attributes) {
HRESULT hr = attributes->SetUnknown(MF_CAPTURE_ENGINE_D3D_MANAGER,
mf_dxgi_device_manager_.Get());
RETURN_ON_HR_FAILURE(
hr, "Failed to set MF_CAPTURE_ENGINE_D3D_MANAGER attribute", hr);
return S_OK;
}
HRESULT DXGIDeviceManager::RegisterInSourceReaderAttributes(
IMFAttributes* attributes) {
HRESULT hr = attributes->SetUnknown(MF_SOURCE_READER_D3D_MANAGER,
mf_dxgi_device_manager_.Get());
RETURN_ON_HR_FAILURE(
hr, "Failed to set MF_SOURCE_READER_D3D_MANAGER attribute", hr);
return S_OK;
}
HRESULT DXGIDeviceManager::RegisterWithMediaSource(
Microsoft::WRL::ComPtr<IMFMediaSource> media_source) {
Microsoft::WRL::ComPtr<IMFMediaSourceEx> source_ext;
HRESULT hr = media_source.As(&source_ext);
RETURN_ON_HR_FAILURE(hr, "Failed to query IMFMediaSourceEx", hr);
hr = source_ext->SetD3DManager(mf_dxgi_device_manager_.Get());
RETURN_ON_HR_FAILURE(hr, "Failed to set D3D manager", hr);
return S_OK;
}
Microsoft::WRL::ComPtr<ID3D11Device> DXGIDeviceManager::GetDevice() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DXGIDeviceScopedHandle device_handle(mf_dxgi_device_manager_.Get());
return device_handle.GetDevice();
}
Microsoft::WRL::ComPtr<IMFDXGIDeviceManager>
DXGIDeviceManager::GetMFDXGIDeviceManager() {
return mf_dxgi_device_manager_;
}
void DXGIDeviceManager::OnGpuInfoUpdate(CHROME_LUID luid) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (luid.HighPart != luid_.HighPart || luid.LowPart != luid_.LowPart) {
luid_ = luid;
Microsoft::WRL::ComPtr<ID3D11Device> device;
ResetDevice(device);
}
}
} // namespace media