/** * FreeRDP: A Remote Desktop Protocol Client * FreeRDP Windows Server * * Copyright 2012 Corey Clayton * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "wf_interface.h" #ifdef WITH_WIN8 #define CINTERFACE #include #include #include #include "wf_dxgi.h" /* Driver types supported */ D3D_DRIVER_TYPE DriverTypes[] = { D3D_DRIVER_TYPE_HARDWARE, D3D_DRIVER_TYPE_WARP, D3D_DRIVER_TYPE_REFERENCE, }; UINT NumDriverTypes = ARRAYSIZE(DriverTypes); /* Feature levels supported */ D3D_FEATURE_LEVEL FeatureLevels[] = { D3D_FEATURE_LEVEL_11_0, D3D_FEATURE_LEVEL_10_1, D3D_FEATURE_LEVEL_10_0, D3D_FEATURE_LEVEL_9_1 }; UINT NumFeatureLevels = ARRAYSIZE(FeatureLevels); D3D_FEATURE_LEVEL FeatureLevel; ID3D11Device* gDevice = NULL; ID3D11DeviceContext* gContext = NULL; IDXGIOutputDuplication* gOutputDuplication = NULL; ID3D11Texture2D* gAcquiredDesktopImage = NULL; IDXGISurface* surf; ID3D11Texture2D* sStage; DXGI_OUTDUPL_FRAME_INFO FrameInfo; int wf_dxgi_init(wfInfo* context) { HRESULT status; UINT dTop, i = 0; DXGI_OUTPUT_DESC desc; IDXGIOutput * pOutput; UINT DriverTypeIndex; IDXGIDevice* DxgiDevice = NULL; IDXGIAdapter* DxgiAdapter = NULL; IDXGIOutput* DxgiOutput = NULL; IDXGIOutput1* DxgiOutput1 = NULL; gAcquiredDesktopImage = NULL; for (DriverTypeIndex = 0; DriverTypeIndex < NumDriverTypes; ++DriverTypeIndex) { status = D3D11CreateDevice(NULL, DriverTypes[DriverTypeIndex], NULL, D3D11_CREATE_DEVICE_DEBUG, FeatureLevels, NumFeatureLevels, D3D11_SDK_VERSION, &gDevice, &FeatureLevel, &gContext); if (SUCCEEDED(status)) break; } if (FAILED(status)) { _tprintf(_T("Failed to create device in InitializeDx\n")); return 1; } status = gDevice->lpVtbl->QueryInterface(gDevice, &IID_IDXGIDevice, (void**) &DxgiDevice); if (FAILED(status)) { _tprintf(_T("Failed to get QI for DXGI Device\n")); return 1; } status = DxgiDevice->lpVtbl->GetParent(DxgiDevice, &IID_IDXGIAdapter, (void**) &DxgiAdapter); DxgiDevice->lpVtbl->Release(DxgiDevice); DxgiDevice = NULL; if (FAILED(status)) { _tprintf(_T("Failed to get parent DXGI Adapter\n")); return 1; } ZeroMemory(&desc, sizeof(desc)); pOutput = NULL; while (DxgiAdapter->lpVtbl->EnumOutputs(DxgiAdapter, i, &pOutput) != DXGI_ERROR_NOT_FOUND) { DXGI_OUTPUT_DESC* pDesc = &desc; status = pOutput->lpVtbl->GetDesc(pOutput, pDesc); if (FAILED(status)) { _tprintf(_T("Failed to get description\n")); return 1; } _tprintf(_T("Output %d: [%s] [%d]\n"), i, pDesc->DeviceName, pDesc->AttachedToDesktop); if (pDesc->AttachedToDesktop) dTop = i; pOutput->lpVtbl->Release(pOutput); ++i; } dTop = 0; status = DxgiAdapter->lpVtbl->EnumOutputs(DxgiAdapter, dTop, &DxgiOutput); DxgiAdapter->lpVtbl->Release(DxgiAdapter); DxgiAdapter = NULL; if (FAILED(status)) { _tprintf(_T("Failed to get output\n")); return 1; } status = DxgiOutput->lpVtbl->QueryInterface(DxgiOutput, &IID_IDXGIOutput1, (void**) &DxgiOutput1); DxgiOutput->lpVtbl->Release(DxgiOutput); DxgiOutput = NULL; if (FAILED(status)) { _tprintf(_T("Failed to get IDXGIOutput1\n")); return 1; } status = DxgiOutput1->lpVtbl->DuplicateOutput(DxgiOutput1, (IUnknown*)gDevice, &gOutputDuplication); DxgiOutput1->lpVtbl->Release(DxgiOutput1); DxgiOutput1 = NULL; if (FAILED(status)) { if (status == DXGI_ERROR_NOT_CURRENTLY_AVAILABLE) { _tprintf(_T("There is already the maximum number of applications using the Desktop Duplication API running, please close one of those applications and then try again.\n")); return 1; } _tprintf(_T("Failed to get duplicate output\n")); return 1; } return 0; } int wf_dxgi_cleanup(wfInfo* wfi) { if (wfi->framesWaiting > 0) { wf_dxgi_releasePixelData(wfi); } if (gAcquiredDesktopImage) { gAcquiredDesktopImage->lpVtbl->Release(gAcquiredDesktopImage); gAcquiredDesktopImage = NULL; } if (gOutputDuplication) { gOutputDuplication->lpVtbl->Release(gOutputDuplication); gOutputDuplication = NULL; } if(gContext) { gContext->lpVtbl->Release(gContext); gContext = NULL; } if(gDevice) { gDevice->lpVtbl->Release(gDevice); gDevice = NULL; } return 0; } int wf_dxgi_nextFrame(wfInfo* wfi, UINT timeout) { HRESULT status; UINT i = 0; UINT DataBufferSize = 0; BYTE* DataBuffer = NULL; IDXGIResource* DesktopResource = NULL; if (wfi->framesWaiting > 0) { wf_dxgi_releasePixelData(wfi); } if (gAcquiredDesktopImage) { gAcquiredDesktopImage->lpVtbl->Release(gAcquiredDesktopImage); gAcquiredDesktopImage = NULL; } status = gOutputDuplication->lpVtbl->AcquireNextFrame(gOutputDuplication, timeout, &FrameInfo, &DesktopResource); if (status == DXGI_ERROR_WAIT_TIMEOUT) { return 1; } if (FAILED(status)) { _tprintf(_T("Failed to acquire next frame\n")); status = gOutputDuplication->lpVtbl->ReleaseFrame(gOutputDuplication); if (FAILED(status)) { _tprintf(_T("Failed to release frame\n")); } return 1; } status = DesktopResource->lpVtbl->QueryInterface(DesktopResource, &IID_ID3D11Texture2D, (void**) &gAcquiredDesktopImage); DesktopResource->lpVtbl->Release(DesktopResource); DesktopResource = NULL; if (FAILED(status)) { return 1; } wfi->framesWaiting = FrameInfo.AccumulatedFrames; return 0; } int wf_dxgi_getPixelData(wfInfo* context, BYTE** data, int* pitch, RECT* invalid) { HRESULT status; D3D11_BOX Box; DXGI_MAPPED_RECT mappedRect; D3D11_TEXTURE2D_DESC tDesc; tDesc.Width = (invalid->right - invalid->left); tDesc.Height = (invalid->bottom - invalid->top); tDesc.MipLevels = 1; tDesc.ArraySize = 1; tDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; tDesc.SampleDesc.Count = 1; tDesc.SampleDesc.Quality = 0; tDesc.Usage = D3D11_USAGE_STAGING; tDesc.BindFlags = 0; tDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ; tDesc.MiscFlags = 0; Box.top = invalid->top; Box.left = invalid->left; Box.right = invalid->right; Box.bottom = invalid->bottom; Box.front = 0; Box.back = 1; status = gDevice->lpVtbl->CreateTexture2D(gDevice, &tDesc, NULL, &sStage); if (FAILED(status)) { _tprintf(_T("Failed to create staging surface\n")); exit(1); return 1; } gContext->lpVtbl->CopySubresourceRegion(gContext, (ID3D11Resource*) sStage, 0,0,0,0, (ID3D11Resource*) gAcquiredDesktopImage, 0, &Box); status = sStage->lpVtbl->QueryInterface(sStage, &IID_IDXGISurface, (void**) &surf); if (FAILED(status)) { _tprintf(_T("Failed to QI staging surface\n")); exit(1); return 1; } surf->lpVtbl->Map(surf, &mappedRect, DXGI_MAP_READ); if (FAILED(status)) { _tprintf(_T("Failed to map staging surface\n")); exit(1); return 1; } *data = mappedRect.pBits; *pitch = mappedRect.Pitch; return 0; } int wf_dxgi_releasePixelData(wfInfo* wfi) { HRESULT status; surf->lpVtbl->Unmap(surf); surf->lpVtbl->Release(surf); surf = NULL; sStage->lpVtbl->Release(sStage); sStage = NULL; status = gOutputDuplication->lpVtbl->ReleaseFrame(gOutputDuplication); if (FAILED(status)) { _tprintf(_T("Failed to release frame\n")); return 1; } wfi->framesWaiting = 0; return 0; } int wf_dxgi_getInvalidRegion(RECT* invalid) { UINT i; HRESULT status; UINT dirty; UINT BufSize; RECT* pRect; BYTE* DirtyRects; UINT DataBufferSize = 0; BYTE* DataBuffer = NULL; if (FrameInfo.AccumulatedFrames == 0) { return 1; } if (FrameInfo.TotalMetadataBufferSize) { if (FrameInfo.TotalMetadataBufferSize > DataBufferSize) { if (DataBuffer) { free(DataBuffer); DataBuffer = NULL; } DataBuffer = (BYTE*) malloc(FrameInfo.TotalMetadataBufferSize); if (!DataBuffer) { DataBufferSize = 0; _tprintf(_T("Failed to allocate memory for metadata\n")); exit(1); } DataBufferSize = FrameInfo.TotalMetadataBufferSize; } BufSize = FrameInfo.TotalMetadataBufferSize; status = gOutputDuplication->lpVtbl->GetFrameMoveRects(gOutputDuplication, BufSize, (DXGI_OUTDUPL_MOVE_RECT*) DataBuffer, &BufSize); if (FAILED(status)) { _tprintf(_T("Failed to get frame move rects\n")); return 1; } DirtyRects = DataBuffer + BufSize; BufSize = FrameInfo.TotalMetadataBufferSize - BufSize; status = gOutputDuplication->lpVtbl->GetFrameDirtyRects(gOutputDuplication, BufSize, (RECT*) DirtyRects, &BufSize); if (FAILED(status)) { _tprintf(_T("Failed to get frame dirty rects\n")); return 1; } dirty = BufSize / sizeof(RECT); pRect = (RECT*) DirtyRects; for(i = 0; i