mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 14:06:27 +00:00 
			
		
		
		
	lock.c: Separate DETECT_DEADLOCKS from DEBUG_THREADS
Previously, DETECT_DEADLOCKS depended on DEBUG_THREADS.
Unfortunately, DEBUG_THREADS adds a lot of lock tracking overhead
to all of the lock lifecycle calls whereas DETECT_DEADLOCKS just
causes the lock calls to loop over trylock in 200us intervals until
the lock is obtained and spits out log messages if it takes more
than 5 seconds.  From a code perspective, the only reason they were
tied together was for logging.  So... The ifdefs in lock.c were
refactored to allow DETECT_DEADLOCKS to be enabled without
also enabling DEBUG_THREADS.
Resolves: #321
UserNote: You no longer need to select DEBUG_THREADS to use
DETECT_DEADLOCKS.  This removes a significant amount of overhead
if you just want to detect possible deadlocks vs needing full
lock tracing.
(cherry picked from commit 04183eb246)
			
			
This commit is contained in:
		
				
					committed by
					
						 Asterisk Development Team
						Asterisk Development Team
					
				
			
			
				
	
			
			
			
						parent
						
							a669ccbc3d
						
					
				
				
					commit
					6415562a14
				
			
							
								
								
									
										56
									
								
								main/lock.c
									
									
									
									
									
								
							
							
						
						
									
										56
									
								
								main/lock.c
									
									
									
									
									
								
							| @@ -53,7 +53,7 @@ static void __attribute__((constructor)) __mtx_init(void) | ||||
| #undef pthread_cond_wait | ||||
| #undef pthread_cond_timedwait | ||||
|  | ||||
| #if defined(DEBUG_THREADS) | ||||
| #if defined(DEBUG_THREADS) || defined(DETECT_DEADLOCKS) | ||||
| #define log_mutex_error(canlog, ...) \ | ||||
| 	do { \ | ||||
| 		if (canlog) { \ | ||||
| @@ -147,8 +147,8 @@ int __ast_pthread_mutex_init(int tracking, const char *filename, int lineno, con | ||||
| 	int res; | ||||
| 	pthread_mutexattr_t  attr; | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| #if defined(AST_MUTEX_INIT_W_CONSTRUCTORS) && defined(CAN_COMPARE_MUTEX_TO_INIT_VALUE) | ||||
| #if defined(DEBUG_THREADS) && defined(AST_MUTEX_INIT_W_CONSTRUCTORS) && \ | ||||
| 	defined(CAN_COMPARE_MUTEX_TO_INIT_VALUE) | ||||
| 	if ((t->mutex) != ((pthread_mutex_t) PTHREAD_MUTEX_INITIALIZER)) { | ||||
| 		int canlog = tracking && strcmp(filename, "logger.c"); | ||||
|  | ||||
| @@ -159,6 +159,7 @@ int __ast_pthread_mutex_init(int tracking, const char *filename, int lineno, con | ||||
| 	} | ||||
| #endif /* AST_MUTEX_INIT_W_CONSTRUCTORS */ | ||||
|  | ||||
| #if defined(DEBUG_THREADS) || defined(DETECT_DEADLOCKS) | ||||
| 	t->track = NULL; | ||||
| 	t->flags.tracking = tracking; | ||||
| 	t->flags.setup = 0; | ||||
| @@ -255,9 +256,12 @@ int __ast_pthread_mutex_lock(const char *filename, int lineno, const char *func, | ||||
| { | ||||
| 	int res; | ||||
|  | ||||
| #if defined(DETECT_DEADLOCKS) || defined(DEBUG_THREADS) | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| #endif | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| 	struct ast_lock_track *lt = ast_get_reentrancy(&t->track, &t->flags, 0); | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| 	struct ast_bt *bt = NULL; | ||||
|  | ||||
| 	if (lt) { | ||||
| @@ -280,7 +284,7 @@ int __ast_pthread_mutex_lock(const char *filename, int lineno, const char *func, | ||||
| 	} | ||||
| #endif /* DEBUG_THREADS */ | ||||
|  | ||||
| #if defined(DETECT_DEADLOCKS) && defined(DEBUG_THREADS) | ||||
| #if defined(DETECT_DEADLOCKS) | ||||
| 	{ | ||||
| 		time_t seconds = time(NULL); | ||||
| 		time_t wait_time, reported_wait = 0; | ||||
| @@ -297,6 +301,7 @@ int __ast_pthread_mutex_lock(const char *filename, int lineno, const char *func, | ||||
| 				if (wait_time > reported_wait && (wait_time % 5) == 0) { | ||||
| 					log_mutex_error(canlog, "%s line %d (%s): Deadlock? waited %d sec for mutex '%s'?\n", | ||||
| 							   filename, lineno, func, (int) wait_time, mutex_name); | ||||
| #ifdef DEBUG_THREADS | ||||
| 					if (lt) { | ||||
| 						ast_reentrancy_lock(lt); | ||||
| #ifdef HAVE_BKTR | ||||
| @@ -310,6 +315,7 @@ int __ast_pthread_mutex_lock(const char *filename, int lineno, const char *func, | ||||
| #endif | ||||
| 						ast_reentrancy_unlock(lt); | ||||
| 					} | ||||
| #endif | ||||
| 					reported_wait = wait_time; | ||||
| 				} | ||||
| 				usleep(200); | ||||
| @@ -666,13 +672,14 @@ int __ast_cond_timedwait(const char *filename, int lineno, const char *func, | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
| int __ast_rwlock_init(int tracking, const char *filename, int lineno, const char *func, const char *rwlock_name, ast_rwlock_t *t) | ||||
| int __ast_rwlock_init(int tracking, const char *filename, int lineno, \ | ||||
| 	const char *func, const char *rwlock_name, ast_rwlock_t *t) | ||||
| { | ||||
| 	int res; | ||||
| 	pthread_rwlockattr_t attr; | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| #if defined(AST_MUTEX_INIT_W_CONSTRUCTORS) && defined(CAN_COMPARE_MUTEX_TO_INIT_VALUE) | ||||
| #if defined(DEBUG_THREADS) && defined(AST_MUTEX_INIT_W_CONSTRUCTORS) && \ | ||||
| 	defined(CAN_COMPARE_MUTEX_TO_INIT_VALUE) | ||||
| 	if (t->lock != ((pthread_rwlock_t) __AST_RWLOCK_INIT_VALUE)) { | ||||
| 		int canlog = tracking && strcmp(filename, "logger.c"); | ||||
|  | ||||
| @@ -683,6 +690,7 @@ int __ast_rwlock_init(int tracking, const char *filename, int lineno, const char | ||||
| 	} | ||||
| #endif /* AST_MUTEX_INIT_W_CONSTRUCTORS */ | ||||
|  | ||||
| #if defined(DEBUG_THREADS) || defined(DETECT_DEADLOCKS) | ||||
| 	t->track = NULL; | ||||
| 	t->flags.tracking = tracking; | ||||
| 	t->flags.setup = 0; | ||||
| @@ -816,13 +824,17 @@ int __ast_rwlock_unlock(const char *filename, int line, const char *func, ast_rw | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
| int __ast_rwlock_rdlock(const char *filename, int line, const char *func, ast_rwlock_t *t, const char *name) | ||||
| int __ast_rwlock_rdlock(const char *filename, int line, const char *func, | ||||
| 	ast_rwlock_t *t, const char *name) | ||||
| { | ||||
| 	int res; | ||||
|  | ||||
| #if defined(DEBUG_THREADS) || defined(DETECT_DEADLOCKS) | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| #endif | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| 	struct ast_lock_track *lt = ast_get_reentrancy(&t->track, &t->flags, 0); | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| 	struct ast_bt *bt = NULL; | ||||
|  | ||||
| 	if (lt) { | ||||
| @@ -845,7 +857,7 @@ int __ast_rwlock_rdlock(const char *filename, int line, const char *func, ast_rw | ||||
| 	} | ||||
| #endif /* DEBUG_THREADS */ | ||||
|  | ||||
| #if defined(DETECT_DEADLOCKS) && defined(DEBUG_THREADS) | ||||
| #if defined(DETECT_DEADLOCKS) | ||||
| 	{ | ||||
| 		time_t seconds = time(NULL); | ||||
| 		time_t wait_time, reported_wait = 0; | ||||
| @@ -856,6 +868,7 @@ int __ast_rwlock_rdlock(const char *filename, int line, const char *func, ast_rw | ||||
| 				if (wait_time > reported_wait && (wait_time % 5) == 0) { | ||||
| 					log_mutex_error(canlog, "%s line %d (%s): Deadlock? waited %d sec for readlock '%s'?\n", | ||||
| 						filename, line, func, (int)wait_time, name); | ||||
| #ifdef DEBUG_THREADS | ||||
| 					if (lt) { | ||||
| 						ast_reentrancy_lock(lt); | ||||
| #ifdef HAVE_BKTR | ||||
| @@ -869,15 +882,16 @@ int __ast_rwlock_rdlock(const char *filename, int line, const char *func, ast_rw | ||||
| #endif | ||||
| 						ast_reentrancy_unlock(lt); | ||||
| 					} | ||||
| #endif | ||||
| 					reported_wait = wait_time; | ||||
| 				} | ||||
| 				usleep(200); | ||||
| 			} | ||||
| 		} while (res == EBUSY); | ||||
| 	} | ||||
| #else /* !DETECT_DEADLOCKS || !DEBUG_THREADS */ | ||||
| #else /* !DETECT_DEADLOCKS */ | ||||
| 	res = pthread_rwlock_rdlock(&t->lock); | ||||
| #endif /* !DETECT_DEADLOCKS || !DEBUG_THREADS */ | ||||
| #endif /* !DETECT_DEADLOCKS */ | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| 	if (!res && lt) { | ||||
| @@ -914,13 +928,17 @@ int __ast_rwlock_rdlock(const char *filename, int line, const char *func, ast_rw | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
| int __ast_rwlock_wrlock(const char *filename, int line, const char *func, ast_rwlock_t *t, const char *name) | ||||
| int __ast_rwlock_wrlock(const char *filename, int line, const char *func, \ | ||||
| 	ast_rwlock_t *t, const char *name) | ||||
| { | ||||
| 	int res; | ||||
|  | ||||
| #if defined(DEBUG_THREADS) || defined(DETECT_DEADLOCKS) | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| #endif | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| 	struct ast_lock_track *lt = ast_get_reentrancy(&t->track, &t->flags, 0); | ||||
| 	int canlog = t->flags.tracking && strcmp(filename, "logger.c"); | ||||
| 	struct ast_bt *bt = NULL; | ||||
|  | ||||
| 	if (lt) { | ||||
| @@ -943,7 +961,7 @@ int __ast_rwlock_wrlock(const char *filename, int line, const char *func, ast_rw | ||||
| 	} | ||||
| #endif /* DEBUG_THREADS */ | ||||
|  | ||||
| #if defined(DETECT_DEADLOCKS) && defined(DEBUG_THREADS) | ||||
| #ifdef DETECT_DEADLOCKS | ||||
| 	{ | ||||
| 		time_t seconds = time(NULL); | ||||
| 		time_t wait_time, reported_wait = 0; | ||||
| @@ -954,6 +972,7 @@ int __ast_rwlock_wrlock(const char *filename, int line, const char *func, ast_rw | ||||
| 				if (wait_time > reported_wait && (wait_time % 5) == 0) { | ||||
| 					log_mutex_error(canlog, "%s line %d (%s): Deadlock? waited %d sec for writelock '%s'?\n", | ||||
| 						filename, line, func, (int)wait_time, name); | ||||
| #ifdef DEBUG_THREADS | ||||
| 					if (lt) { | ||||
| 						ast_reentrancy_lock(lt); | ||||
| #ifdef HAVE_BKTR | ||||
| @@ -967,15 +986,16 @@ int __ast_rwlock_wrlock(const char *filename, int line, const char *func, ast_rw | ||||
| #endif | ||||
| 						ast_reentrancy_unlock(lt); | ||||
| 					} | ||||
| #endif | ||||
| 					reported_wait = wait_time; | ||||
| 				} | ||||
| 				usleep(200); | ||||
| 			} | ||||
| 		} while (res == EBUSY); | ||||
| 	} | ||||
| #else /* !DETECT_DEADLOCKS || !DEBUG_THREADS */ | ||||
| #else /* !DETECT_DEADLOCKS */ | ||||
| 	res = pthread_rwlock_wrlock(&t->lock); | ||||
| #endif /* !DETECT_DEADLOCKS || !DEBUG_THREADS */ | ||||
| #endif /* !DETECT_DEADLOCKS */ | ||||
|  | ||||
| #ifdef DEBUG_THREADS | ||||
| 	if (!res && lt) { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user