Commit 74a308cf authored by Peter Eisentraut's avatar Peter Eisentraut

Use explicit_bzero

Use the explicit_bzero() function in places where it is important that
security information such as passwords is cleared from memory.  There
might be other places where it could be useful; this is just an
initial collection.

For platforms that don't have explicit_bzero(), provide various
fallback implementations.  (explicit_bzero() itself isn't standard,
but as Linux/glibc, FreeBSD, and OpenBSD have it, it's the most common
spelling, so it makes sense to make that the invocation point.)

Discussion: https://www.postgresql.org/message-id/flat/42d26bde-5d5b-c90d-87ae-6cab875f73be%402ndquadrant.com
parent ae060a52
...@@ -15087,7 +15087,7 @@ fi ...@@ -15087,7 +15087,7 @@ fi
LIBS_including_readline="$LIBS" LIBS_including_readline="$LIBS"
LIBS=`echo "$LIBS" | sed -e 's/-ledit//g' -e 's/-lreadline//g'` LIBS=`echo "$LIBS" | sed -e 's/-ledit//g' -e 's/-lreadline//g'`
for ac_func in cbrt clock_gettime copyfile fdatasync getifaddrs getpeerucred getrlimit mbstowcs_l memmove poll posix_fallocate ppoll pstat pthread_is_threaded_np readlink setproctitle setproctitle_fast setsid shm_open strchrnul strsignal symlink sync_file_range uselocale utime utimes wcstombs_l for ac_func in cbrt clock_gettime copyfile fdatasync getifaddrs getpeerucred getrlimit mbstowcs_l memset_s memmove poll posix_fallocate ppoll pstat pthread_is_threaded_np readlink setproctitle setproctitle_fast setsid shm_open strchrnul strsignal symlink sync_file_range uselocale utime utimes wcstombs_l
do : do :
as_ac_var=`$as_echo "ac_cv_func_$ac_func" | $as_tr_sh` as_ac_var=`$as_echo "ac_cv_func_$ac_func" | $as_tr_sh`
ac_fn_c_check_func "$LINENO" "$ac_func" "$as_ac_var" ac_fn_c_check_func "$LINENO" "$ac_func" "$as_ac_var"
...@@ -15739,6 +15739,19 @@ esac ...@@ -15739,6 +15739,19 @@ esac
fi fi
ac_fn_c_check_func "$LINENO" "explicit_bzero" "ac_cv_func_explicit_bzero"
if test "x$ac_cv_func_explicit_bzero" = xyes; then :
$as_echo "#define HAVE_EXPLICIT_BZERO 1" >>confdefs.h
else
case " $LIBOBJS " in
*" explicit_bzero.$ac_objext "* ) ;;
*) LIBOBJS="$LIBOBJS explicit_bzero.$ac_objext"
;;
esac
fi
ac_fn_c_check_func "$LINENO" "fls" "ac_cv_func_fls" ac_fn_c_check_func "$LINENO" "fls" "ac_cv_func_fls"
if test "x$ac_cv_func_fls" = xyes; then : if test "x$ac_cv_func_fls" = xyes; then :
$as_echo "#define HAVE_FLS 1" >>confdefs.h $as_echo "#define HAVE_FLS 1" >>confdefs.h
......
...@@ -1594,6 +1594,7 @@ AC_CHECK_FUNCS(m4_normalize([ ...@@ -1594,6 +1594,7 @@ AC_CHECK_FUNCS(m4_normalize([
getpeerucred getpeerucred
getrlimit getrlimit
mbstowcs_l mbstowcs_l
memset_s
memmove memmove
poll poll
posix_fallocate posix_fallocate
...@@ -1691,6 +1692,7 @@ fi ...@@ -1691,6 +1692,7 @@ fi
AC_REPLACE_FUNCS(m4_normalize([ AC_REPLACE_FUNCS(m4_normalize([
dlopen dlopen
explicit_bzero
fls fls
getopt getopt
getrusage getrusage
......
...@@ -87,6 +87,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf, ...@@ -87,6 +87,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf,
{ {
if (ferror(fh)) if (ferror(fh))
{ {
explicit_bzero(buf, size);
ereport(loglevel, ereport(loglevel,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not read from command \"%s\": %m", errmsg("could not read from command \"%s\": %m",
...@@ -98,6 +99,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf, ...@@ -98,6 +99,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf,
pclose_rc = ClosePipeStream(fh); pclose_rc = ClosePipeStream(fh);
if (pclose_rc == -1) if (pclose_rc == -1)
{ {
explicit_bzero(buf, size);
ereport(loglevel, ereport(loglevel,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not close pipe to external command: %m"))); errmsg("could not close pipe to external command: %m")));
...@@ -105,6 +107,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf, ...@@ -105,6 +107,7 @@ run_ssl_passphrase_command(const char *prompt, bool is_server_start, char *buf,
} }
else if (pclose_rc != 0) else if (pclose_rc != 0)
{ {
explicit_bzero(buf, size);
ereport(loglevel, ereport(loglevel,
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("command \"%s\" failed", errmsg("command \"%s\" failed",
......
...@@ -195,6 +195,9 @@ ...@@ -195,6 +195,9 @@
/* Define to 1 if you have the <editline/readline.h> header file. */ /* Define to 1 if you have the <editline/readline.h> header file. */
#undef HAVE_EDITLINE_READLINE_H #undef HAVE_EDITLINE_READLINE_H
/* Define to 1 if you have the `explicit_bzero' function. */
#undef HAVE_EXPLICIT_BZERO
/* Define to 1 if you have the `fdatasync' function. */ /* Define to 1 if you have the `fdatasync' function. */
#undef HAVE_FDATASYNC #undef HAVE_FDATASYNC
...@@ -395,6 +398,9 @@ ...@@ -395,6 +398,9 @@
/* Define to 1 if you have the <memory.h> header file. */ /* Define to 1 if you have the <memory.h> header file. */
#undef HAVE_MEMORY_H #undef HAVE_MEMORY_H
/* Define to 1 if you have the `memset_s' function. */
#undef HAVE_MEMSET_S
/* Define to 1 if the system has the type `MINIDUMP_TYPE'. */ /* Define to 1 if the system has the type `MINIDUMP_TYPE'. */
#undef HAVE_MINIDUMP_TYPE #undef HAVE_MINIDUMP_TYPE
......
...@@ -153,6 +153,9 @@ ...@@ -153,6 +153,9 @@
/* Define to 1 if you have the <editline/readline.h> header file. */ /* Define to 1 if you have the <editline/readline.h> header file. */
/* #undef HAVE_EDITLINE_READLINE_H */ /* #undef HAVE_EDITLINE_READLINE_H */
/* Define to 1 if you have the `explicit_bzero' function. */
/* #undef HAVE_EXPLICIT_BZERO */
/* Define to 1 if you have the `fdatasync' function. */ /* Define to 1 if you have the `fdatasync' function. */
/* #undef HAVE_FDATASYNC */ /* #undef HAVE_FDATASYNC */
...@@ -283,6 +286,9 @@ ...@@ -283,6 +286,9 @@
/* Define to 1 if you have the <memory.h> header file. */ /* Define to 1 if you have the <memory.h> header file. */
#define HAVE_MEMORY_H 1 #define HAVE_MEMORY_H 1
/* Define to 1 if you have the `memset_s' function. */
/* #undef HAVE_MEMSET_S */
/* Define to 1 if the system has the type `MINIDUMP_TYPE'. */ /* Define to 1 if the system has the type `MINIDUMP_TYPE'. */
#define HAVE_MINIDUMP_TYPE 1 #define HAVE_MINIDUMP_TYPE 1
......
...@@ -378,6 +378,10 @@ extern int isinf(double x); ...@@ -378,6 +378,10 @@ extern int isinf(double x);
#endif /* __clang__ && !__cplusplus */ #endif /* __clang__ && !__cplusplus */
#endif /* !HAVE_ISINF */ #endif /* !HAVE_ISINF */
#ifndef HAVE_EXPLICIT_BZERO
extern void explicit_bzero(void *buf, size_t len);
#endif
#ifndef HAVE_STRTOF #ifndef HAVE_STRTOF
extern float strtof(const char *nptr, char **endptr); extern float strtof(const char *nptr, char **endptr);
#endif #endif
......
...@@ -3885,8 +3885,11 @@ freePGconn(PGconn *conn) ...@@ -3885,8 +3885,11 @@ freePGconn(PGconn *conn)
if (conn->connhost[i].port != NULL) if (conn->connhost[i].port != NULL)
free(conn->connhost[i].port); free(conn->connhost[i].port);
if (conn->connhost[i].password != NULL) if (conn->connhost[i].password != NULL)
{
explicit_bzero(conn->connhost[i].password, strlen(conn->connhost[i].password));
free(conn->connhost[i].password); free(conn->connhost[i].password);
} }
}
free(conn->connhost); free(conn->connhost);
} }
...@@ -3919,7 +3922,10 @@ freePGconn(PGconn *conn) ...@@ -3919,7 +3922,10 @@ freePGconn(PGconn *conn)
if (conn->pguser) if (conn->pguser)
free(conn->pguser); free(conn->pguser);
if (conn->pgpass) if (conn->pgpass)
{
explicit_bzero(conn->pgpass, strlen(conn->pgpass));
free(conn->pgpass); free(conn->pgpass);
}
if (conn->pgpassfile) if (conn->pgpassfile)
free(conn->pgpassfile); free(conn->pgpassfile);
if (conn->keepalives) if (conn->keepalives)
...@@ -6931,6 +6937,7 @@ passwordFromFile(const char *hostname, const char *port, const char *dbname, ...@@ -6931,6 +6937,7 @@ passwordFromFile(const char *hostname, const char *port, const char *dbname,
if (!ret) if (!ret)
{ {
/* Out of memory. XXX: an error message would be nice. */ /* Out of memory. XXX: an error message would be nice. */
explicit_bzero(buf, sizeof(buf));
return NULL; return NULL;
} }
...@@ -6947,6 +6954,7 @@ passwordFromFile(const char *hostname, const char *port, const char *dbname, ...@@ -6947,6 +6954,7 @@ passwordFromFile(const char *hostname, const char *port, const char *dbname,
} }
fclose(fp); fclose(fp);
explicit_bzero(buf, sizeof(buf));
return NULL; return NULL;
#undef LINELEN #undef LINELEN
......
/*-------------------------------------------------------------------------
*
* explicit_bzero.c
*
* Portions Copyright (c) 1996-2019, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
*
* IDENTIFICATION
* src/port/explicit_bzero.c
*
*-------------------------------------------------------------------------
*/
#include "c.h"
#if defined(HAVE_MEMSET_S)
void
explicit_bzero(void *buf, size_t len)
{
(void) memset_s(buf, len, 0, len);
}
#elif defined(WIN32)
void
explicit_bzero(void *buf, size_t len)
{
(void) SecureZeroMemory(buf, len);
}
#else
/*
* Indirect call through a volatile pointer to hopefully avoid dead-store
* optimisation eliminating the call. (Idea taken from OpenSSH.) We can't
* assume bzero() is present either, so for simplicity we define our own.
*/
static void
bzero2(void *buf, size_t len)
{
memset(buf, 0, len);
}
static void (* volatile bzero_p)(void *, size_t) = bzero2;
void
explicit_bzero(void *buf, size_t len)
{
bzero_p(buf, len);
}
#endif
...@@ -93,7 +93,7 @@ sub mkvcbuild ...@@ -93,7 +93,7 @@ sub mkvcbuild
$solution = CreateSolution($vsVersion, $config); $solution = CreateSolution($vsVersion, $config);
our @pgportfiles = qw( our @pgportfiles = qw(
chklocale.c fls.c fseeko.c getrusage.c inet_aton.c random.c chklocale.c explicit_bzero.c fls.c fseeko.c getrusage.c inet_aton.c random.c
srandom.c getaddrinfo.c gettimeofday.c inet_net_ntop.c kill.c open.c srandom.c getaddrinfo.c gettimeofday.c inet_net_ntop.c kill.c open.c
erand48.c snprintf.c strlcat.c strlcpy.c dirmod.c noblock.c path.c erand48.c snprintf.c strlcat.c strlcpy.c dirmod.c noblock.c path.c
dirent.c dlopen.c getopt.c getopt_long.c dirent.c dlopen.c getopt.c getopt_long.c
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment