mirror of
https://github.com/MidnightCommander/mc
synced 2024-12-22 04:22:34 +03:00
Ticket #3617: (mc_open): handle varargs mode_t promotion issue.
On systems where 'mode_t' is smaller than 'int', doing 'va_arg (ap, mode_t)' is wrong because of C's "default argument promotions". GCC 4 creates crashing code in this case. The "va_arg" page of Gnulib's manual describes the problem and a simple solution: https://www.gnu.org/software/gnulib/manual/html_node/va_005farg.html However, since that solution reportedly (see thread at next link) still causes GCC to print warnings (for no good reason; perhaps this was fixed in newer GCCs), we pick a solution that defines a PROMOTED_MODE_T at the configuration stage: https://lists.gnu.org/archive/html/bug-gnulib/2009-05/msg00231.html (We take our 'mode_t.m4' from the most recent Gnulib source.) (If any of the URLs above no longer works, simply search the web for the mentioned words.)
This commit is contained in:
parent
2a4614c535
commit
0e89375772
@ -3,6 +3,7 @@ m4_include([m4.include/ax_path_lib_pcre.m4])
|
|||||||
m4_include([m4.include/dx_doxygen.m4])
|
m4_include([m4.include/dx_doxygen.m4])
|
||||||
m4_include([m4.include/mc-cflags.m4])
|
m4_include([m4.include/mc-cflags.m4])
|
||||||
m4_include([m4.include/mc-check-search-type.m4])
|
m4_include([m4.include/mc-check-search-type.m4])
|
||||||
|
m4_include([m4.include/mode_t.m4])
|
||||||
m4_include([m4.include/ls-mntd-fs.m4])
|
m4_include([m4.include/ls-mntd-fs.m4])
|
||||||
m4_include([m4.include/fstypename.m4])
|
m4_include([m4.include/fstypename.m4])
|
||||||
m4_include([m4.include/fsusage.m4])
|
m4_include([m4.include/fsusage.m4])
|
||||||
|
@ -177,6 +177,7 @@ AC_CHECK_SIZEOF(uintmax_t)
|
|||||||
AC_TYPE_OFF_T
|
AC_TYPE_OFF_T
|
||||||
AC_CHECK_SIZEOF(off_t)
|
AC_CHECK_SIZEOF(off_t)
|
||||||
AC_TYPE_MODE_T
|
AC_TYPE_MODE_T
|
||||||
|
gl_PROMOTED_TYPE_MODE_T
|
||||||
AC_TYPE_PID_T
|
AC_TYPE_PID_T
|
||||||
AC_TYPE_UID_T
|
AC_TYPE_UID_T
|
||||||
|
|
||||||
|
@ -200,7 +200,10 @@ mc_open (const vfs_path_t * vpath, int flags, ...)
|
|||||||
{
|
{
|
||||||
va_list ap;
|
va_list ap;
|
||||||
va_start (ap, flags);
|
va_start (ap, flags);
|
||||||
mode = va_arg (ap, mode_t);
|
/* We have to use PROMOTED_MODE_T instead of mode_t. Doing 'va_arg (ap, mode_t)'
|
||||||
|
* fails on systems where 'mode_t' is smaller than 'int' because of C's "default
|
||||||
|
* argument promotions". */
|
||||||
|
mode = va_arg (ap, PROMOTED_MODE_T);
|
||||||
va_end (ap);
|
va_end (ap);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
26
m4.include/mode_t.m4
Normal file
26
m4.include/mode_t.m4
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
# mode_t.m4 serial 2
|
||||||
|
dnl Copyright (C) 2009-2016 Free Software Foundation, Inc.
|
||||||
|
dnl This file is free software; the Free Software Foundation
|
||||||
|
dnl gives unlimited permission to copy and/or distribute it,
|
||||||
|
dnl with or without modifications, as long as this notice is preserved.
|
||||||
|
|
||||||
|
# For using mode_t, it's sufficient to use AC_TYPE_MODE_T and
|
||||||
|
# include <sys/types.h>.
|
||||||
|
|
||||||
|
# Define PROMOTED_MODE_T to the type that is the result of "default argument
|
||||||
|
# promotion" (ISO C 6.5.2.2.(6)) of the type mode_t.
|
||||||
|
AC_DEFUN([gl_PROMOTED_TYPE_MODE_T],
|
||||||
|
[
|
||||||
|
AC_REQUIRE([AC_TYPE_MODE_T])
|
||||||
|
AC_CACHE_CHECK([for promoted mode_t type], [gl_cv_promoted_mode_t], [
|
||||||
|
dnl Assume mode_t promotes to 'int' if and only if it is smaller than 'int',
|
||||||
|
dnl and to itself otherwise. This assumption is not guaranteed by the ISO C
|
||||||
|
dnl standard, but we don't know of any real-world counterexamples.
|
||||||
|
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[#include <sys/types.h>]],
|
||||||
|
[[typedef int array[2 * (sizeof (mode_t) < sizeof (int)) - 1];]])],
|
||||||
|
[gl_cv_promoted_mode_t='int'],
|
||||||
|
[gl_cv_promoted_mode_t='mode_t'])
|
||||||
|
])
|
||||||
|
AC_DEFINE_UNQUOTED([PROMOTED_MODE_T], [$gl_cv_promoted_mode_t],
|
||||||
|
[Define to the type that is the result of default argument promotions of type mode_t.])
|
||||||
|
])
|
Loading…
Reference in New Issue
Block a user