On Thu, Aug 06, 2020 at 05:23:07PM -0700, Junio C Hamano wrote:
quoted hunk ↗ jump to hunk
diff --git a/git-compat-util.h b/git-compat-util.h
index 5637114b8d..7a0fb7a045 100644
--- a/git-compat-util.h
+++ b/git-compat-util.h
@@ -252,8 +252,10 @@ typedef unsigned long uintptr_t;
#ifdef PRECOMPOSE_UNICODE
#include "compat/precompose_utf8.h"
#else
-#define precompose_str(in,i_nfd2nfc)
-#define precompose_argv(c,v)
+static inline void precompose_argv(int argc, const char **argv)
+{
+ ; /* nothing */
+}
This one looks safe and seems like an improvement, since it's our own
function that we're redefining.
But this one for example:
-#define flockfile(fh)
-#define funlockfile(fh)
+static inline void flockfile(FILE *fh)
+{
+ ; /* nothing */
+}
+static inline void funlockfile(FILE *fh)
+{
+ ; /* nothing */
+}
is re-defining a name that's usually reserved for the system (at least
on POSIX systems). For most systems that define it, we'd actually use
the system version (and not compile this code at all). But there may be
systems where we choose not to (either the system version is deficient,
or we're testing the fallback code on a more-capable system, or our
#ifndef check isn't sufficient on that system for some reason).
If the system defines it as a macro, we'd probably get a garbled
compiler error as the macro is expanded here. Adding #undef flockfile,
etc beforehand would help. I'm not sure if the current code might give
us a macro redefinition warning on such a system.
If the system defines it as a function, we'd probably get redefinition
warnings.
So...I dunno. Those are all theoretical complaints. But I also think
what it's buying is not very big:
- unlike precompose_argv(), modern POSIX-compliant systems (which we
all tend to develop on) don't hit this fallback code. So your
average developer is likely to see any problems here.
- this is really the tip of the portability iceberg anyway. In the
example that motivated this thread, you were catching failures to
adjust to strvec. But in code like this:
#ifdef FOO
void some_func(int x, const char *y)
{
struct argv_array whatever = ARGV_ARRAY_INIT;
...
}
#else
void some_func(int x, const char *y)
{
/* do nothing */
}
#endif
You'd still fail to notice the problem if you're not compiling with
-DFOO. I think we have to accept that the compiler on a given
platform won't be able to catch everything. That's why we have CI on
many platforms, plus people building and reporting problems on their
own systems.
quoted hunk ↗ jump to hunk
@@ -270,7 +272,9 @@ struct itimerval {
#endif
#ifdef NO_SETITIMER
-#define setitimer(which,value,ovalue)
+static inline int setitimer(int which, const struct itimerval *value, struct itimerval *newvalue) {
+ ; /* nothing */
+}
#endif
Same reasoning applies to this one, plus the added bonus that we'd need
that struct type defined. brian mentioned using "void *". That works as
long as the system doesn't define setitimer() itself with the real
types. Another option is to forward declare "struct itimerval" (which
_should_ be OK even if the system does so, since our forward declaration
has no details). But again, I'm not sure if it's worth the hassle.
-Peff