| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * Copyright (C) 1999 - 2005, Digium, Inc. | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  |  * Mark Spencer <markster@digium.com> | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  |  * This program is free software, distributed under the terms of | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  | /*! \file
 | 
					
						
							| 
									
										
										
										
											2005-12-30 21:18:06 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * \author Mark Spencer <markster@digium.com> | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-10-24 20:12:06 +00:00
										 |  |  |  * \brief Local Proxy Channel | 
					
						
							| 
									
										
										
										
											2005-09-14 20:46:50 +00:00
										 |  |  |  *  | 
					
						
							| 
									
										
										
										
											2005-11-06 15:09:47 +00:00
										 |  |  |  * \ingroup channel_drivers | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-07 18:54:56 +00:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ASTERISK_FILE_VERSION(__FILE__, "$Revision$") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-06-06 21:09:59 +00:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | #include <sys/signal.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-21 06:02:45 +00:00
										 |  |  | #include "asterisk/lock.h"
 | 
					
						
							|  |  |  | #include "asterisk/channel.h"
 | 
					
						
							|  |  |  | #include "asterisk/config.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/pbx.h"
 | 
					
						
							|  |  |  | #include "asterisk/sched.h"
 | 
					
						
							|  |  |  | #include "asterisk/io.h"
 | 
					
						
							|  |  |  | #include "asterisk/acl.h"
 | 
					
						
							|  |  |  | #include "asterisk/callerid.h"
 | 
					
						
							|  |  |  | #include "asterisk/file.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/app.h"
 | 
					
						
							|  |  |  | #include "asterisk/musiconhold.h"
 | 
					
						
							|  |  |  | #include "asterisk/manager.h"
 | 
					
						
							| 
									
										
										
										
											2006-02-01 23:05:28 +00:00
										 |  |  | #include "asterisk/stringfields.h"
 | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | #include "asterisk/devicestate.h"
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-23 19:59:43 +00:00
										 |  |  | /*** DOCUMENTATION
 | 
					
						
							|  |  |  | 	<manager name="LocalOptimizeAway" language="en_US"> | 
					
						
							|  |  |  | 		<synopsis> | 
					
						
							|  |  |  | 			Optimize away a local channel when possible. | 
					
						
							|  |  |  | 		</synopsis> | 
					
						
							|  |  |  | 		<syntax> | 
					
						
							|  |  |  | 			<xi:include xpointer="xpointer(/docs/manager[@name='Login']/syntax/parameter[@name='ActionID'])" /> | 
					
						
							|  |  |  | 			<parameter name="Channel" required="true"> | 
					
						
							|  |  |  | 				<para>The channel name to optimize away.</para> | 
					
						
							|  |  |  | 			</parameter> | 
					
						
							|  |  |  | 		</syntax> | 
					
						
							|  |  |  | 		<description> | 
					
						
							|  |  |  | 			<para>A local channel created with "/n" will not automatically optimize away. | 
					
						
							|  |  |  | 			Calling this command on the local channel will clear that flag and allow | 
					
						
							|  |  |  | 			it to optimize away if it's bridged or when it becomes bridged.</para> | 
					
						
							|  |  |  | 		</description> | 
					
						
							|  |  |  | 	</manager> | 
					
						
							|  |  |  |  ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | static const char tdesc[] = "Local Proxy Channel Driver"; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define IS_OUTBOUND(a,b) (a == b->chan ? 1 : 0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | static struct ast_jb_conf g_jb_conf = { | 
					
						
							|  |  |  | 	.flags = 0, | 
					
						
							|  |  |  | 	.max_size = -1, | 
					
						
							|  |  |  | 	.resync_threshold = -1, | 
					
						
							|  |  |  | 	.impl = "", | 
					
						
							| 
									
										
										
										
											2010-03-02 19:08:38 +00:00
										 |  |  | 	.target_extra = -1, | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | static struct ast_channel *local_request(const char *type, format_t format, const struct ast_channel *requestor, void *data, int *cause); | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | static int local_digit_begin(struct ast_channel *ast, char digit); | 
					
						
							| 
									
										
											  
											
												Merged revisions 51311 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.4
........
r51311 | russell | 2007-01-19 11:49:38 -0600 (Fri, 19 Jan 2007) | 23 lines
Merge the changes from the /team/group/vldtmf_fixup branch.
The main bug being addressed here is a problem introduced when two SIP
channels using SIP INFO dtmf have their media directly bridged.  So, when a
DTMF END frame comes into Asterisk from an incoming INFO message, Asterisk
would try to emulate a digit of some length by first sending a DTMF BEGIN
frame and sending a DTMF END later timed off of incoming audio.  However,
since there was no audio coming in, the DTMF_END was never generated.  This
caused DTMF based features to no longer work.
To fix this, the core now knows when a channel doesn't care about DTMF BEGIN
frames (such as a SIP channel sending INFO dtmf).  If this is the case, then
Asterisk will not emulate a digit of some length, and will instead just pass
through the single DTMF END event.
Channel drivers also now get passed the length of the digit to their digit_end
callback.  This improves SIP INFO support even further by enabling us to put
the real digit duration in the INFO message instead of a hard coded 250ms.
Also, for an incoming INFO message, the duration is read from the frame and
passed into the core instead of just getting ignored.
(issue #8597, maybe others...)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@51314 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-01-19 18:06:03 +00:00
										 |  |  | static int local_digit_end(struct ast_channel *ast, char digit, unsigned int duration); | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | static int local_call(struct ast_channel *ast, 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); | 
					
						
							| 
									
										
										
										
											2006-05-10 12:24:11 +00:00
										 |  |  | static int local_indicate(struct ast_channel *ast, int condition, const void *data, size_t datalen); | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | static int local_fixup(struct ast_channel *oldchan, struct ast_channel *newchan); | 
					
						
							| 
									
										
										
										
											2005-03-28 20:48:24 +00:00
										 |  |  | static int local_sendhtml(struct ast_channel *ast, int subclass, const char *data, int datalen); | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | static int local_sendtext(struct ast_channel *ast, const char *text); | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | static int local_devicestate(void *data); | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | static struct ast_channel *local_bridgedchannel(struct ast_channel *chan, struct ast_channel *bridge); | 
					
						
							| 
									
										
										
										
											2010-07-29 14:03:59 +00:00
										 |  |  | static int local_queryoption(struct ast_channel *ast, int option, void *data, int *datalen); | 
					
						
							| 
									
										
										
										
											2010-09-10 22:15:47 +00:00
										 |  |  | static int local_setoption(struct ast_channel *chan, int option, void *data, int datalen); | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | /* PBX interface structure for channel registration */ | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | static const struct ast_channel_tech local_tech = { | 
					
						
							| 
									
										
										
										
											2006-02-01 23:05:28 +00:00
										 |  |  | 	.type = "Local", | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	.description = tdesc, | 
					
						
							|  |  |  | 	.capabilities = -1, | 
					
						
							|  |  |  | 	.requester = local_request, | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | 	.send_digit_begin = local_digit_begin, | 
					
						
							|  |  |  | 	.send_digit_end = local_digit_end, | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	.call = local_call, | 
					
						
							|  |  |  | 	.hangup = local_hangup, | 
					
						
							|  |  |  | 	.answer = local_answer, | 
					
						
							|  |  |  | 	.read = local_read, | 
					
						
							|  |  |  | 	.write = local_write, | 
					
						
							| 
									
										
										
										
											2006-05-19 08:03:28 +00:00
										 |  |  | 	.write_video = local_write, | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	.exception = local_read, | 
					
						
							|  |  |  | 	.indicate = local_indicate, | 
					
						
							|  |  |  | 	.fixup = local_fixup, | 
					
						
							|  |  |  | 	.send_html = local_sendhtml, | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | 	.send_text = local_sendtext, | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | 	.devicestate = local_devicestate, | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | 	.bridged_channel = local_bridgedchannel, | 
					
						
							| 
									
										
										
										
											2010-07-29 14:03:59 +00:00
										 |  |  | 	.queryoption = local_queryoption, | 
					
						
							| 
									
										
										
										
											2010-09-10 22:15:47 +00:00
										 |  |  | 	.setoption = local_setoption, | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-29 12:20:16 +00:00
										 |  |  | /*! \brief the local pvt structure for all channels
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	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 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | struct local_pvt { | 
					
						
							| 
									
										
										
										
											2009-10-29 12:20:16 +00:00
										 |  |  | 	ast_mutex_t lock;			/*!< Channel private lock */ | 
					
						
							|  |  |  | 	unsigned int flags;                     /*!< Private flags */ | 
					
						
							|  |  |  | 	char context[AST_MAX_CONTEXT];		/*!< Context to call */ | 
					
						
							|  |  |  | 	char exten[AST_MAX_EXTENSION];		/*!< Extension to call */ | 
					
						
							|  |  |  | 	int reqformat;				/*!< Requested format */ | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | 	struct ast_jb_conf jb_conf;		/*!< jitterbuffer configuration for this local channel */ | 
					
						
							| 
									
										
										
										
											2009-10-29 12:20:16 +00:00
										 |  |  | 	struct ast_channel *owner;		/*!< Master Channel - Bridging happens here */ | 
					
						
							|  |  |  | 	struct ast_channel *chan;		/*!< Outbound channel - PBX is run here */ | 
					
						
							|  |  |  | 	struct ast_module_user *u_owner;	/*!< reference to keep the module loaded while in use */ | 
					
						
							|  |  |  | 	struct ast_module_user *u_chan;		/*!< reference to keep the module loaded while in use */ | 
					
						
							|  |  |  | 	AST_LIST_ENTRY(local_pvt) list;		/*!< Next entity */ | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | #define LOCAL_GLARE_DETECT    (1 << 0) /*!< Detect glare on hangup */
 | 
					
						
							|  |  |  | #define LOCAL_CANCEL_QUEUE    (1 << 1) /*!< Cancel queue */
 | 
					
						
							|  |  |  | #define LOCAL_ALREADY_MASQED  (1 << 2) /*!< Already masqueraded */
 | 
					
						
							|  |  |  | #define LOCAL_LAUNCHED_PBX    (1 << 3) /*!< PBX was launched */
 | 
					
						
							|  |  |  | #define LOCAL_NO_OPTIMIZATION (1 << 4) /*!< Do not optimize using masquerading */
 | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | #define LOCAL_BRIDGE          (1 << 5) /*!< Report back the "true" channel as being bridged to */
 | 
					
						
							| 
									
										
										
										
											2009-10-29 18:13:42 +00:00
										 |  |  | #define LOCAL_MOH_PASSTHRU    (1 << 6) /*!< Pass through music on hold start/stop frames */
 | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | static AST_LIST_HEAD_STATIC(locals, local_pvt); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-10 22:15:47 +00:00
										 |  |  | static int local_setoption(struct ast_channel *chan, int option, void * data, int datalen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 	struct local_pvt *p; | 
					
						
							|  |  |  | 	struct ast_channel *otherchan; | 
					
						
							|  |  |  | 	ast_chan_write_info_t *write_info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (option != AST_OPTION_CHANNEL_WRITE) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	write_info = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (write_info->version != AST_CHAN_WRITE_INFO_T_VERSION) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "The chan_write_info_t type has changed, and this channel hasn't been updated!\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | startover: | 
					
						
							|  |  |  | 	ast_channel_lock(chan); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p = chan->tech_pvt; | 
					
						
							|  |  |  | 	if (!p) { | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not update other side of %s, local_pvt went away.\n", chan->name); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (ast_mutex_trylock(&p->lock)) { | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		sched_yield(); | 
					
						
							|  |  |  | 		ast_channel_lock(chan); | 
					
						
							|  |  |  | 		p = chan->tech_pvt; | 
					
						
							|  |  |  | 		if (!p) { | 
					
						
							|  |  |  | 			ast_channel_unlock(chan); | 
					
						
							|  |  |  | 			ast_log(LOG_WARNING, "Could not update other side of %s, local_pvt went away.\n", chan->name); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	otherchan = (write_info->chan == p->owner) ? p->chan : p->owner; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!otherchan || otherchan == write_info->chan) { | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Could not update other side of %s, other side went away.\n", chan->name); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_channel_trylock(otherchan)) { | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		goto startover; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = write_info->write_fn(otherchan, write_info->function, write_info->data, write_info->value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_channel_unlock(otherchan); | 
					
						
							|  |  |  | 	ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 	ast_channel_unlock(chan); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | /*! \brief Adds devicestate to local channels */ | 
					
						
							|  |  |  | static int local_devicestate(void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	char *exten = ast_strdupa(data); | 
					
						
							| 
									
										
										
										
											2006-09-26 20:20:34 +00:00
										 |  |  | 	char *context = NULL, *opts = NULL; | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
										
										
											2007-05-02 15:46:49 +00:00
										 |  |  | 	struct local_pvt *lp; | 
					
						
							| 
									
										
										
										
											2006-06-16 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	if (!(context = strchr(exten, '@'))) { | 
					
						
							| 
									
										
										
										
											2006-06-16 12:55:18 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Someone used Local/%s somewhere without a @context. This is bad.\n", exten); | 
					
						
							|  |  |  | 		return AST_DEVICE_INVALID;	 | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-06-16 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	*context++ = '\0'; | 
					
						
							| 
									
										
										
										
											2006-06-16 12:55:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-26 20:20:34 +00:00
										 |  |  | 	/* Strip options if they exist */ | 
					
						
							|  |  |  | 	if ((opts = strchr(context, '/'))) | 
					
						
							|  |  |  | 		*opts = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-14 19:39:12 +00:00
										 |  |  | 	ast_debug(3, "Checking if extension %s@%s exists (devicestate)\n", exten, context); | 
					
						
							| 
									
										
										
										
											2007-05-02 15:46:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | 	res = ast_exists_extension(NULL, context, exten, 1, NULL); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	if (!res)		 | 
					
						
							| 
									
										
										
										
											2006-06-14 22:39:19 +00:00
										 |  |  | 		return AST_DEVICE_INVALID; | 
					
						
							| 
									
										
										
										
											2007-05-02 15:46:49 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	res = AST_DEVICE_NOT_INUSE; | 
					
						
							|  |  |  | 	AST_LIST_LOCK(&locals); | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE(&locals, lp, list) { | 
					
						
							|  |  |  | 		if (!strcmp(exten, lp->exten) && !strcmp(context, lp->context) && lp->owner) { | 
					
						
							|  |  |  | 			res = AST_DEVICE_INUSE; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	AST_LIST_UNLOCK(&locals); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2006-05-19 07:56:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-03 17:17:27 +00:00
										 |  |  | /*!
 | 
					
						
							|  |  |  |  * \note Assumes the pvt is no longer in the pvts list | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | static struct local_pvt *local_pvt_destroy(struct local_pvt *pvt) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_mutex_destroy(&pvt->lock); | 
					
						
							|  |  |  | 	ast_free(pvt); | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | /*! \brief Return the bridged channel of a Local channel */ | 
					
						
							|  |  |  | static struct ast_channel *local_bridgedchannel(struct ast_channel *chan, struct ast_channel *bridge) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct local_pvt *p = bridge->tech_pvt; | 
					
						
							|  |  |  | 	struct ast_channel *bridged = bridge; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-09 22:50:22 +00:00
										 |  |  | 	if (!p) { | 
					
						
							|  |  |  | 		ast_debug(1, "Asked for bridged channel on '%s'/'%s', returning <none>\n", | 
					
						
							|  |  |  | 			chan->name, bridge->name); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_test_flag(p, LOCAL_BRIDGE)) { | 
					
						
							|  |  |  | 		/* Find the opposite channel */ | 
					
						
							|  |  |  | 		bridged = (bridge == p->owner ? p->chan : p->owner); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/* Now see if the opposite channel is bridged to anything */ | 
					
						
							| 
									
										
										
										
											2008-04-14 15:36:02 +00:00
										 |  |  | 		if (!bridged) { | 
					
						
							|  |  |  | 			bridged = bridge; | 
					
						
							|  |  |  | 		} else if (bridged->_bridge) { | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | 			bridged = bridged->_bridge; | 
					
						
							| 
									
										
										
										
											2008-04-14 15:36:02 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return bridged; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-29 14:03:59 +00:00
										 |  |  | static int local_queryoption(struct ast_channel *ast, int option, void *data, int *datalen) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							|  |  |  | 	struct ast_channel *chan, *bridged; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!p) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (option != AST_OPTION_T38_STATE) { | 
					
						
							|  |  |  | 		/* AST_OPTION_T38_STATE is the only supported option at this time */ | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 	chan = IS_OUTBOUND(ast, p) ? p->owner : p->chan; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | try_again: | 
					
						
							|  |  |  | 	if (!chan) { | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_channel_trylock(chan)) { | 
					
						
							|  |  |  | 		DEADLOCK_AVOIDANCE(&p->lock); | 
					
						
							|  |  |  | 		chan = IS_OUTBOUND(ast, p) ? p->owner : p->chan; | 
					
						
							|  |  |  | 		goto try_again; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bridged = ast_bridged_channel(chan); | 
					
						
							|  |  |  | 	if (!bridged) { | 
					
						
							|  |  |  | 		/* can't query channel unless we are bridged */ | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_channel_trylock(bridged)) { | 
					
						
							|  |  |  | 		ast_channel_unlock(chan); | 
					
						
							|  |  |  | 		DEADLOCK_AVOIDANCE(&p->lock); | 
					
						
							|  |  |  | 		chan = IS_OUTBOUND(ast, p) ? p->owner : p->chan; | 
					
						
							|  |  |  | 		goto try_again; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	res = ast_channel_queryoption(bridged, option, data, datalen, 0); | 
					
						
							|  |  |  | 	ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 	ast_channel_unlock(chan); | 
					
						
							|  |  |  | 	ast_channel_unlock(bridged); | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | static int local_queue_frame(struct local_pvt *p, int isoutbound, struct ast_frame *f,  | 
					
						
							|  |  |  | 	struct ast_channel *us, int us_locked) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct ast_channel *other = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-12-02 15:52:37 +00:00
										 |  |  | 	/* Recalculate outbound channel */ | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	other = isoutbound ? p->owner : p->chan; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-12-02 15:52:37 +00:00
										 |  |  | 	if (!other) { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-01-28 17:21:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-19 15:54:06 +00:00
										 |  |  | 	/* do not queue frame if generator is on both local channels */ | 
					
						
							| 
									
										
										
										
											2009-03-16 15:50:55 +00:00
										 |  |  | 	if (us && us->generator && other->generator) { | 
					
						
							| 
									
										
										
										
											2009-01-19 15:54:06 +00:00
										 |  |  | 		return 0; | 
					
						
							| 
									
										
										
										
											2009-03-16 15:50:55 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Set glare detection */ | 
					
						
							|  |  |  | 	ast_set_flag(p, LOCAL_GLARE_DETECT); | 
					
						
							| 
									
										
										
										
											2009-01-19 15:54:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-28 17:21:24 +00:00
										 |  |  | 	/* Ensure that we have both channels locked */ | 
					
						
							| 
									
										
										
										
											2008-02-19 20:06:23 +00:00
										 |  |  | 	while (other && ast_channel_trylock(other)) { | 
					
						
							| 
									
										
										
										
											2010-07-03 02:36:31 +00:00
										 |  |  | 		int res; | 
					
						
							|  |  |  | 		if ((res = ast_mutex_unlock(&p->lock))) { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR, "chan_local bug! '&p->lock' was not locked when entering local_queue_frame! (%s)\n", strerror(res)); | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		if (us && us_locked) { | 
					
						
							| 
									
										
										
										
											2008-06-27 12:28:38 +00:00
										 |  |  | 			do { | 
					
						
							|  |  |  | 				CHANNEL_DEADLOCK_AVOIDANCE(us); | 
					
						
							|  |  |  | 			} while (ast_mutex_trylock(&p->lock)); | 
					
						
							| 
									
										
										
										
											2008-06-25 02:34:11 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			usleep(1); | 
					
						
							| 
									
										
										
										
											2008-06-27 12:28:38 +00:00
										 |  |  | 			ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-02-19 20:06:23 +00:00
										 |  |  | 		other = isoutbound ? p->owner : p->chan; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-01-28 17:21:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-16 15:50:55 +00:00
										 |  |  | 	/* Since glare detection only occurs within this function, and because
 | 
					
						
							|  |  |  | 	 * a pvt flag cannot be set without having the pvt lock, this is the only | 
					
						
							|  |  |  | 	 * location where we could detect a cancelling of the queue. */ | 
					
						
							|  |  |  | 	if (ast_test_flag(p, LOCAL_CANCEL_QUEUE)) { | 
					
						
							|  |  |  | 		/* We had a glare on the hangup.  Forget all this business,
 | 
					
						
							|  |  |  | 		return and destroy p.  */ | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		p = local_pvt_destroy(p); | 
					
						
							| 
									
										
										
										
											2009-08-31 16:18:33 +00:00
										 |  |  | 		if (other) { | 
					
						
							|  |  |  | 			ast_channel_unlock(other); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-03-16 15:50:55 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-19 20:06:23 +00:00
										 |  |  | 	if (other) { | 
					
						
							| 
									
										
										
										
											2010-03-01 17:11:31 +00:00
										 |  |  | 		if (f->frametype == AST_FRAME_CONTROL && f->subclass.integer == AST_CONTROL_RINGING) { | 
					
						
							|  |  |  | 			ast_setstate(other, AST_STATE_RINGING); | 
					
						
							| 
									
										
											  
											
												Change channel state on local channels for busy,answer,ring.
  
Previously local channels channel state never changed. This became problematic
when the state of the other side of the local channel was lost, for example
during a masquerade. Changing the state of the local channel allows for the
scenario to be detected when the channel state is set to ringing, but the peer
isn't ringing. The specific problem scenario is described in 164201. Although
this was noted on one of the issues, here is the tested dialplan verified to
work:
exten => 9700,1,Dial(Local/*9700@default&Local/0009700@default)
exten => *9700,1,Set(GLOBAL(TESTCHAN)=${CHANNEL:0:${MATH(${LEN(${CHANNEL})}-1):0:2}}1)
exten => *9700,n,wait(3) ;3 works, 1 did not
exten => *9700,n,Dial(SIP/5001)
exten => 0009700,1,Wait(1) ;1 works, 3 did not
exten => 0009700,n,ChannelRedirect(${TESTCHAN},parkedcalls,701,1)
(closes issue #14992)
Reported by: davidw
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@246070 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-02-10 16:47:37 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-02-01 17:53:39 +00:00
										 |  |  | 		ast_queue_frame(other, f); | 
					
						
							| 
									
										
										
										
											2008-02-19 20:06:23 +00:00
										 |  |  | 		ast_channel_unlock(other); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-01-28 17:21:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 	ast_clear_flag(p, LOCAL_GLARE_DETECT); | 
					
						
							| 
									
										
										
										
											2008-01-28 17:21:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int local_answer(struct ast_channel *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	int res = -1; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	if (isoutbound) { | 
					
						
							|  |  |  | 		/* Pass along answer since somebody answered us */ | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 		struct ast_frame answer = { AST_FRAME_CONTROL, { AST_CONTROL_ANSWER } }; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		res = local_queue_frame(p, isoutbound, &answer, ast, 1); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	} else | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Huh?  Local is being asked to answer?\n"); | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 	if (!res) | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-02 23:46:45 +00:00
										 |  |  | /*!
 | 
					
						
							|  |  |  |  * \internal | 
					
						
							|  |  |  |  * \note This function assumes that we're only called from the "outbound" local channel side | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void check_bridge(struct local_pvt *p) | 
					
						
							| 
									
										
										
										
											2003-04-06 06:11:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-01-21 23:56:40 +00:00
										 |  |  | 	struct ast_channel_monitor *tmp; | 
					
						
							| 
									
										
										
										
											2007-03-01 22:23:26 +00:00
										 |  |  | 	if (ast_test_flag(p, LOCAL_ALREADY_MASQED) || ast_test_flag(p, LOCAL_NO_OPTIMIZATION) || !p->chan || !p->owner || (p->chan->_bridge != ast_bridged_channel(p->chan))) | 
					
						
							| 
									
										
										
										
											2005-10-19 03:45:49 +00:00
										 |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2006-06-12 16:07:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* only do the masquerade if we are being called on the outbound channel,
 | 
					
						
							|  |  |  | 	   if it has been bridged to another channel and if there are no pending | 
					
						
							|  |  |  | 	   frames on the owner channel (because they would be transferred to the | 
					
						
							|  |  |  | 	   outbound channel during the masquerade) | 
					
						
							|  |  |  | 	*/ | 
					
						
							| 
									
										
										
										
											2010-04-02 23:46:45 +00:00
										 |  |  | 	if (p->chan->_bridge /* Not ast_bridged_channel!  Only go one step! */ && AST_LIST_EMPTY(&p->owner->readq)) { | 
					
						
							| 
									
										
										
										
											2003-04-06 06:11:25 +00:00
										 |  |  | 		/* Masquerade bridged channel into owner */ | 
					
						
							| 
									
										
										
										
											2003-08-14 13:12:39 +00:00
										 |  |  | 		/* Lock everything we need, one by one, and give up if
 | 
					
						
							|  |  |  | 		   we can't get everything.  Remember, we'll get another | 
					
						
							|  |  |  | 		   chance in just a little bit */ | 
					
						
							| 
									
										
										
										
											2007-09-17 22:59:36 +00:00
										 |  |  | 		if (!ast_channel_trylock(p->chan->_bridge)) { | 
					
						
							| 
									
										
										
										
											2007-08-01 15:39:54 +00:00
										 |  |  | 			if (!ast_check_hangup(p->chan->_bridge)) { | 
					
						
							| 
									
										
										
										
											2007-09-17 22:59:36 +00:00
										 |  |  | 				if (!ast_channel_trylock(p->owner)) { | 
					
						
							| 
									
										
										
										
											2007-08-01 15:39:54 +00:00
										 |  |  | 					if (!ast_check_hangup(p->owner)) { | 
					
						
							| 
									
										
										
										
											2008-11-18 18:31:08 +00:00
										 |  |  | 						if (p->owner->monitor && !p->chan->_bridge->monitor) { | 
					
						
							| 
									
										
										
										
											2008-01-21 23:56:40 +00:00
										 |  |  | 							/* If a local channel is being monitored, we don't want a masquerade
 | 
					
						
							|  |  |  | 							 * to cause the monitor to go away. Since the masquerade swaps the monitors, | 
					
						
							|  |  |  | 							 * pre-swapping the monitors before the masquerade will ensure that the monitor | 
					
						
							|  |  |  | 							 * ends up where it is expected. | 
					
						
							|  |  |  | 							 */ | 
					
						
							|  |  |  | 							tmp = p->owner->monitor; | 
					
						
							|  |  |  | 							p->owner->monitor = p->chan->_bridge->monitor; | 
					
						
							|  |  |  | 							p->chan->_bridge->monitor = tmp; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2008-11-18 18:31:08 +00:00
										 |  |  | 						if (p->chan->audiohooks) { | 
					
						
							|  |  |  | 							struct ast_audiohook_list *audiohooks_swapper; | 
					
						
							|  |  |  | 							audiohooks_swapper = p->chan->audiohooks; | 
					
						
							|  |  |  | 							p->chan->audiohooks = p->owner->audiohooks; | 
					
						
							|  |  |  | 							p->owner->audiohooks = audiohooks_swapper; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2010-08-09 20:46:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						/* If any Caller ID was set, preserve it after masquerade like above. We must check
 | 
					
						
							|  |  |  | 						 * to see if Caller ID was set because otherwise we'll mistakingly copy info not | 
					
						
							|  |  |  | 						 * set from the dialplan and will overwrite the real channel Caller ID. The reason | 
					
						
							|  |  |  | 						 * for this whole preswapping action is because the Caller ID is set on the channel | 
					
						
							|  |  |  | 						 * thread (which is the to be masqueraded away local channel) before both local | 
					
						
							|  |  |  | 						 * channels are optimized away. | 
					
						
							|  |  |  | 						 */ | 
					
						
							| 
									
										
										
										
											2010-08-09 23:04:59 +00:00
										 |  |  | 						if (p->owner->caller.id.name.valid || p->owner->caller.id.number.valid | 
					
						
							|  |  |  | 							|| p->owner->caller.id.subaddress.valid || p->owner->caller.ani.name.valid | 
					
						
							|  |  |  | 							|| p->owner->caller.ani.number.valid || p->owner->caller.ani.subaddress.valid) { | 
					
						
							| 
									
										
										
										
											2010-08-09 20:46:50 +00:00
										 |  |  | 							struct ast_party_caller tmp; | 
					
						
							|  |  |  | 							tmp = p->owner->caller; | 
					
						
							|  |  |  | 							p->owner->caller = p->chan->_bridge->caller; | 
					
						
							|  |  |  | 							p->chan->_bridge->caller = tmp; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2010-08-09 23:04:59 +00:00
										 |  |  | 						if (p->owner->redirecting.from.name.valid || p->owner->redirecting.from.number.valid | 
					
						
							|  |  |  | 							|| p->owner->redirecting.from.subaddress.valid || p->owner->redirecting.to.name.valid | 
					
						
							|  |  |  | 							|| p->owner->redirecting.to.number.valid || p->owner->redirecting.to.subaddress.valid) { | 
					
						
							|  |  |  | 							struct ast_party_redirecting tmp; | 
					
						
							|  |  |  | 							tmp = p->owner->redirecting; | 
					
						
							|  |  |  | 							p->owner->redirecting = p->chan->_bridge->redirecting; | 
					
						
							|  |  |  | 							p->chan->_bridge->redirecting = tmp; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						if (p->owner->dialed.number.str || p->owner->dialed.subaddress.valid) { | 
					
						
							|  |  |  | 							struct ast_party_dialed tmp; | 
					
						
							|  |  |  | 							tmp = p->owner->dialed; | 
					
						
							|  |  |  | 							p->owner->dialed = p->chan->_bridge->dialed; | 
					
						
							|  |  |  | 							p->chan->_bridge->dialed = tmp; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-09 20:46:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-17 21:14:18 +00:00
										 |  |  | 						ast_app_group_update(p->chan, p->owner); | 
					
						
							| 
									
										
										
										
											2005-10-19 03:45:49 +00:00
										 |  |  | 						ast_channel_masquerade(p->owner, p->chan->_bridge); | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 						ast_set_flag(p, LOCAL_ALREADY_MASQED); | 
					
						
							| 
									
										
										
										
											2005-10-19 03:45:49 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2007-09-17 22:59:36 +00:00
										 |  |  | 					ast_channel_unlock(p->owner); | 
					
						
							| 
									
										
										
										
											2005-10-19 03:45:49 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2007-09-17 22:59:36 +00:00
										 |  |  | 				ast_channel_unlock(p->chan->_bridge); | 
					
						
							| 
									
										
										
										
											2003-08-14 13:12:39 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-06 06:11:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | static struct ast_frame  *local_read(struct ast_channel *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-01-31 17:57:12 +00:00
										 |  |  | 	return &ast_null_frame; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int local_write(struct ast_channel *ast, struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	int res = -1; | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	/* Just queue for delivery to the other side */ | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2010-04-02 23:46:45 +00:00
										 |  |  | 	if (isoutbound && f && (f->frametype == AST_FRAME_VOICE || f->frametype == AST_FRAME_VIDEO)) | 
					
						
							|  |  |  | 		check_bridge(p); | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 	if (!ast_test_flag(p, LOCAL_ALREADY_MASQED)) | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		res = local_queue_frame(p, isoutbound, f, ast, 1); | 
					
						
							| 
									
										
										
										
											2005-03-05 04:35:23 +00:00
										 |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2007-06-14 19:39:12 +00:00
										 |  |  | 		ast_debug(1, "Not posting to queue since already masked on '%s'\n", ast->name); | 
					
						
							| 
									
										
										
										
											2005-03-05 04:35:23 +00:00
										 |  |  | 		res = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 	if (!res) | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | static int local_fixup(struct ast_channel *oldchan, struct ast_channel *newchan) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = newchan->tech_pvt; | 
					
						
							| 
									
										
										
										
											2006-11-15 22:32:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	if ((p->owner != oldchan) && (p->chan != oldchan)) { | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Old channel wasn't %p but was %p/%p\n", oldchan, p->owner, p->chan); | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (p->owner == oldchan) | 
					
						
							|  |  |  | 		p->owner = newchan; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 		p->chan = newchan; | 
					
						
							| 
									
										
										
										
											2010-09-22 23:20:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Do not let a masquerade cause a Local channel to be bridged to itself! */ | 
					
						
							| 
									
										
										
										
											2010-09-24 16:11:19 +00:00
										 |  |  | 	if (!ast_check_hangup(newchan) && (p->owner->_bridge == p->chan || p->chan->_bridge == p->owner)) { | 
					
						
							| 
									
										
										
										
											2010-09-22 23:20:27 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "You can not bridge a Local channel to itself!\n"); | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		ast_queue_hangup(newchan); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-04-06 22:17:32 +00:00
										 |  |  | 	ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-10 12:24:11 +00:00
										 |  |  | static int local_indicate(struct ast_channel *ast, int condition, const void *data, size_t datalen) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2006-11-07 20:16:02 +00:00
										 |  |  | 	int res = 0; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	struct ast_frame f = { AST_FRAME_CONTROL, }; | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 	/* If this is an MOH hold or unhold, do it on the Local channel versus real channel */ | 
					
						
							| 
									
										
										
										
											2009-10-29 18:13:42 +00:00
										 |  |  | 	if (!ast_test_flag(p, LOCAL_MOH_PASSTHRU) && condition == AST_CONTROL_HOLD) { | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 		ast_moh_start(ast, data, NULL); | 
					
						
							| 
									
										
										
										
											2009-10-29 18:13:42 +00:00
										 |  |  | 	} else if (!ast_test_flag(p, LOCAL_MOH_PASSTHRU) && condition == AST_CONTROL_UNHOLD) { | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 		ast_moh_stop(ast); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 	} 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.*/ | 
					
						
							| 
									
										
										
										
											2009-05-28 18:48:56 +00:00
										 |  |  | 		ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 		isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							|  |  |  | 		if (isoutbound) { | 
					
						
							|  |  |  | 			this_channel = p->chan; | 
					
						
							|  |  |  | 			the_other_channel = p->owner; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			this_channel = p->owner; | 
					
						
							|  |  |  | 			the_other_channel = p->chan; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (the_other_channel) { | 
					
						
							|  |  |  | 			unsigned char frame_data[1024]; | 
					
						
							|  |  |  | 			if (condition == AST_CONTROL_CONNECTED_LINE) { | 
					
						
							| 
									
										
											  
											
												Enhancements to connected line and redirecting work.
From reviewboard:
Digium has a commercial customer who has made extensive use of the connected party and
redirecting information present in later versions of Asterisk Business Edition and which
is to be in the upcoming 1.8 release. Through their use of the feature, new problems and solutions
have come about. This patch adds several enhancements to maximize usage of the connected party
and redirecting information functionality.
First, Asterisk trunk already had connected line interception macros. These macros allow you to
manipulate connected line information before it was sent out to its target. This patch adds the
same feature except for redirecting information instead.
Second, the ast_callerid and ast_party_id structures have been enhanced to provide a "tag." This
tag can be set with func_callerid, func_connectedline, func_redirecting, and in the case of DAHDI,
mISDN, and SIP channels, can be set in a configuration file. The idea behind the callerid tag is
that it can be set to whatever value the administrator likes. Later, when running connected line
and redirecting macros, the admin can read the tag off the appropriate structure to determine what
action to take. You can think of this sort of like a channel variable, except that instead of having
the variable associated with a channel, the variable is associated with a specific identity within
Asterisk.
Third, app_dial has two new options, s and u. The s option lets a dialplan writer force a specific
caller ID tag to be placed on the outgoing channel. The u option allows the dialplan writer to force
a specific calling presentation value on the outgoing channel.
Fourth, there is a new control frame subclass called AST_CONTROL_READ_ACTION added. This was added
to correct a very specific situation. In the case of SIP semi-attended (blond) transfers, the party
being transferred would not have the opportunity to run a connected line interception macro to
possibly alter the transfer target's connected line information. The issue here was that during a
blond transfer, the SIP transfer code has no bridged channel on which to queue the connected line
update. The way this was corrected was to add this new control frame subclass. Now, we queue an
AST_CONTROL_READ_ACTION frame on the channel on which the connected line interception macro should
be run. When ast_read is called to read the frame, ast_read responds by calling a callback function
associated with the specific read action the control frame describes. In this case, the action taken
is to run the connected line interception macro on the transferee's channel.
Review: https://reviewboard.asterisk.org/r/652/
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@263541 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-05-17 15:36:31 +00:00
										 |  |  | 				if (isoutbound) { | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 					ast_connected_line_copy_to_caller(&the_other_channel->caller, &this_channel->connected); | 
					
						
							| 
									
										
											  
											
												Enhancements to connected line and redirecting work.
From reviewboard:
Digium has a commercial customer who has made extensive use of the connected party and
redirecting information present in later versions of Asterisk Business Edition and which
is to be in the upcoming 1.8 release. Through their use of the feature, new problems and solutions
have come about. This patch adds several enhancements to maximize usage of the connected party
and redirecting information functionality.
First, Asterisk trunk already had connected line interception macros. These macros allow you to
manipulate connected line information before it was sent out to its target. This patch adds the
same feature except for redirecting information instead.
Second, the ast_callerid and ast_party_id structures have been enhanced to provide a "tag." This
tag can be set with func_callerid, func_connectedline, func_redirecting, and in the case of DAHDI,
mISDN, and SIP channels, can be set in a configuration file. The idea behind the callerid tag is
that it can be set to whatever value the administrator likes. Later, when running connected line
and redirecting macros, the admin can read the tag off the appropriate structure to determine what
action to take. You can think of this sort of like a channel variable, except that instead of having
the variable associated with a channel, the variable is associated with a specific identity within
Asterisk.
Third, app_dial has two new options, s and u. The s option lets a dialplan writer force a specific
caller ID tag to be placed on the outgoing channel. The u option allows the dialplan writer to force
a specific calling presentation value on the outgoing channel.
Fourth, there is a new control frame subclass called AST_CONTROL_READ_ACTION added. This was added
to correct a very specific situation. In the case of SIP semi-attended (blond) transfers, the party
being transferred would not have the opportunity to run a connected line interception macro to
possibly alter the transfer target's connected line information. The issue here was that during a
blond transfer, the SIP transfer code has no bridged channel on which to queue the connected line
update. The way this was corrected was to add this new control frame subclass. Now, we queue an
AST_CONTROL_READ_ACTION frame on the channel on which the connected line interception macro should
be run. When ast_read is called to read the frame, ast_read responds by calling a callback function
associated with the specific read action the control frame describes. In this case, the action taken
is to run the connected line interception macro on the transferee's channel.
Review: https://reviewboard.asterisk.org/r/652/
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@263541 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-05-17 15:36:31 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 				f.datalen = ast_connected_line_build_data(frame_data, sizeof(frame_data), &this_channel->connected, NULL); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 				f.datalen = ast_redirecting_build_data(frame_data, sizeof(frame_data), &this_channel->redirecting, NULL); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 			f.subclass.integer = condition; | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 			f.data.ptr = frame_data; | 
					
						
							|  |  |  | 			if (!(res = local_queue_frame(p, isoutbound, &f, ast, 1))) { | 
					
						
							|  |  |  | 				ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2009-06-01 14:19:49 +00:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		/* Queue up a frame representing the indication as a control frame */ | 
					
						
							|  |  |  | 		ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 		isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 		f.subclass.integer = condition; | 
					
						
							| 
									
										
										
										
											2008-05-22 16:29:54 +00:00
										 |  |  | 		f.data.ptr = (void*)data; | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 		f.datalen = datalen; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		if (!(res = local_queue_frame(p, isoutbound, &f, ast, 1))) | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 			ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2006-11-07 20:12:30 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | static int local_digit_begin(struct ast_channel *ast, char digit) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	int res = -1; | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | 	struct ast_frame f = { AST_FRAME_DTMF_BEGIN, }; | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-08-13 15:25:16 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 	f.subclass.integer = digit; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 	if (!(res = local_queue_frame(p, isoutbound, &f, ast, 0))) | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 51311 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.4
........
r51311 | russell | 2007-01-19 11:49:38 -0600 (Fri, 19 Jan 2007) | 23 lines
Merge the changes from the /team/group/vldtmf_fixup branch.
The main bug being addressed here is a problem introduced when two SIP
channels using SIP INFO dtmf have their media directly bridged.  So, when a
DTMF END frame comes into Asterisk from an incoming INFO message, Asterisk
would try to emulate a digit of some length by first sending a DTMF BEGIN
frame and sending a DTMF END later timed off of incoming audio.  However,
since there was no audio coming in, the DTMF_END was never generated.  This
caused DTMF based features to no longer work.
To fix this, the core now knows when a channel doesn't care about DTMF BEGIN
frames (such as a SIP channel sending INFO dtmf).  If this is the case, then
Asterisk will not emulate a digit of some length, and will instead just pass
through the single DTMF END event.
Channel drivers also now get passed the length of the digit to their digit_end
callback.  This improves SIP INFO support even further by enabling us to put
the real digit duration in the INFO message instead of a hard coded 250ms.
Also, for an incoming INFO message, the duration is read from the frame and
passed into the core instead of just getting ignored.
(issue #8597, maybe others...)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@51314 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-01-19 18:06:03 +00:00
										 |  |  | static int local_digit_end(struct ast_channel *ast, char digit, unsigned int duration) | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							|  |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	struct ast_frame f = { AST_FRAME_DTMF_END, }; | 
					
						
							|  |  |  | 	int isoutbound; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-08-31 01:59:02 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 	f.subclass.integer = digit; | 
					
						
							| 
									
										
											  
											
												Merged revisions 51311 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.4
........
r51311 | russell | 2007-01-19 11:49:38 -0600 (Fri, 19 Jan 2007) | 23 lines
Merge the changes from the /team/group/vldtmf_fixup branch.
The main bug being addressed here is a problem introduced when two SIP
channels using SIP INFO dtmf have their media directly bridged.  So, when a
DTMF END frame comes into Asterisk from an incoming INFO message, Asterisk
would try to emulate a digit of some length by first sending a DTMF BEGIN
frame and sending a DTMF END later timed off of incoming audio.  However,
since there was no audio coming in, the DTMF_END was never generated.  This
caused DTMF based features to no longer work.
To fix this, the core now knows when a channel doesn't care about DTMF BEGIN
frames (such as a SIP channel sending INFO dtmf).  If this is the case, then
Asterisk will not emulate a digit of some length, and will instead just pass
through the single DTMF END event.
Channel drivers also now get passed the length of the digit to their digit_end
callback.  This improves SIP INFO support even further by enabling us to put
the real digit duration in the INFO message instead of a hard coded 250ms.
Also, for an incoming INFO message, the duration is read from the frame and
passed into the core instead of just getting ignored.
(issue #8597, maybe others...)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@51314 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-01-19 18:06:03 +00:00
										 |  |  | 	f.len = duration; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 	if (!(res = local_queue_frame(p, isoutbound, &f, ast, 0))) | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | static int local_sendtext(struct ast_channel *ast, const char *text) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							|  |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	struct ast_frame f = { AST_FRAME_TEXT, }; | 
					
						
							|  |  |  | 	int isoutbound; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2008-05-22 16:29:54 +00:00
										 |  |  | 	f.data.ptr = (char *) text; | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | 	f.datalen = strlen(text) + 1; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 	if (!(res = local_queue_frame(p, isoutbound, &f, ast, 0))) | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2006-05-19 08:20:01 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-03-28 20:48:24 +00:00
										 |  |  | static int local_sendhtml(struct ast_channel *ast, int subclass, const char *data, int datalen) | 
					
						
							| 
									
										
										
										
											2005-02-28 06:06:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2005-02-28 06:06:42 +00:00
										 |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	struct ast_frame f = { AST_FRAME_HTML, }; | 
					
						
							|  |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2005-02-28 06:06:42 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 	f.subclass.integer = subclass; | 
					
						
							| 
									
										
										
										
											2008-05-22 16:29:54 +00:00
										 |  |  | 	f.data.ptr = (char *)data; | 
					
						
							| 
									
										
										
										
											2005-02-28 06:06:42 +00:00
										 |  |  | 	f.datalen = datalen; | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 	if (!(res = local_queue_frame(p, isoutbound, &f, ast, 0))) | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2005-02-28 06:06:42 +00:00
										 |  |  | 	return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Initiate new call, part of PBX interface 
 | 
					
						
							|  |  |  |  * 	dest is the dial string */ | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | static int local_call(struct ast_channel *ast, char *dest, int timeout) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2003-08-19 19:36:52 +00:00
										 |  |  | 	int res; | 
					
						
							| 
									
										
											  
											
												Merged revisions 7265-7266,7268-7275 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.2
........
r7265 | oej | 2005-12-01 17:18:14 -0600 (Thu, 01 Dec 2005) | 2 lines
Changing bug report address to the Asterisk issue tracker
........
r7266 | kpfleming | 2005-12-01 17:18:29 -0600 (Thu, 01 Dec 2005) | 3 lines
Makefile 'update' target now supports updating from Subversion repositories (issue #5875)
remove support for 'patches' subdirectory, it's no longer useful
........
r7268 | kpfleming | 2005-12-01 17:34:58 -0600 (Thu, 01 Dec 2005) | 2 lines
ensure channel's scheduling context is freed (issue #5788)
........
r7269 | kpfleming | 2005-12-01 17:49:44 -0600 (Thu, 01 Dec 2005) | 2 lines
don't block waiting for the Festival server forever when it goes away (issue #5882)
........
r7270 | kpfleming | 2005-12-01 18:26:12 -0600 (Thu, 01 Dec 2005) | 2 lines
allow variables to exist on both 'halves' of the Local channel (issue #5810)
........
r7271 | kpfleming | 2005-12-01 18:28:48 -0600 (Thu, 01 Dec 2005) | 2 lines
protect agent_bridgedchannel() from segfaulting when there is no bridged channel (issue #5879)
........
r7272 | kpfleming | 2005-12-01 18:39:00 -0600 (Thu, 01 Dec 2005) | 3 lines
properly handle password changes when mailbox is last line of config file and not followed by a newline (issue #5870)
reformat password changing code to conform to coding guidelines (issue #5870)
........
r7273 | kpfleming | 2005-12-01 18:42:40 -0600 (Thu, 01 Dec 2005) | 2 lines
allow previous context-searching behavior to be used if desired (issue #5899)
........
r7274 | kpfleming | 2005-12-01 18:51:15 -0600 (Thu, 01 Dec 2005) | 2 lines
inherit channel variables into channels created by Page() application (issue #5888)
........
r7275 | oej | 2005-12-01 18:52:13 -0600 (Thu, 01 Dec 2005) | 2 lines
Bug #5907. Improve SIP INFO DTMF debugging output. (1.2 & Trunk)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@7276 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2005-12-02 01:01:11 +00:00
										 |  |  | 	struct ast_var_t *varptr = NULL, *new; | 
					
						
							|  |  |  | 	size_t len, namelen; | 
					
						
							| 
									
										
											  
											
												Merge Call completion support into trunk.
From Reviewboard:
CCSS stands for Call Completion Supplementary Services. An admittedly out-of-date
overview of the architecture can be found in the file doc/CCSS_architecture.pdf
in the CCSS branch. Off the top of my head, the big differences between what is
implemented and what is in the document are as follows:
1. We did not end up modifying the Hangup application at all.
2. The document states that a single call completion monitor may be used across
   multiple calls to the same device. This proved to not be such a good idea
   when implementing protocol-specific monitors, and so we ended up using one
   monitor per-device per-call.
3. There are some configuration options which were conceived after the document
   was written. These are documented in the ccss.conf.sample that is on this
   review request.
		      
For some basic understanding of terminology used throughout this code, see the
ccss.tex document that is on this review.
This implements CCBS and CCNR in several flavors.
First up is a "generic" implementation, which can work over any channel technology
provided that the channel technology can accurately report device state. Call
completion is requested using the dialplan application CallCompletionRequest and can
be canceled using CallCompletionCancel. Device state subscriptions are used in order
to monitor the state of called parties.
Next, there is a SIP-specific implementation of call completion. This method uses the
methods outlined in draft-ietf-bliss-call-completion-06 to implement call completion
using SIP signaling. There are a few things to note here:
* The agent/monitor terminology used throughout Asterisk sometimes is the reverse of
  what is defined in the referenced draft.
* Implementation of the draft required support for SIP PUBLISH. I attempted to write
  this in a generic-enough fashion such that if someone were to want to write PUBLISH
  support for other event packages, such as dialog-state or presence, most of the effort
  would be in writing callbacks specific to the event package.
* A subportion of supporting PUBLISH reception was that we had to implement a PIDF
  parser. The PIDF support added is a bit minimal. I first wrote a validation
  routine to ensure that the PIDF document is formatted properly. The rest of the
  PIDF reading is done in-line in the call-completion-specific PUBLISH-handling
  code. In other words, while there is PIDF support here, it is not in any state
  where it could easily be applied to other event packages as is.
Finally, there are a variety of ISDN-related call completion protocols supported. These
were written by Richard Mudgett, and as such I can't really say much about their
implementation. There are notes in the CHANGES file that indicate the ISDN protocols
over which call completion is supported.
Review: https://reviewboard.asterisk.org/r/523
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@256528 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-04-09 15:31:32 +00:00
										 |  |  | 	char *reduced_dest = ast_strdupa(dest); | 
					
						
							|  |  |  | 	char *slash; | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* If you value your sanity, please don't look at this code */ | 
					
						
							|  |  |  | start_over: | 
					
						
							|  |  |  | 	while (ast_channel_trylock(p->chan)) { | 
					
						
							|  |  |  | 		ast_channel_unlock(p->owner); | 
					
						
							|  |  |  | 		usleep(1); | 
					
						
							|  |  |  | 		ast_channel_lock(p->owner); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* p->owner and p->chan are locked now. Let's get p locked */ | 
					
						
							|  |  |  | 	if (ast_mutex_trylock(&p->lock)) { | 
					
						
							|  |  |  | 		/* @#$&$@ */ | 
					
						
							|  |  |  | 		ast_channel_unlock(p->chan); | 
					
						
							|  |  |  | 		ast_channel_unlock(p->owner); | 
					
						
							|  |  |  | 		usleep(1); | 
					
						
							|  |  |  | 		ast_channel_lock(p->owner); | 
					
						
							|  |  |  | 		goto start_over; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-10-02 00:58:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-08-17 17:45:01 +00:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Note that cid_num and cid_name aren't passed in the ast_channel_alloc | 
					
						
							|  |  |  | 	 * call, so it's done here instead. | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 	 * | 
					
						
							|  |  |  | 	 * All these failure points just return -1. The individual strings will | 
					
						
							|  |  |  | 	 * be cleared when we destroy the channel. | 
					
						
							| 
									
										
										
										
											2007-08-17 17:45:01 +00:00
										 |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 	ast_party_redirecting_copy(&p->chan->redirecting, &p->owner->redirecting); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 	ast_party_dialed_copy(&p->chan->dialed, &p->owner->dialed); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 	ast_connected_line_copy_to_caller(&p->chan->caller, &p->owner->connected); | 
					
						
							|  |  |  | 	ast_connected_line_copy_from_caller(&p->chan->connected, &p->owner->caller); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-01 23:05:28 +00:00
										 |  |  | 	ast_string_field_set(p->chan, language, p->owner->language); | 
					
						
							|  |  |  | 	ast_string_field_set(p->chan, accountcode, p->owner->accountcode); | 
					
						
							| 
									
										
										
										
											2008-12-31 19:34:28 +00:00
										 |  |  | 	ast_string_field_set(p->chan, musicclass, p->owner->musicclass); | 
					
						
							| 
									
										
										
										
											2008-04-24 19:54:57 +00:00
										 |  |  | 	ast_cdr_update(p->chan); | 
					
						
							| 
									
										
											  
											
												Merged revisions 7265-7266,7268-7275 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.2
........
r7265 | oej | 2005-12-01 17:18:14 -0600 (Thu, 01 Dec 2005) | 2 lines
Changing bug report address to the Asterisk issue tracker
........
r7266 | kpfleming | 2005-12-01 17:18:29 -0600 (Thu, 01 Dec 2005) | 3 lines
Makefile 'update' target now supports updating from Subversion repositories (issue #5875)
remove support for 'patches' subdirectory, it's no longer useful
........
r7268 | kpfleming | 2005-12-01 17:34:58 -0600 (Thu, 01 Dec 2005) | 2 lines
ensure channel's scheduling context is freed (issue #5788)
........
r7269 | kpfleming | 2005-12-01 17:49:44 -0600 (Thu, 01 Dec 2005) | 2 lines
don't block waiting for the Festival server forever when it goes away (issue #5882)
........
r7270 | kpfleming | 2005-12-01 18:26:12 -0600 (Thu, 01 Dec 2005) | 2 lines
allow variables to exist on both 'halves' of the Local channel (issue #5810)
........
r7271 | kpfleming | 2005-12-01 18:28:48 -0600 (Thu, 01 Dec 2005) | 2 lines
protect agent_bridgedchannel() from segfaulting when there is no bridged channel (issue #5879)
........
r7272 | kpfleming | 2005-12-01 18:39:00 -0600 (Thu, 01 Dec 2005) | 3 lines
properly handle password changes when mailbox is last line of config file and not followed by a newline (issue #5870)
reformat password changing code to conform to coding guidelines (issue #5870)
........
r7273 | kpfleming | 2005-12-01 18:42:40 -0600 (Thu, 01 Dec 2005) | 2 lines
allow previous context-searching behavior to be used if desired (issue #5899)
........
r7274 | kpfleming | 2005-12-01 18:51:15 -0600 (Thu, 01 Dec 2005) | 2 lines
inherit channel variables into channels created by Page() application (issue #5888)
........
r7275 | oej | 2005-12-01 18:52:13 -0600 (Thu, 01 Dec 2005) | 2 lines
Bug #5907. Improve SIP INFO DTMF debugging output. (1.2 & Trunk)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@7276 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2005-12-02 01:01:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merge Call completion support into trunk.
From Reviewboard:
CCSS stands for Call Completion Supplementary Services. An admittedly out-of-date
overview of the architecture can be found in the file doc/CCSS_architecture.pdf
in the CCSS branch. Off the top of my head, the big differences between what is
implemented and what is in the document are as follows:
1. We did not end up modifying the Hangup application at all.
2. The document states that a single call completion monitor may be used across
   multiple calls to the same device. This proved to not be such a good idea
   when implementing protocol-specific monitors, and so we ended up using one
   monitor per-device per-call.
3. There are some configuration options which were conceived after the document
   was written. These are documented in the ccss.conf.sample that is on this
   review request.
		      
For some basic understanding of terminology used throughout this code, see the
ccss.tex document that is on this review.
This implements CCBS and CCNR in several flavors.
First up is a "generic" implementation, which can work over any channel technology
provided that the channel technology can accurately report device state. Call
completion is requested using the dialplan application CallCompletionRequest and can
be canceled using CallCompletionCancel. Device state subscriptions are used in order
to monitor the state of called parties.
Next, there is a SIP-specific implementation of call completion. This method uses the
methods outlined in draft-ietf-bliss-call-completion-06 to implement call completion
using SIP signaling. There are a few things to note here:
* The agent/monitor terminology used throughout Asterisk sometimes is the reverse of
  what is defined in the referenced draft.
* Implementation of the draft required support for SIP PUBLISH. I attempted to write
  this in a generic-enough fashion such that if someone were to want to write PUBLISH
  support for other event packages, such as dialog-state or presence, most of the effort
  would be in writing callbacks specific to the event package.
* A subportion of supporting PUBLISH reception was that we had to implement a PIDF
  parser. The PIDF support added is a bit minimal. I first wrote a validation
  routine to ensure that the PIDF document is formatted properly. The rest of the
  PIDF reading is done in-line in the call-completion-specific PUBLISH-handling
  code. In other words, while there is PIDF support here, it is not in any state
  where it could easily be applied to other event packages as is.
Finally, there are a variety of ISDN-related call completion protocols supported. These
were written by Richard Mudgett, and as such I can't really say much about their
implementation. There are notes in the CHANGES file that indicate the ISDN protocols
over which call completion is supported.
Review: https://reviewboard.asterisk.org/r/523
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@256528 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-04-09 15:31:32 +00:00
										 |  |  | 	ast_channel_cc_params_init(p->chan, ast_channel_get_cc_config_params(p->owner)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-14 15:48:36 +00:00
										 |  |  | 	if (!ast_exists_extension(NULL, p->chan->context, p->chan->exten, 1, | 
					
						
							|  |  |  | 		S_COR(p->owner->caller.id.number.valid, p->owner->caller.id.number.str, NULL))) { | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | 		ast_log(LOG_NOTICE, "No such extension/context %s@%s while calling Local channel\n", p->chan->exten, p->chan->context); | 
					
						
							| 
									
										
										
										
											2008-10-30 19:45:52 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2009-06-12 15:37:30 +00:00
										 |  |  | 		ast_channel_unlock(p->chan); | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-29 17:08:22 +00:00
										 |  |  | 	/* Make sure we inherit the ANSWERED_ELSEWHERE flag if it's set on the queue/dial call request in the dialplan */ | 
					
						
							|  |  |  | 	if (ast_test_flag(ast, AST_FLAG_ANSWERED_ELSEWHERE)) { | 
					
						
							|  |  |  | 		ast_set_flag(p->chan, AST_FLAG_ANSWERED_ELSEWHERE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merged revisions 7265-7266,7268-7275 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.2
........
r7265 | oej | 2005-12-01 17:18:14 -0600 (Thu, 01 Dec 2005) | 2 lines
Changing bug report address to the Asterisk issue tracker
........
r7266 | kpfleming | 2005-12-01 17:18:29 -0600 (Thu, 01 Dec 2005) | 3 lines
Makefile 'update' target now supports updating from Subversion repositories (issue #5875)
remove support for 'patches' subdirectory, it's no longer useful
........
r7268 | kpfleming | 2005-12-01 17:34:58 -0600 (Thu, 01 Dec 2005) | 2 lines
ensure channel's scheduling context is freed (issue #5788)
........
r7269 | kpfleming | 2005-12-01 17:49:44 -0600 (Thu, 01 Dec 2005) | 2 lines
don't block waiting for the Festival server forever when it goes away (issue #5882)
........
r7270 | kpfleming | 2005-12-01 18:26:12 -0600 (Thu, 01 Dec 2005) | 2 lines
allow variables to exist on both 'halves' of the Local channel (issue #5810)
........
r7271 | kpfleming | 2005-12-01 18:28:48 -0600 (Thu, 01 Dec 2005) | 2 lines
protect agent_bridgedchannel() from segfaulting when there is no bridged channel (issue #5879)
........
r7272 | kpfleming | 2005-12-01 18:39:00 -0600 (Thu, 01 Dec 2005) | 3 lines
properly handle password changes when mailbox is last line of config file and not followed by a newline (issue #5870)
reformat password changing code to conform to coding guidelines (issue #5870)
........
r7273 | kpfleming | 2005-12-01 18:42:40 -0600 (Thu, 01 Dec 2005) | 2 lines
allow previous context-searching behavior to be used if desired (issue #5899)
........
r7274 | kpfleming | 2005-12-01 18:51:15 -0600 (Thu, 01 Dec 2005) | 2 lines
inherit channel variables into channels created by Page() application (issue #5888)
........
r7275 | oej | 2005-12-01 18:52:13 -0600 (Thu, 01 Dec 2005) | 2 lines
Bug #5907. Improve SIP INFO DTMF debugging output. (1.2 & Trunk)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@7276 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2005-12-02 01:01:11 +00:00
										 |  |  | 	/* copy the channel variables from the incoming channel to the outgoing channel */ | 
					
						
							|  |  |  | 	/* Note that due to certain assumptions, they MUST be in the same order */ | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE(&p->owner->varshead, varptr, entries) { | 
					
						
							|  |  |  | 		namelen = strlen(varptr->name); | 
					
						
							|  |  |  | 		len = sizeof(struct ast_var_t) + namelen + strlen(varptr->value) + 2; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 		if ((new = ast_calloc(1, len))) { | 
					
						
							| 
									
										
											  
											
												Merged revisions 7265-7266,7268-7275 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.2
........
r7265 | oej | 2005-12-01 17:18:14 -0600 (Thu, 01 Dec 2005) | 2 lines
Changing bug report address to the Asterisk issue tracker
........
r7266 | kpfleming | 2005-12-01 17:18:29 -0600 (Thu, 01 Dec 2005) | 3 lines
Makefile 'update' target now supports updating from Subversion repositories (issue #5875)
remove support for 'patches' subdirectory, it's no longer useful
........
r7268 | kpfleming | 2005-12-01 17:34:58 -0600 (Thu, 01 Dec 2005) | 2 lines
ensure channel's scheduling context is freed (issue #5788)
........
r7269 | kpfleming | 2005-12-01 17:49:44 -0600 (Thu, 01 Dec 2005) | 2 lines
don't block waiting for the Festival server forever when it goes away (issue #5882)
........
r7270 | kpfleming | 2005-12-01 18:26:12 -0600 (Thu, 01 Dec 2005) | 2 lines
allow variables to exist on both 'halves' of the Local channel (issue #5810)
........
r7271 | kpfleming | 2005-12-01 18:28:48 -0600 (Thu, 01 Dec 2005) | 2 lines
protect agent_bridgedchannel() from segfaulting when there is no bridged channel (issue #5879)
........
r7272 | kpfleming | 2005-12-01 18:39:00 -0600 (Thu, 01 Dec 2005) | 3 lines
properly handle password changes when mailbox is last line of config file and not followed by a newline (issue #5870)
reformat password changing code to conform to coding guidelines (issue #5870)
........
r7273 | kpfleming | 2005-12-01 18:42:40 -0600 (Thu, 01 Dec 2005) | 2 lines
allow previous context-searching behavior to be used if desired (issue #5899)
........
r7274 | kpfleming | 2005-12-01 18:51:15 -0600 (Thu, 01 Dec 2005) | 2 lines
inherit channel variables into channels created by Page() application (issue #5888)
........
r7275 | oej | 2005-12-01 18:52:13 -0600 (Thu, 01 Dec 2005) | 2 lines
Bug #5907. Improve SIP INFO DTMF debugging output. (1.2 & Trunk)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@7276 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2005-12-02 01:01:11 +00:00
										 |  |  | 			memcpy(new, varptr, len); | 
					
						
							|  |  |  | 			new->value = &(new->name[0]) + namelen + 1; | 
					
						
							|  |  |  | 			AST_LIST_INSERT_TAIL(&p->chan->varshead, new, entries); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
											  
											
												Merged revisions 90735 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.4
........
r90735 | mmichelson | 2007-12-03 17:12:17 -0600 (Mon, 03 Dec 2007) | 22 lines
A big one...
This is the merge of the forward-loop branch. The main change here is that call-forwards can no longer loop.
This is accomplished by creating a datastore on the calling channel which has a linked list of all devices
dialed. If a forward happens, then the local channel which is created inherits the datastore. If, through this
progression of forwards and datastore inheritance, a device is attempted to be dialed a second time, it will simply
be skipped and a warning message will be printed to the CLI. After the dialing has been completed, the datastore
is detached from the channel and destroyed.
This change also introduces some side effects to the code which I shall enumerate here:
1. Datastore inheritance has been backported from trunk into 1.4
2. A large chunk of code has been removed from app_dial. This chunk is the section of code
   which handles the call forward case after the channel has been requested but before it has
   been called. This was removed because call-forwarding still works fine without it, it makes the
   code less error-prone should it need changing, and it made this set of changes much less painful
   to just have the forwarding handled in one place in each module.
3. Two new files, global_datastores.h and .c have been added. These are necessary since the datastore
   which is attached to the channel may be created and attached in either app_dial or app_queue, so they
   need a common place to find the datastore info. This approach was taken in case similar datastores are
   needed in the future, there will be a common place to add them.
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@90873 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2007-12-04 17:08:36 +00:00
										 |  |  | 	ast_channel_datastore_inherit(p->owner, p->chan); | 
					
						
							| 
									
										
											  
											
												Merge Call completion support into trunk.
From Reviewboard:
CCSS stands for Call Completion Supplementary Services. An admittedly out-of-date
overview of the architecture can be found in the file doc/CCSS_architecture.pdf
in the CCSS branch. Off the top of my head, the big differences between what is
implemented and what is in the document are as follows:
1. We did not end up modifying the Hangup application at all.
2. The document states that a single call completion monitor may be used across
   multiple calls to the same device. This proved to not be such a good idea
   when implementing protocol-specific monitors, and so we ended up using one
   monitor per-device per-call.
3. There are some configuration options which were conceived after the document
   was written. These are documented in the ccss.conf.sample that is on this
   review request.
		      
For some basic understanding of terminology used throughout this code, see the
ccss.tex document that is on this review.
This implements CCBS and CCNR in several flavors.
First up is a "generic" implementation, which can work over any channel technology
provided that the channel technology can accurately report device state. Call
completion is requested using the dialplan application CallCompletionRequest and can
be canceled using CallCompletionCancel. Device state subscriptions are used in order
to monitor the state of called parties.
Next, there is a SIP-specific implementation of call completion. This method uses the
methods outlined in draft-ietf-bliss-call-completion-06 to implement call completion
using SIP signaling. There are a few things to note here:
* The agent/monitor terminology used throughout Asterisk sometimes is the reverse of
  what is defined in the referenced draft.
* Implementation of the draft required support for SIP PUBLISH. I attempted to write
  this in a generic-enough fashion such that if someone were to want to write PUBLISH
  support for other event packages, such as dialog-state or presence, most of the effort
  would be in writing callbacks specific to the event package.
* A subportion of supporting PUBLISH reception was that we had to implement a PIDF
  parser. The PIDF support added is a bit minimal. I first wrote a validation
  routine to ensure that the PIDF document is formatted properly. The rest of the
  PIDF reading is done in-line in the call-completion-specific PUBLISH-handling
  code. In other words, while there is PIDF support here, it is not in any state
  where it could easily be applied to other event packages as is.
Finally, there are a variety of ISDN-related call completion protocols supported. These
were written by Richard Mudgett, and as such I can't really say much about their
implementation. There are notes in the CHANGES file that indicate the ISDN protocols
over which call completion is supported.
Review: https://reviewboard.asterisk.org/r/523
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@256528 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-04-09 15:31:32 +00:00
										 |  |  | 	/* 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 ((slash = strrchr(reduced_dest, '/'))) { | 
					
						
							|  |  |  | 		*slash = '\0'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_set_cc_interfaces_chanvar(p->chan, reduced_dest); | 
					
						
							| 
									
										
											  
											
												Merged revisions 7265-7266,7268-7275 via svnmerge from 
https://origsvn.digium.com/svn/asterisk/branches/1.2
........
r7265 | oej | 2005-12-01 17:18:14 -0600 (Thu, 01 Dec 2005) | 2 lines
Changing bug report address to the Asterisk issue tracker
........
r7266 | kpfleming | 2005-12-01 17:18:29 -0600 (Thu, 01 Dec 2005) | 3 lines
Makefile 'update' target now supports updating from Subversion repositories (issue #5875)
remove support for 'patches' subdirectory, it's no longer useful
........
r7268 | kpfleming | 2005-12-01 17:34:58 -0600 (Thu, 01 Dec 2005) | 2 lines
ensure channel's scheduling context is freed (issue #5788)
........
r7269 | kpfleming | 2005-12-01 17:49:44 -0600 (Thu, 01 Dec 2005) | 2 lines
don't block waiting for the Festival server forever when it goes away (issue #5882)
........
r7270 | kpfleming | 2005-12-01 18:26:12 -0600 (Thu, 01 Dec 2005) | 2 lines
allow variables to exist on both 'halves' of the Local channel (issue #5810)
........
r7271 | kpfleming | 2005-12-01 18:28:48 -0600 (Thu, 01 Dec 2005) | 2 lines
protect agent_bridgedchannel() from segfaulting when there is no bridged channel (issue #5879)
........
r7272 | kpfleming | 2005-12-01 18:39:00 -0600 (Thu, 01 Dec 2005) | 3 lines
properly handle password changes when mailbox is last line of config file and not followed by a newline (issue #5870)
reformat password changing code to conform to coding guidelines (issue #5870)
........
r7273 | kpfleming | 2005-12-01 18:42:40 -0600 (Thu, 01 Dec 2005) | 2 lines
allow previous context-searching behavior to be used if desired (issue #5899)
........
r7274 | kpfleming | 2005-12-01 18:51:15 -0600 (Thu, 01 Dec 2005) | 2 lines
inherit channel variables into channels created by Page() application (issue #5888)
........
r7275 | oej | 2005-12-01 18:52:13 -0600 (Thu, 01 Dec 2005) | 2 lines
Bug #5907. Improve SIP INFO DTMF debugging output. (1.2 & Trunk)
........
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@7276 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2005-12-02 01:01:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	/* Start switch on sub channel */ | 
					
						
							| 
									
										
										
										
											2007-07-05 13:28:41 +00:00
										 |  |  | 	if (!(res = ast_pbx_start(p->chan))) | 
					
						
							|  |  |  | 		ast_set_flag(p, LOCAL_LAUNCHED_PBX); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-08-19 19:36:52 +00:00
										 |  |  | 	ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2009-04-03 22:41:46 +00:00
										 |  |  | 	ast_channel_unlock(p->chan); | 
					
						
							| 
									
										
										
										
											2003-08-19 19:36:52 +00:00
										 |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Hangup a call through the local proxy channel */ | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | static int local_hangup(struct ast_channel *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	struct local_pvt *p = ast->tech_pvt; | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	int isoutbound; | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | 	struct ast_frame f = { AST_FRAME_CONTROL, { AST_CONTROL_HANGUP }, .data.uint32 = ast->hangupcause }; | 
					
						
							| 
									
										
										
										
											2003-05-19 23:37:38 +00:00
										 |  |  | 	struct ast_channel *ochan = NULL; | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 	int glaredetect = 0, res = 0; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-16 18:32:27 +00:00
										 |  |  | 	if (!p) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-27 16:00:05 +00:00
										 |  |  | 	ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2008-06-27 12:28:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-06-22 04:54:52 +00:00
										 |  |  | 	isoutbound = IS_OUTBOUND(ast, p); | 
					
						
							| 
									
										
										
										
											2009-01-29 17:08:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (p->chan && ast_test_flag(ast, AST_FLAG_ANSWERED_ELSEWHERE)) { | 
					
						
							|  |  |  | 		ast_set_flag(p->chan, AST_FLAG_ANSWERED_ELSEWHERE); | 
					
						
							|  |  |  | 		ast_debug(2, "This local call has the ANSWERED_ELSEWHERE flag set.\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-08-19 19:36:52 +00:00
										 |  |  | 	if (isoutbound) { | 
					
						
							| 
									
										
										
										
											2006-03-31 10:13:22 +00:00
										 |  |  | 		const char *status = pbx_builtin_getvar_helper(p->chan, "DIALSTATUS"); | 
					
						
							| 
									
										
										
										
											2008-01-16 17:21:49 +00:00
										 |  |  | 		if ((status) && (p->owner)) { | 
					
						
							|  |  |  | 			/* Deadlock avoidance */ | 
					
						
							| 
									
										
										
										
											2008-02-20 21:41:31 +00:00
										 |  |  | 			while (p->owner && ast_channel_trylock(p->owner)) { | 
					
						
							| 
									
										
										
										
											2008-01-16 17:21:49 +00:00
										 |  |  | 				ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2010-04-28 21:20:03 +00:00
										 |  |  | 				if (p->chan) { | 
					
						
							|  |  |  | 					ast_channel_unlock(p->chan); | 
					
						
							| 
									
										
										
										
											2008-04-24 20:06:06 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-01-16 17:21:49 +00:00
										 |  |  | 				usleep(1); | 
					
						
							| 
									
										
										
										
											2010-04-28 21:20:03 +00:00
										 |  |  | 				if (p->chan) { | 
					
						
							|  |  |  | 					ast_channel_lock(p->chan); | 
					
						
							| 
									
										
										
										
											2008-04-24 20:06:06 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2008-01-16 17:21:49 +00:00
										 |  |  | 				ast_mutex_lock(&p->lock); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-02-20 21:41:31 +00:00
										 |  |  | 			if (p->owner) { | 
					
						
							|  |  |  | 				pbx_builtin_setvar_helper(p->owner, "CHANLOCALSTATUS", status); | 
					
						
							|  |  |  | 				ast_channel_unlock(p->owner); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2008-01-16 17:21:49 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 		p->chan = NULL; | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 		ast_clear_flag(p, LOCAL_LAUNCHED_PBX); | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 		ast_module_user_remove(p->u_chan); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_module_user_remove(p->u_owner); | 
					
						
							| 
									
										
										
										
											2008-10-14 17:38:06 +00:00
										 |  |  | 		while (p->chan && ast_channel_trylock(p->chan)) { | 
					
						
							| 
									
										
										
										
											2010-04-28 21:20:03 +00:00
										 |  |  | 				ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 				if (p->owner) { | 
					
						
							|  |  |  | 					ast_channel_unlock(p->owner); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				usleep(1); | 
					
						
							|  |  |  | 				if (p->owner) { | 
					
						
							|  |  |  | 					ast_channel_lock(p->owner); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2008-10-14 17:38:06 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-04-28 21:20:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-13 19:44:53 +00:00
										 |  |  | 		p->owner = NULL; | 
					
						
							| 
									
										
										
										
											2008-06-20 20:17:04 +00:00
										 |  |  | 		if (p->chan) { | 
					
						
							|  |  |  | 			ast_queue_hangup(p->chan); | 
					
						
							| 
									
										
										
										
											2008-10-14 17:38:06 +00:00
										 |  |  | 			ast_channel_unlock(p->chan); | 
					
						
							| 
									
										
										
										
											2008-06-20 20:17:04 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2003-05-19 23:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	ast->tech_pvt = NULL; | 
					
						
							| 
									
										
										
										
											2004-12-11 04:46:49 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	if (!p->owner && !p->chan) { | 
					
						
							|  |  |  | 		/* Okay, done with the private part now, too. */ | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 		glaredetect = ast_test_flag(p, LOCAL_GLARE_DETECT); | 
					
						
							| 
									
										
										
										
											2003-07-01 03:32:14 +00:00
										 |  |  | 		/* If we have a queue holding, don't actually destroy p yet, but
 | 
					
						
							|  |  |  | 		   let local_queue do it. */ | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 		if (glaredetect) | 
					
						
							|  |  |  | 			ast_set_flag(p, LOCAL_CANCEL_QUEUE); | 
					
						
							| 
									
										
										
										
											2003-04-08 13:50:04 +00:00
										 |  |  | 		/* Remove from list */ | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 		AST_LIST_LOCK(&locals); | 
					
						
							|  |  |  | 		AST_LIST_REMOVE(&locals, p, list); | 
					
						
							|  |  |  | 		AST_LIST_UNLOCK(&locals); | 
					
						
							| 
									
										
										
										
											2004-11-06 21:33:01 +00:00
										 |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-04-08 13:50:04 +00:00
										 |  |  | 		/* And destroy */ | 
					
						
							| 
									
										
										
										
											2004-06-22 17:42:14 +00:00
										 |  |  | 		if (!glaredetect) { | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | 			p = local_pvt_destroy(p); | 
					
						
							| 
									
										
										
										
											2004-06-22 17:42:14 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 	if (p->chan && !ast_test_flag(p, LOCAL_LAUNCHED_PBX)) | 
					
						
							| 
									
										
										
										
											2003-05-19 23:37:38 +00:00
										 |  |  | 		/* Need to actually hangup since there is no PBX */ | 
					
						
							|  |  |  | 		ochan = p->chan; | 
					
						
							|  |  |  | 	else | 
					
						
							| 
									
										
										
										
											2008-05-13 21:18:55 +00:00
										 |  |  | 		res = local_queue_frame(p, isoutbound, &f, NULL, 1); | 
					
						
							| 
									
										
										
										
											2007-04-09 02:44:37 +00:00
										 |  |  | 	if (!res) | 
					
						
							|  |  |  | 		ast_mutex_unlock(&p->lock); | 
					
						
							| 
									
										
										
										
											2003-05-19 23:37:38 +00:00
										 |  |  | 	if (ochan) | 
					
						
							|  |  |  | 		ast_hangup(ochan); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Create a call structure */ | 
					
						
							| 
									
										
										
										
											2006-04-21 17:53:39 +00:00
										 |  |  | static struct local_pvt *local_alloc(const char *data, int format) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct local_pvt *tmp = NULL; | 
					
						
							|  |  |  | 	char *c = NULL, *opts = NULL; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	if (!(tmp = ast_calloc(1, sizeof(*tmp)))) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Initialize private structure information */ | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	ast_mutex_init(&tmp->lock); | 
					
						
							| 
									
										
										
										
											2006-04-21 17:53:39 +00:00
										 |  |  | 	ast_copy_string(tmp->exten, data, sizeof(tmp->exten)); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | 	memcpy(&tmp->jb_conf, &g_jb_conf, sizeof(tmp->jb_conf)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	/* Look for options */ | 
					
						
							|  |  |  | 	if ((opts = strchr(tmp->exten, '/'))) { | 
					
						
							| 
									
										
										
										
											2006-04-21 17:53:39 +00:00
										 |  |  | 		*opts++ = '\0'; | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 		if (strchr(opts, 'n')) | 
					
						
							| 
									
										
										
										
											2006-09-07 21:11:07 +00:00
										 |  |  | 			ast_set_flag(tmp, LOCAL_NO_OPTIMIZATION); | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | 		if (strchr(opts, 'j')) { | 
					
						
							|  |  |  | 			if (ast_test_flag(tmp, LOCAL_NO_OPTIMIZATION)) | 
					
						
							|  |  |  | 				ast_set_flag(&tmp->jb_conf, AST_JB_ENABLED); | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				ast_log(LOG_ERROR, "You must use the 'n' option for chan_local " | 
					
						
							|  |  |  | 					"to use the 'j' option to enable the jitterbuffer\n"); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-04-10 20:28:40 +00:00
										 |  |  | 		if (strchr(opts, 'b')) { | 
					
						
							|  |  |  | 			ast_set_flag(tmp, LOCAL_BRIDGE); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-10-29 18:13:42 +00:00
										 |  |  | 		if (strchr(opts, 'm')) { | 
					
						
							|  |  |  | 			ast_set_flag(tmp, LOCAL_MOH_PASSTHRU); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Look for a context */ | 
					
						
							|  |  |  | 	if ((c = strchr(tmp->exten, '@'))) | 
					
						
							| 
									
										
										
										
											2006-04-21 17:53:39 +00:00
										 |  |  | 		*c++ = '\0'; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-04-21 17:53:39 +00:00
										 |  |  | 	ast_copy_string(tmp->context, c ? c : "default", sizeof(tmp->context)); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	tmp->reqformat = format; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 	/* We can't do this check here, because we don't know the CallerID yet, and
 | 
					
						
							|  |  |  | 	 * the CallerID could potentially affect what step is actually taken (or | 
					
						
							|  |  |  | 	 * even if that step exists). */ | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	if (!ast_exists_extension(NULL, tmp->context, tmp->exten, 1, NULL)) { | 
					
						
							|  |  |  | 		ast_log(LOG_NOTICE, "No such extension/context %s@%s creating local channel\n", tmp->exten, tmp->context); | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | 		tmp = local_pvt_destroy(tmp); | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 		/* Add to list */ | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 		AST_LIST_LOCK(&locals); | 
					
						
							|  |  |  | 		AST_LIST_INSERT_HEAD(&locals, tmp, list); | 
					
						
							|  |  |  | 		AST_LIST_UNLOCK(&locals); | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-10-06 16:52:14 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-01-11 01:20:29 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return tmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Start new local channel */ | 
					
						
							| 
									
										
										
										
											2009-06-26 15:28:53 +00:00
										 |  |  | static struct ast_channel *local_new(struct local_pvt *p, int state, const char *linkedid) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct ast_channel *tmp = NULL, *tmp2 = NULL; | 
					
						
							|  |  |  | 	int randnum = ast_random() & 0xffff, fmt = 0; | 
					
						
							| 
									
										
										
										
											2007-05-14 14:13:45 +00:00
										 |  |  | 	const char *t; | 
					
						
							|  |  |  | 	int ama; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	/* Allocate two new Asterisk channels */ | 
					
						
							| 
									
										
										
										
											2007-05-14 14:13:45 +00:00
										 |  |  | 	/* safe accountcode */ | 
					
						
							|  |  |  | 	if (p->owner && p->owner->accountcode) | 
					
						
							|  |  |  | 		t = p->owner->accountcode; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		t = ""; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p->owner) | 
					
						
							|  |  |  | 		ama = p->owner->amaflags; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		ama = 0; | 
					
						
							| 
									
										
										
										
											2009-06-26 15:28:53 +00:00
										 |  |  | 	if (!(tmp = ast_channel_alloc(1, state, 0, 0, t, p->exten, p->context, linkedid, ama, "Local/%s@%s-%04x;1", p->exten, p->context, randnum))  | 
					
						
							|  |  |  | 		|| !(tmp2 = ast_channel_alloc(1, AST_STATE_RING, 0, 0, t, p->exten, p->context, linkedid, ama, "Local/%s@%s-%04x;2", p->exten, p->context, randnum))) { | 
					
						
							| 
									
										
											  
											
												Convert the ast_channel data structure over to the astobj2 framework.
There is a lot that could be said about this, but the patch is a big 
improvement for performance, stability, code maintainability, 
and ease of future code development.
The channel list is no longer an unsorted linked list.  The main container 
for channels is an astobj2 hash table.  All of the code related to searching 
for channels or iterating active channels has been rewritten.  Let n be 
the number of active channels.  Iterating the channel list has gone from 
O(n^2) to O(n).  Searching for a channel by name went from O(n) to O(1).  
Searching for a channel by extension is still O(n), but uses a new method 
for doing so, which is more efficient.
The ast_channel object is now a reference counted object.  The benefits 
here are plentiful.  Some benefits directly related to issues in the 
previous code include:
1) When threads other than the channel thread owning a channel wanted 
   access to a channel, it had to hold the lock on it to ensure that it didn't 
   go away.  This is no longer a requirement.  Holding a reference is 
   sufficient.
2) There are places that now require less dealing with channel locks.
3) There are places where channel locks are held for much shorter periods 
   of time.
4) There are places where dealing with more than one channel at a time becomes 
   _MUCH_ easier.  ChanSpy is a great example of this.  Writing code in the 
   future that deals with multiple channels will be much easier.
Some additional information regarding channel locking and reference count 
handling can be found in channel.h, where a new section has been added that 
discusses some of the rules associated with it.
Mark Michelson also assisted with the development of this patch.  He did the 
conversion of ChanSpy and introduced a new API, ast_autochan, which makes it 
much easier to deal with holding on to a channel pointer for an extended period 
of time and having it get automatically updated if the channel gets masqueraded.
Mark was also a huge help in the code review process.
Thanks to David Vossel for his assistance with this branch, as well.  David 
did the conversion of the DAHDIScan application by making it become a wrapper 
for ChanSpy internally.
The changes come from the svn/asterisk/team/russell/ast_channel_ao2 branch.
Review: http://reviewboard.digium.com/r/203/
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@190423 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2009-04-24 14:04:26 +00:00
										 |  |  | 		if (tmp) { | 
					
						
							|  |  |  | 			tmp = ast_channel_release(tmp); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Unable to allocate channel structure(s)\n"); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
											  
											
												Convert the ast_channel data structure over to the astobj2 framework.
There is a lot that could be said about this, but the patch is a big 
improvement for performance, stability, code maintainability, 
and ease of future code development.
The channel list is no longer an unsorted linked list.  The main container 
for channels is an astobj2 hash table.  All of the code related to searching 
for channels or iterating active channels has been rewritten.  Let n be 
the number of active channels.  Iterating the channel list has gone from 
O(n^2) to O(n).  Searching for a channel by name went from O(n) to O(1).  
Searching for a channel by extension is still O(n), but uses a new method 
for doing so, which is more efficient.
The ast_channel object is now a reference counted object.  The benefits 
here are plentiful.  Some benefits directly related to issues in the 
previous code include:
1) When threads other than the channel thread owning a channel wanted 
   access to a channel, it had to hold the lock on it to ensure that it didn't 
   go away.  This is no longer a requirement.  Holding a reference is 
   sufficient.
2) There are places that now require less dealing with channel locks.
3) There are places where channel locks are held for much shorter periods 
   of time.
4) There are places where dealing with more than one channel at a time becomes 
   _MUCH_ easier.  ChanSpy is a great example of this.  Writing code in the 
   future that deals with multiple channels will be much easier.
Some additional information regarding channel locking and reference count 
handling can be found in channel.h, where a new section has been added that 
discusses some of the rules associated with it.
Mark Michelson also assisted with the development of this patch.  He did the 
conversion of ChanSpy and introduced a new API, ast_autochan, which makes it 
much easier to deal with holding on to a channel pointer for an extended period 
of time and having it get automatically updated if the channel gets masqueraded.
Mark was also a huge help in the code review process.
Thanks to David Vossel for his assistance with this branch, as well.  David 
did the conversion of the DAHDIScan application by making it become a wrapper 
for ChanSpy internally.
The changes come from the svn/asterisk/team/russell/ast_channel_ao2 branch.
Review: http://reviewboard.digium.com/r/203/
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@190423 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2009-04-24 14:04:26 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	tmp2->tech = tmp->tech = &local_tech; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 	tmp->nativeformats = p->reqformat; | 
					
						
							|  |  |  | 	tmp2->nativeformats = p->reqformat; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Determine our read/write format and set it on each channel */ | 
					
						
							|  |  |  | 	fmt = ast_best_codec(p->reqformat); | 
					
						
							|  |  |  | 	tmp->writeformat = fmt; | 
					
						
							|  |  |  | 	tmp2->writeformat = fmt; | 
					
						
							|  |  |  | 	tmp->rawwriteformat = fmt; | 
					
						
							|  |  |  | 	tmp2->rawwriteformat = fmt; | 
					
						
							|  |  |  | 	tmp->readformat = fmt; | 
					
						
							|  |  |  | 	tmp2->readformat = fmt; | 
					
						
							|  |  |  | 	tmp->rawreadformat = fmt; | 
					
						
							|  |  |  | 	tmp2->rawreadformat = fmt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 	tmp->tech_pvt = p; | 
					
						
							|  |  |  | 	tmp2->tech_pvt = p; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 	p->owner = tmp; | 
					
						
							|  |  |  | 	p->chan = tmp2; | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | 	p->u_owner = ast_module_user_add(p->owner); | 
					
						
							|  |  |  | 	p->u_chan = ast_module_user_add(p->chan); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 	ast_copy_string(tmp->context, p->context, sizeof(tmp->context)); | 
					
						
							|  |  |  | 	ast_copy_string(tmp2->context, p->context, sizeof(tmp2->context)); | 
					
						
							|  |  |  | 	ast_copy_string(tmp2->exten, p->exten, sizeof(tmp->exten)); | 
					
						
							|  |  |  | 	tmp->priority = 1; | 
					
						
							|  |  |  | 	tmp2->priority = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-09 15:10:14 +00:00
										 |  |  | 	ast_jb_configure(tmp, &p->jb_conf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return tmp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Part of PBX interface */ | 
					
						
							| 
									
										
										
										
											2009-11-04 14:05:12 +00:00
										 |  |  | static struct ast_channel *local_request(const char *type, format_t format, const struct ast_channel *requestor, void *data, int *cause) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct local_pvt *p = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	struct ast_channel *chan = NULL; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	/* Allocate a new private structure and then Asterisk channel */ | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | 	if ((p = local_alloc(data, format))) { | 
					
						
							| 
									
										
										
										
											2009-06-26 15:28:53 +00:00
										 |  |  | 		if (!(chan = local_new(p, AST_STATE_DOWN, requestor ? requestor->linkedid : NULL))) { | 
					
						
							| 
									
										
										
										
											2008-03-03 17:17:27 +00:00
										 |  |  | 			AST_LIST_LOCK(&locals); | 
					
						
							|  |  |  | 			AST_LIST_REMOVE(&locals, p, list); | 
					
						
							|  |  |  | 			AST_LIST_UNLOCK(&locals); | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | 			p = local_pvt_destroy(p); | 
					
						
							| 
									
										
										
										
											2008-03-03 17:17:27 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-07-20 17:22:36 +00:00
										 |  |  | 		if (chan && ast_channel_cc_params_init(chan, requestor ? ast_channel_get_cc_config_params((struct ast_channel *)requestor) : NULL)) { | 
					
						
							| 
									
										
											  
											
												Merge Call completion support into trunk.
From Reviewboard:
CCSS stands for Call Completion Supplementary Services. An admittedly out-of-date
overview of the architecture can be found in the file doc/CCSS_architecture.pdf
in the CCSS branch. Off the top of my head, the big differences between what is
implemented and what is in the document are as follows:
1. We did not end up modifying the Hangup application at all.
2. The document states that a single call completion monitor may be used across
   multiple calls to the same device. This proved to not be such a good idea
   when implementing protocol-specific monitors, and so we ended up using one
   monitor per-device per-call.
3. There are some configuration options which were conceived after the document
   was written. These are documented in the ccss.conf.sample that is on this
   review request.
		      
For some basic understanding of terminology used throughout this code, see the
ccss.tex document that is on this review.
This implements CCBS and CCNR in several flavors.
First up is a "generic" implementation, which can work over any channel technology
provided that the channel technology can accurately report device state. Call
completion is requested using the dialplan application CallCompletionRequest and can
be canceled using CallCompletionCancel. Device state subscriptions are used in order
to monitor the state of called parties.
Next, there is a SIP-specific implementation of call completion. This method uses the
methods outlined in draft-ietf-bliss-call-completion-06 to implement call completion
using SIP signaling. There are a few things to note here:
* The agent/monitor terminology used throughout Asterisk sometimes is the reverse of
  what is defined in the referenced draft.
* Implementation of the draft required support for SIP PUBLISH. I attempted to write
  this in a generic-enough fashion such that if someone were to want to write PUBLISH
  support for other event packages, such as dialog-state or presence, most of the effort
  would be in writing callbacks specific to the event package.
* A subportion of supporting PUBLISH reception was that we had to implement a PIDF
  parser. The PIDF support added is a bit minimal. I first wrote a validation
  routine to ensure that the PIDF document is formatted properly. The rest of the
  PIDF reading is done in-line in the call-completion-specific PUBLISH-handling
  code. In other words, while there is PIDF support here, it is not in any state
  where it could easily be applied to other event packages as is.
Finally, there are a variety of ISDN-related call completion protocols supported. These
were written by Richard Mudgett, and as such I can't really say much about their
implementation. There are notes in the CHANGES file that indicate the ISDN protocols
over which call completion is supported.
Review: https://reviewboard.asterisk.org/r/523
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@256528 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2010-04-09 15:31:32 +00:00
										 |  |  | 			chan = ast_channel_release(chan); | 
					
						
							|  |  |  | 			p = local_pvt_destroy(p); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2008-03-03 17:06:35 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	return chan; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief CLI command "local show channels" */ | 
					
						
							| 
									
										
										
										
											2007-09-18 22:43:45 +00:00
										 |  |  | static char *locals_show(struct ast_cli_entry *e, int cmd, struct ast_cli_args *a) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct local_pvt *p = NULL; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-18 22:43:45 +00:00
										 |  |  | 	switch (cmd) { | 
					
						
							|  |  |  | 	case CLI_INIT: | 
					
						
							|  |  |  | 		e->command = "local show channels"; | 
					
						
							|  |  |  | 		e->usage = | 
					
						
							|  |  |  | 			"Usage: local show channels\n" | 
					
						
							|  |  |  | 			"       Provides summary information on active local proxy channels.\n"; | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	case CLI_GENERATE: | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (a->argc != 3) | 
					
						
							|  |  |  | 		return CLI_SHOWUSAGE; | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 	AST_LIST_LOCK(&locals); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	if (!AST_LIST_EMPTY(&locals)) { | 
					
						
							|  |  |  | 		AST_LIST_TRAVERSE(&locals, p, list) { | 
					
						
							|  |  |  | 			ast_mutex_lock(&p->lock); | 
					
						
							| 
									
										
										
										
											2007-09-18 22:43:45 +00:00
										 |  |  | 			ast_cli(a->fd, "%s -- %s@%s\n", p->owner ? p->owner->name : "<unowned>", p->exten, p->context); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 			ast_mutex_unlock(&p->lock); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else | 
					
						
							| 
									
										
										
										
											2007-09-18 22:43:45 +00:00
										 |  |  | 		ast_cli(a->fd, "No local channels in use\n"); | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 	AST_LIST_UNLOCK(&locals); | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-18 22:43:45 +00:00
										 |  |  | 	return CLI_SUCCESS; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | static struct ast_cli_entry cli_local[] = { | 
					
						
							| 
									
										
										
										
											2007-10-22 20:05:18 +00:00
										 |  |  | 	AST_CLI_DEFINE(locals_show, "List status of local channels"), | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-23 19:59:43 +00:00
										 |  |  | static int manager_optimize_away(struct mansession *s, const struct message *m) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const char *channel; | 
					
						
							|  |  |  | 	struct local_pvt *p, *tmp = NULL; | 
					
						
							|  |  |  | 	struct ast_channel *c; | 
					
						
							|  |  |  | 	int found = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	channel = astman_get_header(m, "Channel"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (ast_strlen_zero(channel)) { | 
					
						
							|  |  |  | 		astman_send_error(s, m, "'Channel' not specified."); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c = ast_channel_get_by_name(channel); | 
					
						
							|  |  |  | 	if (!c) { | 
					
						
							|  |  |  | 		astman_send_error(s, m, "Channel does not exist."); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p = c->tech_pvt; | 
					
						
							|  |  |  | 	ast_channel_unref(c); | 
					
						
							|  |  |  | 	c = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (AST_LIST_LOCK(&locals)) { | 
					
						
							|  |  |  | 		astman_send_error(s, m, "Unable to lock the monitor"); | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_LIST_TRAVERSE(&locals, tmp, list) { | 
					
						
							|  |  |  | 		if (tmp == p) { | 
					
						
							|  |  |  | 			ast_mutex_lock(&tmp->lock); | 
					
						
							|  |  |  | 			found = 1; | 
					
						
							|  |  |  | 			ast_clear_flag(tmp, LOCAL_NO_OPTIMIZATION); | 
					
						
							|  |  |  | 			ast_mutex_unlock(&tmp->lock); | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	AST_LIST_UNLOCK(&locals); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (found) { | 
					
						
							|  |  |  | 		astman_send_ack(s, m, "Queued channel to be optimized away"); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		astman_send_error(s, m, "Unable to find channel"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Load module into PBX, register channel */ | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int load_module(void) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	/* Make sure we can register our channel type */ | 
					
						
							|  |  |  | 	if (ast_channel_register(&local_tech)) { | 
					
						
							| 
									
										
										
										
											2006-02-01 23:05:28 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "Unable to register channel class 'Local'\n"); | 
					
						
							| 
									
										
										
										
											2007-10-26 21:37:02 +00:00
										 |  |  | 		return AST_MODULE_LOAD_FAILURE; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_register_multiple(cli_local, sizeof(cli_local) / sizeof(struct ast_cli_entry)); | 
					
						
							| 
									
										
										
										
											2010-06-23 19:59:43 +00:00
										 |  |  | 	ast_manager_register_xml("LocalOptimizeAway", EVENT_FLAG_SYSTEM|EVENT_FLAG_CALL, manager_optimize_away); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-26 21:37:02 +00:00
										 |  |  | 	return AST_MODULE_LOAD_SUCCESS; | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-01-20 00:18:42 +00:00
										 |  |  | /*! \brief Unload the local proxy channel from Asterisk */ | 
					
						
							| 
									
										
										
										
											2006-08-21 02:11:39 +00:00
										 |  |  | static int unload_module(void) | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2006-09-07 19:50:59 +00:00
										 |  |  | 	struct local_pvt *p = NULL; | 
					
						
							| 
									
										
										
										
											2005-07-25 18:09:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	/* First, take us out of the channel loop */ | 
					
						
							| 
									
										
										
										
											2006-09-18 19:54:18 +00:00
										 |  |  | 	ast_cli_unregister_multiple(cli_local, sizeof(cli_local) / sizeof(struct ast_cli_entry)); | 
					
						
							| 
									
										
										
										
											2010-06-23 19:59:43 +00:00
										 |  |  | 	ast_manager_unregister("LocalOptimizeAway"); | 
					
						
							| 
									
										
										
										
											2005-03-04 06:47:24 +00:00
										 |  |  | 	ast_channel_unregister(&local_tech); | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 	if (!AST_LIST_LOCK(&locals)) { | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 		/* Hangup all interfaces if they have an owner */ | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 		AST_LIST_TRAVERSE(&locals, p, list) { | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 			if (p->owner) | 
					
						
							|  |  |  | 				ast_softhangup(p->owner, AST_SOFTHANGUP_APPUNLOAD); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2006-02-10 00:20:43 +00:00
										 |  |  | 		AST_LIST_UNLOCK(&locals); | 
					
						
							| 
									
										
										
										
											2003-04-06 02:41:42 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "Unable to lock the monitor\n"); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	}		 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-20 19:35:02 +00:00
										 |  |  | AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_LOAD_ORDER, "Local Proxy Channel (Note: used internally by other modules)", | 
					
						
							|  |  |  | 		.load = load_module, | 
					
						
							|  |  |  | 		.unload = unload_module, | 
					
						
							|  |  |  | 		.load_pri = AST_MODPRI_CHANNEL_DRIVER, | 
					
						
							|  |  |  | 	); |