Files
cbh/ElaWidgetTools/DeveloperComponents/ElaDxgi.cpp
2025-09-20 01:41:33 +08:00

271 lines
9.5 KiB
C++

#include "ElaDxgi.h"
#ifdef Q_OS_WIN
#include <QDateTime>
#include <QDebug>
ElaDxgi::ElaDxgi(QObject *parent) : QObject(parent) {
_pIsInitSuccess = false;
_pIsGrabActive = false;
_pGrabFrameRate = 120;
_pTimeoutMsValue = 50;
_pIsGrabStoped = true;
_pIsGrabCenter = false;
}
ElaDxgi::~ElaDxgi() { releaseInterface(); }
bool ElaDxgi::initialize(int dxID, int outputID) {
_pIsInitSuccess = false;
_pDxDeviceID = dxID;
_pOutputDeviceID = outputID;
releaseInterface();
ID3D11Device *d3dDevice = nullptr;
ID3D11DeviceContext *d3dContext = nullptr;
D3D_FEATURE_LEVEL feat = D3D_FEATURE_LEVEL_11_0;
HRESULT hr = D3D11CreateDevice(nullptr, D3D_DRIVER_TYPE_HARDWARE, nullptr, 0, nullptr, 0, D3D11_SDK_VERSION, &d3dDevice, &feat, &d3dContext);
if (FAILED(hr)) {
_pLastError = "Failed to D3D11CreateDevice ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
if (d3dDevice) {
d3dDevice->Release();
}
if (d3dContext) {
d3dContext->Release();
}
return false;
}
IDXGIFactory *dxgiFactory = nullptr;
CreateDXGIFactory(__uuidof(dxgiFactory), reinterpret_cast<void **>(&dxgiFactory));
IDXGIAdapter *dxgiAdapter = nullptr;
QVector<IDXGIAdapter *> dxgiAdapterVector;
for (uint i = 0; dxgiFactory->EnumAdapters(i, &dxgiAdapter) != DXGI_ERROR_NOT_FOUND; ++i) {
dxgiAdapterVector.append(dxgiAdapter);
}
dxgiFactory->Release();
dxgiAdapter = nullptr;
_pDxDeviceList.clear();
for (int i = 0; i < dxgiAdapterVector.count(); i++) {
IDXGIAdapter *adapt = dxgiAdapterVector.at(i);
DXGI_ADAPTER_DESC desc;
adapt->GetDesc(&desc);
_pDxDeviceList.append(QString::fromWCharArray(desc.Description));
}
if (dxID >= 0 && dxID < dxgiAdapterVector.count()) {
dxgiAdapter = dxgiAdapterVector.at(dxID);
}
if (!dxgiAdapter) {
_pLastError = "Failed to found gpu";
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
for (auto adapter : dxgiAdapterVector) {
adapter->Release();
}
return false;
}
IDXGIOutput *dxgiOutput = nullptr;
QVector<IDXGIOutput *> dxgiOutputVector;
for (uint i = 0; dxgiAdapter->EnumOutputs(i, &dxgiOutput) != DXGI_ERROR_NOT_FOUND; ++i) {
dxgiOutputVector.append(dxgiOutput);
}
for (auto adapter : dxgiAdapterVector) {
adapter->Release();
}
dxgiOutput = nullptr;
_pOutputDeviceList.clear();
for (int i = 0; i < dxgiOutputVector.count(); i++) {
const auto &output = dxgiOutputVector.at(i);
DXGI_OUTPUT_DESC desc;
output->GetDesc(&desc);
_pOutputDeviceList.append(QString::fromWCharArray(desc.DeviceName));
}
if (outputID >= 0 && outputID < dxgiOutputVector.count()) {
dxgiOutput = dxgiOutputVector.at(outputID);
}
if (!dxgiOutput) {
_pLastError = "Failed to found screen!";
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
for (auto output : dxgiOutputVector) {
output->Release();
}
return false;
}
IDXGIOutput6 *dxgiOutput6 = nullptr;
hr = dxgiOutput->QueryInterface(__uuidof(IDXGIOutput6), reinterpret_cast<void **>(&dxgiOutput6));
for (auto output : dxgiOutputVector) {
output->Release();
}
if (FAILED(hr)) {
_pLastError = "Failed to QueryInterface IDXGIOutput6 ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
dxgiOutput6->Release();
return false;
}
UINT supportedFormatsCount = 1; // 支持的格式数
DXGI_FORMAT supportedFormats[1] = {DXGI_FORMAT_B8G8R8A8_UNORM};
hr = dxgiOutput6->DuplicateOutput1(d3dDevice, 0, supportedFormatsCount, supportedFormats, &_duplication);
dxgiOutput6->Release();
if (FAILED(hr)) {
_pLastError = "Failed to DuplicateOutput ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
if (_duplication) {
_duplication->Release();
_duplication = nullptr;
}
return false;
}
_device = d3dDevice;
_context = d3dContext;
_pIsInitSuccess = true;
return true;
}
QImage ElaDxgi::getGrabImage() const {
QImage grabImage(_imageBits, _descWidth, _descHeight, QImage::Format_ARGB32);
if (_pIsGrabCenter) {
return grabImage.copy(
QRect((_descWidth - _pGrabArea.width()) / 2, (_descHeight - _pGrabArea.height()) / 2, _pGrabArea.width(), _pGrabArea.height()));
} else {
return grabImage.copy(_pGrabArea);
}
}
void ElaDxgi::onGrabScreen() {
if (!_duplication || !_device || !_context) {
setIsGrabStoped(true);
return;
}
_pIsGrabStoped = false;
while (_pIsGrabActive) {
IDXGIResource *desktopRes = nullptr;
DXGI_OUTDUPL_FRAME_INFO frameInfo;
while (true) {
if (!_pIsGrabActive) {
setIsGrabStoped(true);
return;
}
_grabTimer.start();
_duplication->ReleaseFrame();
HRESULT hr = _duplication->AcquireNextFrame(_pTimeoutMsValue, &frameInfo, &desktopRes);
if (FAILED(hr)) {
if (hr != DXGI_ERROR_WAIT_TIMEOUT) {
if (desktopRes) {
desktopRes->Release();
desktopRes = nullptr;
}
while (true) {
if (!_duplication || !_device || !_context) {
setIsGrabStoped(true);
return;
}
if (initialize(_pDxDeviceID, _pOutputDeviceID)) {
break;
}
}
}
continue;
}
if (frameInfo.LastPresentTime.QuadPart) {
break;
}
}
D3D11_TEXTURE2D_DESC desc;
ID3D11Texture2D *textrueRes = nullptr;
HRESULT hr = desktopRes->QueryInterface(__uuidof(ID3D11Texture2D), reinterpret_cast<void **>(&textrueRes));
desktopRes->Release();
if (FAILED(hr)) {
qDebug() << "Failed to ID3D11Texture2D result =" << QString::number(uint(hr), 16);
continue;
}
textrueRes->GetDesc(&desc);
D3D11_TEXTURE2D_DESC texDesc;
ZeroMemory(&texDesc, sizeof(texDesc));
texDesc.Width = desc.Width;
texDesc.Height = desc.Height;
texDesc.MipLevels = 1;
texDesc.ArraySize = 1;
texDesc.SampleDesc.Count = 1;
texDesc.SampleDesc.Quality = 0;
texDesc.Usage = D3D11_USAGE_STAGING;
texDesc.Format = desc.Format;
texDesc.BindFlags = 0;
texDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
texDesc.MiscFlags = 0;
_device->CreateTexture2D(&texDesc, nullptr, &_texture);
_context->CopyResource(_texture, textrueRes);
IDXGISurface1 *surface = nullptr;
hr = _texture->QueryInterface(__uuidof(IDXGISurface1), reinterpret_cast<void **>(&surface));
if (FAILED(hr)) {
qDebug() << "Failed to QueryInterface IDXGISurface1 ErrorCode =" << QString::number(uint(hr), 16);
continue;
}
DXGI_MAPPED_RECT map;
surface->Map(&map, DXGI_MAP_READ);
_imageBits = static_cast<uchar *>(map.pBits);
_descWidth = desc.Width;
_descHeight = desc.Height;
surface->Unmap();
surface->Release();
_texture->Release();
QImage grabImage(_imageBits, _descWidth, _descHeight, QImage::Format_ARGB32);
if (_pIsGrabCenter) {
Q_EMIT grabScreenOver(std::move(grabImage.copy(
QRect((_descWidth - _pGrabArea.width()) / 2, (_descHeight - _pGrabArea.height()) / 2, _pGrabArea.width(), _pGrabArea.height()))));
} else {
Q_EMIT grabScreenOver(std::move(grabImage.copy(_pGrabArea)));
}
if (_lastGrabTime == 0) {
_lastGrabTime = _grabTimer.elapsed(); // 毫秒
} else {
_lastGrabTime = (_grabTimer.elapsed() + _lastGrabTime) / 2;
}
_cpuSleepTime = (1000 - _lastGrabTime * _pGrabFrameRate) / _pGrabFrameRate;
cpuSleep(_cpuSleepTime * 1000);
// qDebug() << _cpuSleepTime << _lastGrabTime;
}
setIsGrabStoped(true);
}
void ElaDxgi::releaseInterface() {
if (_duplication) {
_duplication->Release();
_duplication = nullptr;
}
if (_device) {
_device->Release();
_device = nullptr;
}
if (_context) {
_context->Release();
_context = nullptr;
}
}
void ElaDxgi::cpuSleep(qint64 usec) {
if (usec <= 0) {
return;
}
LARGE_INTEGER cpuFerq;
LARGE_INTEGER startTime;
LARGE_INTEGER endTime;
QueryPerformanceFrequency(&cpuFerq);
QueryPerformanceCounter(&startTime);
while (true) {
QueryPerformanceCounter(&endTime);
if (((endTime.QuadPart - startTime.QuadPart) * 1000000) / cpuFerq.QuadPart > usec) {
break;
}
}
}
#endif