Fix Solaris problems with ttyname_r.

* m4/gnupg-ttyname.m4: New.  Based on ttyname_r from gnulib.
* src/ttyname_r.c (_gpgme_ttyname_r): Rename from ttyname_r.
Implement hacks required for Solaris and possible other non-fully
Posix systems.
* src/util.h: Include unistd.h.  Redefine ttyname_r depending on
REPLACE_TTYNAME_R and put it into the gpgme name space.
--

Unfortunately we cant not use the ttyname_r replacement from gnulib
because we want to keep GPGME LGPLv2+.
This commit is contained in:
Werner Koch 2012-01-19 18:43:10 +01:00
parent c96778297f
commit 2f304957f5
4 changed files with 157 additions and 56 deletions

View File

@ -194,7 +194,6 @@ AM_CONDITIONAL(HAVE_DOSISH_SYSTEM, test "$have_dosish_system" = yes)
if test "$have_w32_system" = yes; then if test "$have_w32_system" = yes; then
AC_DEFINE(HAVE_W32_SYSTEM,1, AC_DEFINE(HAVE_W32_SYSTEM,1,
[Defined if we run on any kind of W32 API based system]) [Defined if we run on any kind of W32 API based system])
ACSUBST
fi fi
AM_CONDITIONAL(HAVE_W32_SYSTEM, test "$have_w32_system" = yes) AM_CONDITIONAL(HAVE_W32_SYSTEM, test "$have_w32_system" = yes)
@ -286,7 +285,7 @@ if test "$ac_cv_func_vasprintf" != yes; then
fi fi
# Try to find a thread-safe version of ttyname(). # Try to find a thread-safe version of ttyname().
AC_REPLACE_FUNCS(ttyname_r) gnupg_REPLACE_TTYNAME_R
if test "$ac_cv_func_ttyname_r" != yes; then if test "$ac_cv_func_ttyname_r" != yes; then
AC_MSG_WARN([ AC_MSG_WARN([
*** ***

View File

@ -1,16 +1,35 @@
# ttyname_r.m4 serial 8 # gnupg-ttyname.m4
dnl Copyright (C) 2010-2012 Free Software Foundation, Inc. # Copyright (C) 2010-2012 Free Software Foundation, Inc.
dnl This file is free software; the Free Software Foundation # This file is free software; the Free Software Foundation
dnl gives unlimited permission to copy and/or distribute it, # gives unlimited permission to copy and/or distribute it,
dnl with or without modifications, as long as this notice is preserved. # with or without modifications, as long as this notice is preserved.
#
# This file is based on gnulib/m4/ttyname_r.m4 serial 8.
#
AC_DEFUN([gl_FUNC_TTYNAME_R],
# gnupg_REPLACE_TTYNAME_R
#
# This macro is an extended version of AC_REPLACE_FUNCS(ttyname_r).
# It takes peculiarities in the implementation of ttyname_r in account.
#
# The macro HAVE_TTYNAME_R will be defined to 1 if the function
# exists; it will be defined to 0 if it does not exists or no
# declaration is available.
#
# The macro HAVE_POSIXDECL_TTYNAME_R is defined if ttyname_r conforms
# to the Posix declaration.
#
# The macro HAVE_BROKEN_TTYNAME_R is defined it ttyname_r does not work
# correctly with the supplied buffer size. If this is defined the function
# will also be replaced.
#
# The macro REPLACE_TTYNAME_R is defined if ttyname_r is a replacement
# function. This macro is useful for the definition of the prototype.
#
AC_DEFUN([gnupg_REPLACE_TTYNAME_R],
[ [
AC_REQUIRE([gl_UNISTD_H_DEFAULTS]) AC_CHECK_HEADERS([unistd.h])
dnl Persuade Solaris <unistd.h> to provide the POSIX compliant declaration of
dnl ttyname_r().
AC_REQUIRE([gl_USE_SYSTEM_EXTENSIONS])
AC_CHECK_DECLS_ONCE([ttyname_r]) AC_CHECK_DECLS_ONCE([ttyname_r])
if test $ac_cv_have_decl_ttyname_r = no; then if test $ac_cv_have_decl_ttyname_r = no; then
@ -20,23 +39,27 @@ AC_DEFUN([gl_FUNC_TTYNAME_R],
AC_CHECK_FUNCS([ttyname_r]) AC_CHECK_FUNCS([ttyname_r])
if test $ac_cv_func_ttyname_r = no; then if test $ac_cv_func_ttyname_r = no; then
HAVE_TTYNAME_R=0 HAVE_TTYNAME_R=0
AC_LIBOBJ([ttyname_r])
AC_DEFINE([REPLACE_TTYNAME_R],[1],
[Define to 1 if ttyname_r is a replacement function.])
else else
HAVE_TTYNAME_R=1 HAVE_TTYNAME_R=1
dnl On MacOS X 10.4 (and Solaris 10 without gl_USE_SYSTEM_EXTENSIONS) dnl On MacOS X 10.4 (and Solaris 10 without __EXTENSIONS__)
dnl the return type is 'char *', not 'int'. dnl the return type is 'char *', not 'int'.
AC_CACHE_CHECK([whether ttyname_r is compatible with its POSIX signature], AC_CACHE_CHECK([whether ttyname_r is compatible with its POSIX signature],
[gl_cv_func_ttyname_r_posix], [gnupg_cv_func_ttyname_r_posix],
[AC_COMPILE_IFELSE( [AC_COMPILE_IFELSE(
[AC_LANG_PROGRAM( [AC_LANG_PROGRAM(
[[#include <stddef.h> [[#include <stddef.h>
#include <unistd.h>]], #include <unistd.h>]],
[[*ttyname_r (0, NULL, 0);]]) [[*ttyname_r (0, NULL, 0);]])
], ],
[gl_cv_func_ttyname_r_posix=no], [gnupg_cv_func_ttyname_r_posix=no],
[gl_cv_func_ttyname_r_posix=yes]) [gnupg_cv_func_ttyname_r_posix=yes])
]) ])
if test $gl_cv_func_ttyname_r_posix = no; then if test $gnupg_cv_func_ttyname_r_posix = no; then
REPLACE_TTYNAME_R=1 AC_LIBOBJ([ttyname_r])
AC_DEFINE([REPLACE_TTYNAME_R],[1])
else else
AC_DEFINE([HAVE_POSIXDECL_TTYNAME_R], [1], AC_DEFINE([HAVE_POSIXDECL_TTYNAME_R], [1],
[Define if the ttyname_r function has a POSIX compliant declaration.]) [Define if the ttyname_r function has a POSIX compliant declaration.])
@ -47,18 +70,18 @@ AC_DEFUN([gl_FUNC_TTYNAME_R],
dnl buffer is large enough. dnl buffer is large enough.
AC_REQUIRE([AC_CANONICAL_HOST]) AC_REQUIRE([AC_CANONICAL_HOST])
AC_CACHE_CHECK([whether ttyname_r works with small buffers], AC_CACHE_CHECK([whether ttyname_r works with small buffers],
[gl_cv_func_ttyname_r_works], [gnupg_cv_func_ttyname_r_works],
[ [
dnl Initial guess, used when cross-compiling or when /dev/tty cannot dnl Initial guess, used when cross-compiling or when /dev/tty cannot
dnl be opened. dnl be opened.
changequote(,)dnl changequote(,)dnl
case "$host_os" in case "$host_os" in
# Guess no on Solaris. # Guess no on Solaris.
solaris*) gl_cv_func_ttyname_r_works="guessing no" ;; solaris*) gnupg_cv_func_ttyname_r_works="guessing no" ;;
# Guess no on OSF/1. # Guess no on OSF/1.
osf*) gl_cv_func_ttyname_r_works="guessing no" ;; osf*) gnupg_cv_func_ttyname_r_works="guessing no" ;;
# Guess yes otherwise. # Guess yes otherwise.
*) gl_cv_func_ttyname_r_works="guessing yes" ;; *) gnupg_cv_func_ttyname_r_works="guessing yes" ;;
esac esac
changequote([,])dnl changequote([,])dnl
AC_RUN_IFELSE( AC_RUN_IFELSE(
@ -81,21 +104,20 @@ main (void)
result |= 18; result |= 18;
return result; return result;
}]])], }]])],
[gl_cv_func_ttyname_r_works=yes], [gnupg_cv_func_ttyname_r_works=yes],
[case $? in [case $? in
17 | 18) gl_cv_func_ttyname_r_works=no ;; 17 | 18) gnupg_cv_func_ttyname_r_works=no ;;
esac], esac],
[:]) [:])
]) ])
case "$gl_cv_func_ttyname_r_works" in case "$gnupg_cv_func_ttyname_r_works" in
*yes) ;; *yes) ;;
*) REPLACE_TTYNAME_R=1 ;; *) AC_LIBOBJ([ttyname_r])
AC_DEFINE([REPLACE_TTYNAME_R],[1])
AC_DEFINE([HAVE_BROKEN_TTYNAME_R], [1],
[Define if ttyname_r is does not work with small buffers])
;;
esac esac
fi fi
fi fi
]) ])
# Prerequisites of lib/ttyname_r.c.
AC_DEFUN([gl_PREREQ_TTYNAME_R], [
AC_CHECK_FUNCS([ttyname])
])

View File

@ -1,5 +1,5 @@
/* ttyname_r.c - A ttyname_r() replacement. /* ttyname_r.c - A ttyname_r() replacement.
Copyright (C) 2003, 2004 g10 Code GmbH Copyright (C) 2003, 2004, 2012 g10 Code GmbH
This file is part of GPGME. This file is part of GPGME.
@ -14,9 +14,8 @@
Lesser General Public License for more details. Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public You should have received a copy of the GNU Lesser General Public
License along with this program; if not, write to the Free Software License along with this program; if not, see <http://www.gnu.org/licenses/>.
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA */
02111-1307, USA. */
#if HAVE_CONFIG_H #if HAVE_CONFIG_H
#include <config.h> #include <config.h>
@ -29,29 +28,103 @@
#endif #endif
#ifdef __GNUC__ #if !HAVE_TTYNAME_R && defined(__GNUC__)
# warning ttyname is not thread-safe, and ttyname_r is missing # warning ttyname is not thread-safe, and ttyname_r is missing
#endif #endif
int int
ttyname_r (int fd, char *buf, size_t buflen) _gpgme_ttyname_r (int fd, char *buf, size_t buflen)
{ {
#if HAVE_TTYNAME_R
# if HAVE_BROKEN_TTYNAME_R
/* Solaris fails if BUFLEN is less than 128. OSF/1 5.1 completely
ignores BUFLEN. We use a large buffer to woraround this. */
{
char largebuf[512];
size_t namelen;
int rc;
# if HAVE_POSIXDECL_TTYNAME_R
if (buflen < sizeof (largebuf))
{
rc = ttyname_r (fd, largebuf, (int)sizeof (largebuf));
if (!rc)
{
namelen = strlen (largebuf) + 1;
if (namelen > buflen)
rc = ERANGE;
else
memcpy (buf, largebuf, namelen);
}
}
else
rc = ttyname_r (fd, buf, (int)buflen);
# else /*!HAVE_POSIXDECL_TTYNAME_R*/
char *name;
if (buflen < sizeof (largebuf))
name = ttyname_r (fd, largebuf, (int)sizeof (largebuf));
else
name = ttyname_r (fd, buf, (int)buflen);
rc = name? 0 : (errno? errno : -1);
if (!rc && buf != name)
{
namelen = strlen (name) + 1;
if (namelen > buflen)
rc = ERANGE;
else
memmove (buf, name, namelen);
}
# endif
return rc;
}
# else /*!HAVE_BROKEN_TTYNAME_R*/
{
int rc;
# if HAVE_POSIXDECL_TTYNAME_R
rc = ttyname_r (fd, buf, buflen);
# else /*!HAVE_POSIXDECL_TTYNAME_R*/
char *name;
size_t namelen;
name = ttyname_r (fd, buf, (int)buflen);
rc = name? 0 : (errno? errno : -1);
if (!rc && buf != name)
{
namelen = strlen (name) + 1;
if (namelen > buflen)
rc = ERANGE;
else
memmove (buf, name, namelen);
}
# endif
return rc;
}
# endif /*!HAVE_BROKEN_TTYNAME_R*/
#else /*!HAVE_TTYNAME_R*/
char *tty; char *tty;
#if HAVE_W32_SYSTEM # if HAVE_W32_SYSTEM
/* We use this default one for now. AFAICS we only need it to be /* We use this default one for now. AFAICS we only need it to be
passed to gpg and in turn to pinentry. Providing a replacement passed to gpg and in turn to pinentry. Providing a replacement
is needed because elsewhere we bail out on error. If we is needed because elsewhere we bail out on error. If we
eventually implement a pinentry for Windows it is uinlikely that eventually implement a pinentry for Windows it is inlikely that
we need a real tty at all. */ we need a real tty at all. */
tty = "/dev/tty"; tty = "/dev/tty";
#else # else
tty = ttyname (fd); tty = ttyname (fd);
if (!tty) if (!tty)
return errno; return errno? errno : -1;
#endif # endif
strncpy (buf, tty, buflen); strncpy (buf, tty, buflen);
buf[buflen - 1] = '\0'; buf[buflen - 1] = '\0';
return (strlen (tty) >= buflen) ? ERANGE : 0; return (strlen (tty) >= buflen) ? ERANGE : 0;
#endif /*!HAVE_TTYNAME_R*/
} }

View File

@ -34,6 +34,10 @@
#ifdef HAVE_SYS_TYPES_H #ifdef HAVE_SYS_TYPES_H
# include <sys/types.h> # include <sys/types.h>
#endif #endif
/* We must see the symbol ttyname_r before a redefinition. */
#ifdef HAVE_UNISTD_H
# include <unistd.h>
#endif
#include "gpgme.h" #include "gpgme.h"
@ -78,11 +82,14 @@ int vasprintf (char **result, const char *format, va_list args);
int asprintf (char **result, const char *format, ...); int asprintf (char **result, const char *format, ...);
#endif #endif
#ifndef HAVE_TTYNAME_R #if REPLACE_TTYNAME_R
int ttyname_r (int fd, char *buf, size_t buflen); int _gpgme_ttyname_r (int fd, char *buf, size_t buflen);
#endif #undef ttyname_r
#define ttyname_r(a,b,c) _gpgme_ttyname_r ((a), (b), (c))
#endif #endif
#endif /*HAVE_CONFIG_H*/
/*-- conversion.c --*/ /*-- conversion.c --*/
/* Convert two hexadecimal digits from STR to the value they /* Convert two hexadecimal digits from STR to the value they