[asterisk-commits] russell: branch russell/messaging r311292 - /team/russell/messaging/main/

SVN commits to the Asterisk project asterisk-commits at lists.digium.com
Thu Mar 17 17:10:40 CDT 2011


Author: russell
Date: Thu Mar 17 17:10:35 2011
New Revision: 311292

URL: http://svnview.digium.com/svn/asterisk?view=rev&rev=311292
Log:
Use a new mechanism for ensuring that message techs are unload safe.

This is now handled in the same way as format attribute interfaces in format.c.

Modified:
    team/russell/messaging/main/message.c

Modified: team/russell/messaging/main/message.c
URL: http://svnview.digium.com/svn/asterisk/team/russell/messaging/main/message.c?view=diff&rev=311292&r1=311291&r2=311292
==============================================================================
--- team/russell/messaging/main/message.c (original)
+++ team/russell/messaging/main/message.c Thu Mar 17 17:10:35 2011
@@ -97,16 +97,16 @@
 				<variable name="MESSAGE_SEND_STATUS">
 					<para>This is the time from dialing a channel until when it is disconnected.</para>
 					<value name="INVALID_PROTOCOL">
-						<para>No handler for the technology part of the URI was found.</para>
+						No handler for the technology part of the URI was found.
 					</value>
 					<value name="INVALID_URI">
-						<para>The protocol handler reported that the URI was not valid.</para>
+						The protocol handler reported that the URI was not valid.
 					</value>
 					<value name="SUCCESS">
-						<para>Successfully passed on to the protocol handler, but delivery has not necessarily been guaranteed.</para>
+						Successfully passed on to the protocol handler, but delivery has not necessarily been guaranteed.
 					</value>
 					<value name="FAILURE">
-						<para>The protocol handler reported that it was unabled to deliver the message for some reason.</para>
+						The protocol handler reported that it was unabled to deliver the message for some reason.
 					</value>
 				</variable>
 			</variablelist>
@@ -128,6 +128,13 @@
 
 struct ast_msg_tech_holder {
 	const struct ast_msg_tech *tech;
+	/*! 
+	 * \brief A rwlock for this object
+	 *
+	 * a read/write lock must be used to protect the wrapper instead
+	 * of the ao2 lock. A rdlock must be held to read tech_holder->tech.
+	 */
+	ast_rwlock_t tech_lock;
 };
 
 static struct ao2_container *msg_techs;
@@ -621,31 +628,34 @@
 
 static int msg_tech_hash(const void *obj, const int flags)
 {
-	const struct ast_msg_tech_holder *tech_holder = obj;
-
-	return ast_str_case_hash(tech_holder->tech->name);
+	struct ast_msg_tech_holder *tech_holder = (struct ast_msg_tech_holder *) obj;
+	int res = 0;
+
+	ast_rwlock_rdlock(&tech_holder->tech_lock);
+	if (tech_holder->tech) {
+		res = ast_str_case_hash(tech_holder->tech->name);
+	}
+	ast_rwlock_unlock(&tech_holder->tech_lock);
+
+	return res;
 }
 
 static int msg_tech_cmp(void *obj, void *arg, int flags)
 {
-	const struct ast_msg_tech_holder *tech_holder = obj;
+	struct ast_msg_tech_holder *tech_holder = obj;
 	const struct ast_msg_tech_holder *tech_holder2 = arg;
-
-	return strcasecmp(tech_holder->tech->name, tech_holder2->tech->name) ?  0 : CMP_MATCH | CMP_STOP;
-}
-
-static int msg_tech_cmp_locked(void *obj, void *arg, int flags)
-{
-	int res;
-	struct ast_msg_tech_holder *tech_holder = obj;
-
-	res = msg_tech_cmp(obj, arg, flags);
-
-	if (res == (CMP_MATCH | CMP_STOP)) {
-		ao2_lock(tech_holder);
-	}
-
-	return res;
+	int res = 1;
+
+	ast_rwlock_rdlock(&tech_holder->tech_lock);
+	/*
+	 * tech_holder2 is a temporary fake tech_holder.
+	 */
+	if (tech_holder->tech) {
+		res = strcasecmp(tech_holder->tech->name, tech_holder2->tech->name) ? 0 : CMP_MATCH | CMP_STOP;
+	}
+	ast_rwlock_unlock(&tech_holder->tech_lock);
+
+	return res ? 0 : CMP_MATCH | CMP_STOP;
 }
 
 /*!
@@ -659,7 +669,7 @@
 	char *tech_name;
 	struct ast_msg_tech_holder *tech_holder = NULL;
 	char *parse;
-	int res;
+	int res = -1;
 	AST_DECLARE_APP_ARGS(args,
 		AST_APP_ARG(to);
 		AST_APP_ARG(from);
@@ -704,8 +714,7 @@
 			.tech = &tmp_msg_tech,
 		};
 
-		tech_holder = ao2_callback(msg_techs, OBJ_POINTER,
-					msg_tech_cmp_locked, &tmp_tech_holder);
+		tech_holder = ao2_find(msg_techs, &tmp_tech_holder, OBJ_POINTER);
 	}
 
 	if (!tech_holder) {
@@ -720,14 +729,17 @@
 	 * that they could change.
 	 */
 	ao2_lock(msg);
-	res = tech_holder->tech->msg_send(msg, args.to, args.from);
+	ast_rwlock_rdlock(&tech_holder->tech_lock);
+	if (tech_holder->tech) {
+		res = tech_holder->tech->msg_send(msg, args.to, args.from);
+	}
+	ast_rwlock_unlock(&tech_holder->tech_lock);
 	ao2_unlock(msg);
 
 	pbx_builtin_setvar_helper(chan, "MESSAGE_SEND_STATUS", res ? "FAILURE" : "SUCCESS");
 
 exit_cleanup:
 	if (tech_holder) {
-		ao2_unlock(tech_holder);
 		ao2_ref(tech_holder, -1);
 		tech_holder = NULL;
 	}
@@ -755,6 +767,7 @@
 		return -1;
 	}
 
+	ast_rwlock_init(&tech_holder->tech_lock);
 	tech_holder->tech = tech;
 
 	ao2_link(msg_techs, tech_holder);
@@ -781,15 +794,9 @@
 		return -1;
 	}
 
-	/*
-	 * The code that sends messages using a message technology ensures that it
-	 * locks the tech_holder before it returns a reference to it from the
-	 * container.  By just grabbing and releasing the lock here, we ensure that
-	 * we wait until any currently running message send code finishes before
-	 * we proceed with unregistration.
-	 */
-	ao2_lock(tech_holder);
-	ao2_unlock(tech_holder);
+	ast_rwlock_wrlock(&tech_holder->tech_lock);
+	tech_holder->tech = NULL;
+	ast_rwlock_unlock(&tech_holder->tech_lock);
 
 	ao2_ref(tech_holder, -1);
 	tech_holder = NULL;




More information about the asterisk-commits mailing list