/*-------------------------------------------------------------------------
*
* test_thread_funcs.c
- * libc thread test program
+ * libc thread test program
*
* Portions Copyright (c) 1996-2003, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California
*
- * $PostgreSQL: pgsql/src/tools/thread/thread_test.c,v 1.20 2004/04/23 18:15:55 momjian Exp $
+ * $PostgreSQL: pgsql/src/tools/thread/thread_test.c,v 1.21 2004/04/23 20:35:50 momjian Exp $
*
* This program tests to see if your standard libc functions use
* pthread_setspecific()/pthread_getspecific() to be thread-safe.
#include "postgres.h"
-void func_call_1(void);
-void func_call_2(void);
+void func_call_1(void);
+void func_call_2(void);
-char myhostname[MAXHOSTNAMELEN];
-
-volatile int errno1_set = 0;
-volatile int errno2_set = 0;
+pthread_mutex_t init_mutex = PTHREAD_MUTEX_INITIALIZER;
volatile int thread1_done = 0;
volatile int thread2_done = 0;
-char *strerror_p1;
-char *strerror_p2;
+volatile int errno1_set = 0;
+volatile int errno2_set = 0;
+#ifndef HAVE_STRERROR_R
+char *strerror_p1;
+char *strerror_p2;
+bool strerror_threadsafe = false;
+#endif
+
+#ifndef HAVE_GETPWUID_R
struct passwd *passwd_p1;
struct passwd *passwd_p2;
+bool getpwuid_threadsafe = false;
+#endif
+#if !defined(HAVE_GETADDRINFO) && !defined(HAVE_GETHOSTBYNAME_R)
struct hostent *hostent_p1;
struct hostent *hostent_p2;
+char myhostname[MAXHOSTNAMELEN];
+bool gethostbyname_threadsafe = false;
+#endif
-bool gethostbyname_threadsafe = false;
-bool getpwuid_threadsafe = false;
-bool strerror_threadsafe = false;
-bool platform_is_threadsafe = true;
-
-pthread_mutex_t init_mutex = PTHREAD_MUTEX_INITIALIZER;
+bool platform_is_threadsafe = true;
-int main(int argc, char *argv[])
+int
+main(int argc, char *argv[])
{
- pthread_t thread1,
- thread2;
+ pthread_t thread1,
+ thread2;
if (argc > 1)
{
- fprintf(stderr, "Usage: %s\n", argv[0]);
- return 1;
+ fprintf(stderr, "Usage: %s\n", argv[0]);
+ return 1;
}
+#if !defined(HAVE_GETADDRINFO) && !defined(HAVE_GETHOSTBYNAME_R)
if (gethostname(myhostname, MAXHOSTNAMELEN) != 0)
{
- fprintf(stderr, "can not get local hostname, exiting\n");
- exit(1);
+ fprintf(stderr, "can not get local hostname, exiting\n");
+ exit(1);
}
-
- printf("\
-Make sure you have added any needed 'PTHREAD_CFLAGS' and 'PTHREAD_LIBS'\n\
-defines to your template/$port file before compiling this program.\n\n"
-);
+#endif
/* Hold lock until we are ready for the child threads to exit. */
- pthread_mutex_lock(&init_mutex);
-
- pthread_create(&thread1, NULL, (void * (*)(void *)) func_call_1, NULL);
- pthread_create(&thread2, NULL, (void * (*)(void *)) func_call_2, NULL);
+ pthread_mutex_lock(&init_mutex);
+
+ pthread_create(&thread1, NULL, (void *(*) (void *)) func_call_1, NULL);
+ pthread_create(&thread2, NULL, (void *(*) (void *)) func_call_2, NULL);
while (thread1_done == 0 || thread2_done == 0)
- sched_yield(); /* if this is a portability problem, remove it */
+ sched_yield(); /* if this is a portability problem,
+ * remove it */
- fprintf(stderr, "errno is thread-safe\n");
-
+ fprintf(stderr, "Your errno is thread-safe.\n");
+
+#ifndef HAVE_STRERROR_R
if (strerror_p1 != strerror_p2)
strerror_threadsafe = true;
+#endif
+#ifndef HAVE_GETPWUID_R
if (passwd_p1 != passwd_p2)
getpwuid_threadsafe = true;
+#endif
+#if !defined(HAVE_GETADDRINFO) && !defined(HAVE_GETHOSTBYNAME_R)
if (hostent_p1 != hostent_p2)
gethostbyname_threadsafe = true;
+#endif
pthread_mutex_unlock(&init_mutex); /* let children exit */
-
+
pthread_join(thread1, NULL); /* clean up children */
pthread_join(thread2, NULL);
- printf("\n");
-
#ifdef HAVE_STRERROR_R
- printf("Your system has sterror_r(), so it doesn't use strerror().\n");
+ printf("Your system has sterror_r(); it does not need strerror().\n");
#else
printf("Your system uses strerror() which is ");
if (strerror_threadsafe)
printf("thread-safe.\n");
else
{
- platform_is_threadsafe = false;
printf("not thread-safe.\n");
+ platform_is_threadsafe = false;
}
#endif
#ifdef HAVE_GETPWUID_R
- printf("Your system has getpwuid_r(), so it doesn't use getpwuid().\n");
+ printf("Your system has getpwuid_r(); it does not need getpwuid().\n");
#else
printf("Your system uses getpwuid() which is ");
if (getpwuid_threadsafe)
printf("thread-safe.\n");
else
{
- platform_is_threadsafe = false;
printf("not thread-safe.\n");
+ platform_is_threadsafe = false;
}
#endif
#ifdef HAVE_GETADDRINFO
- printf("Your system has getaddrinfo(), so it doesn't use gethostbyname()\n"
- " or gethostbyname_r().\n");
+ printf("Your system has getaddrinfo(); it does not need gethostbyname()\n"
+ " or gethostbyname_r().\n");
#else
#ifdef HAVE_GETHOSTBYNAME_R
- printf("Your system has gethostbyname_r(), so it doesn't use gethostbyname().\n");
+ printf("Your system has gethostbyname_r(); it does not need gethostbyname().\n");
#else
printf("Your system uses gethostbyname which is ");
if (gethostbyname_threadsafe)
printf("thread-safe.\n");
else
{
- platform_is_threadsafe = false;
printf("not thread-safe.\n");
+ platform_is_threadsafe = false;
}
#endif
#endif
- if (!platform_is_threadsafe)
+ if (platform_is_threadsafe)
{
- printf("\n** YOUR PLATFORM IS NOT THREADSAFE **\n");
- return 1;
+ printf("\nYour platform is thread-safe.\n");
+ return 0;
}
else
{
- printf("\nYOUR PLATFORM IS THREADSAFE\n");
- return 0;
+ printf("\n** YOUR PLATFORM IS NOT THREAD-SAFE. **\n");
+ return 1;
}
}
-void func_call_1(void) {
- void *p;
-
+void
+func_call_1(void)
+{
+#if !defined(HAVE_GETPWUID_R) || \
+ (!defined(HAVE_GETADDRINFO) && \
+ !defined(HAVE_GETHOSTBYNAME_R))
+ void *p;
+#endif
+
unlink("/tmp/thread_test.1");
/* create, then try to fail on exclusive create open */
if (open("/tmp/thread_test.1", O_RDWR | O_CREAT, 0600) < 0 ||
open("/tmp/thread_test.1", O_RDWR | O_CREAT | O_EXCL, 0600) >= 0)
{
- fprintf(stderr, "Could not create file in /tmp or\n");
- fprintf(stderr, "could not generate failure for create file in /tmp, exiting\n");
- exit(1);
+ fprintf(stderr, "Could not create file in /tmp or\n");
+ fprintf(stderr, "could not generate failure for create file in /tmp, exiting\n");
+ exit(1);
}
+
/*
- * Wait for other thread to set errno.
- * We can't use thread-specific locking here because it might
- * affect errno.
+ * Wait for other thread to set errno. We can't use thread-specific
+ * locking here because it might affect errno.
*/
errno1_set = 1;
while (errno2_set == 0)
sched_yield();
if (errno != EEXIST)
{
- fprintf(stderr, "errno not thread-safe; exiting\n");
- unlink("/tmp/thread_test.1");
- exit(1);
+ fprintf(stderr, "errno not thread-safe; exiting\n");
+ unlink("/tmp/thread_test.1");
+ exit(1);
}
unlink("/tmp/thread_test.1");
-
+
+#ifndef HAVE_STRERROR_R
strerror_p1 = strerror(EACCES);
+
/*
- * If strerror() uses sys_errlist, the pointer might change for different
- * errno values, so we don't check to see if it varies within the thread.
+ * If strerror() uses sys_errlist, the pointer might change for
+ * different errno values, so we don't check to see if it varies
+ * within the thread.
*/
+#endif
+#ifndef HAVE_GETPWUID_R
passwd_p1 = getpwuid(0);
p = getpwuid(1);
if (passwd_p1 != p)
{
printf("Your getpwuid() changes the static memory area between calls\n");
- passwd_p1 = NULL; /* force thread-safe failure report */
+ passwd_p1 = NULL; /* force thread-safe failure report */
}
+#endif
+#if !defined(HAVE_GETADDRINFO) && !defined(HAVE_GETHOSTBYNAME_R)
/* threads do this in opposite order */
hostent_p1 = gethostbyname(myhostname);
p = gethostbyname("localhost");
if (hostent_p1 != p)
{
printf("Your gethostbyname() changes the static memory area between calls\n");
- hostent_p1 = NULL; /* force thread-safe failure report */
+ hostent_p1 = NULL; /* force thread-safe failure report */
}
+#endif
thread1_done = 1;
pthread_mutex_lock(&init_mutex); /* wait for parent to test */
}
-void func_call_2(void) {
- void *p;
+void
+func_call_2(void)
+{
+#if !defined(HAVE_GETPWUID_R) || \
+ (!defined(HAVE_GETADDRINFO) && \
+ !defined(HAVE_GETHOSTBYNAME_R))
+ void *p;
+#endif
unlink("/tmp/thread_test.2");
/* open non-existant file */
if (open("/tmp/thread_test.2", O_RDONLY, 0600) >= 0)
{
- fprintf(stderr, "Read-only open succeeded without create, exiting\n");
- exit(1);
+ fprintf(stderr, "Read-only open succeeded without create, exiting\n");
+ exit(1);
}
+
/*
- * Wait for other thread to set errno.
- * We can't use thread-specific locking here because it might
- * affect errno.
+ * Wait for other thread to set errno. We can't use thread-specific
+ * locking here because it might affect errno.
*/
errno2_set = 1;
while (errno1_set == 0)
sched_yield();
if (errno != ENOENT)
{
- fprintf(stderr, "errno not thread-safe; exiting\n");
- unlink("/tmp/thread_test.A");
- exit(1);
+ fprintf(stderr, "errno not thread-safe; exiting\n");
+ unlink("/tmp/thread_test.A");
+ exit(1);
}
unlink("/tmp/thread_test.2");
-
+
+#ifndef HAVE_STRERROR_R
strerror_p2 = strerror(EINVAL);
+
/*
- * If strerror() uses sys_errlist, the pointer might change for different
- * errno values, so we don't check to see if it varies within the thread.
+ * If strerror() uses sys_errlist, the pointer might change for
+ * different errno values, so we don't check to see if it varies
+ * within the thread.
*/
+#endif
+#ifndef HAVE_GETPWUID_R
passwd_p2 = getpwuid(2);
p = getpwuid(3);
if (passwd_p2 != p)
{
printf("Your getpwuid() changes the static memory area between calls\n");
- passwd_p2 = NULL; /* force thread-safe failure report */
+ passwd_p2 = NULL; /* force thread-safe failure report */
}
+#endif
+#if !defined(HAVE_GETADDRINFO) && !defined(HAVE_GETHOSTBYNAME_R)
/* threads do this in opposite order */
hostent_p2 = gethostbyname("localhost");
p = gethostbyname(myhostname);
if (hostent_p2 != p)
{
printf("Your gethostbyname() changes the static memory area between calls\n");
- hostent_p2 = NULL; /* force thread-safe failure report */
+ hostent_p2 = NULL; /* force thread-safe failure report */
}
+#endif
thread2_done = 1;
pthread_mutex_lock(&init_mutex); /* wait for parent to test */