forked from pool/graphviz
68 lines
3.0 KiB
Diff
68 lines
3.0 KiB
Diff
From 87cc54644e40fccc0651b7fedc137b3dd02b4514 Mon Sep 17 00:00:00 2001
|
||
From: Matthew Fernandez <matthew.fernandez@gmail.com>
|
||
Date: Mon, 21 Mar 2022 08:09:55 -0700
|
||
Subject: [PATCH] smyrna Init: squash -Wincompatible-pointer-types
|
||
MIME-Version: 1.0
|
||
Content-Type: text/plain; charset=UTF-8
|
||
Content-Transfer-Encoding: 8bit
|
||
|
||
OpenGL has an unorthodox API wherein the `gluTessCallback` function’s prototype
|
||
indicates it takes a `void(*)(void)`, but its docs¹ explain that the type
|
||
actually varies depending on the second argument. As a result, the compiler
|
||
(correctly) warns that some of these `gluTessCallback` calls are passing
|
||
function pointers that do not have the same ABI. Presumably this works out
|
||
because at the end of the day a function pointer is just some bits in C and the
|
||
OpenGL implementation branches on the `which` argument and invokes the pointer
|
||
correctly. But if OpenGL really wanted to discard type safety this way, it is
|
||
not clear to me why they did not make the function pointer argument a `void*`.
|
||
Anyway, this commit squashes the compiler warnings which emerge when enabling
|
||
this in the CMake build system, failing the build.
|
||
|
||
Gitlab: related to #1836
|
||
|
||
¹ This is not the authoritative source, but Microsoft’s docs for their
|
||
implementation provide a good explanation.
|
||
https://docs.microsoft.com/en-us/windows/win32/opengl/glutess
|
||
---
|
||
cmd/smyrna/polytess.c | 25 ++++++++++++++++++++-----
|
||
1 file changed, 20 insertions(+), 5 deletions(-)
|
||
|
||
--- cmd/smyrna/polytess.c
|
||
+++ cmd/smyrna/polytess.c 2024-09-06 09:04:16.790734526 +0000
|
||
@@ -39,15 +39,30 @@ static void CALLBACK vertexCallback(GLvo
|
||
|
||
}
|
||
|
||
-static GLUtesselator* Init()
|
||
+// OpenGL’s `gluTessCallback` function has a prototype indicating it takes a
|
||
+// `void(*)(void)`. But its documentation describes passing in various function
|
||
+// pointers with differing calling conventions. To use this API while also
|
||
+// pacifying all the various build environments, we need this rather silly
|
||
+// wrapper.
|
||
+#ifdef _MSC_VER
|
||
+// MSVC is of the (correct) opinion that casting between function pointers of
|
||
+// incompatible calling conventions is unacceptable behavior…
|
||
+#define MAKE_GLU_CALLBACK(f) f
|
||
+#else
|
||
+// …nevertheless other compilers insist we cast or they believe we have made a
|
||
+// typo
|
||
+#define MAKE_GLU_CALLBACK(f) ((void (*)(void))(f))
|
||
+#endif
|
||
+
|
||
+static GLUtesselator* Init(void)
|
||
{
|
||
// Create a new tessellation object
|
||
GLUtesselator* tobj = gluNewTess();
|
||
// Set callback functions
|
||
- gluTessCallback(tobj, GLU_TESS_VERTEX, &vertexCallback);
|
||
- gluTessCallback(tobj, GLU_TESS_BEGIN, &glBegin);
|
||
- gluTessCallback(tobj, GLU_TESS_END, &glEnd);
|
||
- gluTessCallback(tobj, GLU_TESS_COMBINE,&combineCallback);
|
||
+ gluTessCallback(tobj, GLU_TESS_VERTEX, MAKE_GLU_CALLBACK(vertexCallback));
|
||
+ gluTessCallback(tobj, GLU_TESS_BEGIN, MAKE_GLU_CALLBACK(glBegin));
|
||
+ gluTessCallback(tobj, GLU_TESS_END, MAKE_GLU_CALLBACK(glEnd));
|
||
+ gluTessCallback(tobj, GLU_TESS_COMBINE, MAKE_GLU_CALLBACK(combineCallback));
|
||
return tobj;
|
||
}
|
||
|