2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2012 Dolphin Emulator Project
|
2015-05-18 01:08:10 +02:00
|
|
|
// Licensed under GPLv2+
|
2013-04-17 23:43:35 -04:00
|
|
|
// Refer to the license.txt file included.
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-03-12 15:33:41 -04:00
|
|
|
#include <string>
|
|
|
|
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Core/Host.h"
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-11-01 10:33:02 -04:00
|
|
|
#include "VideoBackends/OGL/GLInterface/WGL.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
|
|
|
|
#include "VideoCommon/RenderBase.h"
|
|
|
|
#include "VideoCommon/VertexShaderManager.h"
|
|
|
|
#include "VideoCommon/VideoConfig.h"
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-03-09 21:14:26 +01:00
|
|
|
static HDC hDC = nullptr; // Private GDI Device Context
|
|
|
|
static HGLRC hRC = nullptr; // Permanent Rendering Context
|
|
|
|
static HINSTANCE dllHandle = nullptr; // Handle to OpenGL32.dll
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2013-12-31 00:03:49 +01:00
|
|
|
// typedef from wglext.h
|
|
|
|
typedef BOOL(WINAPI * PFNWGLSWAPINTERVALEXTPROC) (int interval);
|
2014-03-09 21:14:26 +01:00
|
|
|
static PFNWGLSWAPINTERVALEXTPROC wglSwapIntervalEXT = nullptr;
|
2013-12-31 00:03:49 +01:00
|
|
|
|
2013-01-24 10:31:08 -06:00
|
|
|
void cInterfaceWGL::SwapInterval(int Interval)
|
|
|
|
{
|
2013-12-31 00:03:49 +01:00
|
|
|
if (wglSwapIntervalEXT)
|
2013-01-24 10:31:08 -06:00
|
|
|
wglSwapIntervalEXT(Interval);
|
|
|
|
else
|
|
|
|
ERROR_LOG(VIDEO, "No support for SwapInterval (framerate clamped to monitor refresh rate).");
|
|
|
|
}
|
2012-12-26 00:34:09 -06:00
|
|
|
void cInterfaceWGL::Swap()
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
|
|
|
SwapBuffers(hDC);
|
|
|
|
}
|
|
|
|
|
2014-03-17 18:17:12 -05:00
|
|
|
void* cInterfaceWGL::GetFuncAddress(const std::string& name)
|
2013-12-30 07:22:50 -06:00
|
|
|
{
|
2014-01-18 14:10:24 +00:00
|
|
|
void* func = (void*)wglGetProcAddress((LPCSTR)name.c_str());
|
2014-03-09 21:14:26 +01:00
|
|
|
if (func == nullptr)
|
2014-01-18 14:10:24 +00:00
|
|
|
func = (void*)GetProcAddress(dllHandle, (LPCSTR)name.c_str());
|
2014-03-09 21:14:26 +01:00
|
|
|
return func;
|
2013-12-30 07:22:50 -06:00
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// Draw messages on top of the screen
|
|
|
|
bool cInterfaceWGL::PeekMessages()
|
|
|
|
{
|
|
|
|
// TODO: peekmessage
|
|
|
|
MSG msg;
|
|
|
|
while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
|
|
|
|
{
|
|
|
|
if (msg.message == WM_QUIT)
|
|
|
|
return FALSE;
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessage(&msg);
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create rendering window.
|
2014-02-16 23:51:41 -05:00
|
|
|
// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
|
2014-08-06 00:44:21 -04:00
|
|
|
bool cInterfaceWGL::Create(void *window_handle)
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2014-06-15 00:23:43 +02:00
|
|
|
if (window_handle == nullptr)
|
|
|
|
return false;
|
|
|
|
|
2014-08-05 02:51:32 -04:00
|
|
|
HWND window_handle_reified = reinterpret_cast<HWND>(window_handle);
|
|
|
|
RECT window_rect = {0};
|
|
|
|
|
|
|
|
if (!GetClientRect(window_handle_reified, &window_rect))
|
|
|
|
return false;
|
2012-12-17 15:01:52 -06:00
|
|
|
|
|
|
|
// Control window size and picture scaling
|
2014-08-05 02:51:32 -04:00
|
|
|
int twidth = (window_rect.right - window_rect.left);
|
|
|
|
int theight = (window_rect.bottom - window_rect.top);
|
|
|
|
s_backbuffer_width = twidth;
|
|
|
|
s_backbuffer_height = theight;
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-08-05 02:51:32 -04:00
|
|
|
m_window_handle = window_handle_reified;
|
2014-06-15 00:23:43 +02:00
|
|
|
|
2014-01-18 14:10:24 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
dllHandle = LoadLibrary(TEXT("OpenGL32.dll"));
|
|
|
|
#endif
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
PIXELFORMATDESCRIPTOR pfd = // pfd Tells Windows How We Want Things To Be
|
|
|
|
{
|
|
|
|
sizeof(PIXELFORMATDESCRIPTOR), // Size Of This Pixel Format Descriptor
|
|
|
|
1, // Version Number
|
|
|
|
PFD_DRAW_TO_WINDOW | // Format Must Support Window
|
|
|
|
PFD_SUPPORT_OPENGL | // Format Must Support OpenGL
|
|
|
|
PFD_DOUBLEBUFFER, // Must Support Double Buffering
|
|
|
|
PFD_TYPE_RGBA, // Request An RGBA Format
|
|
|
|
32, // Select Our Color Depth
|
|
|
|
0, 0, 0, 0, 0, 0, // Color Bits Ignored
|
|
|
|
0, // 8bit Alpha Buffer
|
|
|
|
0, // Shift Bit Ignored
|
|
|
|
0, // No Accumulation Buffer
|
|
|
|
0, 0, 0, 0, // Accumulation Bits Ignored
|
2013-10-29 01:23:17 -04:00
|
|
|
0, // 0Bit Z-Buffer (Depth Buffer)
|
2013-10-29 19:19:56 +01:00
|
|
|
0, // 0bit Stencil Buffer
|
2012-12-17 15:01:52 -06:00
|
|
|
0, // No Auxiliary Buffer
|
|
|
|
PFD_MAIN_PLANE, // Main Drawing Layer
|
|
|
|
0, // Reserved
|
|
|
|
0, 0, 0 // Layer Masks Ignored
|
|
|
|
};
|
|
|
|
|
2013-12-31 00:03:49 +01:00
|
|
|
int PixelFormat; // Holds The Results After Searching For A Match
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-08-30 17:01:19 -04:00
|
|
|
if (!(hDC = GetDC(window_handle_reified)))
|
|
|
|
{
|
2012-12-17 15:01:52 -06:00
|
|
|
PanicAlert("(1) Can't create an OpenGL Device context. Fail.");
|
|
|
|
return false;
|
|
|
|
}
|
2014-08-30 17:01:19 -04:00
|
|
|
|
|
|
|
if (!(PixelFormat = ChoosePixelFormat(hDC, &pfd)))
|
|
|
|
{
|
2012-12-17 15:01:52 -06:00
|
|
|
PanicAlert("(2) Can't find a suitable PixelFormat.");
|
|
|
|
return false;
|
|
|
|
}
|
2014-08-30 17:01:19 -04:00
|
|
|
|
|
|
|
if (!SetPixelFormat(hDC, PixelFormat, &pfd))
|
|
|
|
{
|
2012-12-17 15:01:52 -06:00
|
|
|
PanicAlert("(3) Can't set the PixelFormat.");
|
|
|
|
return false;
|
|
|
|
}
|
2014-08-30 17:01:19 -04:00
|
|
|
|
|
|
|
if (!(hRC = wglCreateContext(hDC)))
|
|
|
|
{
|
2012-12-17 15:01:52 -06:00
|
|
|
PanicAlert("(4) Can't create an OpenGL rendering context.");
|
|
|
|
return false;
|
|
|
|
}
|
2014-01-23 08:04:33 -06:00
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cInterfaceWGL::MakeCurrent()
|
|
|
|
{
|
2014-06-27 23:50:51 +02:00
|
|
|
bool success = wglMakeCurrent(hDC, hRC) ? true : false;
|
2014-02-24 12:45:02 +01:00
|
|
|
if (success)
|
|
|
|
{
|
|
|
|
// Grab the swap interval function pointer
|
|
|
|
wglSwapIntervalEXT = (PFNWGLSWAPINTERVALEXTPROC)GLInterface->GetFuncAddress("wglSwapIntervalEXT");
|
|
|
|
}
|
|
|
|
return success;
|
2013-04-11 03:32:07 +02:00
|
|
|
}
|
|
|
|
|
2014-06-27 23:50:51 +02:00
|
|
|
bool cInterfaceWGL::ClearCurrent()
|
|
|
|
{
|
|
|
|
return wglMakeCurrent(hDC, nullptr) ? true : false;
|
|
|
|
}
|
|
|
|
|
2012-12-17 15:01:52 -06:00
|
|
|
// Update window width, size and etc. Called from Render.cpp
|
|
|
|
void cInterfaceWGL::Update()
|
|
|
|
{
|
|
|
|
RECT rcWindow;
|
2014-08-05 02:51:32 -04:00
|
|
|
GetClientRect(m_window_handle, &rcWindow);
|
2012-12-17 15:01:52 -06:00
|
|
|
|
|
|
|
// Get the new window width and height
|
2014-08-05 02:51:32 -04:00
|
|
|
s_backbuffer_width = (rcWindow.right - rcWindow.left);
|
|
|
|
s_backbuffer_height = (rcWindow.bottom - rcWindow.top);
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close backend
|
|
|
|
void cInterfaceWGL::Shutdown()
|
|
|
|
{
|
|
|
|
if (hRC)
|
|
|
|
{
|
2014-03-09 21:14:26 +01:00
|
|
|
if (!wglMakeCurrent(nullptr, nullptr))
|
2012-12-17 15:01:52 -06:00
|
|
|
NOTICE_LOG(VIDEO, "Could not release drawing context.");
|
|
|
|
|
|
|
|
if (!wglDeleteContext(hRC))
|
2013-04-01 00:10:54 -04:00
|
|
|
ERROR_LOG(VIDEO, "Attempt to release rendering context failed.");
|
2012-12-17 15:01:52 -06:00
|
|
|
|
2014-03-09 21:14:26 +01:00
|
|
|
hRC = nullptr;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
|
|
|
|
2014-08-05 02:51:32 -04:00
|
|
|
if (hDC && !ReleaseDC(m_window_handle, hDC))
|
2012-12-17 15:01:52 -06:00
|
|
|
{
|
2013-04-01 00:10:54 -04:00
|
|
|
ERROR_LOG(VIDEO, "Attempt to release device context failed.");
|
2014-03-09 21:14:26 +01:00
|
|
|
hDC = nullptr;
|
2012-12-17 15:01:52 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|