[svn-commits] rmudgett: branch rmudgett/bridge_phase r387627 - in /team/rmudgett/bridge_pha...

SVN commits to the Digium repositories svn-commits at lists.digium.com
Fri May 3 19:08:24 CDT 2013


Author: rmudgett
Date: Fri May  3 19:08:21 2013
New Revision: 387627

URL: http://svnview.digium.com/svn/asterisk?view=rev&rev=387627
Log:
Initial separation of local channel code from unreal local channel derivative framework.

Still to do:
* Rename ast_bridge_local_optimized_out() to ast_bridge_unreal_optimized_out().
* Create ast_local_setup_bridge().
* Create ast_local_setup_masquerade().
* Create ast_local_setup_dialplan().
* Separate local_call() from unreal channel start.
* Separate local_alloc() from unreal private setup.
* Separate local_new() from unreal channel setup.
* Create ast_unreal_new().
* Separate local_request() from unreal channel setup.
* Move channels/chan_local.c to main/core_local.c and make it not dynamically loadable.

Added:
    team/rmudgett/bridge_phase/include/asterisk/core_local.h   (with props)
Modified:
    team/rmudgett/bridge_phase/channels/chan_local.c

Modified: team/rmudgett/bridge_phase/channels/chan_local.c
URL: http://svnview.digium.com/svn/asterisk/team/rmudgett/bridge_phase/channels/chan_local.c?view=diff&rev=387627&r1=387626&r2=387627
==============================================================================
--- team/rmudgett/bridge_phase/channels/chan_local.c (original)
+++ team/rmudgett/bridge_phase/channels/chan_local.c Fri May  3 19:08:21 2013
@@ -55,6 +55,7 @@
 #include "asterisk/devicestate.h"
 #include "asterisk/astobj2.h"
 #include "asterisk/bridging.h"
+#include "asterisk/core_local.h"
 
 /*** DOCUMENTATION
 	<manager name="LocalOptimizeAway" language="en_US">
@@ -77,57 +78,36 @@
 
 static const char tdesc[] = "Local Proxy Channel Driver";
 
-#define IS_OUTBOUND(a,b) (a == b->chan ? 1 : 0)
-
 static struct ao2_container *locals;
 
 static unsigned int name_sequence = 0;
 
-static struct ast_jb_conf g_jb_conf = {
-	.flags = 0,
-	.max_size = -1,
-	.resync_threshold = -1,
-	.impl = "",
-	.target_extra = -1,
-};
-
 static struct ast_channel *local_request(const char *type, struct ast_format_cap *cap, const struct ast_channel *requestor, const char *data, int *cause);
-static int local_digit_begin(struct ast_channel *ast, char digit);
-static int local_digit_end(struct ast_channel *ast, char digit, unsigned int duration);
 static int local_call(struct ast_channel *ast, const char *dest, int timeout);
 static int local_hangup(struct ast_channel *ast);
-static int local_answer(struct ast_channel *ast);
-static struct ast_frame *local_read(struct ast_channel *ast);
-static int local_write(struct ast_channel *ast, struct ast_frame *f);
-static int local_indicate(struct ast_channel *ast, int condition, const void *data, size_t datalen);
-static int local_fixup(struct ast_channel *oldchan, struct ast_channel *newchan);
-static int local_sendhtml(struct ast_channel *ast, int subclass, const char *data, int datalen);
-static int local_sendtext(struct ast_channel *ast, const char *text);
 static int local_devicestate(const char *data);
-static int local_queryoption(struct ast_channel *ast, int option, void *data, int *datalen);
-static int local_setoption(struct ast_channel *chan, int option, void *data, int datalen);
 
 /* PBX interface structure for channel registration */
 static struct ast_channel_tech local_tech = {
 	.type = "Local",
 	.description = tdesc,
 	.requester = local_request,
-	.send_digit_begin = local_digit_begin,
-	.send_digit_end = local_digit_end,
+	.send_digit_begin = ast_unreal_digit_begin,
+	.send_digit_end = ast_unreal_digit_end,
 	.call = local_call,
 	.hangup = local_hangup,
-	.answer = local_answer,
-	.read = local_read,
-	.write = local_write,
-	.write_video = local_write,
-	.exception = local_read,
-	.indicate = local_indicate,
-	.fixup = local_fixup,
-	.send_html = local_sendhtml,
-	.send_text = local_sendtext,
+	.answer = ast_unreal_answer,
+	.read = ast_unreal_read,
+	.write = ast_unreal_write,
+	.write_video = ast_unreal_write,
+	.exception = ast_unreal_read,
+	.indicate = ast_unreal_indicate,
+	.fixup = ast_unreal_fixup,
+	.send_html = ast_unreal_sendhtml,
+	.send_text = ast_unreal_sendtext,
 	.devicestate = local_devicestate,
-	.queryoption = local_queryoption,
-	.setoption = local_setoption,
+	.queryoption = ast_unreal_queryoption,
+	.setoption = ast_unreal_setoption,
 };
 
 /*!
@@ -135,32 +115,16 @@
  *
  * The local channel pvt has two ast_chan objects - the "owner" and the "next channel", the outbound channel
  *
- * ast_chan owner -> local_pvt -> ast_chan chan -> yet-another-pvt-depending-on-channel-type
+ * ast_chan owner -> ast_local_pvt -> ast_chan chan
  */
-struct local_pvt {
-	struct ast_channel *owner;      /*!< Master Channel - Bridging happens here */
-	struct ast_channel *chan;       /*!< Outbound channel - PBX is run here */
-	struct ast_format_cap *reqcap;  /*!< Requested format capabilities */
-	struct ast_jb_conf jb_conf;     /*!< jitterbuffer configuration for this local channel */
-	unsigned int flags;             /*!< Private flags */
+struct ast_local_pvt {
+	/*! Unreal channel driver base class values. */
+	struct ast_unreal_pvt base;
 	char context[AST_MAX_CONTEXT];  /*!< Context to call */
 	char exten[AST_MAX_EXTENSION];  /*!< Extension to call */
 };
 
-#define LOCAL_LAUNCHED_PBX    (1 << 0) /*!< PBX was launched */
-#define LOCAL_NO_OPTIMIZATION (1 << 1) /*!< Do not optimize using masquerading */
-#define LOCAL_MOH_PASSTHRU    (1 << 2) /*!< Pass through music on hold start/stop frames */
-
-/*!
- * \brief Send a pvt in with no locks held and get all locks
- *
- * \note NO locks should be held prior to calling this function
- * \note The pvt must have a ref held before calling this function
- * \note if outchan or outowner is set != NULL after calling this function
- *       those channels are locked and reffed.
- * \note Batman.
- */
-static void awesome_locking(struct local_pvt *p, struct ast_channel **outchan, struct ast_channel **outowner)
+void ast_unreal_lock_all(struct ast_unreal_pvt *p, struct ast_channel **outchan, struct ast_channel **outowner)
 {
 	struct ast_channel *chan = NULL;
 	struct ast_channel *owner = NULL;
@@ -209,11 +173,42 @@
 	*outchan = p->chan;
 }
 
+struct ast_channel *ast_local_get_peer(struct ast_channel *ast)
+{
+	struct ast_local_pvt *p = ast_channel_tech_pvt(ast);
+	struct ast_local_pvt *found;
+	struct ast_channel *peer;
+
+	if (!p) {
+		return NULL;
+	}
+
+	found = p ? ao2_find(locals, p, 0) : NULL;
+	if (!found) {
+		/* ast is either not a local channel or it has alredy been hungup */
+		return NULL;
+	}
+	ao2_lock(found);
+	if (ast == p->base.owner) {
+		peer = p->base.chan;
+	} else if (ast == p->base.chan) {
+		peer = p->base.owner;
+	} else {
+		peer = NULL;
+	}
+	if (peer) {
+		ast_channel_ref(peer);
+	}
+	ao2_unlock(found);
+	ao2_ref(found, -1);
+	return peer;
+}
+
 /* Called with ast locked */
-static int local_setoption(struct ast_channel *ast, int option, void *data, int datalen)
+int ast_unreal_setoption(struct ast_channel *ast, int option, void *data, int datalen)
 {
 	int res = 0;
-	struct local_pvt *p;
+	struct ast_unreal_pvt *p;
 	struct ast_channel *otherchan = NULL;
 	ast_chan_write_info_t *write_info;
 
@@ -230,7 +225,7 @@
 
 	if (!strcmp(write_info->function, "CHANNEL")
 		&& !strncasecmp(write_info->data, "hangup_handler_", 15)) {
-		/* Block CHANNEL(hangup_handler_xxx) writes to the other local channel. */
+		/* Block CHANNEL(hangup_handler_xxx) writes to the other unreal channel. */
 		return 0;
 	}
 
@@ -271,11 +266,12 @@
 /*! \brief Adds devicestate to local channels */
 static int local_devicestate(const char *data)
 {
+	int is_inuse = 0;
+	int res = AST_DEVICE_INVALID;
 	char *exten = ast_strdupa(data);
 	char *context;
 	char *opts;
-	int res;
-	struct local_pvt *lp;
+	struct ast_local_pvt *lp;
 	struct ao2_iterator it;
 
 	/* Strip options if they exist */
@@ -292,23 +288,17 @@
 	}
 	*context++ = '\0';
 
-	ast_debug(3, "Checking if extension %s@%s exists (devicestate)\n", exten, context);
-	res = ast_exists_extension(NULL, context, exten, 1, NULL);
-	if (!res) {
-		return AST_DEVICE_INVALID;
-	}
-
-	res = AST_DEVICE_NOT_INUSE;
-
 	it = ao2_iterator_init(locals, 0);
 	for (; (lp = ao2_iterator_next(&it)); ao2_ref(lp, -1)) {
-		int is_inuse;
-
 		ao2_lock(lp);
-		is_inuse = !strcmp(exten, lp->exten)
-			&& !strcmp(context, lp->context)
-			&& lp->owner
-			&& ast_test_flag(lp, LOCAL_LAUNCHED_PBX);
+		if (!strcmp(exten, lp->exten)
+			&& !strcmp(context, lp->context)) {
+			res = AST_DEVICE_NOT_INUSE;
+			if (lp->base.owner
+				&& ast_test_flag(&lp->base, AST_UNREAL_CARETAKER_THREAD)) {
+				is_inuse = 1;
+			}
+		}
 		ao2_unlock(lp);
 		if (is_inuse) {
 			res = AST_DEVICE_INUSE;
@@ -318,13 +308,20 @@
 	}
 	ao2_iterator_destroy(&it);
 
+	if (res == AST_DEVICE_INVALID) {
+		ast_debug(3, "Checking if extension %s@%s exists (devicestate)\n", exten, context);
+		if (ast_exists_extension(NULL, context, exten, 1, NULL)) {
+			res = AST_DEVICE_NOT_INUSE;
+		}
+	}
+
 	return res;
 }
 
 /* Called with ast locked */
-static int local_queryoption(struct ast_channel *ast, int option, void *data, int *datalen)
-{
-	struct local_pvt *p;
+int ast_unreal_queryoption(struct ast_channel *ast, int option, void *data, int *datalen)
+{
+	struct ast_unreal_pvt *p;
 	struct ast_channel *peer;
 	struct ast_channel *other;
 	int res = 0;
@@ -340,7 +337,7 @@
 	}
 
 	ao2_lock(p);
-	other = IS_OUTBOUND(ast, p) ? p->owner : p->chan;
+	other = AST_UNREAL_IS_OUTBOUND(ast, p) ? p->owner : p->chan;
 	if (!other) {
 		ao2_unlock(p);
 		return -1;
@@ -363,16 +360,16 @@
 /*!
  * \brief queue a frame onto either the p->owner or p->chan
  *
- * \note the local_pvt MUST have it's ref count bumped before entering this function and
+ * \note the ast_unreal_pvt MUST have it's ref count bumped before entering this function and
  * decremented after this function is called.  This is a side effect of the deadlock
  * avoidance that is necessary to lock 2 channels and a tech_pvt.  Without a ref counted
- * local_pvt, it is impossible to guarantee it will not be destroyed by another thread
+ * ast_unreal_pvt, it is impossible to guarantee it will not be destroyed by another thread
  * during deadlock avoidance.
  */
-static int local_queue_frame(struct local_pvt *p, int isoutbound, struct ast_frame *f,
+static int unreal_queue_frame(struct ast_unreal_pvt *p, int isoutbound, struct ast_frame *f,
 	struct ast_channel *us, int us_locked)
 {
-	struct ast_channel *other = NULL;
+	struct ast_channel *other;
 
 	/* Recalculate outbound channel */
 	other = isoutbound ? p->owner : p->chan;
@@ -380,7 +377,7 @@
 		return 0;
 	}
 
-	/* do not queue frame if generator is on both local channels */
+	/* do not queue frame if generator is on both unreal channels */
 	if (us && ast_channel_generator(us) && ast_channel_generator(other)) {
 		return 0;
 	}
@@ -407,9 +404,9 @@
 	return 0;
 }
 
-static int local_answer(struct ast_channel *ast)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_answer(struct ast_channel *ast)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int isoutbound;
 	int res = -1;
 
@@ -419,14 +416,15 @@
 
 	ao2_ref(p, 1);
 	ao2_lock(p);
-	isoutbound = IS_OUTBOUND(ast, p);
+	isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 	if (isoutbound) {
 		/* Pass along answer since somebody answered us */
 		struct ast_frame answer = { AST_FRAME_CONTROL, { AST_CONTROL_ANSWER } };
 
-		res = local_queue_frame(p, isoutbound, &answer, ast, 1);
+		res = unreal_queue_frame(p, isoutbound, &answer, ast, 1);
 	} else {
-		ast_log(LOG_WARNING, "Huh?  Local is being asked to answer?\n");
+		ast_log(LOG_WARNING, "Huh?  %s is being asked to answer?\n",
+			ast_channel_name(ast));
 	}
 	ao2_unlock(p);
 	ao2_ref(p, -1);
@@ -435,25 +433,26 @@
 
 /*!
  * \internal
- * \brief Check and optimize out the local channels between bridges.
+ * \brief Check and optimize out the unreal channels between bridges.
  * \since 12.0.0
  *
- * \param ast Channel writing a frame into the local channels.
+ * \param ast Channel writing a frame into the unreal channels.
  * \param p Local channel private.
  *
  * \note It is assumed that ast is locked.
  * \note It is assumed that p is locked.
  *
- * \retval 0 if local channels were not optimized out.
- * \retval non-zero if local channels were optimized out.
+ * \retval 0 if unreal channels were not optimized out.
+ * \retval non-zero if unreal channels were optimized out.
  */
-static int got_optimized_out(struct ast_channel *ast, struct local_pvt *p)
+static int got_optimized_out(struct ast_channel *ast, struct ast_unreal_pvt *p)
 {
 	/* Do a few conditional checks early on just to see if this optimization is possible */
-	if (ast_test_flag(p, LOCAL_NO_OPTIMIZATION) || !p->chan || !p->owner) {
+	if (ast_test_flag(p, AST_UNREAL_NO_OPTIMIZATION) || !p->chan || !p->owner) {
 		return 0;
 	}
 	if (ast == p->owner) {
+/* BUGBUG need to rename ast_bridge_local_optimized_out() to ast_bridge_unreal_optimized_out(). */
 		return ast_bridge_local_optimized_out(p->owner, p->chan);
 	}
 	if (ast == p->chan) {
@@ -463,14 +462,14 @@
 	return 0;
 }
 
-static struct ast_frame  *local_read(struct ast_channel *ast)
+struct ast_frame  *ast_unreal_read(struct ast_channel *ast)
 {
 	return &ast_null_frame;
 }
 
-static int local_write(struct ast_channel *ast, struct ast_frame *f)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_write(struct ast_channel *ast, struct ast_frame *f)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = -1;
 
 	if (!p) {
@@ -488,7 +487,7 @@
 		}
 		/* fall through */
 	default:
-		res = local_queue_frame(p, IS_OUTBOUND(ast, p), f, ast, 1);
+		res = unreal_queue_frame(p, AST_UNREAL_IS_OUTBOUND(ast, p), f, ast, 1);
 		break;
 	}
 	ao2_unlock(p);
@@ -497,9 +496,9 @@
 	return res;
 }
 
-static int local_fixup(struct ast_channel *oldchan, struct ast_channel *newchan)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(newchan);
+int ast_unreal_fixup(struct ast_channel *oldchan, struct ast_channel *newchan)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(newchan);
 	struct ast_bridge *bridge_owner;
 	struct ast_bridge *bridge_chan;
 
@@ -539,9 +538,9 @@
 	return 0;
 }
 
-static int local_indicate(struct ast_channel *ast, int condition, const void *data, size_t datalen)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_indicate(struct ast_channel *ast, int condition, const void *data, size_t datalen)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = 0;
 	struct ast_frame f = { AST_FRAME_CONTROL, };
 	int isoutbound;
@@ -550,26 +549,31 @@
 		return -1;
 	}
 
-	ao2_ref(p, 1); /* ref for local_queue_frame */
+	ao2_ref(p, 1); /* ref for unreal_queue_frame */
 
 	/* If this is an MOH hold or unhold, do it on the Local channel versus real channel */
-	if (!ast_test_flag(p, LOCAL_MOH_PASSTHRU) && condition == AST_CONTROL_HOLD) {
+	if (!ast_test_flag(p, AST_UNREAL_MOH_PASSTHRU) && condition == AST_CONTROL_HOLD) {
 		ast_moh_start(ast, data, NULL);
-	} else if (!ast_test_flag(p, LOCAL_MOH_PASSTHRU) && condition == AST_CONTROL_UNHOLD) {
+	} else if (!ast_test_flag(p, AST_UNREAL_MOH_PASSTHRU) && condition == AST_CONTROL_UNHOLD) {
 		ast_moh_stop(ast);
 	} else if (condition == AST_CONTROL_CONNECTED_LINE || condition == AST_CONTROL_REDIRECTING) {
 		struct ast_channel *this_channel;
 		struct ast_channel *the_other_channel;
 
-		/* A connected line update frame may only contain a partial amount of data, such
-		 * as just a source, or just a ton, and not the full amount of information. However,
-		 * the collected information is all stored in the outgoing channel's connectedline
-		 * structure, so when receiving a connected line update on an outgoing local channel,
-		 * we need to transmit the collected connected line information instead of whatever
-		 * happens to be in this control frame. The same applies for redirecting information, which
-		 * is why it is handled here as well.*/
+		/*
+		 * A connected line update frame may only contain a partial
+		 * amount of data, such as just a source, or just a ton, and not
+		 * the full amount of information.  However, the collected
+		 * information is all stored in the outgoing channel's
+		 * connectedline structure, so when receiving a connected line
+		 * update on an outgoing unreal channel, we need to transmit the
+		 * collected connected line information instead of whatever
+		 * happens to be in this control frame.  The same applies for
+		 * redirecting information, which is why it is handled here as
+		 * well.
+		 */
 		ao2_lock(p);
-		isoutbound = IS_OUTBOUND(ast, p);
+		isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 		if (isoutbound) {
 			this_channel = p->chan;
 			the_other_channel = p->owner;
@@ -590,7 +594,7 @@
 			}
 			f.subclass.integer = condition;
 			f.data.ptr = frame_data;
-			res = local_queue_frame(p, isoutbound, &f, ast, 1);
+			res = unreal_queue_frame(p, isoutbound, &f, ast, 1);
 		}
 		ao2_unlock(p);
 	} else {
@@ -598,15 +602,15 @@
 		ao2_lock(p);
 		/*
 		 * Block -1 stop tones events if we are to be optimized out.  We
-		 * don't need a flurry of these events on a local channel chain
+		 * don't need a flurry of these events on a unreal channel chain
 		 * when initially connected to slow the optimization process.
 		 */
-		if (0 <= condition || ast_test_flag(p, LOCAL_NO_OPTIMIZATION)) {
-			isoutbound = IS_OUTBOUND(ast, p);
+		if (0 <= condition || ast_test_flag(p, AST_UNREAL_NO_OPTIMIZATION)) {
+			isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 			f.subclass.integer = condition;
 			f.data.ptr = (void *) data;
 			f.datalen = datalen;
-			res = local_queue_frame(p, isoutbound, &f, ast, 1);
+			res = unreal_queue_frame(p, isoutbound, &f, ast, 1);
 
 			if (!res && condition == AST_CONTROL_T38_PARAMETERS
 				&& datalen == sizeof(struct ast_control_t38_parameters)) {
@@ -626,9 +630,9 @@
 	return res;
 }
 
-static int local_digit_begin(struct ast_channel *ast, char digit)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_digit_begin(struct ast_channel *ast, char digit)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = -1;
 	struct ast_frame f = { AST_FRAME_DTMF_BEGIN, };
 	int isoutbound;
@@ -637,20 +641,20 @@
 		return -1;
 	}
 
-	ao2_ref(p, 1); /* ref for local_queue_frame */
+	ao2_ref(p, 1); /* ref for unreal_queue_frame */
 	ao2_lock(p);
-	isoutbound = IS_OUTBOUND(ast, p);
+	isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 	f.subclass.integer = digit;
-	res = local_queue_frame(p, isoutbound, &f, ast, 0);
+	res = unreal_queue_frame(p, isoutbound, &f, ast, 0);
 	ao2_unlock(p);
 	ao2_ref(p, -1);
 
 	return res;
 }
 
-static int local_digit_end(struct ast_channel *ast, char digit, unsigned int duration)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_digit_end(struct ast_channel *ast, char digit, unsigned int duration)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = -1;
 	struct ast_frame f = { AST_FRAME_DTMF_END, };
 	int isoutbound;
@@ -659,21 +663,21 @@
 		return -1;
 	}
 
-	ao2_ref(p, 1); /* ref for local_queue_frame */
+	ao2_ref(p, 1); /* ref for unreal_queue_frame */
 	ao2_lock(p);
-	isoutbound = IS_OUTBOUND(ast, p);
+	isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 	f.subclass.integer = digit;
 	f.len = duration;
-	res = local_queue_frame(p, isoutbound, &f, ast, 0);
+	res = unreal_queue_frame(p, isoutbound, &f, ast, 0);
 	ao2_unlock(p);
 	ao2_ref(p, -1);
 
 	return res;
 }
 
-static int local_sendtext(struct ast_channel *ast, const char *text)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_sendtext(struct ast_channel *ast, const char *text)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = -1;
 	struct ast_frame f = { AST_FRAME_TEXT, };
 	int isoutbound;
@@ -682,20 +686,20 @@
 		return -1;
 	}
 
-	ao2_ref(p, 1); /* ref for local_queue_frame */
+	ao2_ref(p, 1); /* ref for unreal_queue_frame */
 	ao2_lock(p);
-	isoutbound = IS_OUTBOUND(ast, p);
+	isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 	f.data.ptr = (char *) text;
 	f.datalen = strlen(text) + 1;
-	res = local_queue_frame(p, isoutbound, &f, ast, 0);
+	res = unreal_queue_frame(p, isoutbound, &f, ast, 0);
 	ao2_unlock(p);
 	ao2_ref(p, -1);
 	return res;
 }
 
-static int local_sendhtml(struct ast_channel *ast, int subclass, const char *data, int datalen)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+int ast_unreal_sendhtml(struct ast_channel *ast, int subclass, const char *data, int datalen)
+{
+	struct ast_unreal_pvt *p = ast_channel_tech_pvt(ast);
 	int res = -1;
 	struct ast_frame f = { AST_FRAME_HTML, };
 	int isoutbound;
@@ -704,24 +708,28 @@
 		return -1;
 	}
 
-	ao2_ref(p, 1); /* ref for local_queue_frame */
+	ao2_ref(p, 1); /* ref for unreal_queue_frame */
 	ao2_lock(p);
-	isoutbound = IS_OUTBOUND(ast, p);
+	isoutbound = AST_UNREAL_IS_OUTBOUND(ast, p);
 	f.subclass.integer = subclass;
 	f.data.ptr = (char *)data;
 	f.datalen = datalen;
-	res = local_queue_frame(p, isoutbound, &f, ast, 0);
+	res = unreal_queue_frame(p, isoutbound, &f, ast, 0);
 	ao2_unlock(p);
 	ao2_ref(p, -1);
 
 	return res;
 }
 
+/* BUGBUG need to create ast_local_setup_bridge(). */
+/* BUGBUG need to create ast_local_setup_masquerade(). */
+/* BUGBUG need to create ast_local_setup_dialplan(). */
+/* BUGBUG need to separate local_call() from unreal channel start. */
 /*! \brief Initiate new call, part of PBX interface
  *         dest is the dial string */
 static int local_call(struct ast_channel *ast, const char *dest, int timeout)
 {
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
+	struct ast_local_pvt *p = ast_channel_tech_pvt(ast);
 	int pvt_locked = 0;
 
 	struct ast_channel *owner = NULL;
@@ -733,6 +741,7 @@
 	char *slash;
 	const char *exten;
 	const char *context;
+	const char *owner_cid;
 
 	if (!p) {
 		return -1;
@@ -743,7 +752,7 @@
 	ao2_ref(p, 1);
 	ast_channel_unlock(ast);
 
-	awesome_locking(p, &chan, &owner);
+	ast_unreal_lock_all(&p->base, &chan, &owner);
 	pvt_locked = 1;
 
 	if (owner != ast) {
@@ -791,9 +800,11 @@
 		}
 	}
 	ast_channel_datastore_inherit(owner, chan);
-	/* If the local channel has /n or /b on the end of it,
-	 * we need to lop that off for our argument to setting
-	 * up the CC_INTERFACES variable
+
+	/*
+	 * If the local channel has /n on the end of it, we need to lop
+	 * that off for our argument to setting up the CC_INTERFACES
+	 * variable.
 	 */
 	if ((slash = strrchr(reduced_dest, '/'))) {
 		*slash = '\0';
@@ -808,11 +819,18 @@
 
 	ast_channel_unlock(chan);
 
-	if (!ast_exists_extension(chan, context, exten, 1,
-		S_COR(ast_channel_caller(owner)->id.number.valid, ast_channel_caller(owner)->id.number.str, NULL))) {
+	owner_cid = S_COR(ast_channel_caller(owner)->id.number.valid,
+		ast_channel_caller(owner)->id.number.str, NULL);
+	if (owner_cid) {
+		owner_cid = ast_strdupa(owner_cid);
+	}
+	ast_channel_unlock(owner);
+	owner = ast_channel_unref(owner);
+
+	if (!ast_exists_extension(chan, context, exten, 1, owner_cid)) {
 		ast_log(LOG_NOTICE, "No such extension/context %s@%s while calling Local channel\n", exten, context);
 		res = -1;
-		chan = ast_channel_unref(chan); /* we already unlocked it, so clear it here so the cleanup label won't touch it. */
+		chan = ast_channel_unref(chan); /* we already unlocked it, clear it here so the cleanup label won't touch it. */
 		goto return_cleanup;
 	}
 
@@ -849,20 +867,20 @@
 		"Context: %s\r\n"
 		"Exten: %s\r\n"
 		"LocalOptimization: %s\r\n",
-		ast_channel_name(p->owner), ast_channel_name(p->chan),
-		ast_channel_uniqueid(p->owner), ast_channel_uniqueid(p->chan),
+		ast_channel_name(p->base.owner), ast_channel_name(p->base.chan),
+		ast_channel_uniqueid(p->base.owner), ast_channel_uniqueid(p->base.chan),
 		p->context, p->exten,
-		ast_test_flag(p, LOCAL_NO_OPTIMIZATION) ? "Yes" : "No");
+		ast_test_flag(&p->base, AST_UNREAL_NO_OPTIMIZATION) ? "Yes" : "No");
 
 
 	/* Start switch on sub channel */
 	res = ast_pbx_start(chan);
 	if (!res) {
 		ao2_lock(p);
-		ast_set_flag(p, LOCAL_LAUNCHED_PBX);
+		ast_set_flag(&p->base, AST_UNREAL_CARETAKER_THREAD);
 		ao2_unlock(p);
 	}
-	chan = ast_channel_unref(chan); /* chan is already unlocked, clear it here so the cleanup lable won't touch it. */
+	chan = ast_channel_unref(chan); /* we already unlocked it, clear it here so the cleanup label won't touch it. */
 
 return_cleanup:
 	if (p) {
@@ -873,7 +891,7 @@
 	}
 	if (chan) {
 		ast_channel_unlock(chan);
-		chan = ast_channel_unref(chan);
+		ast_channel_unref(chan);
 	}
 
 	/* owner is supposed to be == to ast,  if it
@@ -883,7 +901,7 @@
 			ast_channel_unlock(owner);
 			ast_channel_lock(ast);
 		}
-		owner = ast_channel_unref(owner);
+		ast_channel_unref(owner);
 	} else {
 		/* we have to exit with ast locked */
 		ast_channel_lock(ast);
@@ -892,176 +910,200 @@
 	return res;
 }
 
-/*! \brief Hangup a call through the local proxy channel */
-static int local_hangup(struct ast_channel *ast)
-{
-	struct local_pvt *p = ast_channel_tech_pvt(ast);
-	int isoutbound;
+int ast_unreal_hangup(struct ast_unreal_pvt *p, struct ast_channel *ast)
+{
 	int hangup_chan = 0;
 	int res = 0;
-	struct ast_frame f = { AST_FRAME_CONTROL, { AST_CONTROL_HANGUP }, .data.uint32 = ast_channel_hangupcause(ast) };
+	int cause;
 	struct ast_channel *owner = NULL;
 	struct ast_channel *chan = NULL;
 
-	if (!p) {
-		return -1;
-	}
-
-	/* give the pvt a ref since we are unlocking the channel. */
-	ao2_ref(p, 1);
-
-	/* the pvt isn't going anywhere, we gave it a ref */
+	/* the pvt isn't going anywhere, it has a ref */
 	ast_channel_unlock(ast);
 
 	/* lock everything */
-	awesome_locking(p, &chan, &owner);
+	ast_unreal_lock_all(p, &chan, &owner);
 
 	if (ast != chan && ast != owner) {
 		res = -1;
-		goto local_hangup_cleanup;
-	}
-
-	isoutbound = IS_OUTBOUND(ast, p); /* just comparing pointer of ast */
-
-	if (p->chan && ast_channel_hangupcause(ast) == AST_CAUSE_ANSWERED_ELSEWHERE) {
-		ast_channel_hangupcause_set(p->chan, AST_CAUSE_ANSWERED_ELSEWHERE);
-		ast_debug(2, "This local call has AST_CAUSE_ANSWERED_ELSEWHERE set.\n");
-	}
-
-	if (isoutbound) {
-		const char *status = pbx_builtin_getvar_helper(p->chan, "DIALSTATUS");
-
-		if (status && p->owner) {
-			ast_channel_hangupcause_set(p->owner, ast_channel_hangupcause(p->chan));
-			pbx_builtin_setvar_helper(p->owner, "CHANLOCALSTATUS", status);
-		}
-
-		ast_clear_flag(p, LOCAL_LAUNCHED_PBX);
+		goto unreal_hangup_cleanup;
+	}
+
+	cause = ast_channel_hangupcause(ast);
+
+	if (ast == p->chan) {
+		/* Outgoing side is hanging up. */
+		ast_clear_flag(p, AST_UNREAL_CARETAKER_THREAD);
 		p->chan = NULL;
+		if (p->owner) {
+			const char *status = pbx_builtin_getvar_helper(p->chan, "DIALSTATUS");
+
+			if (status) {
+				ast_channel_hangupcause_set(p->owner, cause);
+				pbx_builtin_setvar_helper(p->owner, "CHANLOCALSTATUS", status);
+			}
+			ast_queue_hangup_with_cause(p->owner, cause);
+		}
 	} else {
+		/* Owner side is hanging up. */
+		p->owner = NULL;
 		if (p->chan) {
-			ast_queue_hangup(p->chan);
-		}
-		p->owner = NULL;
-	}
-
-	ast_channel_tech_pvt_set(ast, NULL); /* this is one of our locked channels, doesn't matter which */
-
-	if (!p->owner && !p->chan) {
-		ao2_unlock(p);
-
-		ao2_unlink(locals, p);
-		ao2_ref(p, -1);
-		p = NULL;
-		res = 0;
-		goto local_hangup_cleanup;
-	}
-	if (p->chan && !ast_test_flag(p, LOCAL_LAUNCHED_PBX)) {
-		/* Need to actually hangup since there is no PBX */
-		hangup_chan = 1;
-	} else {
-		local_queue_frame(p, isoutbound, &f, NULL, 0);
-	}
-
-local_hangup_cleanup:
-	if (p) {
-		ao2_unlock(p);
-		ao2_ref(p, -1);
-	}
+			if (cause == AST_CAUSE_ANSWERED_ELSEWHERE) {
+				ast_channel_hangupcause_set(p->chan, AST_CAUSE_ANSWERED_ELSEWHERE);
+				ast_debug(2, "%s has AST_CAUSE_ANSWERED_ELSEWHERE set.\n",
+					ast_channel_name(p->chan));
+			}
+			if (!ast_test_flag(p, AST_UNREAL_CARETAKER_THREAD)) {
+				/*
+				 * Need to actually hangup p->chan since nothing else is taking
+				 * care of it.
+				 */
+				hangup_chan = 1;
+			} else {
+				ast_queue_hangup_with_cause(p->chan, cause);
+			}
+		}
+	}
+
+	/* this is one of our locked channels, doesn't matter which */
+	ast_channel_tech_pvt_set(ast, NULL);
+
+unreal_hangup_cleanup:
+	ao2_unlock(p);
 	if (owner) {
 		ast_channel_unlock(owner);
-		owner = ast_channel_unref(owner);
+		ast_channel_unref(owner);
 	}
 	if (chan) {
 		ast_channel_unlock(chan);
 		if (hangup_chan) {
 			ast_hangup(chan);
 		}
-		chan = ast_channel_unref(chan);
-	}
-
-	/* leave with the same stupid channel locked that came in */
+		ast_channel_unref(chan);
+	}
+
+	/* leave with the channel locked that came in */
 	ast_channel_lock(ast);
+
+	return res;
+}
+
+/*! \brief Hangup a call through the local proxy channel */
+static int local_hangup(struct ast_channel *ast)
+{
+	struct ast_local_pvt *p = ast_channel_tech_pvt(ast);
+	int res;
+
+	if (!p) {
+		return -1;
+	}
+
+	/* give the pvt a ref to fulfill calling requirements. */
+	ao2_ref(p, +1);
+	res = ast_unreal_hangup(&p->base, ast);
+	if (!res) {
+		int unlink;
+
+		ao2_lock(p);
+		unlink = !p->base.owner && !p->base.chan;
+		ao2_unlock(p);
+		if (unlink) {
+			ao2_unlink(locals, p);
+		}
+	}
+	ao2_ref(p, -1);
+
 	return res;
 }
 
 /*!
  * \internal
- * \brief struct local_pvt destructor.
- *
- * \param vdoomed Void local_pvt to destroy.
+ * \brief struct ast_unreal_pvt destructor.
+ *
+ * \param vdoomed Void ast_local_pvt to destroy.
  *
  * \return Nothing
  */
 static void local_pvt_destructor(void *vdoomed)
 {
-	struct local_pvt *doomed = vdoomed;
-
-	doomed->reqcap = ast_format_cap_destroy(doomed->reqcap);
+	struct ast_local_pvt *doomed = vdoomed;
+
+	doomed->base.reqcap = ast_format_cap_destroy(doomed->base.reqcap);
 
 	ast_module_unref(ast_module_info->self);
 }
 
+/* BUGBUG need to separate local_alloc() from unreal private setup. */
 /*! \brief Create a call structure */
-static struct local_pvt *local_alloc(const char *data, struct ast_format_cap *cap)
-{
-	struct local_pvt *tmp = NULL;
+static struct ast_local_pvt *local_alloc(const char *data, struct ast_format_cap *cap)
+{
+	struct ast_local_pvt *pvt;
 	char *parse;
-	char *c = NULL;
-	char *opts = NULL;
-
-	if (!(tmp = ao2_alloc(sizeof(*tmp), local_pvt_destructor))) {
+	char *context;
+	char *opts;
+
+	static const struct ast_jb_conf jb_conf = {
+		.flags = 0,
+		.max_size = -1,
+		.resync_threshold = -1,
+		.impl = "",
+		.target_extra = -1,
+	};
+
+	if (!(pvt = ao2_alloc(sizeof(*pvt), local_pvt_destructor))) {
 		return NULL;
 	}
-	if (!(tmp->reqcap = ast_format_cap_dup(cap))) {
-		ao2_ref(tmp, -1);
+	if (!(pvt->base.reqcap = ast_format_cap_dup(cap))) {
+		ao2_ref(pvt, -1);
 		return NULL;
 	}
 
+	memcpy(&pvt->base.jb_conf, &jb_conf, sizeof(pvt->base.jb_conf));
+
 	ast_module_ref(ast_module_info->self);
 
-	/* Initialize private structure information */
 	parse = ast_strdupa(data);
-
-	memcpy(&tmp->jb_conf, &g_jb_conf, sizeof(tmp->jb_conf));
 
 	/* Look for options */
 	if ((opts = strchr(parse, '/'))) {
 		*opts++ = '\0';
 		if (strchr(opts, 'n'))
-			ast_set_flag(tmp, LOCAL_NO_OPTIMIZATION);
+			ast_set_flag(&pvt->base, AST_UNREAL_NO_OPTIMIZATION);
 		if (strchr(opts, 'j')) {
-			if (ast_test_flag(tmp, LOCAL_NO_OPTIMIZATION))
-				ast_set_flag(&tmp->jb_conf, AST_JB_ENABLED);
+			if (ast_test_flag(&pvt->base, AST_UNREAL_NO_OPTIMIZATION))
+				ast_set_flag(&pvt->base.jb_conf, AST_JB_ENABLED);
 			else {
 				ast_log(LOG_ERROR, "You must use the 'n' option with the 'j' option to enable the jitter buffer\n");
 			}
 		}
 		if (strchr(opts, 'm')) {
-			ast_set_flag(tmp, LOCAL_MOH_PASSTHRU);
+			ast_set_flag(&pvt->base, AST_UNREAL_MOH_PASSTHRU);
 		}
 	}
 
 	/* Look for a context */
-	if ((c = strchr(parse, '@'))) {
-		*c++ = '\0';
-	}
-
-	ast_copy_string(tmp->context, c ? c : "default", sizeof(tmp->context));
-	ast_copy_string(tmp->exten, parse, sizeof(tmp->exten));
-
-	ao2_link(locals, tmp);
-
-	return tmp; /* this is returned with a ref */
-}
-
+	if ((context = strchr(parse, '@'))) {
+		*context++ = '\0';
+	}
+
+	ast_copy_string(pvt->context, S_OR(context, "default"), sizeof(pvt->context));
+	ast_copy_string(pvt->exten, parse, sizeof(pvt->exten));
+	snprintf(pvt->base.name, sizeof(pvt->base.name), "%s@%s", pvt->exten, pvt->context);
+
+	ao2_link(locals, pvt);
+
+	return pvt; /* this is returned with a ref */
+}
+
+/* BUGBUG need to separate local_new() from unreal channel setup. */
 /*! \brief Start new local channel */
-static struct ast_channel *local_new(struct local_pvt *p, int state, const char *linkedid, struct ast_callid *callid)
+static struct ast_channel *local_new(struct ast_local_pvt *p, int state, const char *linkedid, struct ast_callid *callid)
 {
 	struct ast_channel *owner;
 	struct ast_channel *chan;
 	struct ast_format fmt;
 	int generated_seqno = ast_atomic_fetchadd_int((int *)&name_sequence, +1);
+	const struct ast_channel_tech *tech = &local_tech;
 
 	/*
 	 * Allocate two new Asterisk channels
@@ -1072,10 +1114,10 @@
 	 */
 	if (!(owner = ast_channel_alloc(1, state, NULL, NULL, NULL,
 			p->exten, p->context, linkedid, 0,
-			"Local/%s@%s-%08x;1", p->exten, p->context, generated_seqno))
+			"%s/%s-%08x;1", tech->type, p->base.name, generated_seqno))
 		|| !(chan = ast_channel_alloc(1, AST_STATE_RING, NULL, NULL, NULL,
 			p->exten, p->context, ast_channel_linkedid(owner), 0,
-			"Local/%s@%s-%08x;2", p->exten, p->context, generated_seqno))) {
+			"%s/%s-%08x;2", tech->type, p->base.name, generated_seqno))) {
 		if (owner) {
 			owner = ast_channel_release(owner);
 		}
@@ -1088,16 +1130,16 @@
 		ast_channel_callid_set(chan, callid);
 	}
 
-	ast_channel_tech_set(owner, &local_tech);
-	ast_channel_tech_set(chan, &local_tech);
+	ast_channel_tech_set(owner, tech);
+	ast_channel_tech_set(chan, tech);
 	ast_channel_tech_pvt_set(owner, p);
 	ast_channel_tech_pvt_set(chan, p);
 
-	ast_format_cap_copy(ast_channel_nativeformats(owner), p->reqcap);
-	ast_format_cap_copy(ast_channel_nativeformats(chan), p->reqcap);
+	ast_format_cap_copy(ast_channel_nativeformats(owner), p->base.reqcap);
+	ast_format_cap_copy(ast_channel_nativeformats(chan), p->base.reqcap);
 
 	/* Determine our read/write format and set it on each channel */
-	ast_best_codec(p->reqcap, &fmt);
+	ast_best_codec(p->base.reqcap, &fmt);
 	ast_format_copy(ast_channel_writeformat(owner), &fmt);
 	ast_format_copy(ast_channel_writeformat(chan), &fmt);
 	ast_format_copy(ast_channel_rawwriteformat(owner), &fmt);
@@ -1110,18 +1152,20 @@
 	ast_set_flag(ast_channel_flags(owner), AST_FLAG_DISABLE_DEVSTATE_CACHE);
 	ast_set_flag(ast_channel_flags(chan), AST_FLAG_DISABLE_DEVSTATE_CACHE);
 
-	p->owner = owner;
-	p->chan = chan;
-
-	ast_jb_configure(owner, &p->jb_conf);
+	p->base.owner = owner;
+	p->base.chan = chan;
+
+	ast_jb_configure(owner, &p->base.jb_conf);
 
 	return owner;
 }
 
+/* BUGBUG need to create ast_unreal_new(). */
+/* BUGBUG need to separate local_request() from unreal channel setup. */
 /*! \brief Part of PBX interface */
 static struct ast_channel *local_request(const char *type, struct ast_format_cap *cap, const struct ast_channel *requestor, const char *data, int *cause)
 {
-	struct local_pvt *p;
+	struct ast_local_pvt *p;
 	struct ast_channel *chan;
 	struct ast_callid *callid = ast_read_threadstorage_callid();
 
@@ -1136,8 +1180,8 @@
 		ao2_unlink(locals, p);
 	} else if (ast_channel_cc_params_init(chan, requestor ? ast_channel_get_cc_config_params((struct ast_channel *)requestor) : NULL)) {
 		ao2_unlink(locals, p);
-		p->owner = ast_channel_release(p->owner);
-		p->chan = ast_channel_release(p->chan);
+		p->base.owner = ast_channel_release(p->base.owner);
+		p->base.chan = ast_channel_release(p->base.chan);
 		chan = NULL;
 	}
 	ao2_ref(p, -1); /* kill the ref from the alloc */
@@ -1154,7 +1198,7 @@
 /*! \brief CLI command "local show channels" */
 static char *locals_show(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a)
 {
-	struct local_pvt *p = NULL;
+	struct ast_local_pvt *p;
 	struct ao2_iterator it;
 
 	switch (cmd) {
@@ -1180,7 +1224,9 @@
 	it = ao2_iterator_init(locals, 0);
 	while ((p = ao2_iterator_next(&it))) {
 		ao2_lock(p);
-		ast_cli(a->fd, "%s -- %s@%s\n", p->owner ? ast_channel_name(p->owner) : "<unowned>", p->exten, p->context);
+		ast_cli(a->fd, "%s -- %s\n",
+			p->base.owner ? ast_channel_name(p->base.owner) : "<unowned>",
+			p->base.name);
 		ao2_unlock(p);
 		ao2_ref(p, -1);
 	}
@@ -1196,8 +1242,8 @@
 static int manager_optimize_away(struct mansession *s, const struct message *m)
 {
 	const char *channel;
-	struct local_pvt *p;
-	struct local_pvt *found;
+	struct ast_local_pvt *p;
+	struct ast_local_pvt *found;
 	struct ast_channel *chan;
 
 	channel = astman_get_header(m, "Channel");
@@ -1218,7 +1264,7 @@
 	found = p ? ao2_find(locals, p, 0) : NULL;
 	if (found) {
 		ao2_lock(found);
-		ast_clear_flag(found, LOCAL_NO_OPTIMIZATION);
+		ast_clear_flag(&found->base, AST_UNREAL_NO_OPTIMIZATION);
 		ao2_unlock(found);
 		ao2_ref(found, -1);
 		astman_send_ack(s, m, "Queued channel to be optimized away");
@@ -1235,6 +1281,7 @@
 	return (obj == arg) ? CMP_MATCH : 0;
 }
 
+/* BUGBUG need to move this file to main/core_local.c and make it not dynamically loadable. */
 /*! \brief Load module into PBX, register channel */
 static int load_module(void)
 {
@@ -1256,7 +1303,7 @@
 		ast_format_cap_destroy(local_tech.capabilities);
 		return AST_MODULE_LOAD_FAILURE;
 	}
-	ast_cli_register_multiple(cli_local, sizeof(cli_local) / sizeof(struct ast_cli_entry));
+	ast_cli_register_multiple(cli_local, ARRAY_LEN(cli_local));
 	ast_manager_register_xml("LocalOptimizeAway", EVENT_FLAG_SYSTEM|EVENT_FLAG_CALL, manager_optimize_away);
 
 	return AST_MODULE_LOAD_SUCCESS;
@@ -1265,18 +1312,18 @@
 /*! \brief Unload the local proxy channel from Asterisk */
 static int unload_module(void)
 {
-	struct local_pvt *p = NULL;
+	struct ast_local_pvt *p;
 	struct ao2_iterator it;
 
 	/* First, take us out of the channel loop */
-	ast_cli_unregister_multiple(cli_local, sizeof(cli_local) / sizeof(struct ast_cli_entry));
+	ast_cli_unregister_multiple(cli_local, ARRAY_LEN(cli_local));
 	ast_manager_unregister("LocalOptimizeAway");
 	ast_channel_unregister(&local_tech);
 
 	it = ao2_iterator_init(locals, 0);
 	while ((p = ao2_iterator_next(&it))) {
-		if (p->owner) {
-			ast_softhangup(p->owner, AST_SOFTHANGUP_APPUNLOAD);
+		if (p->base.owner) {
+			ast_softhangup(p->base.owner, AST_SOFTHANGUP_APPUNLOAD);
 		}
 		ao2_ref(p, -1);
 	}

Added: team/rmudgett/bridge_phase/include/asterisk/core_local.h
URL: http://svnview.digium.com/svn/asterisk/team/rmudgett/bridge_phase/include/asterisk/core_local.h?view=auto&rev=387627
==============================================================================
--- team/rmudgett/bridge_phase/include/asterisk/core_local.h (added)
+++ team/rmudgett/bridge_phase/include/asterisk/core_local.h Fri May  3 19:08:21 2013
@@ -1,0 +1,120 @@
+/*
+ * Asterisk -- An open source telephony toolkit.
+ *
+ * Copyright (C) 2013 Digium, Inc.
+ *
+ * Richard Mudgett <rmudgett at digium.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 Local proxy channel and other unreal channel derivatives framework.
+ *
+ * \author Richard Mudgett <rmudgett at digium.com>
+ *
+ * See Also:
+ * \arg \ref AstCREDITS
+ */
+
+#ifndef _ASTERISK_CORE_LOCAL_H
+#define _ASTERISK_CORE_LOCAL_H
+
+#include "asterisk/channel.h"
+
+#if defined(__cplusplus) || defined(c_plusplus)
+extern "C" {
+#endif
+
+/* ------------------------------------------------------------------- */
+
+/*!
+ * \brief The base pvt structure for local channel derivatives.
+ *

[... 91 lines stripped ...]



More information about the svn-commits mailing list