summaryrefslogtreecommitdiff
path: root/usr/src/lib/libadutils/common/adutils_threadfuncs.c
diff options
context:
space:
mode:
authorJulian Pullen <Julian.Pullen@Sun.COM>2009-10-02 14:03:51 +0100
committerJulian Pullen <Julian.Pullen@Sun.COM>2009-10-02 14:03:51 +0100
commitbd42852645f5ef79a1729096abd7405b72e159f7 (patch)
tree1d6db1b755d015daddd13398a57003494ae01a29 /usr/src/lib/libadutils/common/adutils_threadfuncs.c
parentd5c74c206594778658b2f1e87129ed10e879d953 (diff)
downloadillumos-joyent-bd42852645f5ef79a1729096abd7405b72e159f7.tar.gz
6822695 libadutils does not setup the mutlti threading routines for using LDAP (libldap5)
Diffstat (limited to 'usr/src/lib/libadutils/common/adutils_threadfuncs.c')
-rw-r--r--usr/src/lib/libadutils/common/adutils_threadfuncs.c276
1 files changed, 276 insertions, 0 deletions
diff --git a/usr/src/lib/libadutils/common/adutils_threadfuncs.c b/usr/src/lib/libadutils/common/adutils_threadfuncs.c
new file mode 100644
index 0000000000..21b2d87b03
--- /dev/null
+++ b/usr/src/lib/libadutils/common/adutils_threadfuncs.c
@@ -0,0 +1,276 @@
+/*
+ * CDDL HEADER START
+ *
+ * The contents of this file are subject to the terms of the
+ * Common Development and Distribution License (the "License").
+ * You may not use this file except in compliance with the License.
+ *
+ * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
+ * or http://www.opensolaris.org/os/licensing.
+ * See the License for the specific language governing permissions
+ * and limitations under the License.
+ *
+ * When distributing Covered Code, include this CDDL HEADER in each
+ * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
+ * If applicable, add the following below this CDDL HEADER, with the
+ * fields enclosed by brackets "[]" replaced with your own identifying
+ * information: Portions Copyright [yyyy] [name of copyright owner]
+ *
+ * CDDL HEADER END
+ */
+
+/*
+ * Copyright 2009 Sun Microsystems, Inc. All rights reserved.
+ * Use is subject to license terms.
+ */
+
+/*
+ * Functions for managing thread-local storage for LDAP, and in particular
+ * for managing storage of the LDAP error state.
+ */
+
+#include <ldap.h>
+#include <pthread.h>
+#include <errno.h>
+#include <note.h>
+#include <syslog.h>
+#include <string.h>
+#include "solaris-int.h" /* This is a libladp5 private include file */
+ /* which has the defintion for */
+ /* struct ldap_extra_thread_fns */
+#include "adutils_impl.h"
+
+struct adutils_lderrno {
+ int le_errno;
+ char *le_matched;
+ char *le_errmsg;
+};
+
+static void *adutils_mutex_alloc(void);
+static void adutils_mutex_free(void *mutexp);
+static int adutils_get_errno(void);
+static void adutils_set_errno(int err);
+static void adutils_set_lderrno(int err, char *matched, char *errmsg,
+ void *dummy);
+static int adutils_get_lderrno(char **matched, char **errmsg, void *dummy);
+static void adutils_lderrno_destructor(void *tsd);
+
+static pthread_key_t adutils_lderrno_key = PTHREAD_ONCE_KEY_NP;
+
+static struct ldap_thread_fns thread_fns = {
+ .ltf_mutex_alloc = adutils_mutex_alloc,
+ .ltf_mutex_free = adutils_mutex_free,
+ .ltf_mutex_lock = (int (*)(void *)) pthread_mutex_lock,
+ .ltf_mutex_unlock = (int (*)(void *)) pthread_mutex_unlock,
+ .ltf_get_errno = adutils_get_errno,
+ .ltf_set_errno = adutils_set_errno,
+ .ltf_get_lderrno = adutils_get_lderrno,
+ .ltf_set_lderrno = adutils_set_lderrno,
+ .ltf_lderrno_arg = NULL
+};
+
+struct ldap_extra_thread_fns extra_thread_fns = {
+ .ltf_threadid_fn = (void * (*)(void))pthread_self
+};
+
+
+/*
+ * Set up thread management functions for the specified LDAP session.
+ * Returns either LDAP_SUCCESS or -1.
+ */
+int
+adutils_set_thread_functions(LDAP *ld)
+{
+ int rc;
+
+ if (adutils_lderrno_key == PTHREAD_ONCE_KEY_NP) {
+ if ((rc = pthread_key_create_once_np(&adutils_lderrno_key,
+ adutils_lderrno_destructor)) != 0) {
+ logger(LOG_ERR, "adutils_set_thread_functions() "
+ "pthread_key_create_once_np failed (%s)",
+ strerror(rc));
+ rc = -1;
+ return (rc);
+ }
+ }
+
+ rc = ldap_set_option(ld, LDAP_OPT_THREAD_FN_PTRS,
+ &thread_fns);
+ if (rc != LDAP_SUCCESS) {
+ logger(LOG_ERR,
+ "ldap_set_option LDAP_OPT_THREAD_FN_PTRS failed");
+ return (rc);
+ }
+
+ rc = ldap_set_option(ld, LDAP_OPT_EXTRA_THREAD_FN_PTRS,
+ &extra_thread_fns);
+ if (rc != LDAP_SUCCESS) {
+ logger(LOG_ERR,
+ "ldap_set_option LDAP_OPT_EXTRA_THREAD_FN_PTRS failed");
+ return (rc);
+ }
+ return (rc);
+}
+
+/*
+ * Allocate a mutex.
+ */
+static
+void *
+adutils_mutex_alloc(void)
+{
+ pthread_mutex_t *mutexp;
+ int rc;
+
+ mutexp = malloc(sizeof (pthread_mutex_t));
+ if (mutexp == NULL) {
+ logger(LOG_ERR,
+ "adutils_mutex_alloc: malloc failed (%s)",
+ strerror(errno));
+ return (NULL);
+ }
+
+ rc = pthread_mutex_init(mutexp, NULL);
+ if (rc != 0) {
+ logger(LOG_ERR,
+ "adutils_mutex_alloc: "
+ "pthread_mutex_init failed (%s)",
+ strerror(rc));
+ free(mutexp);
+ return (NULL);
+ }
+ return (mutexp);
+}
+
+/*
+ * Free a mutex.
+ */
+static
+void
+adutils_mutex_free(void *mutexp)
+{
+ (void) pthread_mutex_destroy((pthread_mutex_t *)mutexp);
+ free(mutexp);
+}
+
+/*
+ * Get the thread's local errno.
+ */
+static
+int
+adutils_get_errno(void)
+{
+ return (errno);
+}
+
+/*
+ * Set the thread's local errno.
+ */
+static
+void
+adutils_set_errno(int err)
+{
+ errno = err;
+}
+
+/*
+ * Get a pointer to the thread's local LDAP error state structure.
+ * Lazily allocate the thread-local storage, so that we don't need
+ * initialization when each thread starts.
+ */
+static
+struct adutils_lderrno *
+adutils_get_lderrno_struct(void)
+{
+ struct adutils_lderrno *le;
+ int rc;
+
+ le = pthread_getspecific(adutils_lderrno_key);
+ if (le == NULL) {
+ le = calloc(1, sizeof (*le));
+ if (le == NULL) {
+ logger(LOG_ERR,
+ "adutils_get_lderrno_struct: calloc failed (%s)",
+ strerror(errno));
+ return (NULL);
+ }
+ rc = pthread_setspecific(adutils_lderrno_key, le);
+ if (rc != 0) {
+ logger(LOG_ERR,
+ "adutils_get_lderrno_struct: "
+ "pthread_setspecific failed (%s)",
+ strerror(rc));
+ free(le);
+ return (NULL);
+ }
+ }
+
+ return (le);
+}
+
+/*
+ * Store an error report in the thread's local LDAP error state structure.
+ */
+static
+void
+adutils_set_lderrno(int err, char *matched, char *errmsg, void *dummy)
+{
+ NOTE(ARGUNUSED(dummy))
+ struct adutils_lderrno *le;
+
+ le = adutils_get_lderrno_struct();
+ if (le != NULL) {
+ le->le_errno = err;
+ if (le->le_matched != NULL)
+ ldap_memfree(le->le_matched);
+ le->le_matched = matched;
+ if (le->le_errmsg != NULL)
+ ldap_memfree(le->le_errmsg);
+ le->le_errmsg = errmsg;
+ }
+}
+
+/*
+ * Retrieve an error report from the thread's local LDAP error state structure.
+ */
+static
+int
+adutils_get_lderrno(char **matched, char **errmsg, void *dummy)
+{
+ NOTE(ARGUNUSED(dummy))
+ struct adutils_lderrno *le;
+ static struct adutils_lderrno empty = { LDAP_SUCCESS, NULL, NULL };
+
+ le = adutils_get_lderrno_struct();
+ if (le == NULL)
+ le = &empty;
+
+ if (matched != NULL)
+ *matched = le->le_matched;
+ if (errmsg != NULL)
+ *errmsg = le->le_errmsg;
+ return (le->le_errno);
+}
+
+/*
+ * Free the thread's local LDAP error state structure.
+ */
+static
+void
+adutils_lderrno_destructor(void *tsd)
+{
+ struct adutils_lderrno *le = tsd;
+
+ if (le == NULL)
+ return;
+
+ if (le->le_matched != NULL) {
+ ldap_memfree(le->le_matched);
+ le->le_matched = NULL;
+ }
+ if (le->le_errmsg != NULL) {
+ ldap_memfree(le->le_errmsg);
+ le->le_errmsg = NULL;
+ }
+ free(le);
+}