mirror of
https://gitlab.isc.org/isc-projects/bind9
synced 2025-08-22 10:10:06 +00:00
hp implementation requires an object for each thread accessing a hazard pointer. previous implementation had a hardcoded HP_MAX_THREAD value of 128, which failed on machines with lots of CPU cores (named uses 3n threads). We make isc__hp_max_threads configurable at startup, with the value set to 4*named_g_cpus. It's also important for this value not to be too big as we do linear searches on a list.
140 lines
4.5 KiB
C
140 lines
4.5 KiB
C
/*
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
|
*
|
|
* 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 http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
* information regarding copyright ownership.
|
|
*/
|
|
|
|
/*
|
|
* Hazard Pointer implementation.
|
|
*
|
|
* This work is based on C++ code available from:
|
|
* https://github.com/pramalhe/ConcurrencyFreaks/
|
|
*
|
|
* Copyright (c) 2014-2016, Pedro Ramalhete, Andreia Correia
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* * Neither the name of Concurrency Freaks nor the
|
|
* names of its contributors may be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
|
|
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
|
* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER>
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
|
* THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <isc/atomic.h>
|
|
#include <isc/string.h>
|
|
#include <isc/mem.h>
|
|
#include <isc/types.h>
|
|
#include <isc/util.h>
|
|
|
|
/*%
|
|
* Hazard pointers are a mechanism for protecting objects in memory
|
|
* from being deleted by other threads while in use. This allows
|
|
* safe lock-free data structures.
|
|
*
|
|
* This is an adaptation of the ConcurrencyFreaks implementation in C.
|
|
* More details available at https://github.com/pramalhe/ConcurrencyFreaks,
|
|
* in the file HazardPointers.hpp.
|
|
*/
|
|
|
|
typedef void
|
|
(isc_hp_deletefunc_t)(void *);
|
|
|
|
void
|
|
isc_hp_init(int max_threads);
|
|
/*%<
|
|
* Initialize hazard pointer constants - isc__hp_max_threads. If more threads
|
|
* will try to access hp it will assert.
|
|
*/
|
|
|
|
isc_hp_t *
|
|
isc_hp_new(isc_mem_t *mctx, size_t max_hps, isc_hp_deletefunc_t *deletefunc);
|
|
/*%<
|
|
* Create a new hazard pointer array of size 'max_hps' (or a reasonable
|
|
* default value if 'max_hps' is 0). The function 'deletefunc' will be
|
|
* used to delete objects protected by hazard pointers when it becomes
|
|
* safe to retire them.
|
|
*/
|
|
|
|
void
|
|
isc_hp_destroy(isc_hp_t *hp);
|
|
/*%<
|
|
* Destroy a hazard pointer array and clean up all objects protected
|
|
* by hazard pointers.
|
|
*/
|
|
|
|
void
|
|
isc_hp_clear(isc_hp_t *hp);
|
|
/*%<
|
|
* Clear all hazard pointers in the array for the current thread.
|
|
*
|
|
* Progress condition: wait-free bounded (by max_hps)
|
|
*/
|
|
|
|
void
|
|
isc_hp_clear_one(isc_hp_t *hp, int ihp);
|
|
/*%<
|
|
* Clear a specified hazard pointer in the array for the current thread.
|
|
*
|
|
* Progress condition: wait-free population oblivious.
|
|
*/
|
|
|
|
uintptr_t
|
|
isc_hp_protect(isc_hp_t *hp, int ihp, atomic_uintptr_t *atom);
|
|
/*%<
|
|
* Protect an object referenced by 'atom' with a hazard pointer for the
|
|
* current thread.
|
|
*
|
|
* Progress condition: lock-free.
|
|
*/
|
|
|
|
uintptr_t
|
|
isc_hp_protect_ptr(isc_hp_t *hp, int ihp, atomic_uintptr_t ptr);
|
|
/*%<
|
|
* This returns the same value that is passed as ptr, which is sometimes
|
|
* useful.
|
|
*
|
|
* Progress condition: wait-free population oblivious.
|
|
*/
|
|
|
|
uintptr_t
|
|
isc_hp_protect_release(isc_hp_t *hp, int ihp, atomic_uintptr_t ptr);
|
|
/*%<
|
|
* Same as isc_hp_protect_ptr(), but explicitly uses memory_order_release.
|
|
*
|
|
* Progress condition: wait-free population oblivious.
|
|
*/
|
|
|
|
void
|
|
isc_hp_retire(isc_hp_t *hp, uintptr_t ptr);
|
|
/*%<
|
|
* Retire an object that is no longer in use by any thread, calling
|
|
* the delete function that was specified in isc_hp_new().
|
|
*
|
|
* Progress condition: wait-free bounded (by the number of threads squared)
|
|
*/
|