123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262 |
- /*
- * Asterisk -- An open source telephony toolkit.
- *
- * Copyright (C) 2006, Digium, Inc.
- *
- * Russell Bryant <russell@digium.com>
- *
- * See http://www.asterisk.org for more information about
- * the Asterisk project. Please do not directly contact
- * any of the maintainers of this project for assistance;
- * the project provides a web site, mailing lists and IRC
- * channels for your use.
- *
- * This program is free software, distributed under the terms of
- * the GNU General Public License Version 2. See the LICENSE file
- * at the top of the source tree.
- */
- /*!
- * \file threadstorage.h
- * \author Russell Bryant <russell@digium.com>
- * \brief Definitions to aid in the use of thread local storage
- *
- * \arg \ref AstThreadStorage
- */
- /*!
- * \page AstThreadStorage The Asterisk Thread Storage API
- *
- *
- * The POSIX threads (pthreads) API provides the ability to define thread
- * specific data. The functions and structures defined here are intended
- * to centralize the code that is commonly used when using thread local
- * storage.
- *
- * The motivation for using this code in Asterisk is for situations where
- * storing data on a thread-specific basis can provide some amount of
- * performance benefit. For example, there are some call types in Asterisk
- * where ast_frame structures must be allocated very rapidly (easily 50, 100,
- * 200 times a second). Instead of doing the equivalent of that many calls
- * to malloc() and free() per second, thread local storage is used to keep a
- * list of unused frame structures so that they can be continuously reused.
- *
- * - \ref threadstorage.h
- */
- #ifndef ASTERISK_THREADSTORAGE_H
- #define ASTERISK_THREADSTORAGE_H
- #include "asterisk/utils.h"
- #include "asterisk/inline_api.h"
- /*!
- * \brief data for a thread locally stored variable
- */
- struct ast_threadstorage {
- pthread_once_t once; /*!< Ensure that the key is only initialized by one thread */
- pthread_key_t key; /*!< The key used to retrieve this thread's data */
- void (*key_init)(void); /*!< The function that initializes the key */
- int (*custom_init)(void *); /*!< Custom initialization function specific to the object */
- };
- #if defined(DEBUG_THREADLOCALS)
- void __ast_threadstorage_object_add(void *key, size_t len, const char *file, const char *function, unsigned int line);
- void __ast_threadstorage_object_remove(void *key);
- void __ast_threadstorage_object_replace(void *key_old, void *key_new, size_t len);
- #define THREADSTORAGE_RAW_CLEANUP(v) {}
- #else
- #define THREADSTORAGE_RAW_CLEANUP NULL
- #endif /* defined(DEBUG_THREADLOCALS) */
- /*!
- * \brief Define a thread storage variable
- *
- * \param name The name of the thread storage object
- *
- * This macro would be used to declare an instance of thread storage in a file.
- *
- * Example usage:
- * \code
- * AST_THREADSTORAGE(my_buf);
- * \endcode
- */
- #define AST_THREADSTORAGE(name) \
- AST_THREADSTORAGE_CUSTOM_SCOPE(name, NULL, ast_free_ptr, static)
- #define AST_THREADSTORAGE_PUBLIC(name) \
- AST_THREADSTORAGE_CUSTOM_SCOPE(name, NULL, ast_free_ptr,)
- #define AST_THREADSTORAGE_EXTERNAL(name) \
- extern struct ast_threadstorage name
- #define AST_THREADSTORAGE_RAW(name) \
- AST_THREADSTORAGE_CUSTOM_SCOPE(name, NULL, THREADSTORAGE_RAW_CLEANUP,)
- /*!
- * \brief Define a thread storage variable, with custom initialization and cleanup
- *
- * \param a The name of the thread storage object
- * \param b This is a custom function that will be called after each thread specific
- * object is allocated, with the allocated block of memory passed
- * as the argument.
- * \param c This is a custom function that will be called instead of ast_free
- * when the thread goes away. Note that if this is used, it *MUST*
- * call free on the allocated memory.
- *
- * Example usage:
- * \code
- * AST_THREADSTORAGE_CUSTOM(my_buf, my_init, my_cleanup);
- * \endcode
- */
- #define AST_THREADSTORAGE_CUSTOM(a,b,c) AST_THREADSTORAGE_CUSTOM_SCOPE(a,b,c,static)
- #if defined(PTHREAD_ONCE_INIT_NEEDS_BRACES)
- # define AST_PTHREAD_ONCE_INIT { PTHREAD_ONCE_INIT }
- #else
- # define AST_PTHREAD_ONCE_INIT PTHREAD_ONCE_INIT
- #endif
- #if !defined(DEBUG_THREADLOCALS)
- #define AST_THREADSTORAGE_CUSTOM_SCOPE(name, c_init, c_cleanup, scope) \
- static void __init_##name(void); \
- scope struct ast_threadstorage name = { \
- .once = AST_PTHREAD_ONCE_INIT, \
- .key_init = __init_##name, \
- .custom_init = c_init, \
- }; \
- static void __init_##name(void) \
- { \
- pthread_key_create(&(name).key, c_cleanup); \
- }
- #else /* defined(DEBUG_THREADLOCALS) */
- #define AST_THREADSTORAGE_CUSTOM_SCOPE(name, c_init, c_cleanup, scope) \
- static void __init_##name(void); \
- scope struct ast_threadstorage name = { \
- .once = AST_PTHREAD_ONCE_INIT, \
- .key_init = __init_##name, \
- .custom_init = c_init, \
- }; \
- static void __cleanup_##name(void *data) \
- { \
- __ast_threadstorage_object_remove(data); \
- c_cleanup(data); \
- } \
- static void __init_##name(void) \
- { \
- pthread_key_create(&(name).key, __cleanup_##name); \
- }
- #endif /* defined(DEBUG_THREADLOCALS) */
- /*!
- * \brief Retrieve thread storage
- *
- * \param ts This is a pointer to the thread storage structure declared by using
- * the AST_THREADSTORAGE macro. If declared with
- * AST_THREADSTORAGE(my_buf), then this argument would be (&my_buf).
- * \param init_size This is the amount of space to be allocated the first time
- * this thread requests its data. Thus, this should be the size that the
- * code accessing this thread storage is assuming the size to be.
- *
- * \return This function will return the thread local storage associated with
- * the thread storage management variable passed as the first argument.
- * The result will be NULL in the case of a memory allocation error.
- *
- * Example usage:
- * \code
- * AST_THREADSTORAGE(my_buf);
- * #define MY_BUF_SIZE 128
- * ...
- * void my_func(const char *fmt, ...)
- * {
- * void *buf;
- *
- * if (!(buf = ast_threadstorage_get(&my_buf, MY_BUF_SIZE)))
- * return;
- * ...
- * }
- * \endcode
- */
- #if !defined(DEBUG_THREADLOCALS)
- AST_INLINE_API(
- void *ast_threadstorage_get(struct ast_threadstorage *ts, size_t init_size),
- {
- void *buf;
- pthread_once(&ts->once, ts->key_init);
- if (!(buf = pthread_getspecific(ts->key))) {
- if (!(buf = ast_calloc(1, init_size))) {
- return NULL;
- }
- if (ts->custom_init && ts->custom_init(buf)) {
- ast_free(buf);
- return NULL;
- }
- pthread_setspecific(ts->key, buf);
- }
- return buf;
- }
- )
- #else /* defined(DEBUG_THREADLOCALS) */
- AST_INLINE_API(
- void *__ast_threadstorage_get(struct ast_threadstorage *ts, size_t init_size, const char *file, const char *function, unsigned int line),
- {
- void *buf;
- pthread_once(&ts->once, ts->key_init);
- if (!(buf = pthread_getspecific(ts->key))) {
- if (!(buf = ast_calloc(1, init_size))) {
- return NULL;
- }
- if (ts->custom_init && ts->custom_init(buf)) {
- ast_free(buf);
- return NULL;
- }
- pthread_setspecific(ts->key, buf);
- __ast_threadstorage_object_add(buf, init_size, file, function, line);
- }
- return buf;
- }
- )
- #define ast_threadstorage_get(ts, init_size) __ast_threadstorage_get(ts, init_size, __FILE__, __PRETTY_FUNCTION__, __LINE__)
- #endif /* defined(DEBUG_THREADLOCALS) */
- /*!
- * \brief Retrieve a raw pointer from threadstorage.
- * \param ts Threadstorage object to operate on.
- *
- * \return A pointer associated with the current thread, NULL
- * if no pointer is associated yet.
- *
- * \note This should only be used on threadstorage declared
- * by AST_THREADSTORAGE_RAW unless you really know what
- * you are doing.
- */
- AST_INLINE_API(
- void *ast_threadstorage_get_ptr(struct ast_threadstorage *ts),
- {
- pthread_once(&ts->once, ts->key_init);
- return pthread_getspecific(ts->key);
- }
- )
- /*!
- * \brief Set a raw pointer from threadstorage.
- * \param ts Threadstorage object to operate on.
- *
- * \retval 0 Success
- * \retval non-zero Failure
- *
- * \note This should only be used on threadstorage declared
- * by AST_THREADSTORAGE_RAW unless you really know what
- * you are doing.
- */
- AST_INLINE_API(
- int ast_threadstorage_set_ptr(struct ast_threadstorage *ts, void *ptr),
- {
- pthread_once(&ts->once, ts->key_init);
- return pthread_setspecific(ts->key, ptr);
- }
- )
- #endif /* ASTERISK_THREADSTORAGE_H */
|