mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 22:30:28 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			263 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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 */
 |