271 lines
9.5 KiB
C++
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
|