mirror of
https://gitlab.isc.org/isc-projects/bind9
synced 2025-09-05 17:15:31 +00:00
Some POSIX threads implementations (e.g. FreeBSD's libthr) allocate memory on the heap when pthread_cond_init() is called. Every call to that function must be accompanied by a corresponding call to pthread_cond_destroy() or else the memory allocated for the condition variable will leak. jemalloc can be used for detecting memory allocations which are not released by a process when it exits. Unfortunately, since jemalloc is also the system allocator on FreeBSD and a special (profiling-enabled) build of jemalloc is required for memory leak detection, this method cannot be used for detecting leaked memory allocated by libthr on a stock FreeBSD installation. However, libthr's behavior can be emulated on any platform by implementing alternative versions of libisc functions for creating and destroying condition variables that allocate memory using malloc() and release it using free(). This enables using jemalloc for detecting missing pthread_cond_destroy() calls on any platform on which it works reliably. When the newly introduced ISC_TRACK_PTHREADS_OBJECTS preprocessor macro is set, allocate isc_condition_t structures on the heap in isc_condition_init() and free them in isc_condition_destroy(). Reuse existing condition variable macros (after renaming them appropriately) for other operations.
69 lines
1.7 KiB
C
69 lines
1.7 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
* information regarding copyright ownership.
|
|
*/
|
|
|
|
/*! \file */
|
|
|
|
#include <errno.h>
|
|
|
|
#include <isc/condition.h>
|
|
#include <isc/strerr.h>
|
|
#include <isc/string.h>
|
|
#include <isc/time.h>
|
|
#include <isc/util.h>
|
|
|
|
isc_result_t
|
|
isc__condition_waituntil(pthread_cond_t *c, pthread_mutex_t *m, isc_time_t *t) {
|
|
int presult;
|
|
isc_result_t result;
|
|
struct timespec ts;
|
|
char strbuf[ISC_STRERRORSIZE];
|
|
|
|
REQUIRE(c != NULL && m != NULL && t != NULL);
|
|
|
|
/*
|
|
* POSIX defines a timespec's tv_sec as time_t.
|
|
*/
|
|
result = isc_time_secondsastimet(t, &ts.tv_sec);
|
|
|
|
/*
|
|
* If we have a range error ts.tv_sec is most probably a signed
|
|
* 32 bit value. Set ts.tv_sec to INT_MAX. This is a kludge.
|
|
*/
|
|
if (result == ISC_R_RANGE) {
|
|
ts.tv_sec = INT_MAX;
|
|
} else if (result != ISC_R_SUCCESS) {
|
|
return (result);
|
|
}
|
|
|
|
/*!
|
|
* POSIX defines a timespec's tv_nsec as long. isc_time_nanoseconds
|
|
* ensures its return value is < 1 billion, which will fit in a long.
|
|
*/
|
|
ts.tv_nsec = (long)isc_time_nanoseconds(t);
|
|
|
|
do {
|
|
presult = pthread_cond_timedwait(c, m, &ts);
|
|
if (presult == 0) {
|
|
return (ISC_R_SUCCESS);
|
|
}
|
|
if (presult == ETIMEDOUT) {
|
|
return (ISC_R_TIMEDOUT);
|
|
}
|
|
} while (presult == EINTR);
|
|
|
|
strerror_r(presult, strbuf, sizeof(strbuf));
|
|
UNEXPECTED_ERROR(__FILE__, __LINE__,
|
|
"pthread_cond_timedwait() returned %s", strbuf);
|
|
return (ISC_R_UNEXPECTED);
|
|
}
|