mirror of
				https://github.com/glfw/glfw.git
				synced 2025-10-25 01:22:32 +00:00 
			
		
		
		
	This updates to a newer version of glad2 and switches to the header-only variant. This also (finally) switches to the newer glad2 loader signature that allows us to pass in glfwGetInstanceProcAddress directly.
		
			
				
	
	
		
			264 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			264 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| //========================================================================
 | |
| // Monitor information tool
 | |
| // Copyright (c) Camilla Löwy <elmindreda@glfw.org>
 | |
| //
 | |
| // 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
 | |
| //
 | |
| //========================================================================
 | |
| 
 | |
| #define GLAD_GL_IMPLEMENTATION
 | |
| #include <glad/gl.h>
 | |
| #define GLFW_INCLUDE_NONE
 | |
| #include <GLFW/glfw3.h>
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <stdlib.h>
 | |
| 
 | |
| #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);
 | |
| }
 | |
| 
 |