mostly-reentrant resolver/getaddrinfo(3)
Brian F. Feldman
green at FreeBSD.org
Mon Feb 9 18:52:54 PST 2004
Alright, here we go! I simplified some things out a bit and used
pthread_once(3) to make things look a little cleaner. The RES_BOGUS flag
was unnecessary, and now single-threaded programs and the first thread of
multi-threaded programs do not incur the allocation of per-thread resolver
storage. I also allocated all the per-thread storage at once because the
user is not privy to that, anyway. I've gotten good feedback so far, and
the latest round of changes at the least "works for me" :) Try it in your
multi-tab Mozilla!
Index: include/resolv.h
===================================================================
RCS file: /usr/ncvs/src/include/resolv.h,v
retrieving revision 1.23
diff -u -r1.23 resolv.h
--- include/resolv.h 7 Dec 2003 12:32:23 -0000 1.23
+++ include/resolv.h 10 Feb 2004 00:55:35 -0000
@@ -200,7 +200,12 @@
char * humanname; /* Its fun name, like "mail exchanger" */
};
-extern struct __res_state _res;
+__BEGIN_DECLS
+extern struct __res_state *___res(void);
+extern struct __res_state_ext *___res_ext(void);
+__END_DECLS
+#define _res (*___res())
+#define _res_ext (*___res_ext())
/* for INET6 */
extern struct __res_state_ext _res_ext;
Index: lib/libc/include/reentrant.h
===================================================================
RCS file: /usr/ncvs/src/lib/libc/include/reentrant.h,v
retrieving revision 1.2
diff -u -r1.2 reentrant.h
--- lib/libc/include/reentrant.h 1 Nov 2002 09:37:17 -0000 1.2
+++ lib/libc/include/reentrant.h 10 Feb 2004 01:11:45 -0000
@@ -94,10 +94,12 @@
#define mutex_t pthread_mutex_t
#define cond_t pthread_cond_t
#define rwlock_t pthread_rwlock_t
+#define once_t pthread_once_t
#define thread_key_t pthread_key_t
#define MUTEX_INITIALIZER PTHREAD_MUTEX_INITIALIZER
#define RWLOCK_INITIALIZER PTHREAD_RWLOCK_INITIALIZER
+#define ONCE_INITIALIZER PTHREAD_ONCE_INIT
#define mutex_init(m, a) _pthread_mutex_init(m, a)
#define mutex_lock(m) if (__isthreaded) \
@@ -127,6 +129,7 @@
#define thr_getspecific(k) _pthread_getspecific(k)
#define thr_sigsetmask(f, n, o) _pthread_sigmask(f, n, o)
+#define thr_once(o, i) _pthread_once(o, i)
#define thr_self() _pthread_self()
#define thr_exit(x) _pthread_exit(x)
#define thr_main() _pthread_main_np()
Index: lib/libc/net/getaddrinfo.c
===================================================================
RCS file: /usr/ncvs/src/lib/libc/net/getaddrinfo.c,v
retrieving revision 1.48
diff -u -r1.48 getaddrinfo.c
--- lib/libc/net/getaddrinfo.c 30 Oct 2003 17:36:53 -0000 1.48
+++ lib/libc/net/getaddrinfo.c 6 Feb 2004 06:20:23 -0000
@@ -1511,6 +1511,7 @@
return 0;
}
+ THREAD_UNLOCK();
switch (_nsdispatch(&result, dtab, NSDB_HOSTS, "getaddrinfo",
default_dns_files, hostname, pai)) {
case NS_TRYAGAIN:
@@ -1524,20 +1525,20 @@
goto free;
case NS_SUCCESS:
error = 0;
+ THREAD_LOCK();
for (cur = result; cur; cur = cur->ai_next) {
GET_PORT(cur, servname);
/* canonname should be filled already */
}
+ THREAD_UNLOCK();
break;
}
- THREAD_UNLOCK();
*res = result;
return 0;
free:
- THREAD_UNLOCK();
if (result)
freeaddrinfo(result);
return error;
@@ -2037,6 +2038,7 @@
memset(&sentinel, 0, sizeof(sentinel));
cur = &sentinel;
+ THREAD_LOCK();
_sethtent();
while ((p = _gethtent(name, pai)) != NULL) {
cur->ai_next = p;
@@ -2044,6 +2046,7 @@
cur = cur->ai_next;
}
_endhtent();
+ THREAD_UNLOCK();
*((struct addrinfo **)rv) = sentinel.ai_next;
if (sentinel.ai_next == NULL)
@@ -2152,9 +2155,12 @@
memset(&sentinel, 0, sizeof(sentinel));
cur = &sentinel;
+ THREAD_LOCK();
if (!__ypdomain) {
- if (_yp_check(&__ypdomain) == 0)
+ if (_yp_check(&__ypdomain) == 0) {
+ THREAD_UNLOCK();
return NS_UNAVAIL;
+ }
}
if (__ypcurrent)
free(__ypcurrent);
@@ -2189,6 +2195,7 @@
cur = cur->ai_next;
}
}
+ THREAD_UNLOCK();
if (sentinel.ai_next == NULL) {
h_errno = HOST_NOT_FOUND;
Index: lib/libc/net/res_init.c
===================================================================
RCS file: /usr/ncvs/src/lib/libc/net/res_init.c,v
retrieving revision 1.31
diff -u -r1.31 res_init.c
--- lib/libc/net/res_init.c 7 Dec 2003 12:32:24 -0000 1.31
+++ lib/libc/net/res_init.c 10 Feb 2004 01:01:04 -0000
@@ -91,7 +91,11 @@
#include <unistd.h>
#include <netdb.h>
+#include "namespace.h"
+#include "reentrant.h"
+#include "un-namespace.h"
#include "res_config.h"
+#include "res_send_private.h"
static void res_setoptions(char *, char *);
@@ -106,16 +110,13 @@
#endif
/*
- * Resolver state default settings.
+ * Check structure for failed per-thread allocations.
*/
-
-struct __res_state _res
-# if defined(__BIND_RES_TEXT)
- = { RES_TIMEOUT, } /* Motorola, et al. */
-# endif
- ;
-
-struct __res_state_ext _res_ext;
+static struct res_per_thread {
+ struct __res_state res_state;
+ struct __res_state_ext res_state_ext;
+ struct __res_send_private res_send_private;
+} _res_per_thread_bogus;
/*
* Set up default settings. If the configuration file exist, the values
@@ -142,6 +143,7 @@
res_init()
{
FILE *fp;
+ struct __res_send_private *rsp;
char *cp, **pp;
int n;
char buf[MAXDNAME];
@@ -157,6 +159,19 @@
#endif
/*
+ * If allocation of memory for this thread's resolver has failed,
+ * return the error to the user.
+ */
+ if (&_res == &_res_per_thread_bogus.res_state)
+ return (-1);
+ rsp = ___res_send_private();
+ rsp->s = -1;
+ rsp->connected = 0;
+ rsp->vc = 0;
+ rsp->af = 0;
+ rsp->Qhook = NULL;
+ rsp->Rhook = NULL;
+ /*
* These three fields used to be statically initialized. This made
* it hard to use this code in a shared library. It is necessary,
* now that we're doing dynamic initialization here, that we preserve
@@ -595,6 +610,88 @@
gettimeofday(&now, NULL);
return (0xffff & (now.tv_sec ^ now.tv_usec ^ getpid()));
+}
+
+/*
+ * Resolver state default settings.
+ */
+
+#undef _res
+#undef _res_ext
+#ifdef __BIND_RES_TEXT
+struct __res_state _res = { RES_TIMEOUT }; /* Motorola, et al. */
+#else
+struct __res_state _res;
+#endif
+struct __res_state_ext _res_ext;
+static struct __res_send_private _res_send_private;
+
+static thread_key_t res_key;
+static once_t res_init_once = ONCE_INITIALIZER;
+static int res_thr_keycreated = 0;
+
+static void
+free_res(void *ptr)
+{
+ struct res_per_thread *myrsp = ptr;
+
+ if (myrsp->res_state.options & RES_INIT)
+ res_close();
+ free(myrsp);
+}
+
+static void
+res_keycreate(void)
+{
+ res_thr_keycreated = thr_keycreate(&res_key, free_res) == 0;
+}
+
+static struct res_per_thread *
+allocate_res(void)
+{
+ struct res_per_thread *myrsp;
+
+ if (thr_once(&res_init_once, res_keycreate) != 0 ||
+ !res_thr_keycreated)
+ return (&_res_per_thread_bogus);
+
+ myrsp = thr_getspecific(res_key);
+ if (myrsp != NULL)
+ return (myrsp);
+ myrsp = calloc(1, sizeof(*myrsp));
+ if (myrsp == NULL)
+ return (&_res_per_thread_bogus);
+#ifdef __BIND_RES_TEXT
+ myrsp->res_state.options = RES_TIMEOUT; /* Motorola, et al. */
+#endif
+ if (thr_setspecific(res_key, myrsp) == 0)
+ return (myrsp);
+ free(myrsp);
+ return (&_res_per_thread_bogus);
+}
+
+struct __res_state *
+___res(void)
+{
+ if (thr_main() != 0)
+ return (&_res);
+ return (&allocate_res()->res_state);
+}
+
+struct __res_state_ext *
+___res_ext(void)
+{
+ if (thr_main() != 0)
+ return (&_res_ext);
+ return (&allocate_res()->res_state_ext);
+}
+
+struct __res_send_private *
+___res_send_private(void)
+{
+ if (thr_main() != 0)
+ return (&_res_send_private);
+ return (&allocate_res()->res_send_private);
}
/*
Index: lib/libc/net/res_send.c
===================================================================
RCS file: /usr/ncvs/src/lib/libc/net/res_send.c,v
retrieving revision 1.46
diff -u -r1.46 res_send.c
--- lib/libc/net/res_send.c 6 Jan 2004 18:45:13 -0000 1.46
+++ lib/libc/net/res_send.c 6 Feb 2004 06:07:50 -0000
@@ -101,14 +101,15 @@
#include "un-namespace.h"
#include "res_config.h"
+#include "res_send_private.h"
-static int s = -1; /* socket used for communications */
-static int connected = 0; /* is the socket connected */
-static int vc = 0; /* is the socket a virtual circuit? */
-static int af = 0; /* address family of socket */
-static res_send_qhook Qhook = NULL;
-static res_send_rhook Rhook = NULL;
+#define s ___res_send_private()->s
+#define connected ___res_send_private()->connected
+#define vc ___res_send_private()->vc
+#define af ___res_send_private()->af
+#define Qhook ___res_send_private()->Qhook
+#define Rhook ___res_send_private()->Rhook
#define CAN_RECONNECT 1
@@ -123,8 +124,6 @@
fprintf args;\
__fp_nquery(query, size, stdout);\
} else {}
-static char abuf[NI_MAXHOST];
-static char pbuf[NI_MAXSERV];
static void Aerror(FILE *, char *, int, struct sockaddr *);
static void Perror(FILE *, char *, int);
@@ -138,6 +137,9 @@
int save = errno;
if (_res.options & RES_DEBUG) {
+ char abuf[NI_MAXHOST];
+ char pbuf[NI_MAXSERV];
+
if (getnameinfo(address, address->sa_len, abuf, sizeof(abuf),
pbuf, sizeof(pbuf),
NI_NUMERICHOST|NI_NUMERICSERV|NI_WITHSCOPEID) != 0) {
@@ -388,6 +390,7 @@
*/
for (try = 0; try < _res.retry; try++) {
for (ns = 0; ns < _res.nscount; ns++) {
+ char abuf[NI_MAXHOST];
struct sockaddr *nsap = get_nsaddr(ns);
socklen_t salen;
Index: lib/libc/net/res_send_private.h
===================================================================
RCS file: lib/libc/net/res_send_private.h
diff -N lib/libc/net/res_send_private.h
--- /dev/null 1 Jan 1970 00:00:00 -0000
+++ lib/libc/net/res_send_private.h 9 Feb 2004 02:53:27 -0000
@@ -0,0 +1,10 @@
+struct __res_send_private {
+ int s; /* socket used for communications */
+ int connected; /* is the socket connected */
+ int vc; /* is the socket a virtual circuit? */
+ int af; /* address family of socket */
+ res_send_qhook Qhook;
+ res_send_rhook Rhook;
+};
+
+struct __res_send_private *___res_send_private(void);
--
Brian Fundakowski Feldman \'[ FreeBSD ]''''''''''\
<> green at FreeBSD.org \ The Power to Serve! \
Opinions expressed are my own. \,,,,,,,,,,,,,,,,,,,,,,\
More information about the freebsd-hackers
mailing list