mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 14:06:27 +00:00 
			
		
		
		
	add 'show threads' and 'show profile' commands.
These are momstly debugging tools for developers, a bit documented in the header files (utils.h), although more documentation is definitely necessary. The performance impact is close to zero(*) so there is no need to compile it conditionally. (*) not completely true - thread destruction still needs to search a list _but_ this can be easily optimized if we end up with hundreds of active threads (in which case, though, the problem is clearly elsewhere). git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@19544 65c4cc65-6c06-0410-ace0-fbb531ad65f3
This commit is contained in:
		
							
								
								
									
										211
									
								
								asterisk.c
									
									
									
									
									
								
							
							
						
						
									
										211
									
								
								asterisk.c
									
									
									
									
									
								
							| @@ -74,6 +74,9 @@ | |||||||
| #include <grp.h> | #include <grp.h> | ||||||
| #include <pwd.h> | #include <pwd.h> | ||||||
| #include <sys/stat.h> | #include <sys/stat.h> | ||||||
|  | #ifdef linux | ||||||
|  | #include <sys/prctl.h> | ||||||
|  | #endif | ||||||
| #include <regex.h> | #include <regex.h> | ||||||
|  |  | ||||||
| #ifdef linux | #ifdef linux | ||||||
| @@ -272,6 +275,208 @@ void ast_unregister_file_version(const char *file) | |||||||
| 		free(find); | 		free(find); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | struct thread_list_t { | ||||||
|  | 	AST_LIST_ENTRY(thread_list_t) list; | ||||||
|  | 	char *name; | ||||||
|  | 	pthread_t id; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static AST_LIST_HEAD_STATIC(thread_list, thread_list_t); | ||||||
|  |  | ||||||
|  | static char show_threads_help[] = | ||||||
|  | "Usage: show threads\n" | ||||||
|  | "       List threads currently active in the system.\n"; | ||||||
|  |  | ||||||
|  | void ast_register_thread(char *name) | ||||||
|  | {  | ||||||
|  | 	struct thread_list_t *new = ast_calloc(1, sizeof(*new)); | ||||||
|  |  | ||||||
|  | 	if (!new) | ||||||
|  | 		return; | ||||||
|  | 	new->id = pthread_self(); | ||||||
|  | 	new->name = name; /* this was a copy already */ | ||||||
|  | 	AST_LIST_LOCK(&thread_list); | ||||||
|  | 	AST_LIST_INSERT_HEAD(&thread_list, new, list); | ||||||
|  | 	AST_LIST_UNLOCK(&thread_list); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void ast_unregister_thread(void *id) | ||||||
|  | { | ||||||
|  | 	struct thread_list_t *x; | ||||||
|  |  | ||||||
|  | 	AST_LIST_LOCK(&thread_list); | ||||||
|  | 	AST_LIST_TRAVERSE_SAFE_BEGIN(&thread_list, x, list) { | ||||||
|  | 		if ((void *)x->id == id) { | ||||||
|  | 			AST_LIST_REMOVE_CURRENT(&thread_list, list); | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	AST_LIST_TRAVERSE_SAFE_END; | ||||||
|  | 	AST_LIST_UNLOCK(&thread_list); | ||||||
|  | 	if (x) { | ||||||
|  | 		free(x->name); | ||||||
|  | 		free(x); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int handle_show_threads(int fd, int argc, char *argv[]) | ||||||
|  | { | ||||||
|  | 	int count = 0; | ||||||
|  | 	struct thread_list_t *cur; | ||||||
|  |  | ||||||
|  | 	AST_LIST_LOCK(&thread_list); | ||||||
|  | 	AST_LIST_TRAVERSE(&thread_list, cur, list) { | ||||||
|  | 		ast_cli(fd, "%p %s\n", (void *)cur->id, cur->name); | ||||||
|  | 		count++; | ||||||
|  | 	} | ||||||
|  |         AST_LIST_UNLOCK(&thread_list); | ||||||
|  | 	ast_cli(fd, "%d threads listed.\n", count); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | struct profile_entry { | ||||||
|  | 	const char *name; | ||||||
|  | 	uint64_t	scale;	/* if non-zero, values are scaled by this */ | ||||||
|  | 	int64_t	mark; | ||||||
|  | 	int64_t	value; | ||||||
|  | 	int64_t	events; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | struct profile_data { | ||||||
|  | 	int entries; | ||||||
|  | 	int max_size; | ||||||
|  | 	struct profile_entry e[0]; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | static struct profile_data *prof_data; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * allocates a counter with a given name and scale. | ||||||
|  |  * Returns the identifier of the counter. | ||||||
|  |  */ | ||||||
|  | int ast_add_profile(const char *name, uint64_t scale) | ||||||
|  | { | ||||||
|  | 	int l = sizeof(struct profile_data); | ||||||
|  | 	int n = 10;	/* default entries */ | ||||||
|  |  | ||||||
|  | 	if (prof_data == NULL) { | ||||||
|  | 		prof_data = ast_calloc(1, l + n*sizeof(struct profile_entry)); | ||||||
|  | 		if (prof_data == NULL) | ||||||
|  | 			return -1; | ||||||
|  | 		prof_data->entries = 0; | ||||||
|  | 		prof_data->max_size = n; | ||||||
|  | 	} | ||||||
|  | 	if (prof_data->entries >= prof_data->max_size) { | ||||||
|  | 		void *p; | ||||||
|  | 		n = prof_data->max_size + 20; | ||||||
|  | 		p = ast_realloc(prof_data, l + n*sizeof(struct profile_entry)); | ||||||
|  | 		if (p == NULL) | ||||||
|  | 			return -1; | ||||||
|  | 		prof_data = p; | ||||||
|  | 		prof_data->max_size = n; | ||||||
|  | 	} | ||||||
|  | 	n = prof_data->entries++; | ||||||
|  | 	prof_data->e[n].name = ast_strdup(name); | ||||||
|  | 	prof_data->e[n].value = 0; | ||||||
|  | 	prof_data->e[n].events = 0; | ||||||
|  | 	prof_data->e[n].mark = 0; | ||||||
|  | 	prof_data->e[n].scale = scale; | ||||||
|  | 	return n; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int64_t ast_profile(int i, int64_t delta) | ||||||
|  | { | ||||||
|  | 	if (!prof_data || i < 0 || i > prof_data->entries)	/* invalid index */ | ||||||
|  | 		return 0; | ||||||
|  | 	if (prof_data->e[i].scale > 1) | ||||||
|  | 		delta /= prof_data->e[i].scale; | ||||||
|  | 	prof_data->e[i].value += delta; | ||||||
|  | 	prof_data->e[i].events++; | ||||||
|  | 	return prof_data->e[i].value; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if defined ( __i386__) && (defined(__FreeBSD__) || defined(linux)) | ||||||
|  | #if defined(__FreeBSD__) | ||||||
|  | #include <machine/cpufunc.h> | ||||||
|  | #elif defined(linux) | ||||||
|  | static __inline u_int64_t | ||||||
|  | rdtsc(void) | ||||||
|  | {  | ||||||
|  | 	uint64_t rv; | ||||||
|  |  | ||||||
|  | 	__asm __volatile(".byte 0x0f, 0x31" : "=A" (rv)); | ||||||
|  | 	return (rv); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | #else	/* supply a dummy function on other platforms */ | ||||||
|  | xxx | ||||||
|  | static __inline u_int64_t | ||||||
|  | rdtsc(void) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | int64_t ast_mark(int i, int startstop) | ||||||
|  | { | ||||||
|  | 	if (!prof_data || i < 0 || i > prof_data->entries) /* invalid index */ | ||||||
|  | 		return 0; | ||||||
|  | 	if (startstop == 1) | ||||||
|  | 		prof_data->e[i].mark = rdtsc(); | ||||||
|  | 	else { | ||||||
|  | 		prof_data->e[i].mark = (rdtsc() - prof_data->e[i].mark); | ||||||
|  | 		if (prof_data->e[i].scale > 1) | ||||||
|  | 			prof_data->e[i].mark /= prof_data->e[i].scale; | ||||||
|  | 		prof_data->e[i].value += prof_data->e[i].mark; | ||||||
|  | 		prof_data->e[i].events++; | ||||||
|  | 	} | ||||||
|  | 	return prof_data->e[i].mark; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static int handle_show_profile(int fd, int argc, char *argv[]) | ||||||
|  | { | ||||||
|  | 	int i, min, max; | ||||||
|  | 	char *search = NULL; | ||||||
|  |  | ||||||
|  | 	if (prof_data == NULL) | ||||||
|  | 		return 0; | ||||||
|  |  | ||||||
|  | 	min = 0; | ||||||
|  | 	max = prof_data->entries; | ||||||
|  | 	if  (argc >= 3) { /* specific entries */ | ||||||
|  | 		if (isdigit(argv[2][0])) { | ||||||
|  | 			min = atoi(argv[2]); | ||||||
|  | 			if (argc == 4 && strcmp(argv[3], "-")) | ||||||
|  | 				max = atoi(argv[3]); | ||||||
|  | 		} else | ||||||
|  | 			search = argv[2]; | ||||||
|  | 	} | ||||||
|  | 	if (max > prof_data->entries) | ||||||
|  | 		max = prof_data->entries; | ||||||
|  | 	if (!strcmp(argv[0], "clear")) { | ||||||
|  | 		for (i= min; i < max; i++) { | ||||||
|  | 			if (!search || strstr(prof_data->e[i].name, search)) { | ||||||
|  | 				prof_data->e[i].value = 0; | ||||||
|  | 				prof_data->e[i].events = 0; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	ast_cli(fd, "profile values (%d, allocated %d)\n-------------------\n", | ||||||
|  | 		prof_data->entries, prof_data->max_size); | ||||||
|  | 	for (i = min; i < max; i++) { | ||||||
|  | 		struct profile_entry *e = &prof_data->e[i]; | ||||||
|  | 		if (!search || strstr(prof_data->e[i].name, search)) | ||||||
|  | 		    ast_cli(fd, "%6d: [%8ld] %10ld %12lld %12lld %s\n", | ||||||
|  | 			i, | ||||||
|  | 			(long)e->scale, | ||||||
|  | 			(long)e->events, (long long)e->value, | ||||||
|  | 			(long long)(e->events ? e->value / e->events : e->value), | ||||||
|  | 			e->name); | ||||||
|  | 	} | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| static char show_version_files_help[] =  | static char show_version_files_help[] =  | ||||||
| "Usage: show version files [like <pattern>]\n" | "Usage: show version files [like <pattern>]\n" | ||||||
| "       Shows the revision numbers of the files used to build this copy of Asterisk.\n" | "       Shows the revision numbers of the files used to build this copy of Asterisk.\n" | ||||||
| @@ -1231,6 +1436,12 @@ static struct ast_cli_entry core_cli[] = { | |||||||
| #if !defined(LOW_MEMORY) | #if !defined(LOW_MEMORY) | ||||||
| 	{ { "show", "version", "files", NULL }, handle_show_version_files, | 	{ { "show", "version", "files", NULL }, handle_show_version_files, | ||||||
| 	  "Show versions of files used to build Asterisk", show_version_files_help, complete_show_version_files }, | 	  "Show versions of files used to build Asterisk", show_version_files_help, complete_show_version_files }, | ||||||
|  | 	{ { "show", "threads", NULL }, handle_show_threads, | ||||||
|  | 	  "Show running threads", show_threads_help, NULL }, | ||||||
|  | 	{ { "show", "profile", NULL }, handle_show_profile, | ||||||
|  | 	  "Show profiling info"}, | ||||||
|  | 	{ { "clear", "profile", NULL }, handle_show_profile, | ||||||
|  | 	  "Clear profiling info"}, | ||||||
| #endif /* ! LOW_MEMORY */ | #endif /* ! LOW_MEMORY */ | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -15,6 +15,8 @@ | |||||||
|  * \brief Asterisk main include file. File version handling, generic pbx functions. |  * \brief Asterisk main include file. File version handling, generic pbx functions. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | #include "asterisk/compat.h" | ||||||
|  |  | ||||||
| #ifndef _ASTERISK_H | #ifndef _ASTERISK_H | ||||||
| #define _ASTERISK_H | #define _ASTERISK_H | ||||||
|  |  | ||||||
| @@ -107,6 +109,24 @@ void ast_register_file_version(const char *file, const char *version); | |||||||
|  */ |  */ | ||||||
| void ast_unregister_file_version(const char *file); | void ast_unregister_file_version(const char *file); | ||||||
|  |  | ||||||
|  | /*! | ||||||
|  |  * \brief support for event profiling | ||||||
|  |  * (note, this must be documented a lot more) | ||||||
|  |  * ast_add_profile allocates a generic 'counter' with a given name, | ||||||
|  |  * which can be shown with the command 'show profile <name>' | ||||||
|  |  * | ||||||
|  |  * The counter accumulates positive or negative values supplied by | ||||||
|  |  * ast_add_profile(), dividing them by the 'scale' value passed in the | ||||||
|  |  * create call, and also counts the number of 'events'. | ||||||
|  |  * Values can also be taked by the TSC counter on ia32 architectures, | ||||||
|  |  * in which case you can mark the start of an event calling ast_mark(id, 1) | ||||||
|  |  * and then the end of the event with ast_mark(id, 0). | ||||||
|  |  * For non-i386 architectures, these two calls return 0. | ||||||
|  |  */ | ||||||
|  | int ast_add_profile(const char *, uint64_t scale); | ||||||
|  | int64_t ast_profile(int, int64_t); | ||||||
|  | int64_t ast_mark(int, int start1_stop0); | ||||||
|  |  | ||||||
| /*! | /*! | ||||||
|  * \brief Register/unregister a source code file with the core. |  * \brief Register/unregister a source code file with the core. | ||||||
|  * \param file the source file name |  * \param file the source file name | ||||||
| @@ -129,6 +149,20 @@ void ast_unregister_file_version(const char *file); | |||||||
|  * revision number. |  * revision number. | ||||||
|  */ |  */ | ||||||
| #if defined(__GNUC__) && !defined(LOW_MEMORY) | #if defined(__GNUC__) && !defined(LOW_MEMORY) | ||||||
|  | #ifdef MTX_PROFILE | ||||||
|  | #define	HAVE_MTX_PROFILE	/* used in lock.h */ | ||||||
|  | #define ASTERISK_FILE_VERSION(file, version) \ | ||||||
|  | 	static int mtx_prof = -1;       /* profile mutex */	\ | ||||||
|  | 	static void __attribute__((constructor)) __register_file_version(void) \ | ||||||
|  | 	{ \ | ||||||
|  | 		mtx_prof = ast_add_profile("mtx_lock_" file, 0);	\ | ||||||
|  | 		ast_register_file_version(file, version); \ | ||||||
|  | 	} \ | ||||||
|  | 	static void __attribute__((destructor)) __unregister_file_version(void) \ | ||||||
|  | 	{ \ | ||||||
|  | 		ast_unregister_file_version(file); \ | ||||||
|  | 	} | ||||||
|  | #else | ||||||
| #define ASTERISK_FILE_VERSION(file, version) \ | #define ASTERISK_FILE_VERSION(file, version) \ | ||||||
| 	static void __attribute__((constructor)) __register_file_version(void) \ | 	static void __attribute__((constructor)) __register_file_version(void) \ | ||||||
| 	{ \ | 	{ \ | ||||||
| @@ -138,6 +172,7 @@ void ast_unregister_file_version(const char *file); | |||||||
| 	{ \ | 	{ \ | ||||||
| 		ast_unregister_file_version(file); \ | 		ast_unregister_file_version(file); \ | ||||||
| 	} | 	} | ||||||
|  | #endif | ||||||
| #elif !defined(LOW_MEMORY) /* ! __GNUC__  && ! LOW_MEMORY*/ | #elif !defined(LOW_MEMORY) /* ! __GNUC__  && ! LOW_MEMORY*/ | ||||||
| #define ASTERISK_FILE_VERSION(file, x) static const char __file_version[] = x; | #define ASTERISK_FILE_VERSION(file, x) static const char __file_version[] = x; | ||||||
| #else /* LOW_MEMORY */ | #else /* LOW_MEMORY */ | ||||||
|   | |||||||
| @@ -80,6 +80,7 @@ int unsetenv(const char *name); | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #ifdef __linux__ | #ifdef __linux__ | ||||||
|  | #include <inttypes.h> | ||||||
| #define HAVE_STRCASESTR | #define HAVE_STRCASESTR | ||||||
| #define HAVE_STRNDUP | #define HAVE_STRNDUP | ||||||
| #define HAVE_STRNLEN | #define HAVE_STRNLEN | ||||||
|   | |||||||
| @@ -56,6 +56,23 @@ | |||||||
| #ifndef _ASTERISK_LOCK_H | #ifndef _ASTERISK_LOCK_H | ||||||
| #define _ASTERISK_LOCK_H | #define _ASTERISK_LOCK_H | ||||||
|  |  | ||||||
|  | /* internal macro to profile mutexes. Only computes the delay on | ||||||
|  |  * non-blocking calls. | ||||||
|  |  */ | ||||||
|  | #ifndef	HAVE_MTX_PROFILE | ||||||
|  | #define	__MTX_PROF	/* nothing */ | ||||||
|  | #else | ||||||
|  | #define	__MTX_PROF	{			\ | ||||||
|  | 	int i;					\ | ||||||
|  | 	/* profile only non-blocking events */	\ | ||||||
|  | 	ast_mark(mtx_prof, 1);			\ | ||||||
|  | 	i = pthread_mutex_trylock(pmutex);	\ | ||||||
|  | 	ast_mark(mtx_prof, 0);			\ | ||||||
|  | 	if (!i)					\ | ||||||
|  | 		return i;			\ | ||||||
|  | 	} | ||||||
|  | #endif	/* HAVE_MTX_PROFILE */ | ||||||
|  |  | ||||||
| #include <pthread.h> | #include <pthread.h> | ||||||
| #include <netdb.h> | #include <netdb.h> | ||||||
| #include <time.h> | #include <time.h> | ||||||
| @@ -75,7 +92,7 @@ | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #ifdef BSD | #ifdef BSD | ||||||
| #ifdef __GNUC__ | #if 0 && defined( __GNUC__) | ||||||
| #define AST_MUTEX_INIT_W_CONSTRUCTORS | #define AST_MUTEX_INIT_W_CONSTRUCTORS | ||||||
| #else | #else | ||||||
| #define AST_MUTEX_INIT_ON_FIRST_USE | #define AST_MUTEX_INIT_ON_FIRST_USE | ||||||
| @@ -264,7 +281,13 @@ static inline int __ast_pthread_mutex_lock(const char *filename, int lineno, con | |||||||
| 		time_t seconds = time(NULL); | 		time_t seconds = time(NULL); | ||||||
| 		time_t current; | 		time_t current; | ||||||
| 		do { | 		do { | ||||||
|  | #ifdef	HAVE_MTX_PROFILE | ||||||
|  | 			ast_mark(mtx_prof, 1); | ||||||
|  | #endif | ||||||
| 			res = pthread_mutex_trylock(&t->mutex); | 			res = pthread_mutex_trylock(&t->mutex); | ||||||
|  | #ifdef	HAVE_MTX_PROFILE | ||||||
|  | 			ast_mark(mtx_prof, 0); | ||||||
|  | #endif | ||||||
| 			if (res == EBUSY) { | 			if (res == EBUSY) { | ||||||
| 				current = time(NULL); | 				current = time(NULL); | ||||||
| 				if ((current - seconds) && (!((current - seconds) % 5))) { | 				if ((current - seconds) && (!((current - seconds) % 5))) { | ||||||
| @@ -279,6 +302,12 @@ static inline int __ast_pthread_mutex_lock(const char *filename, int lineno, con | |||||||
| 		} while (res == EBUSY); | 		} while (res == EBUSY); | ||||||
| 	} | 	} | ||||||
| #else | #else | ||||||
|  | #ifdef	HAVE_MTX_PROFILE | ||||||
|  | 	ast_mark(mtx_prof, 1); | ||||||
|  | 	res = pthread_mutex_trylock(&t->mutex); | ||||||
|  | 	ast_mark(mtx_prof, 0); | ||||||
|  | 	if (res) | ||||||
|  | #endif | ||||||
| 	res = pthread_mutex_lock(&t->mutex); | 	res = pthread_mutex_lock(&t->mutex); | ||||||
| #endif /* DETECT_DEADLOCKS */ | #endif /* DETECT_DEADLOCKS */ | ||||||
|  |  | ||||||
| @@ -581,6 +610,7 @@ static void  __attribute__ ((destructor)) fini_##mutex(void) \ | |||||||
|  |  | ||||||
| static inline int ast_mutex_lock(ast_mutex_t *pmutex) | static inline int ast_mutex_lock(ast_mutex_t *pmutex) | ||||||
| { | { | ||||||
|  | 	__MTX_PROF | ||||||
| 	return pthread_mutex_lock(pmutex); | 	return pthread_mutex_lock(pmutex); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -601,8 +631,10 @@ static inline int ast_mutex_lock(ast_mutex_t *pmutex) | |||||||
| { | { | ||||||
| 	if (*pmutex == (ast_mutex_t)AST_MUTEX_KIND) | 	if (*pmutex == (ast_mutex_t)AST_MUTEX_KIND) | ||||||
| 		ast_mutex_init(pmutex); | 		ast_mutex_init(pmutex); | ||||||
|  | 	__MTX_PROF | ||||||
| 	return pthread_mutex_lock(pmutex); | 	return pthread_mutex_lock(pmutex); | ||||||
| } | } | ||||||
|  |  | ||||||
| static inline int ast_mutex_trylock(ast_mutex_t *pmutex) | static inline int ast_mutex_trylock(ast_mutex_t *pmutex) | ||||||
| { | { | ||||||
| 	if (*pmutex == (ast_mutex_t)AST_MUTEX_KIND) | 	if (*pmutex == (ast_mutex_t)AST_MUTEX_KIND) | ||||||
| @@ -616,6 +648,7 @@ static inline int ast_mutex_trylock(ast_mutex_t *pmutex) | |||||||
|  |  | ||||||
| static inline int ast_mutex_lock(ast_mutex_t *pmutex) | static inline int ast_mutex_lock(ast_mutex_t *pmutex) | ||||||
| { | { | ||||||
|  | 	__MTX_PROF | ||||||
| 	return pthread_mutex_lock(pmutex); | 	return pthread_mutex_lock(pmutex); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -225,8 +225,14 @@ static force_inline int inaddrcmp(const struct sockaddr_in *sin1, const struct s | |||||||
| } | } | ||||||
|  |  | ||||||
| #define AST_STACKSIZE 256 * 1024 | #define AST_STACKSIZE 256 * 1024 | ||||||
| #define ast_pthread_create(a,b,c,d) ast_pthread_create_stack(a,b,c,d,0) |  | ||||||
| int ast_pthread_create_stack(pthread_t *thread, pthread_attr_t *attr, void *(*start_routine)(void *), void *data, size_t stacksize); | void ast_register_thread(char *name); | ||||||
|  | void ast_unregister_thread(void *id); | ||||||
|  |  | ||||||
|  | #define ast_pthread_create(a,b,c,d) ast_pthread_create_stack(a,b,c,d,0, \ | ||||||
|  | 	 __FILE__, __FUNCTION__, __LINE__, #c) | ||||||
|  | int ast_pthread_create_stack(pthread_t *thread, pthread_attr_t *attr, void *(*start_routine)(void *), void *data, size_t stacksize, | ||||||
|  | 	const char *file, const char *caller, int line, const char *start_fn); | ||||||
|  |  | ||||||
| /*! | /*! | ||||||
| 	\brief Process a string to find and replace characters | 	\brief Process a string to find and replace characters | ||||||
|   | |||||||
							
								
								
									
										47
									
								
								utils.c
									
									
									
									
									
								
							
							
						
						
									
										47
									
								
								utils.c
									
									
									
									
									
								
							| @@ -509,8 +509,42 @@ int ast_utils_init(void) | |||||||
| #undef pthread_create /* For ast_pthread_create function only */ | #undef pthread_create /* For ast_pthread_create function only */ | ||||||
| #endif /* !__linux__ */ | #endif /* !__linux__ */ | ||||||
|  |  | ||||||
| int ast_pthread_create_stack(pthread_t *thread, pthread_attr_t *attr, void *(*start_routine)(void *), void *data, size_t stacksize) | /* | ||||||
|  |  * support for 'show threads'. The start routine is wrapped by | ||||||
|  |  * dummy_start(), so that ast_register_thread() and | ||||||
|  |  * ast_unregister_thread() know the thread identifier. | ||||||
|  |  */ | ||||||
|  | struct thr_arg { | ||||||
|  | 	void *(*start_routine)(void *); | ||||||
|  | 	void *data; | ||||||
|  | 	char *name; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * on OS/X, pthread_cleanup_push() and pthread_cleanup_pop() | ||||||
|  |  * are odd macros which start and end a block, so they _must_ be | ||||||
|  |  * used in pairs (the latter with a '1' argument to call the | ||||||
|  |  * handler on exit. | ||||||
|  |  * On BSD we don't need this, but we keep it for compatibility with the MAC. | ||||||
|  |  */ | ||||||
|  | static void *dummy_start(void *data) | ||||||
| { | { | ||||||
|  | 	void *ret; | ||||||
|  | 	struct thr_arg a = *((struct thr_arg *)data);	/* make a local copy */ | ||||||
|  |  | ||||||
|  | 	free(data); | ||||||
|  | 	ast_register_thread(a.name); | ||||||
|  | 	pthread_cleanup_push(ast_unregister_thread, (void *)pthread_self());	/* on unregister */ | ||||||
|  | 	ret = a.start_routine(a.data); | ||||||
|  | 	pthread_cleanup_pop(1); | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int ast_pthread_create_stack(pthread_t *thread, pthread_attr_t *attr, void *(*start_routine)(void *), void *data, size_t stacksize, | ||||||
|  | 	const char *file, const char *caller, int line, const char *start_fn) | ||||||
|  | { | ||||||
|  | 	struct thr_arg *a; | ||||||
|  |  | ||||||
| 	pthread_attr_t lattr; | 	pthread_attr_t lattr; | ||||||
| 	if (!attr) { | 	if (!attr) { | ||||||
| 		pthread_attr_init(&lattr); | 		pthread_attr_init(&lattr); | ||||||
| @@ -534,6 +568,17 @@ int ast_pthread_create_stack(pthread_t *thread, pthread_attr_t *attr, void *(*st | |||||||
| 	errno = pthread_attr_setstacksize(attr, stacksize); | 	errno = pthread_attr_setstacksize(attr, stacksize); | ||||||
| 	if (errno) | 	if (errno) | ||||||
| 		ast_log(LOG_WARNING, "pthread_attr_setstacksize returned non-zero: %s\n", strerror(errno)); | 		ast_log(LOG_WARNING, "pthread_attr_setstacksize returned non-zero: %s\n", strerror(errno)); | ||||||
|  | 	a = ast_malloc(sizeof(*a)); | ||||||
|  | 	if (!a) | ||||||
|  | 		ast_log(LOG_WARNING, "no memory, thread %s will not be listed\n", start_fn); | ||||||
|  | 	else {	/* remap parameters */ | ||||||
|  | 		a->start_routine = start_routine; | ||||||
|  | 		a->data = data; | ||||||
|  | 		start_routine = dummy_start; | ||||||
|  | 		asprintf(&a->name, "%-20s started at [%5d] %s %s()", | ||||||
|  | 			start_fn, line, file, caller); | ||||||
|  | 		data = a; | ||||||
|  | 	} | ||||||
| 	return pthread_create(thread, attr, start_routine, data); /* We're in ast_pthread_create, so it's okay */ | 	return pthread_create(thread, attr, start_routine, data); /* We're in ast_pthread_create, so it's okay */ | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -84,6 +84,25 @@ void ast_unregister_file_version(const char *file) | |||||||
| { | { | ||||||
| } | } | ||||||
|  |  | ||||||
|  | int ast_add_profile(const char *, uint64_t scale); | ||||||
|  | int ast_add_profile(const char *s, uint64_t scale) | ||||||
|  | { | ||||||
|  | 	return -1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int64_t ast_profile(int, int64_t); | ||||||
|  | int64_t ast_profile(int key, int64_t val) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | int64_t ast_mark(int, int start1_stop0); | ||||||
|  | int64_t ast_mark(int key, int start1_stop0) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /* end of dummy functions */ | ||||||
|  |  | ||||||
| static struct ast_chan *find_chan(char *name) | static struct ast_chan *find_chan(char *name) | ||||||
| { | { | ||||||
| 	struct ast_chan *chan; | 	struct ast_chan *chan; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user