/* * Implementation of safe memory reclamation scheme using * quiescent states. * * This is derived from the "GUS" safe memory reclamation technique * in FreeBSD written by Jeffrey Roberson. It is heavily modified. Any bugs * in this code are likely due to the modifications. * * The original copyright is preserved below. * * Copyright (c) 2019,2020 Jeffrey Roberson <[email protected]> * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice unmodified, this list of conditions, and the following * disclaimer. * 2. 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. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 THE AUTHOR 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. */ #include "Python.h" #include "pycore_initconfig.h" // _PyStatus_NO_MEMORY() #include "pycore_lock.h" // PyMutex_Lock() #include "pycore_qsbr.h" #include "pycore_pystate.h" // _PyThreadState_GET() // Starting size of the array of qsbr thread states #define MIN_ARRAY_SIZE … // For _Py_qsbr_deferred_advance(): the number of deferrals before advancing // the write sequence. #define QSBR_DEFERRED_LIMIT … // Allocate a QSBR thread state from the freelist static struct _qsbr_thread_state * qsbr_allocate(struct _qsbr_shared *shared) { … } // Initialize (or reintialize) the freelist of QSBR thread states static void initialize_new_array(struct _qsbr_shared *shared) { … } // Grow the array of QSBR thread states. Returns 0 on success, -1 on failure. static int grow_thread_array(struct _qsbr_shared *shared) { … } uint64_t _Py_qsbr_advance(struct _qsbr_shared *shared) { … } uint64_t _Py_qsbr_deferred_advance(struct _qsbr_thread_state *qsbr) { … } static uint64_t qsbr_poll_scan(struct _qsbr_shared *shared) { … } bool _Py_qsbr_poll(struct _qsbr_thread_state *qsbr, uint64_t goal) { … } void _Py_qsbr_attach(struct _qsbr_thread_state *qsbr) { … } void _Py_qsbr_detach(struct _qsbr_thread_state *qsbr) { … } Py_ssize_t _Py_qsbr_reserve(PyInterpreterState *interp) { … } void _Py_qsbr_register(_PyThreadStateImpl *tstate, PyInterpreterState *interp, Py_ssize_t index) { … } void _Py_qsbr_unregister(PyThreadState *tstate) { … } void _Py_qsbr_fini(PyInterpreterState *interp) { … } void _Py_qsbr_after_fork(_PyThreadStateImpl *tstate) { … }