| // Copyright 2016 The SwiftShader Authors. All Rights Reserved. |
| // |
| // 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. |
| |
| #include "FrameBufferGDI.hpp" |
| |
| #include "Common/Debug.hpp" |
| |
| namespace sw |
| { |
| extern bool forceWindowed; |
| |
| FrameBufferGDI::FrameBufferGDI(HWND windowHandle, int width, int height, bool fullscreen, bool topLeftOrigin) : FrameBufferWin(windowHandle, width, height, fullscreen, topLeftOrigin) |
| { |
| if(!windowed) |
| { |
| SetWindowPos(windowHandle, HWND_TOPMOST, 0, 0, width, height, SWP_SHOWWINDOW); |
| |
| DEVMODE deviceMode; |
| deviceMode.dmSize = sizeof(DEVMODE); |
| deviceMode.dmPelsWidth= width; |
| deviceMode.dmPelsHeight = height; |
| deviceMode.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT; |
| |
| ChangeDisplaySettings(&deviceMode, CDS_FULLSCREEN); |
| } |
| |
| init(this->windowHandle); |
| |
| format = FORMAT_X8R8G8B8; |
| } |
| |
| FrameBufferGDI::~FrameBufferGDI() |
| { |
| release(); |
| |
| if(!windowed) |
| { |
| ChangeDisplaySettings(0, 0); |
| |
| RECT clientRect; |
| RECT windowRect; |
| GetClientRect(windowHandle, &clientRect); |
| GetWindowRect(windowHandle, &windowRect); |
| int windowWidth = width + (windowRect.right - windowRect.left) - (clientRect.right - clientRect.left); |
| int windowHeight = height + (windowRect.bottom - windowRect.top) - (clientRect.bottom - clientRect.top); |
| int desktopWidth = GetSystemMetrics(SM_CXSCREEN); |
| int desktopHeight = GetSystemMetrics(SM_CYSCREEN); |
| SetWindowPos(windowHandle, HWND_TOP, desktopWidth / 2 - windowWidth / 2, desktopHeight / 2 - windowHeight / 2, windowWidth, windowHeight, SWP_SHOWWINDOW); |
| } |
| } |
| |
| void *FrameBufferGDI::lock() |
| { |
| stride = width * 4; |
| |
| return framebuffer; |
| } |
| |
| void FrameBufferGDI::unlock() |
| { |
| } |
| |
| void FrameBufferGDI::flip(sw::Surface *source) |
| { |
| blit(source, nullptr, nullptr); |
| } |
| |
| void FrameBufferGDI::blit(sw::Surface *source, const Rect *sourceRect, const Rect *destRect) |
| { |
| copy(source); |
| |
| int sourceLeft = sourceRect ? sourceRect->x0 : 0; |
| int sourceTop = sourceRect ? sourceRect->y0 : 0; |
| int sourceWidth = sourceRect ? sourceRect->x1 - sourceRect->x0 : width; |
| int sourceHeight = sourceRect ? sourceRect->y1 - sourceRect->y0 : height; |
| int destLeft = destRect ? destRect->x0 : 0; |
| int destTop = destRect ? destRect->y0 : 0; |
| int destWidth = destRect ? destRect->x1 - destRect->x0 : bounds.right - bounds.left; |
| int destHeight = destRect ? destRect->y1 - destRect->y0 : bounds.bottom - bounds.top; |
| |
| StretchBlt(windowContext, destLeft, destTop, destWidth, destHeight, bitmapContext, sourceLeft, sourceTop, sourceWidth, sourceHeight, SRCCOPY); |
| } |
| |
| void FrameBufferGDI::flip(HWND windowOverride, sw::Surface *source) |
| { |
| blit(windowOverride, source, nullptr, nullptr); |
| } |
| |
| void FrameBufferGDI::blit(HWND windowOverride, sw::Surface *source, const Rect *sourceRect, const Rect *destRect) |
| { |
| if(windowed && windowOverride != 0 && windowOverride != bitmapWindow) |
| { |
| release(); |
| init(windowOverride); |
| } |
| |
| blit(source, sourceRect, destRect); |
| } |
| |
| void FrameBufferGDI::setGammaRamp(GammaRamp *gammaRamp, bool calibrate) |
| { |
| SetDeviceGammaRamp(windowContext, gammaRamp); |
| } |
| |
| void FrameBufferGDI::getGammaRamp(GammaRamp *gammaRamp) |
| { |
| GetDeviceGammaRamp(windowContext, gammaRamp); |
| } |
| |
| void FrameBufferGDI::screenshot(void *destBuffer) |
| { |
| UNIMPLEMENTED(); |
| } |
| |
| bool FrameBufferGDI::getScanline(bool &inVerticalBlank, unsigned int &scanline) |
| { |
| UNIMPLEMENTED(); |
| |
| return false; |
| } |
| |
| void FrameBufferGDI::init(HWND window) |
| { |
| bitmapWindow = window; |
| |
| windowContext = GetDC(window); |
| bitmapContext = CreateCompatibleDC(windowContext); |
| |
| BITMAPINFO bitmapInfo; |
| memset(&bitmapInfo, 0, sizeof(BITMAPINFO)); |
| bitmapInfo.bmiHeader.biSize = sizeof(BITMAPINFO); |
| bitmapInfo.bmiHeader.biBitCount = 32; |
| bitmapInfo.bmiHeader.biPlanes = 1; |
| bitmapInfo.bmiHeader.biHeight = -height; |
| bitmapInfo.bmiHeader.biWidth = width; |
| bitmapInfo.bmiHeader.biCompression = BI_RGB; |
| |
| bitmap = CreateDIBSection(bitmapContext, &bitmapInfo, DIB_RGB_COLORS, &framebuffer, 0, 0); |
| SelectObject(bitmapContext, bitmap); |
| |
| updateBounds(window); |
| } |
| |
| void FrameBufferGDI::release() |
| { |
| SelectObject(bitmapContext, 0); |
| DeleteObject(bitmap); |
| ReleaseDC(bitmapWindow, windowContext); |
| DeleteDC(bitmapContext); |
| } |
| } |