//======================================================================== // Monitor information tool // Copyright (c) Camilla Löwy // // This software is provided 'as-is', without any express or implied // warranty. In no event will the authors be held liable for any damages // arising from the use of this software. // // Permission is granted to anyone to use this software for any purpose, // including commercial applications, and to alter it and redistribute it // freely, subject to the following restrictions: // // 1. The origin of this software must not be misrepresented; you must not // claim that you wrote the original software. If you use this software // in a product, an acknowledgment in the product documentation would // be appreciated but is not required. // // 2. Altered source versions must be plainly marked as such, and must not // be misrepresented as being the original software. // // 3. This notice may not be removed or altered from any source // distribution. // //======================================================================== // // This test prints monitor and video mode information or verifies video // modes // //======================================================================== #include #define GLFW_INCLUDE_NONE #include #include #include #include #include "getopt.h" enum Mode { LIST_MODE, TEST_MODE }; static void usage(void) { printf("Usage: monitors [-t]\n"); printf(" monitors -h\n"); } static int euclid(int a, int b) { return b ? euclid(b, a % b) : a; } static const char* format_mode(const GLFWvidmode* mode) { static char buffer[512]; const int gcd = euclid(mode->width, mode->height); snprintf(buffer, sizeof(buffer), "%i x %i x %i (%i:%i) (%i %i %i) %i Hz", mode->width, mode->height, mode->redBits + mode->greenBits + mode->blueBits, mode->width / gcd, mode->height / gcd, mode->redBits, mode->greenBits, mode->blueBits, mode->refreshRate); buffer[sizeof(buffer) - 1] = '\0'; return buffer; } static void error_callback(int error, const char* description) { fprintf(stderr, "Error: %s\n", description); } static void framebuffer_size_callback(GLFWwindow* window, int width, int height) { printf("Framebuffer resized to %ix%i\n", width, height); glViewport(0, 0, width, height); } static void key_callback(GLFWwindow* window, int key, int scancode, int action, int mods) { if (key == GLFW_KEY_ESCAPE) glfwSetWindowShouldClose(window, GLFW_TRUE); } static void list_modes(GLFWmonitor* monitor) { int count, x, y, width_mm, height_mm, i; int workarea_x, workarea_y, workarea_width, workarea_height; float xscale, yscale; const GLFWvidmode* mode = glfwGetVideoMode(monitor); const GLFWvidmode* modes = glfwGetVideoModes(monitor, &count); glfwGetMonitorPos(monitor, &x, &y); glfwGetMonitorPhysicalSize(monitor, &width_mm, &height_mm); glfwGetMonitorContentScale(monitor, &xscale, &yscale); glfwGetMonitorWorkarea(monitor, &workarea_x, &workarea_y, &workarea_width, &workarea_height); printf("Name: %s (%s)\n", glfwGetMonitorName(monitor), glfwGetPrimaryMonitor() == monitor ? "primary" : "secondary"); printf("Current mode: %s\n", format_mode(mode)); printf("Virtual position: %i, %i\n", x, y); printf("Content scale: %f x %f\n", xscale, yscale); printf("Physical size: %i x %i mm (%0.2f dpi at %i x %i)\n", width_mm, height_mm, mode->width * 25.4f / width_mm, mode->width, mode->height); printf("Monitor work area: %i x %i starting at %i, %i\n", workarea_width, workarea_height, workarea_x, workarea_y); printf("Modes:\n"); for (i = 0; i < count; i++) { printf("%3u: %s", (unsigned int) i, format_mode(modes + i)); if (memcmp(mode, modes + i, sizeof(GLFWvidmode)) == 0) printf(" (current mode)"); putchar('\n'); } } static void test_modes(GLFWmonitor* monitor) { int i, count; GLFWwindow* window; const GLFWvidmode* modes = glfwGetVideoModes(monitor, &count); for (i = 0; i < count; i++) { const GLFWvidmode* mode = modes + i; GLFWvidmode current; glfwWindowHint(GLFW_RED_BITS, mode->redBits); glfwWindowHint(GLFW_GREEN_BITS, mode->greenBits); glfwWindowHint(GLFW_BLUE_BITS, mode->blueBits); glfwWindowHint(GLFW_REFRESH_RATE, mode->refreshRate); printf("Testing mode %u on monitor %s: %s\n", (unsigned int) i, glfwGetMonitorName(monitor), format_mode(mode)); window = glfwCreateWindow(mode->width, mode->height, "Video Mode Test", glfwGetPrimaryMonitor(), NULL); if (!window) { printf("Failed to enter mode %u: %s\n", (unsigned int) i, format_mode(mode)); continue; } glfwSetFramebufferSizeCallback(window, framebuffer_size_callback); glfwSetKeyCallback(window, key_callback); glfwMakeContextCurrent(window); gladLoadGL(glfwGetProcAddress); glfwSwapInterval(1); glfwSetTime(0.0); while (glfwGetTime() < 5.0) { glClear(GL_COLOR_BUFFER_BIT); glfwSwapBuffers(window); glfwPollEvents(); if (glfwWindowShouldClose(window)) { printf("User terminated program\n"); glfwTerminate(); exit(EXIT_SUCCESS); } } glGetIntegerv(GL_RED_BITS, ¤t.redBits); glGetIntegerv(GL_GREEN_BITS, ¤t.greenBits); glGetIntegerv(GL_BLUE_BITS, ¤t.blueBits); glfwGetWindowSize(window, ¤t.width, ¤t.height); if (current.redBits != mode->redBits || current.greenBits != mode->greenBits || current.blueBits != mode->blueBits) { printf("*** Color bit mismatch: (%i %i %i) instead of (%i %i %i)\n", current.redBits, current.greenBits, current.blueBits, mode->redBits, mode->greenBits, mode->blueBits); } if (current.width != mode->width || current.height != mode->height) { printf("*** Size mismatch: %ix%i instead of %ix%i\n", current.width, current.height, mode->width, mode->height); } printf("Closing window\n"); glfwDestroyWindow(window); window = NULL; glfwPollEvents(); } } int main(int argc, char** argv) { int ch, i, count, mode = LIST_MODE; GLFWmonitor** monitors; while ((ch = getopt(argc, argv, "th")) != -1) { switch (ch) { case 'h': usage(); exit(EXIT_SUCCESS); case 't': mode = TEST_MODE; break; default: usage(); exit(EXIT_FAILURE); } } glfwSetErrorCallback(error_callback); glfwInitHint(GLFW_COCOA_MENUBAR, GLFW_FALSE); if (!glfwInit()) exit(EXIT_FAILURE); monitors = glfwGetMonitors(&count); for (i = 0; i < count; i++) { if (mode == LIST_MODE) list_modes(monitors[i]); else if (mode == TEST_MODE) test_modes(monitors[i]); } glfwTerminate(); exit(EXIT_SUCCESS); }