| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2016, Fairview 5 Engineering, LLC | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * George Joseph <george.joseph@fairview5.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
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \brief Named Locks | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author George Joseph <george.joseph@fairview5.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "asterisk/_private.h"
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | #include "asterisk/named_locks.h"
 | 
					
						
							|  |  |  | #include "asterisk/utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ao2_container *named_locks; | 
					
						
							|  |  |  | #define NAMED_LOCKS_BUCKETS 101
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | struct named_lock_proxy { | 
					
						
							|  |  |  | 	AO2_WEAKPROXY(); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	char key[0]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | struct ast_named_lock { | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-29 23:59:00 -05:00
										 |  |  | AO2_STRING_FIELD_HASH_FN(named_lock_proxy, key) | 
					
						
							|  |  |  | AO2_STRING_FIELD_CMP_FN(named_lock_proxy, key) | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void named_locks_shutdown(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_cleanup(named_locks); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ast_named_locks_init(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-08-03 16:24:39 -05:00
										 |  |  | 	named_locks = ao2_container_alloc_hash(AO2_ALLOC_OPT_LOCK_MUTEX, 0, | 
					
						
							| 
									
										
										
										
											2017-12-29 23:59:00 -05:00
										 |  |  | 		NAMED_LOCKS_BUCKETS, named_lock_proxy_hash_fn, NULL, named_lock_proxy_cmp_fn); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	if (!named_locks) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_register_cleanup(named_locks_shutdown); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | static void named_lock_proxy_cb(void *weakproxy, void *data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_unlink(named_locks, weakproxy); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | struct ast_named_lock *__ast_named_lock_get(const char *filename, int lineno, const char *func, | 
					
						
							|  |  |  | 	enum ast_named_lock_type lock_type, const char *keyspace, const char *key) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-10-09 22:55:44 -04:00
										 |  |  | 	struct named_lock_proxy *proxy; | 
					
						
							|  |  |  | 	struct ast_named_lock *lock; | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	int keylen = strlen(keyspace) + strlen(key) + 2; | 
					
						
							|  |  |  | 	char *concat_key = ast_alloca(keylen); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	sprintf(concat_key, "%s-%s", keyspace, key); /* Safe */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_lock(named_locks); | 
					
						
							| 
									
										
										
										
											2017-10-09 22:55:44 -04:00
										 |  |  | 	lock = __ao2_weakproxy_find(named_locks, concat_key, OBJ_SEARCH_KEY | OBJ_NOLOCK, | 
					
						
							|  |  |  | 		__PRETTY_FUNCTION__, filename, lineno, func); | 
					
						
							|  |  |  | 	if (lock) { | 
					
						
							|  |  |  | 		ast_assert((ao2_options_get(lock) & AO2_ALLOC_OPT_LOCK_MASK) == lock_type); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 		ao2_unlock(named_locks); | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-09 22:55:44 -04:00
										 |  |  | 		return lock; | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	proxy = ao2_t_weakproxy_alloc(sizeof(*proxy) + keylen, NULL, concat_key); | 
					
						
							|  |  |  | 	if (!proxy) { | 
					
						
							|  |  |  | 		goto failure_cleanup; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	lock = __ao2_alloc(sizeof(*lock) + keylen, NULL, lock_type, concat_key, filename, lineno, func); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	if (!lock) { | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 		goto failure_cleanup; | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	/* We have exclusive access to proxy and lock, no need for locking here. */ | 
					
						
							|  |  |  | 	if (ao2_weakproxy_set_object(proxy, lock, OBJ_NOLOCK)) { | 
					
						
							|  |  |  | 		goto failure_cleanup; | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (ao2_weakproxy_subscribe(proxy, named_lock_proxy_cb, NULL, OBJ_NOLOCK)) { | 
					
						
							|  |  |  | 		goto failure_cleanup; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	strcpy(proxy->key, concat_key); /* Safe */ | 
					
						
							|  |  |  | 	ao2_link_flags(named_locks, proxy, OBJ_NOLOCK); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 	ao2_unlock(named_locks); | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	ao2_t_ref(proxy, -1, "Release allocation reference"); | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-18 14:28:57 -04:00
										 |  |  | 	return lock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | failure_cleanup: | 
					
						
							|  |  |  | 	ao2_unlock(named_locks); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_cleanup(proxy); | 
					
						
							|  |  |  | 	ao2_cleanup(lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							| 
									
										
										
										
											2016-03-31 19:04:29 -06:00
										 |  |  | } |