mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-26 14:27:14 +00:00 
			
		
		
		
	chan_sip: Simplify dialog/peer references, improve REF_DEBUG output.
* Replace functions for ref/undef of dialogs and peers with macro's to call ao2_t_bump/ao2_t_cleanup. * Enable passthough of REF_DEBUG caller information to sip_alloc and find_call. ASTERISK-24882 #close Reported by: Corey Farrell Review: https://reviewboard.asterisk.org/r/4189/ ........ Merged revisions 433115 from http://svn.asterisk.org/svn/asterisk/branches/13 git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@433116 65c4cc65-6c06-0410-ace0-fbb531ad65f3
This commit is contained in:
		| @@ -1181,7 +1181,11 @@ static int sip_send_mwi_to_peer(struct sip_peer *peer, int cache_only); | ||||
| static int __sip_autodestruct(const void *data); | ||||
| static int update_call_counter(struct sip_pvt *fup, int event); | ||||
| static int auto_congest(const void *arg); | ||||
| static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *addr, const int intended_method); | ||||
| static struct sip_pvt *__find_call(struct sip_request *req, struct ast_sockaddr *addr, const int intended_method, | ||||
| 	const char *file, int line, const char *func); | ||||
| #define find_call(req, addr, intended_method) \ | ||||
| 	__find_call(req, addr, intended_method, __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
| 
 | ||||
| static void build_route(struct sip_pvt *p, struct sip_request *req, int backwards, int resp); | ||||
| static int build_path(struct sip_pvt *p, struct sip_peer *peer, struct sip_request *req, const char *pathbuf); | ||||
| static enum check_auth_result register_verify(struct sip_pvt *p, struct ast_sockaddr *addr, | ||||
| @@ -2335,30 +2339,6 @@ static struct ast_tcptls_session_args sip_tls_desc = { | ||||
| 	\return Always returns 0 */ | ||||
| #define append_history(p, event, fmt , args... )	append_history_full(p, "%-15s " fmt, event, ## args) | ||||
| 
 | ||||
| struct sip_pvt *dialog_ref_debug(struct sip_pvt *p, const char *tag, char *file, int line, const char *func) | ||||
| { | ||||
| 	if (p) | ||||
| #ifdef REF_DEBUG | ||||
| 		__ao2_ref_debug(p, 1, tag, file, line, func); | ||||
| #else | ||||
| 		ao2_ref(p, 1); | ||||
| #endif | ||||
| 	else | ||||
| 		ast_log(LOG_ERROR, "Attempt to Ref a null pointer\n"); | ||||
| 	return p; | ||||
| } | ||||
| 
 | ||||
| struct sip_pvt *dialog_unref_debug(struct sip_pvt *p, const char *tag, char *file, int line, const char *func) | ||||
| { | ||||
| 	if (p) | ||||
| #ifdef REF_DEBUG | ||||
| 		__ao2_ref_debug(p, -1, tag, file, line, func); | ||||
| #else | ||||
| 		ao2_ref(p, -1); | ||||
| #endif | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| /*! \brief map from an integer value to a string.
 | ||||
|  * If no match is found, return errorstring | ||||
|  */ | ||||
| @@ -3158,41 +3138,6 @@ cleanup: | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| #ifdef REF_DEBUG | ||||
| struct sip_peer *_ref_peer(struct sip_peer *peer, char *tag, char *file, int line, const char *func) | ||||
| { | ||||
| 	if (peer) | ||||
| 		__ao2_ref_debug(peer, 1, tag, file, line, func); | ||||
| 	else | ||||
| 		ast_log(LOG_ERROR, "Attempt to Ref a null peer pointer\n"); | ||||
| 	return peer; | ||||
| } | ||||
| 
 | ||||
| void *_unref_peer(struct sip_peer *peer, char *tag, char *file, int line, const char *func) | ||||
| { | ||||
| 	if (peer) | ||||
| 		__ao2_ref_debug(peer, -1, tag, file, line, func); | ||||
| 	return NULL; | ||||
| } | ||||
| #else | ||||
| /*!
 | ||||
|  * helper functions to unreference various types of objects. | ||||
|  * By handling them this way, we don't have to declare the | ||||
|  * destructor on each call, which removes the chance of errors. | ||||
|  */ | ||||
| void *sip_unref_peer(struct sip_peer *peer, char *tag) | ||||
| { | ||||
| 	ao2_t_ref(peer, -1, tag); | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| struct sip_peer *sip_ref_peer(struct sip_peer *peer, char *tag) | ||||
| { | ||||
| 	ao2_t_ref(peer, 1, tag); | ||||
| 	return peer; | ||||
| } | ||||
| #endif /* REF_DEBUG */ | ||||
| 
 | ||||
| static void peer_sched_cleanup(struct sip_peer *peer) | ||||
| { | ||||
| 	if (peer->pokeexpire != -1) { | ||||
| @@ -8648,13 +8593,18 @@ static void sip_pvt_callid_set(struct sip_pvt *pvt, ast_callid callid) | ||||
|  * Returns a reference to the object so whoever uses it later must | ||||
|  * remember to release the reference. | ||||
|  */ | ||||
| struct sip_pvt *sip_alloc(ast_string_field callid, struct ast_sockaddr *addr, | ||||
| 				 int useglobal_nat, const int intended_method, struct sip_request *req, ast_callid logger_callid) | ||||
| struct sip_pvt *__sip_alloc(ast_string_field callid, struct ast_sockaddr *addr, | ||||
| 				 int useglobal_nat, const int intended_method, struct sip_request *req, ast_callid logger_callid, | ||||
| 				 const char *file, int line, const char *func) | ||||
| { | ||||
| 	struct sip_pvt *p; | ||||
| 
 | ||||
| 	if (!(p = ao2_t_alloc(sizeof(*p), sip_destroy_fn, "allocate a dialog(pvt) struct"))) | ||||
| 	p = __ao2_alloc_debug(sizeof(*p), sip_destroy_fn, | ||||
| 		AO2_ALLOC_OPT_LOCK_MUTEX, "allocate a dialog(pvt) struct", | ||||
| 		file, line, func, 1); | ||||
| 	if (!p) { | ||||
| 		return NULL; | ||||
| 	} | ||||
| 
 | ||||
| 	if (ast_string_field_init(p, 512)) { | ||||
| 		ao2_t_ref(p, -1, "failed to string_field_init, drop p"); | ||||
| @@ -9170,7 +9120,8 @@ static void sip_set_owner(struct sip_pvt *p, struct ast_channel *chan) | ||||
|  * Returns a reference to the sip_pvt object, remember to give it back once done. | ||||
|  *     Called by handle_request_do | ||||
|  */ | ||||
| static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *addr, const int intended_method) | ||||
| static struct sip_pvt *__find_call(struct sip_request *req, struct ast_sockaddr *addr, const int intended_method, | ||||
| 	const char *file, int line, const char *func) | ||||
| { | ||||
| 	char totag[128]; | ||||
| 	char fromtag[128]; | ||||
| @@ -9227,7 +9178,8 @@ static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *a | ||||
| 		struct sip_pvt tmp_dialog = { | ||||
| 			.callid = callid, | ||||
| 		}; | ||||
| 		sip_pvt_ptr = ao2_t_find(dialogs, &tmp_dialog, OBJ_POINTER, "ao2_find in dialogs"); | ||||
| 		sip_pvt_ptr = __ao2_find_debug(dialogs, &tmp_dialog, OBJ_POINTER, | ||||
| 			"find_call in dialogs", file, line, func); | ||||
| 		if (sip_pvt_ptr) {  /* well, if we don't find it-- what IS in there? */ | ||||
| 			/* Found the call */ | ||||
| 			return sip_pvt_ptr; | ||||
| @@ -9241,11 +9193,12 @@ static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *a | ||||
| 		struct sip_pvt *fork_pvt = NULL; | ||||
| 		struct match_req_args args = { 0, }; | ||||
| 		int found; | ||||
| 		struct ao2_iterator *iterator = ao2_t_callback(dialogs, | ||||
| 		struct ao2_iterator *iterator = __ao2_callback_debug(dialogs, | ||||
| 			OBJ_POINTER | OBJ_MULTIPLE, | ||||
| 			dialog_find_multiple, | ||||
| 			&tmp_dialog, | ||||
| 			"pedantic ao2_find in dialogs"); | ||||
| 			"pedantic ao2_find in dialogs", | ||||
| 			file, line, func); | ||||
| 		struct sip_via *via = NULL; | ||||
| 
 | ||||
| 		args.method = req->method; | ||||
| @@ -9290,7 +9243,8 @@ static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *a | ||||
| 				/* This is likely a forked Request that somehow resulted in us receiving multiple parts of the fork.
 | ||||
| 			 	* RFC 3261 section 8.2.2.2, Indicate that we want to merge requests by sending a 482 response. */ | ||||
| 				transmit_response_using_temp(callid, addr, 1, intended_method, req, "482 (Loop Detected)"); | ||||
| 				dialog_unref(sip_pvt_ptr, "pvt did not match incoming SIP msg, unref from search."); | ||||
| 				__ao2_ref_debug(sip_pvt_ptr, -1, "pvt did not match incoming SIP msg, unref from search.", | ||||
| 					file, line, func); | ||||
| 				ao2_iterator_destroy(iterator); | ||||
| 				dialog_unref(fork_pvt, "unref fork_pvt"); | ||||
| 				free_via(via); | ||||
| @@ -9301,7 +9255,8 @@ static struct sip_pvt *find_call(struct sip_request *req, struct ast_sockaddr *a | ||||
| 				/* fall through */ | ||||
| 			case SIP_REQ_NOT_MATCH: | ||||
| 			default: | ||||
| 				dialog_unref(sip_pvt_ptr, "pvt did not match incoming SIP msg, unref from search"); | ||||
| 				__ao2_ref_debug(sip_pvt_ptr, -1, "pvt did not match incoming SIP msg, unref from search", | ||||
| 					file, line, func); | ||||
| 				break; | ||||
| 			} | ||||
| 		} | ||||
|   | ||||
| @@ -29,13 +29,16 @@ | ||||
|  * functions so we keep track of the refcounts. | ||||
|  * To simplify the code, we allow a NULL to be passed to dialog_unref(). | ||||
|  */ | ||||
| #define dialog_ref(arg1,arg2) dialog_ref_debug((arg1),(arg2), __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
| #define dialog_unref(arg1,arg2) dialog_unref_debug((arg1),(arg2), __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
| struct sip_pvt *dialog_ref_debug(struct sip_pvt *p, const char *tag, char *file, int line, const char *func); | ||||
| struct sip_pvt *dialog_unref_debug(struct sip_pvt *p, const char *tag, char *file, int line, const char *func); | ||||
| #define dialog_ref(dialog, tag) ao2_t_bump(dialog, tag) | ||||
| #define dialog_unref(dialog, tag) ({ ao2_t_cleanup(dialog, tag); (NULL); }) | ||||
|  | ||||
| struct sip_pvt *__sip_alloc(ast_string_field callid, struct ast_sockaddr *sin, | ||||
| 				 int useglobal_nat, const int intended_method, struct sip_request *req, ast_callid logger_callid, | ||||
| 				 const char *file, int line, const char *func); | ||||
|  | ||||
| #define sip_alloc(callid, addr, useglobal_nat, intended_method, req, logger_callid) \ | ||||
| 	__sip_alloc(callid, addr, useglobal_nat, intended_method, req, logger_callid, __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
|  | ||||
| struct sip_pvt *sip_alloc(ast_string_field callid, struct ast_sockaddr *sin, | ||||
| 				 int useglobal_nat, const int intended_method, struct sip_request *req, ast_callid logger_callid); | ||||
| void sip_scheddestroy_final(struct sip_pvt *p, int ms); | ||||
| void sip_scheddestroy(struct sip_pvt *p, int ms); | ||||
| int sip_cancel_destroy(struct sip_pvt *p); | ||||
|   | ||||
| @@ -1869,14 +1869,7 @@ void sip_auth_headers(enum sip_auth_type code, char **header, char **respheader) | ||||
| const char *sip_get_header(const struct sip_request *req, const char *name); | ||||
| const char *sip_get_transport(enum ast_transport t); | ||||
|  | ||||
| #ifdef REF_DEBUG | ||||
| #define sip_ref_peer(arg1,arg2) _ref_peer((arg1),(arg2), __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
| #define sip_unref_peer(arg1,arg2) _unref_peer((arg1),(arg2), __FILE__, __LINE__, __PRETTY_FUNCTION__) | ||||
| struct sip_peer *_ref_peer(struct sip_peer *peer, char *tag, char *file, int line, const char *func); | ||||
| void *_unref_peer(struct sip_peer *peer, char *tag, char *file, int line, const char *func); | ||||
| #else | ||||
| struct sip_peer *sip_ref_peer(struct sip_peer *peer, char *tag); | ||||
| void *sip_unref_peer(struct sip_peer *peer, char *tag); | ||||
| #endif /* REF_DEBUG */ | ||||
| #define sip_ref_peer(peer, tag) ao2_t_bump(peer, tag) | ||||
| #define sip_unref_peer(peer, tag) ({ ao2_t_cleanup(peer, tag); (NULL); }) | ||||
|  | ||||
| #endif | ||||
|   | ||||
		Reference in New Issue
	
	Block a user