| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Asterisk -- An open source telephony toolkit. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2009-2010, Digium, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Matthew Nicholson <mnicholson@digium.com> | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  |  * Initial T.38-gateway code | 
					
						
							|  |  |  |  * 2008, Daniel Ferenci <daniel.ferenci@nethemba.com> | 
					
						
							|  |  |  |  * Created by Nethemba s.r.o. http://www.nethemba.com
 | 
					
						
							|  |  |  |  * Sponsored by IPEX a.s. http://www.ipex.cz
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * T.38-gateway integration into asterisk app_fax and rework | 
					
						
							|  |  |  |  * 2008, Gregory Hinton Nietsky <gregory@dnstelecom.co.za> | 
					
						
							|  |  |  |  * dns Telecom http://www.dnstelecom.co.za
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Modified to make T.38-gateway compatible with Asterisk 1.6.2 | 
					
						
							|  |  |  |  * 2010, Anton Verevkin <mymail@verevkin.it> | 
					
						
							|  |  |  |  * ViaNetTV http://www.vianettv.com
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Modified to make T.38-gateway work | 
					
						
							|  |  |  |  * 2010, Klaus Darilion, IPCom GmbH, www.ipcom.at | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software, distributed under the terms of | 
					
						
							|  |  |  |  * the GNU General Public License Version 2. See the LICENSE file | 
					
						
							|  |  |  |  * at the top of the source tree. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \file
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \brief Spandsp T.38 and G.711 FAX Resource | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \author Matthew Nicholson <mnicholson@digium.com> | 
					
						
							| 
									
										
										
										
											2011-07-01 12:45:09 +00:00
										 |  |  |  * \author Gregory H. Nietsky <gregory@distrotech.co.za> | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This module registers the Spandsp FAX technology with the res_fax module. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*** MODULEINFO
 | 
					
						
							| 
									
										
										
										
											2010-03-03 15:39:45 +00:00
										 |  |  | 	<depend>spandsp</depend> | 
					
						
							|  |  |  | 	<depend>res_fax</depend> | 
					
						
							| 
									
										
										
										
											2011-07-14 20:28:54 +00:00
										 |  |  | 	<support_level>extended</support_level> | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | ***/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 09:58:52 -04:00
										 |  |  | /* Needed for spandsp headers */ | 
					
						
							|  |  |  | #define ASTMM_LIBC ASTMM_IGNORE
 | 
					
						
							| 
									
										
										
										
											2015-05-08 13:30:26 -04:00
										 |  |  | #include "asterisk.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #include "asterisk/logger.h"
 | 
					
						
							|  |  |  | #include "asterisk/module.h"
 | 
					
						
							|  |  |  | #include "asterisk/strings.h"
 | 
					
						
							|  |  |  | #include "asterisk/cli.h"
 | 
					
						
							|  |  |  | #include "asterisk/utils.h"
 | 
					
						
							|  |  |  | #include "asterisk/timing.h"
 | 
					
						
							|  |  |  | #include "asterisk/astobj2.h"
 | 
					
						
							|  |  |  | #include "asterisk/res_fax.h"
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | #include "asterisk/channel.h"
 | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | #include "asterisk/format_cache.h"
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 09:58:52 -04:00
										 |  |  | #define SPANDSP_EXPOSE_INTERNAL_STRUCTURES
 | 
					
						
							|  |  |  | #include <spandsp.h>
 | 
					
						
							|  |  |  | #include <spandsp/version.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #define SPANDSP_FAX_SAMPLES 160
 | 
					
						
							|  |  |  | #define SPANDSP_FAX_TIMER_RATE 8000 / SPANDSP_FAX_SAMPLES	/* 50 ticks per second, 20ms, 160 samples per second */
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | #define SPANDSP_ENGAGE_UDPTL_NAT_RETRY 3
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void *spandsp_fax_new(struct ast_fax_session *s, struct ast_fax_tech_token *token); | 
					
						
							|  |  |  | static void spandsp_fax_destroy(struct ast_fax_session *s); | 
					
						
							|  |  |  | static struct ast_frame *spandsp_fax_read(struct ast_fax_session *s); | 
					
						
							|  |  |  | static int spandsp_fax_write(struct ast_fax_session *s, const struct ast_frame *f); | 
					
						
							|  |  |  | static int spandsp_fax_start(struct ast_fax_session *s); | 
					
						
							|  |  |  | static int spandsp_fax_cancel(struct ast_fax_session *s); | 
					
						
							|  |  |  | static int spandsp_fax_switch_to_t38(struct ast_fax_session *s); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | static int spandsp_fax_gateway_start(struct ast_fax_session *s); | 
					
						
							|  |  |  | static int spandsp_fax_gateway_process(struct ast_fax_session *s, const struct ast_frame *f); | 
					
						
							|  |  |  | static void spandsp_fax_gateway_cleanup(struct ast_fax_session *s); | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | static int spandsp_v21_detect(struct ast_fax_session *s, const struct ast_frame *f); | 
					
						
							|  |  |  | static void spandsp_v21_cleanup(struct ast_fax_session *s); | 
					
						
							|  |  |  | static void spandsp_v21_tone(void *data, int code, int level, int delay); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_capabilities(int fd); | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_session(struct ast_fax_session *s, int fd); | 
					
						
							| 
									
										
										
										
											2014-07-18 15:49:46 +00:00
										 |  |  | static void spandsp_manager_fax_session(struct mansession *s, | 
					
						
							|  |  |  | 	const char *id_text, struct ast_fax_session *session); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | static char *spandsp_fax_cli_show_stats(int fd); | 
					
						
							| 
									
										
										
										
											2010-04-26 14:18:15 +00:00
										 |  |  | static char *spandsp_fax_cli_show_settings(int fd); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct ast_fax_tech spandsp_fax_tech = { | 
					
						
							|  |  |  | 	.type = "Spandsp", | 
					
						
							|  |  |  | 	.description = "Spandsp FAX Driver", | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							|  |  |  | 	/* spandsp 0.0.6 */ | 
					
						
							|  |  |  | 	.version = SPANDSP_RELEASE_DATETIME_STRING, | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* spandsp 0.0.5
 | 
					
						
							|  |  |  | 	 * TODO: maybe we should determine the version better way | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	.version = "pre-20090220", | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	.caps = AST_FAX_TECH_AUDIO | AST_FAX_TECH_T38 | AST_FAX_TECH_SEND | 
					
						
							|  |  |  | 		| AST_FAX_TECH_RECEIVE | AST_FAX_TECH_GATEWAY | 
					
						
							|  |  |  | 		| AST_FAX_TECH_V21_DETECT, | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	.new_session = spandsp_fax_new, | 
					
						
							|  |  |  | 	.destroy_session = spandsp_fax_destroy, | 
					
						
							|  |  |  | 	.read = spandsp_fax_read, | 
					
						
							|  |  |  | 	.write = spandsp_fax_write, | 
					
						
							|  |  |  | 	.start_session = spandsp_fax_start, | 
					
						
							|  |  |  | 	.cancel_session = spandsp_fax_cancel, | 
					
						
							|  |  |  | 	.switch_to_t38 = spandsp_fax_switch_to_t38, | 
					
						
							|  |  |  | 	.cli_show_capabilities = spandsp_fax_cli_show_capabilities, | 
					
						
							|  |  |  | 	.cli_show_session = spandsp_fax_cli_show_session, | 
					
						
							| 
									
										
										
										
											2014-07-18 15:49:46 +00:00
										 |  |  | 	.manager_fax_session = spandsp_manager_fax_session, | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	.cli_show_stats = spandsp_fax_cli_show_stats, | 
					
						
							| 
									
										
										
										
											2010-04-26 14:18:15 +00:00
										 |  |  | 	.cli_show_settings = spandsp_fax_cli_show_settings, | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct spandsp_fax_stats { | 
					
						
							|  |  |  | 	int success; | 
					
						
							|  |  |  | 	int nofax; | 
					
						
							|  |  |  | 	int neg_failed; | 
					
						
							|  |  |  | 	int failed_to_train; | 
					
						
							|  |  |  | 	int rx_protocol_error; | 
					
						
							|  |  |  | 	int tx_protocol_error; | 
					
						
							|  |  |  | 	int protocol_error; | 
					
						
							|  |  |  | 	int retries_exceeded; | 
					
						
							|  |  |  | 	int file_error; | 
					
						
							|  |  |  | 	int mem_error; | 
					
						
							|  |  |  | 	int call_dropped; | 
					
						
							|  |  |  | 	int unknown_error; | 
					
						
							|  |  |  | 	int switched; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct { | 
					
						
							|  |  |  | 	ast_mutex_t lock; | 
					
						
							|  |  |  | 	struct spandsp_fax_stats g711; | 
					
						
							|  |  |  | 	struct spandsp_fax_stats t38; | 
					
						
							|  |  |  | } spandsp_global_stats; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct spandsp_pvt { | 
					
						
							|  |  |  | 	unsigned int ist38:1; | 
					
						
							|  |  |  | 	unsigned int isdone:1; | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	enum ast_t38_state ast_t38_state; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	fax_state_t fax_state; | 
					
						
							|  |  |  | 	t38_terminal_state_t t38_state; | 
					
						
							|  |  |  | 	t30_state_t *t30_state; | 
					
						
							|  |  |  | 	t38_core_state_t *t38_core_state; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct spandsp_fax_stats *stats; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	struct spandsp_fax_gw_stats *t38stats; | 
					
						
							|  |  |  | 	t38_gateway_state_t t38_gw_state; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	struct ast_timer *timer; | 
					
						
							|  |  |  | 	AST_LIST_HEAD(frame_queue, ast_frame) read_frames; | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int v21_detected; | 
					
						
							|  |  |  | 	modem_connect_tones_rx_state_t *tone_state; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | static int spandsp_v21_new(struct spandsp_pvt *p); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | static void session_destroy(struct spandsp_pvt *p); | 
					
						
							|  |  |  | static int t38_tx_packet_handler(t38_core_state_t *t38_core_state, void *data, const uint8_t *buf, int len, int count); | 
					
						
							|  |  |  | static int update_stats(struct spandsp_pvt *p, int completion_code); | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | static int spandsp_modems(struct ast_fax_session_details *details); | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | /* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | static void t30_phase_e_handler(void *data, int completion_code); | 
					
						
							|  |  |  | static void spandsp_log(void *user_data, int level, const char *msg); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | /* for spandsp release 0.0.6 */ | 
					
						
							|  |  |  | static void t30_phase_e_handler(t30_state_t *t30_state, void *data, int completion_code); | 
					
						
							|  |  |  | static void spandsp_log(int level, const char *msg); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void set_logging(logging_state_t *state, struct ast_fax_session_details *details); | 
					
						
							|  |  |  | static void set_local_info(t30_state_t *t30_state, struct ast_fax_session_details *details); | 
					
						
							|  |  |  | static void set_file(t30_state_t *t30_state, struct ast_fax_session_details *details); | 
					
						
							|  |  |  | static void set_ecm(t30_state_t *t30_state, struct ast_fax_session_details *details); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void session_destroy(struct spandsp_pvt *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_frame *f; | 
					
						
							| 
									
										
										
										
											2014-09-18 16:24:48 +00:00
										 |  |  | 	t30_state_t *t30_to_terminate; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-18 16:24:48 +00:00
										 |  |  | 	if (p->t30_state) { | 
					
						
							|  |  |  | 		t30_to_terminate = p->t30_state; | 
					
						
							|  |  |  | 	} else if (p->ist38) { | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20080725
 | 
					
						
							|  |  |  | 		t30_to_terminate = &p->t38_state.t30; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		t30_to_terminate = &p->t38_state.t30_state; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20080725
 | 
					
						
							|  |  |  | 		t30_to_terminate = &p->fax_state.t30; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		t30_to_terminate = &p->fax_state.t30_state; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t30_terminate(t30_to_terminate); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	p->isdone = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_timer_close(p->timer); | 
					
						
							| 
									
										
										
										
											2012-11-05 23:10:14 +00:00
										 |  |  | 	p->timer = NULL; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	fax_release(&p->fax_state); | 
					
						
							|  |  |  | 	t38_terminal_release(&p->t38_state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ((f = AST_LIST_REMOVE_HEAD(&p->read_frames, frame_list))) { | 
					
						
							|  |  |  | 		ast_frfree(f); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int t38_tx_packet_handler(t38_core_state_t *t38_core_state, void *data, const uint8_t *buf, int len, int count) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	struct ast_fax_session *s = data; | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	struct ast_frame fax_frame = { | 
					
						
							|  |  |  | 		.frametype = AST_FRAME_MODEM, | 
					
						
							|  |  |  | 		.subclass.integer = AST_MODEM_T38, | 
					
						
							|  |  |  | 		.src = "res_fax_spandsp_t38", | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct ast_frame *f = &fax_frame; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TODO: Asterisk does not provide means of resending the same packet multiple
 | 
					
						
							|  |  |  | 	  times so count is ignored at the moment */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_FRAME_SET_BUFFER(f, buf, 0, len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(f = ast_frisolate(f))) { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		return res; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		ast_set_flag(f, AST_FAX_FRFLAG_GATEWAY); | 
					
						
							|  |  |  | 		if (p->ast_t38_state == T38_STATE_NEGOTIATED) { | 
					
						
							|  |  |  | 			res = ast_write(s->chan, f); | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			res = ast_queue_frame(s->chan, f); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		ast_frfree(f); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* no need to lock, this all runs in the same thread */ | 
					
						
							|  |  |  | 		AST_LIST_INSERT_TAIL(&p->read_frames, f, frame_list); | 
					
						
							| 
									
										
										
										
											2013-01-22 22:19:02 +00:00
										 |  |  | 		res = 0; | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	return res; | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int update_stats(struct spandsp_pvt *p, int completion_code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	switch (completion_code) { | 
					
						
							|  |  |  | 	case T30_ERR_OK: | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->success, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Link problems */ | 
					
						
							|  |  |  | 	case T30_ERR_CEDTONE:            /*! The CED tone exceeded 5s */ | 
					
						
							|  |  |  | 	case T30_ERR_T0_EXPIRED:         /*! Timed out waiting for initial communication */ | 
					
						
							|  |  |  | 	case T30_ERR_T1_EXPIRED:         /*! Timed out waiting for the first message */ | 
					
						
							|  |  |  | 	case T30_ERR_T3_EXPIRED:         /*! Timed out waiting for procedural interrupt */ | 
					
						
							|  |  |  | 	case T30_ERR_HDLC_CARRIER:       /*! The HDLC carrier did not stop in a timely manner */ | 
					
						
							|  |  |  | 	case T30_ERR_CANNOT_TRAIN:       /*! Failed to train with any of the compatible modems */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->failed_to_train, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case T30_ERR_OPER_INT_FAIL:      /*! Operator intervention failed */ | 
					
						
							|  |  |  | 	case T30_ERR_INCOMPATIBLE:       /*! Far end is not compatible */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_INCAPABLE:       /*! Far end is not able to receive */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_INCAPABLE:       /*! Far end is not able to transmit */ | 
					
						
							|  |  |  | 	case T30_ERR_NORESSUPPORT:       /*! Far end cannot receive at the resolution of the image */ | 
					
						
							|  |  |  | 	case T30_ERR_NOSIZESUPPORT:      /*! Far end cannot receive at the size of image */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->neg_failed, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case T30_ERR_UNEXPECTED:         /*! Unexpected message received */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->protocol_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Phase E status values returned to a transmitter */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_BADDCS:          /*! Received bad response to DCS or training */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_BADPG:           /*! Received a DCN from remote after sending a page */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_ECMPHD:          /*! Invalid ECM response received from receiver */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_GOTDCN:          /*! Received a DCN while waiting for a DIS */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_INVALRSP:        /*! Invalid response after sending a page */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_NODIS:           /*! Received other than DIS while waiting for DIS */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_PHBDEAD:         /*! Received no response to DCS, training or TCF */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_PHDDEAD:         /*! No response after sending a page */ | 
					
						
							|  |  |  | 	case T30_ERR_TX_T5EXP:           /*! Timed out waiting for receiver ready (ECM mode) */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->tx_protocol_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Phase E status values returned to a receiver */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_ECMPHD:          /*! Invalid ECM response received from transmitter */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_GOTDCS:          /*! DCS received while waiting for DTC */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_INVALCMD:        /*! Unexpected command after page received */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_NOCARRIER:       /*! Carrier lost during fax receive */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_NOEOL:           /*! Timed out while waiting for EOL (end Of line) */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->rx_protocol_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case T30_ERR_RX_NOFAX:           /*! Timed out while waiting for first line */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->nofax, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXPDCN:        /*! Timer T2 expired while waiting for DCN */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXPD:          /*! Timer T2 expired while waiting for phase D */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXPFAX:        /*! Timer T2 expired while waiting for fax page */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXPMPS:        /*! Timer T2 expired while waiting for next fax page */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXPRR:         /*! Timer T2 expired while waiting for RR command */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_T2EXP:           /*! Timer T2 expired while waiting for NSS, DCS or MCF */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNWHY:          /*! Unexpected DCN while waiting for DCS or DIS */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNDATA:         /*! Unexpected DCN while waiting for image data */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNFAX:          /*! Unexpected DCN while waiting for EOM, EOP or MPS */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNPHD:          /*! Unexpected DCN after EOM or MPS sequence */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNRRD:          /*! Unexpected DCN after RR/RNR sequence */ | 
					
						
							|  |  |  | 	case T30_ERR_RX_DCNNORTN:        /*! Unexpected DCN after requested retransmission */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->rx_protocol_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* TIFF file problems */ | 
					
						
							|  |  |  | 	case T30_ERR_FILEERROR:          /*! TIFF/F file cannot be opened */ | 
					
						
							|  |  |  | 	case T30_ERR_NOPAGE:             /*! TIFF/F page not found */ | 
					
						
							|  |  |  | 	case T30_ERR_BADTIFF:            /*! TIFF/F format is not compatible */ | 
					
						
							|  |  |  | 	case T30_ERR_BADPAGE:            /*! TIFF/F page number tag missing */ | 
					
						
							|  |  |  | 	case T30_ERR_BADTAG:             /*! Incorrect values for TIFF/F tags */ | 
					
						
							|  |  |  | 	case T30_ERR_BADTIFFHDR:         /*! Bad TIFF/F header - incorrect values in fields */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->file_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case T30_ERR_NOMEM:              /*! Cannot allocate memory for more pages */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->mem_error, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* General problems */ | 
					
						
							|  |  |  | 	case T30_ERR_RETRYDCN:           /*! Disconnected after permitted retries */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->retries_exceeded, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case T30_ERR_CALLDROPPED:        /*! The call dropped prematurely */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->call_dropped, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Feature negotiation issues */ | 
					
						
							|  |  |  | 	case T30_ERR_NOPOLL:             /*! Poll not accepted */ | 
					
						
							|  |  |  | 	case T30_ERR_IDENT_UNACCEPTABLE: /*! Far end's ident is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_SUB_UNACCEPTABLE:   /*! Far end's sub-address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_SEP_UNACCEPTABLE:   /*! Far end's selective polling address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_PSA_UNACCEPTABLE:   /*! Far end's polled sub-address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_SID_UNACCEPTABLE:   /*! Far end's sender identification is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_PWD_UNACCEPTABLE:   /*! Far end's password is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_TSA_UNACCEPTABLE:   /*! Far end's transmitting subscriber internet address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_IRA_UNACCEPTABLE:   /*! Far end's internet routing address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_CIA_UNACCEPTABLE:   /*! Far end's calling subscriber internet address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_ISP_UNACCEPTABLE:   /*! Far end's internet selective polling address is not acceptable */ | 
					
						
							|  |  |  | 	case T30_ERR_CSA_UNACCEPTABLE:   /*! Far end's called subscriber internet address is not acceptable */ | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->neg_failed, 1); | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		ast_atomic_fetchadd_int(&p->stats->unknown_error, 1); | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "unknown FAX session result '%d' (%s)\n", completion_code, t30_completion_code_to_str(completion_code)); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Phase E handler callback.
 | 
					
						
							|  |  |  |  * \param t30_state the span t30 state | 
					
						
							|  |  |  |  * \param data this will be the ast_fax_session | 
					
						
							|  |  |  |  * \param completion_code the result of the fax session | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function pulls stats from the spandsp stack and stores them for res_fax | 
					
						
							|  |  |  |  * to use later. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | /* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | static void t30_phase_e_handler(void *data, int completion_code) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | /* for spandsp release 0.0.6 */ | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | static void t30_phase_e_handler(t30_state_t *t30_state, void *data, int completion_code) | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct ast_fax_session *s = data; | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 	char headerinfo[T30_MAX_PAGE_HEADER_INFO + 1]; | 
					
						
							|  |  |  | 	const char *c; | 
					
						
							|  |  |  | 	t30_stats_t stats; | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | 	/* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | 	t30_state_t *t30_state = p->t30_state; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 	ast_debug(5, "FAX session '%u' entering phase E\n", s->id); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	p->isdone = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	update_stats(p, completion_code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t30_get_transfer_statistics(t30_state, &stats); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (completion_code == T30_ERR_OK) { | 
					
						
							|  |  |  | 		ast_string_field_set(s->details, result, "SUCCESS"); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_string_field_set(s->details, result, "FAILED"); | 
					
						
							|  |  |  | 		ast_string_field_set(s->details, error, t30_completion_code_to_str(completion_code)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_string_field_set(s->details, resultstr, t30_completion_code_to_str(completion_code)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 	ast_debug(5, "FAX session '%u' completed with result: %s (%s)\n", s->id, s->details->result, s->details->resultstr); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ((c = t30_get_tx_ident(t30_state))) { | 
					
						
							|  |  |  | 		ast_string_field_set(s->details, localstationid, c); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((c = t30_get_rx_ident(t30_state))) { | 
					
						
							|  |  |  | 		ast_string_field_set(s->details, remotestationid, c); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							|  |  |  | 	s->details->pages_transferred = (s->details->caps & AST_FAX_TECH_RECEIVE) ? stats.pages_rx : stats.pages_tx; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	s->details->pages_transferred = stats.pages_transferred; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_string_field_build(s->details, transfer_rate, "%d", stats.bit_rate); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_string_field_build(s->details, resolution, "%dx%d", stats.x_resolution, stats.y_resolution); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t30_get_tx_page_header_info(t30_state, headerinfo); | 
					
						
							|  |  |  | 	ast_string_field_set(s->details, headerinfo, headerinfo); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Send spandsp log messages to asterisk.
 | 
					
						
							|  |  |  |  * \param level the spandsp logging level | 
					
						
							|  |  |  |  * \param msg the log message | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note This function is a callback function called by spandsp. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | /* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | static void spandsp_log(void *user_data, int level, const char *msg) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | /* for spandsp release 0.0.6 */ | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | static void spandsp_log(int level, const char *msg) | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (level == SPAN_LOG_ERROR) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "%s", msg); | 
					
						
							|  |  |  | 	} else if (level == SPAN_LOG_WARNING) { | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "%s", msg); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2010-05-21 15:15:58 +00:00
										 |  |  | 		ast_fax_log(LOG_DEBUG, msg); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_logging(logging_state_t *state, struct ast_fax_session_details *details) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int level = SPAN_LOG_WARNING; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 09:53:19 +08:00
										 |  |  | 	if (details->option.debug) { | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		level = SPAN_LOG_DEBUG_3; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | 	/* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | 	span_log_set_message_handler(state, spandsp_log, NULL); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* for spandsp release 0.0.6 */ | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	span_log_set_message_handler(state, spandsp_log); | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	span_log_set_level(state, SPAN_LOG_SHOW_SEVERITY | SPAN_LOG_SHOW_PROTOCOL | level); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_local_info(t30_state_t *t30_state, struct ast_fax_session_details *details) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(details->localstationid)) { | 
					
						
							|  |  |  | 		t30_set_tx_ident(t30_state, details->localstationid); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!ast_strlen_zero(details->headerinfo)) { | 
					
						
							|  |  |  | 		t30_set_tx_page_header_info(t30_state, details->headerinfo); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_file(t30_state_t *t30_state, struct ast_fax_session_details *details) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (details->caps & AST_FAX_TECH_RECEIVE) { | 
					
						
							|  |  |  | 		t30_set_rx_file(t30_state, AST_LIST_FIRST(&details->documents)->filename, -1); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* if not AST_FAX_TECH_RECEIVE, assume AST_FAX_TECH_SEND, this
 | 
					
						
							|  |  |  | 		 * should be safe because we ensure either RECEIVE or SEND is | 
					
						
							|  |  |  | 		 * indicated in spandsp_fax_new() */ | 
					
						
							|  |  |  | 		t30_set_tx_file(t30_state, AST_LIST_FIRST(&details->documents)->filename, -1, -1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void set_ecm(t30_state_t *t30_state, struct ast_fax_session_details *details) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-04-26 14:18:15 +00:00
										 |  |  | 	t30_set_ecm_capability(t30_state, details->option.ecm); | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | 	/* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | 	t30_set_supported_compressions(t30_state, T4_COMPRESSION_T4_1D | T4_COMPRESSION_T4_2D | T4_COMPRESSION_T6); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* for spandsp release 0.0.6 */ | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	t30_set_supported_compressions(t30_state, T30_SUPPORT_T4_1D_COMPRESSION | T30_SUPPORT_T4_2D_COMPRESSION | T30_SUPPORT_T6_COMPRESSION); | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | static int spandsp_v21_new(struct spandsp_pvt *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* XXX Here we use MODEM_CONNECT_TONES_FAX_CED_OR_PREAMBLE even though
 | 
					
						
							|  |  |  | 	 * we don't care about CED tones. Using MODEM_CONNECT_TONES_PREAMBLE | 
					
						
							|  |  |  | 	 * doesn't seem to work right all the time. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	p->tone_state = modem_connect_tones_rx_init(NULL, MODEM_CONNECT_TONES_FAX_CED_OR_PREAMBLE, spandsp_v21_tone, p); | 
					
						
							|  |  |  | 	if (!p->tone_state) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | static int spandsp_modems(struct ast_fax_session_details *details) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int modems = 0; | 
					
						
							|  |  |  | 	if (AST_FAX_MODEM_V17 & details->modems) { | 
					
						
							|  |  |  | 		modems |= T30_SUPPORT_V17; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-04-29 13:05:10 -05:00
										 |  |  | 	if (AST_FAX_MODEM_V27TER & details->modems) { | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | 		modems |= T30_SUPPORT_V27TER; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (AST_FAX_MODEM_V29 & details->modems) { | 
					
						
							|  |  |  | 		modems |= T30_SUPPORT_V29; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (AST_FAX_MODEM_V34 & details->modems) { | 
					
						
							|  |  |  | #if defined(T30_SUPPORT_V34)
 | 
					
						
							|  |  |  | 		modems |= T30_SUPPORT_V34; | 
					
						
							|  |  |  | #elif defined(T30_SUPPORT_V34HDX)
 | 
					
						
							|  |  |  | 		modems |= T30_SUPPORT_V34HDX; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		ast_log(LOG_WARNING, "v34 not supported in this version of spandsp\n"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return modems; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief create an instance of the spandsp tech_pvt for a fax session */ | 
					
						
							|  |  |  | static void *spandsp_fax_new(struct ast_fax_session *s, struct ast_fax_tech_token *token) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p; | 
					
						
							|  |  |  | 	int caller_mode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((!(p = ast_calloc(1, sizeof(*p))))) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Cannot initialize the spandsp private FAX technology structure.\n"); | 
					
						
							|  |  |  | 		goto e_return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_V21_DETECT) { | 
					
						
							|  |  |  | 		if (spandsp_v21_new(p)) { | 
					
						
							|  |  |  | 			ast_log(LOG_ERROR, "Cannot initialize the spandsp private v21 technology structure.\n"); | 
					
						
							|  |  |  | 			goto e_return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		s->state = AST_FAX_STATE_ACTIVE; | 
					
						
							|  |  |  | 		return p; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		s->state = AST_FAX_STATE_INITIALIZED; | 
					
						
							|  |  |  | 		return p; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	AST_LIST_HEAD_INIT(&p->read_frames); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (s->details->caps & AST_FAX_TECH_RECEIVE) { | 
					
						
							|  |  |  | 		caller_mode = 0; | 
					
						
							|  |  |  | 	} else if (s->details->caps & AST_FAX_TECH_SEND) { | 
					
						
							|  |  |  | 		caller_mode = 1; | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "Are we sending or receiving? The FAX requirements (capabilities: 0x%X) were not properly set.\n", s->details->caps); | 
					
						
							|  |  |  | 		goto e_free; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(p->timer = ast_timer_open())) { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "Channel '%s' FAX session '%u' failed to create timing source.\n", s->channame, s->id); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		goto e_free; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->fd = ast_timer_fd(p->timer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p->stats = &spandsp_global_stats.g711; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-09 03:21:56 +00:00
										 |  |  | 	if (s->details->caps & (AST_FAX_TECH_T38 | AST_FAX_TECH_AUDIO)) { | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		if ((s->details->caps & AST_FAX_TECH_AUDIO) == 0) { | 
					
						
							|  |  |  | 			/* audio mode was not requested, start in T.38 mode */ | 
					
						
							|  |  |  | 			p->ist38 = 1; | 
					
						
							|  |  |  | 			p->stats = &spandsp_global_stats.t38; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* init t38 stuff */ | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		t38_terminal_init(&p->t38_state, caller_mode, t38_tx_packet_handler, s); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		set_logging(&p->t38_state.logging, s->details); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* init audio stuff */ | 
					
						
							|  |  |  | 		fax_init(&p->fax_state, caller_mode); | 
					
						
							|  |  |  | 		set_logging(&p->fax_state.logging, s->details); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->state = AST_FAX_STATE_INITIALIZED; | 
					
						
							|  |  |  | 	return p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | e_free: | 
					
						
							|  |  |  | 	ast_free(p); | 
					
						
							|  |  |  | e_return: | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | static void spandsp_v21_cleanup(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	modem_connect_tones_rx_free(p->tone_state); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief Destroy a spandsp fax session.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void spandsp_fax_destroy(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		spandsp_fax_gateway_cleanup(s); | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	} else if (s->details->caps & AST_FAX_TECH_V21_DETECT) { | 
					
						
							|  |  |  | 		spandsp_v21_cleanup(s); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		session_destroy(p); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	ast_free(p); | 
					
						
							|  |  |  | 	s->tech_pvt = NULL; | 
					
						
							|  |  |  | 	s->fd = -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief Read a frame from the spandsp fax stack.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct ast_frame *spandsp_fax_read(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 	uint8_t buffer[AST_FRIENDLY_OFFSET + SPANDSP_FAX_SAMPLES * sizeof(uint16_t)]; | 
					
						
							|  |  |  | 	int16_t *buf = (int16_t *) (buffer + AST_FRIENDLY_OFFSET); | 
					
						
							|  |  |  | 	int samples; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct ast_frame fax_frame = { | 
					
						
							|  |  |  | 		.frametype = AST_FRAME_VOICE, | 
					
						
							|  |  |  | 		.src = "res_fax_spandsp_g711", | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 		.subclass.format = ast_format_slin, | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 	struct ast_frame *f = &fax_frame; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-11-05 23:10:14 +00:00
										 |  |  | 	if (ast_timer_ack(p->timer, 1) < 0) { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "Failed to acknowledge timer for FAX session '%u'\n", s->id); | 
					
						
							| 
									
										
										
										
											2012-11-05 23:10:14 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* XXX do we need to lock here? */ | 
					
						
							|  |  |  | 	if (p->isdone) { | 
					
						
							|  |  |  | 		s->state = AST_FAX_STATE_COMPLETE; | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_debug(5, "FAX session '%u' is complete.\n", s->id); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p->ist38) { | 
					
						
							|  |  |  | 		t38_terminal_send_timeout(&p->t38_state, SPANDSP_FAX_SAMPLES); | 
					
						
							|  |  |  | 		if ((f = AST_LIST_REMOVE_HEAD(&p->read_frames, frame_list))) { | 
					
						
							|  |  |  | 			return f; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if ((samples = fax_tx(&p->fax_state, buf, SPANDSP_FAX_SAMPLES)) > 0) { | 
					
						
							|  |  |  | 			f->samples = samples; | 
					
						
							|  |  |  | 			AST_FRAME_SET_BUFFER(f, buffer, AST_FRIENDLY_OFFSET, samples * sizeof(int16_t)); | 
					
						
							|  |  |  | 			return ast_frisolate(f); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return &ast_null_frame; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | static void spandsp_v21_tone(void *data, int code, int level, int delay) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (code == MODEM_CONNECT_TONES_FAX_PREAMBLE) { | 
					
						
							|  |  |  | 		p->v21_detected = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | static int spandsp_v21_detect(struct ast_fax_session *s, const struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 	int16_t *slndata; | 
					
						
							|  |  |  | 	g711_state_t *decoder; | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (p->v21_detected) { | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*invalid frame*/ | 
					
						
							|  |  |  | 	if (!f->data.ptr || !f->datalen) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 	ast_debug(5, "frame={ datalen=%d, samples=%d, mallocd=%d, src=%s, flags=%u, ts=%ld, len=%ld, seqno=%d, data.ptr=%p, subclass.format=%s  }\n", f->datalen, f->samples, f->mallocd, f->src, f->flags, f->ts, f->len, f->seqno, f->data.ptr, ast_format_get_name(f->subclass.format)); | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* slinear frame can be passed to spandsp */ | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 	if (ast_format_cmp(f->subclass.format, ast_format_slin) == AST_FORMAT_CMP_EQUAL) { | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		modem_connect_tones_rx(p->tone_state, f->data.ptr, f->samples); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* alaw/ulaw frame must be converted to slinear before passing to spandsp */ | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 	} else if (ast_format_cmp(f->subclass.format, ast_format_alaw) == AST_FORMAT_CMP_EQUAL || | 
					
						
							|  |  |  | 	           ast_format_cmp(f->subclass.format, ast_format_ulaw) == AST_FORMAT_CMP_EQUAL) { | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		if (!(slndata = ast_malloc(sizeof(*slndata) * f->samples))) { | 
					
						
							|  |  |  | 			return -1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 		decoder = g711_init(NULL, (ast_format_cmp(f->subclass.format, ast_format_alaw) == AST_FORMAT_CMP_EQUAL ? G711_ALAW : G711_ULAW)); | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		g711_decode(decoder, slndata, f->data.ptr, f->samples); | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 		ast_debug(5, "spandsp transcoding frame from %s to slinear for v21 detection\n", ast_format_get_name(f->subclass.format)); | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		modem_connect_tones_rx(p->tone_state, slndata, f->samples); | 
					
						
							|  |  |  | 		g711_release(decoder); | 
					
						
							| 
									
										
										
										
											2014-03-18 11:52:15 +00:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							|  |  |  | 		g711_free(decoder); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		ast_free(slndata); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* frame in other formats cannot be passed to spandsp, it could cause segfault */ | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "Frame format %s not supported, v.21 detection skipped\n", ast_format_get_name(f->subclass.format)); | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (p->v21_detected) { | 
					
						
							|  |  |  | 		s->details->option.v21_detected = 1; | 
					
						
							| 
									
										
										
										
											2014-03-06 02:22:59 +00:00
										 |  |  | 		ast_debug(5, "v.21 detected\n"); | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief Write a frame to the spandsp fax stack.
 | 
					
						
							|  |  |  |  * \param s a fax session | 
					
						
							|  |  |  |  * \param f the frame to write | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note res_fax does not currently use the return value of this function. | 
					
						
							|  |  |  |  * Also the fax_rx() function never fails. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \retval 0 success | 
					
						
							|  |  |  |  * \retval -1 failure | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int spandsp_fax_write(struct ast_fax_session *s, const struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_V21_DETECT) { | 
					
						
							|  |  |  | 		return spandsp_v21_detect(s, f); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		return spandsp_fax_gateway_process(s, f); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	/* XXX do we need to lock here? */ | 
					
						
							|  |  |  | 	if (s->state == AST_FAX_STATE_COMPLETE) { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_log(LOG_WARNING, "FAX session '%u' is in the '%s' state.\n", s->id, ast_fax_state_to_str(s->state)); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p->ist38) { | 
					
						
							|  |  |  | 		return t38_core_rx_ifp_packet(p->t38_core_state, f->data.ptr, f->datalen, f->seqno); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		return fax_rx(&p->fax_state, f->data.ptr, f->samples); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | /*! \brief generate T.30 packets sent to the T.30 leg of gateway
 | 
					
						
							|  |  |  |  * \param chan T.30 channel | 
					
						
							|  |  |  |  * \param data fax session structure | 
					
						
							|  |  |  |  * \param len not used | 
					
						
							|  |  |  |  * \param samples no of samples generated | 
					
						
							|  |  |  |  * \return -1 on failure or 0 on sucess*/ | 
					
						
							|  |  |  | static int spandsp_fax_gw_t30_gen(struct ast_channel *chan, void *data, int len, int samples) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int res = -1; | 
					
						
							|  |  |  | 	struct ast_fax_session *s = data; | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 	uint8_t buffer[AST_FRIENDLY_OFFSET + samples * sizeof(uint16_t)]; | 
					
						
							|  |  |  | 	struct ast_frame *f; | 
					
						
							|  |  |  | 	struct ast_frame t30_frame = { | 
					
						
							|  |  |  | 		.frametype = AST_FRAME_VOICE, | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 		.subclass.format = ast_format_slin, | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		.src = "res_fax_spandsp_g711", | 
					
						
							|  |  |  | 		.samples = samples, | 
					
						
							|  |  |  | 		.flags = AST_FAX_FRFLAG_GATEWAY, | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	AST_FRAME_SET_BUFFER(&t30_frame, buffer, AST_FRIENDLY_OFFSET, t30_frame.samples * sizeof(int16_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!(f = ast_frisolate(&t30_frame))) { | 
					
						
							|  |  |  | 		return p->isdone ? -1 : res; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* generate a T.30 packet */ | 
					
						
							|  |  |  | 	if ((f->samples = t38_gateway_tx(&p->t38_gw_state, f->data.ptr, f->samples))) { | 
					
						
							|  |  |  | 		f->datalen = f->samples * sizeof(int16_t); | 
					
						
							|  |  |  | 		res = ast_write(chan, f); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ast_frfree(f); | 
					
						
							|  |  |  | 	return p->isdone ? -1 : res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief simple routine to allocate data to generator
 | 
					
						
							|  |  |  |  * \param chan channel | 
					
						
							|  |  |  |  * \param params generator data | 
					
						
							|  |  |  |  * \return data to use in generator call*/ | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | static void *spandsp_fax_gw_gen_alloc(struct ast_channel *chan, void *params) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	ao2_ref(params, +1); | 
					
						
							|  |  |  | 	return params; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | static void spandsp_fax_gw_gen_release(struct ast_channel *chan, void *data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	ao2_ref(data, -1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief activate a spandsp gateway based on the information in the given fax session
 | 
					
						
							|  |  |  |  * \param s fax session | 
					
						
							|  |  |  |  * \return -1 on error 0 on sucess*/ | 
					
						
							| 
									
										
										
										
											2015-01-15 17:28:51 +00:00
										 |  |  | static int spandsp_fax_gateway_start(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 	struct ast_fax_t38_parameters *t38_param; | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2013-05-27 01:33:12 +00:00
										 |  |  | 	RAII_VAR(struct ast_channel *, peer, NULL, ao2_cleanup); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	static struct ast_generator t30_gen = { | 
					
						
							| 
									
										
										
										
											2012-04-04 18:08:28 +00:00
										 |  |  | 		.alloc = spandsp_fax_gw_gen_alloc, | 
					
						
							|  |  |  | 		.release = spandsp_fax_gw_gen_release, | 
					
						
							|  |  |  | 		.generate = spandsp_fax_gw_t30_gen, | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20081012
 | 
					
						
							|  |  |  | 	/* for spandsp shaphots 0.0.6 and higher */ | 
					
						
							|  |  |  | 	p->t38_core_state=&p->t38_gw_state.t38x.t38; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* for spandsp release 0.0.5 */ | 
					
						
							|  |  |  | 	p->t38_core_state=&p->t38_gw_state.t38; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!t38_gateway_init(&p->t38_gw_state, t38_tx_packet_handler, s)) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p->ist38 = 1; | 
					
						
							|  |  |  | 	p->ast_t38_state = ast_channel_get_t38_state(s->chan); | 
					
						
							| 
									
										
										
										
											2015-01-15 17:36:37 +00:00
										 |  |  | 	peer = ast_channel_bridge_peer(s->chan); | 
					
						
							|  |  |  | 	if (!peer) { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-08-29 18:28:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* we can be in T38_STATE_NEGOTIATING or T38_STATE_NEGOTIATED when the
 | 
					
						
							|  |  |  | 	 * gateway is started. We treat both states the same. */ | 
					
						
							|  |  |  | 	if (p->ast_t38_state == T38_STATE_NEGOTIATING) { | 
					
						
							|  |  |  | 		p->ast_t38_state = T38_STATE_NEGOTIATED; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	ast_activate_generator(p->ast_t38_state == T38_STATE_NEGOTIATED ? peer : s->chan, &t30_gen , s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	set_logging(&p->t38_gw_state.logging, s->details); | 
					
						
							|  |  |  | 	set_logging(&p->t38_core_state->logging, s->details); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t38_param = (p->ast_t38_state == T38_STATE_NEGOTIATED) ? &s->details->our_t38_parameters : &s->details->their_t38_parameters; | 
					
						
							|  |  |  | 	t38_set_t38_version(p->t38_core_state, t38_param->version); | 
					
						
							|  |  |  | 	t38_gateway_set_ecm_capability(&p->t38_gw_state, s->details->option.ecm); | 
					
						
							|  |  |  | 	t38_set_max_datagram_size(p->t38_core_state, t38_param->max_ifp); | 
					
						
							|  |  |  | 	t38_set_fill_bit_removal(p->t38_core_state, t38_param->fill_bit_removal); | 
					
						
							|  |  |  | 	t38_set_mmr_transcoding(p->t38_core_state, t38_param->transcoding_mmr); | 
					
						
							|  |  |  | 	t38_set_jbig_transcoding(p->t38_core_state, t38_param->transcoding_jbig); | 
					
						
							| 
									
										
										
										
											2017-12-22 09:23:22 -05:00
										 |  |  | 	t38_set_data_rate_management_method(p->t38_core_state, | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			(t38_param->rate_management == AST_T38_RATE_MANAGEMENT_TRANSFERRED_TCF)? 1 : 2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t38_gateway_set_transmit_on_idle(&p->t38_gw_state, TRUE); | 
					
						
							|  |  |  | 	t38_set_sequence_number_handling(p->t38_core_state, TRUE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | 	t38_gateway_set_supported_modems(&p->t38_gw_state, spandsp_modems(s->details)); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-12-22 09:23:22 -05:00
										 |  |  | 	/* engage udptl nat on other side of T38 line
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	 * (Asterisk changes media ports thus we send a few packets to reinitialize | 
					
						
							|  |  |  | 	 * pinholes in NATs and FWs | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	for (i=0; i < SPANDSP_ENGAGE_UDPTL_NAT_RETRY; i++) { | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20091228
 | 
					
						
							|  |  |  | 		t38_core_send_indicator(&p->t38_gw_state.t38x.t38, T38_IND_NO_SIGNAL); | 
					
						
							|  |  |  | #elif SPANDSP_RELEASE_DATE >= 20081012
 | 
					
						
							|  |  |  | 		t38_core_send_indicator(&p->t38_gw_state.t38x.t38, T38_IND_NO_SIGNAL, p->t38_gw_state.t38x.t38.indicator_tx_count); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		t38_core_send_indicator(&p->t38_gw_state.t38, T38_IND_NO_SIGNAL, p->t38_gw_state.t38.indicator_tx_count); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->state = AST_FAX_STATE_ACTIVE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief process a frame from the bridge
 | 
					
						
							|  |  |  |  * \param s fax session | 
					
						
							|  |  |  |  * \param f frame to process | 
					
						
							| 
									
										
										
										
											2021-10-30 21:04:36 -04:00
										 |  |  |  * \return 1 on sucess 0 on incorrect packet*/ | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | static int spandsp_fax_gateway_process(struct ast_fax_session *s, const struct ast_frame *f) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*invalid frame*/ | 
					
						
							|  |  |  | 	if (!f->data.ptr || !f->datalen) { | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Process a IFP packet */ | 
					
						
							|  |  |  | 	if ((f->frametype == AST_FRAME_MODEM) && (f->subclass.integer == AST_MODEM_T38)) { | 
					
						
							|  |  |  | 		return t38_core_rx_ifp_packet(p->t38_core_state, f->data.ptr, f->datalen, f->seqno); | 
					
						
							| 
									
										
											  
											
												media formats: re-architect handling of media for performance improvements
In the old times media formats were represented using a bit field. This was
fast but had a few limitations.
 1. Asterisk was limited in how many formats it could handle.
 2. Formats, being a bit field, could not include any attribute information.
    A format was strictly its type, e.g., "this is ulaw".
This was changed in Asterisk 10 (see
https://wiki.asterisk.org/wiki/display/AST/Media+Architecture+Proposal for
notes on that work) which led to the creation of the ast_format structure.
This structure allowed Asterisk to handle attributes and bundle information
with a format.
Additionally, ast_format_cap was created to act as a container for multiple
formats that, together, formed the capability of some entity. Another
mechanism was added to allow logic to be registered which performed format
attribute negotiation. Everywhere throughout the codebase Asterisk was
changed to use this strategy.
Unfortunately, in software, there is no free lunch. These new capabilities
came at a cost.
Performance analysis and profiling showed that we spend an inordinate
amount of time comparing, copying, and generally manipulating formats and
their related structures. Basic prototyping has shown that a reasonably
large performance improvement could be made in this area. This patch is the
result of that project, which overhauled the media format architecture
and its usage in Asterisk to improve performance.
Generally, the new philosophy for handling formats is as follows:
 * The ast_format structure is reference counted. This removed a large amount
   of the memory allocations and copying that was done in prior versions.
 * In order to prevent race conditions while keeping things performant, the
   ast_format structure is immutable by convention and lock-free. Violate this
   tenet at your peril!
 * Because formats are reference counted, codecs are also reference counted.
   The Asterisk core generally provides built-in codecs and caches the
   ast_format structures created to represent them. Generally, to prevent
   inordinate amounts of module reference bumping, codecs and formats can be
   added at run-time but cannot be removed.
 * All compatibility with the bit field representation of codecs/formats has
   been moved to a compatibility API. The primary user of this representation
   is chan_iax2, which must continue to maintain its bit-field usage of formats
   for interoperability concerns.
 * When a format is negotiated with attributes, or when a format cannot be
   represented by one of the cached formats, a new format object is created or
   cloned from an existing format. That format may have the same codec
   underlying it, but is a different format than a version of the format with
   different attributes or without attributes.
 * While formats are reference counted objects, the reference count maintained
   on the format should be manipulated with care. Formats are generally cached
   and will persist for the lifetime of Asterisk and do not explicitly need
   to have their lifetime modified. An exception to this is when the user of a
   format does not know where the format came from *and* the user may outlive
   the provider of the format. This occurs, for example, when a format is read
   from a channel: the channel may have a format with attributes (hence,
   non-cached) and the user of the format may last longer than the channel (if
   the reference to the channel is released prior to the format's reference).
For more information on this work, see the API design notes:
  https://wiki.asterisk.org/wiki/display/AST/Media+Format+Rewrite
Finally, this work was the culmination of a large number of developer's
efforts. Extra thanks goes to Corey Farrell, who took on a large amount of the
work in the Asterisk core, chan_sip, and was an invaluable resource in peer
reviews throughout this project.
There were a substantial number of patches contributed during this work; the
following issues/patch names simply reflect some of the work (and will cause
the release scripts to give attribution to the individuals who work on them).
Reviews:
 https://reviewboard.asterisk.org/r/3814
 https://reviewboard.asterisk.org/r/3808
 https://reviewboard.asterisk.org/r/3805
 https://reviewboard.asterisk.org/r/3803
 https://reviewboard.asterisk.org/r/3801
 https://reviewboard.asterisk.org/r/3798
 https://reviewboard.asterisk.org/r/3800
 https://reviewboard.asterisk.org/r/3794
 https://reviewboard.asterisk.org/r/3793
 https://reviewboard.asterisk.org/r/3792
 https://reviewboard.asterisk.org/r/3791
 https://reviewboard.asterisk.org/r/3790
 https://reviewboard.asterisk.org/r/3789
 https://reviewboard.asterisk.org/r/3788
 https://reviewboard.asterisk.org/r/3787
 https://reviewboard.asterisk.org/r/3786
 https://reviewboard.asterisk.org/r/3784
 https://reviewboard.asterisk.org/r/3783
 https://reviewboard.asterisk.org/r/3778
 https://reviewboard.asterisk.org/r/3774
 https://reviewboard.asterisk.org/r/3775
 https://reviewboard.asterisk.org/r/3772
 https://reviewboard.asterisk.org/r/3761
 https://reviewboard.asterisk.org/r/3754
 https://reviewboard.asterisk.org/r/3753
 https://reviewboard.asterisk.org/r/3751
 https://reviewboard.asterisk.org/r/3750
 https://reviewboard.asterisk.org/r/3748
 https://reviewboard.asterisk.org/r/3747
 https://reviewboard.asterisk.org/r/3746
 https://reviewboard.asterisk.org/r/3742
 https://reviewboard.asterisk.org/r/3740
 https://reviewboard.asterisk.org/r/3739
 https://reviewboard.asterisk.org/r/3738
 https://reviewboard.asterisk.org/r/3737
 https://reviewboard.asterisk.org/r/3736
 https://reviewboard.asterisk.org/r/3734
 https://reviewboard.asterisk.org/r/3722
 https://reviewboard.asterisk.org/r/3713
 https://reviewboard.asterisk.org/r/3703
 https://reviewboard.asterisk.org/r/3689
 https://reviewboard.asterisk.org/r/3687
 https://reviewboard.asterisk.org/r/3674
 https://reviewboard.asterisk.org/r/3671
 https://reviewboard.asterisk.org/r/3667
 https://reviewboard.asterisk.org/r/3665
 https://reviewboard.asterisk.org/r/3625
 https://reviewboard.asterisk.org/r/3602
 https://reviewboard.asterisk.org/r/3519
 https://reviewboard.asterisk.org/r/3518
 https://reviewboard.asterisk.org/r/3516
 https://reviewboard.asterisk.org/r/3515
 https://reviewboard.asterisk.org/r/3512
 https://reviewboard.asterisk.org/r/3506
 https://reviewboard.asterisk.org/r/3413
 https://reviewboard.asterisk.org/r/3410
 https://reviewboard.asterisk.org/r/3387
 https://reviewboard.asterisk.org/r/3388
 https://reviewboard.asterisk.org/r/3389
 https://reviewboard.asterisk.org/r/3390
 https://reviewboard.asterisk.org/r/3321
 https://reviewboard.asterisk.org/r/3320
 https://reviewboard.asterisk.org/r/3319
 https://reviewboard.asterisk.org/r/3318
 https://reviewboard.asterisk.org/r/3266
 https://reviewboard.asterisk.org/r/3265
 https://reviewboard.asterisk.org/r/3234
 https://reviewboard.asterisk.org/r/3178
ASTERISK-23114 #close
Reported by: mjordan
  media_formats_translation_core.diff uploaded by kharwell (License 6464)
  rb3506.diff uploaded by mjordan (License 6283)
  media_format_app_file.diff uploaded by kharwell (License 6464) 
  misc-2.diff uploaded by file (License 5000)
  chan_mild-3.diff uploaded by file (License 5000) 
  chan_obscure.diff uploaded by file (License 5000) 
  jingle.diff uploaded by file (License 5000) 
  funcs.diff uploaded by file (License 5000) 
  formats.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  bridges.diff uploaded by file (License 5000) 
  mf-codecs-2.diff uploaded by file (License 5000) 
  mf-app_fax.diff uploaded by file (License 5000) 
  mf-apps-3.diff uploaded by file (License 5000) 
  media-formats-3.diff uploaded by file (License 5000) 
ASTERISK-23715
  rb3713.patch uploaded by coreyfarrell (License 5909)
  rb3689.patch uploaded by mjordan (License 6283)
  
ASTERISK-23957
  rb3722.patch uploaded by mjordan (License 6283) 
  mf-attributes-3.diff uploaded by file (License 5000) 
ASTERISK-23958
Tested by: jrose
  rb3822.patch uploaded by coreyfarrell (License 5909) 
  rb3800.patch uploaded by jrose (License 6182)
  chan_sip.diff uploaded by mjordan (License 6283) 
  rb3747.patch uploaded by jrose (License 6182)
ASTERISK-23959 #close
Tested by: sgriepentrog, mjordan, coreyfarrell
  sip_cleanup.diff uploaded by opticron (License 6273)
  chan_sip_caps.diff uploaded by mjordan (License 6283) 
  rb3751.patch uploaded by coreyfarrell (License 5909) 
  chan_sip-3.diff uploaded by file (License 5000) 
ASTERISK-23960 #close
Tested by: opticron
  direct_media.diff uploaded by opticron (License 6273) 
  pjsip-direct-media.diff uploaded by file (License 5000) 
  format_cap_remove.diff uploaded by opticron (License 6273) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  chan_pjsip-2.diff uploaded by file (License 5000) 
ASTERISK-23966 #close
Tested by: rmudgett
  rb3803.patch uploaded by rmudgetti (License 5621)
  chan_dahdi.diff uploaded by file (License 5000) 
  
ASTERISK-24064 #close
Tested by: coreyfarrell, mjordan, opticron, file, rmudgett, sgriepentrog, jrose
  rb3814.patch uploaded by rmudgett (License 5621) 
  moh_cleanup.diff uploaded by opticron (License 6273) 
  bridge_leak.diff uploaded by opticron (License 6273) 
  translate.diff uploaded by file (License 5000) 
  rb3795.patch uploaded by rmudgett (License 5621) 
  tls_fix.diff uploaded by mjordan (License 6283) 
  fax-mf-fix-2.diff uploaded by file (License 5000) 
  rtp_transfer_stuff uploaded by mjordan (License 6283) 
  rb3787.patch uploaded by rmudgett (License 5621) 
  media-formats-explicit-translate-format-3.diff uploaded by file (License 5000) 
  format_cache_case_fix.diff uploaded by opticron (License 6273) 
  rb3774.patch uploaded by rmudgett (License 5621) 
  rb3775.patch uploaded by rmudgett (License 5621) 
  rtp_engine_fix.diff uploaded by opticron (License 6273) 
  rtp_crash_fix.diff uploaded by opticron (License 6273) 
  rb3753.patch uploaded by mjordan (License 6283) 
  rb3750.patch uploaded by mjordan (License 6283) 
  rb3748.patch uploaded by rmudgett (License 5621) 
  media_format_fixes.diff uploaded by opticron (License 6273) 
  rb3740.patch uploaded by mjordan (License 6283) 
  rb3739.patch uploaded by mjordan (License 6283) 
  rb3734.patch uploaded by mjordan (License 6283) 
  rb3689.patch uploaded by mjordan (License 6283) 
  rb3674.patch uploaded by coreyfarrell (License 5909) 
  rb3671.patch uploaded by coreyfarrell (License 5909) 
  rb3667.patch uploaded by coreyfarrell (License 5909) 
  rb3665.patch uploaded by mjordan (License 6283) 
  rb3625.patch uploaded by coreyfarrell (License 5909) 
  rb3602.patch uploaded by coreyfarrell (License 5909) 
  format_compatibility-2.diff uploaded by file (License 5000) 
  core.diff uploaded by file (License 5000) 
  
git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@419044 65c4cc65-6c06-0410-ace0-fbb531ad65f3
											
										 
											2014-07-20 22:06:33 +00:00
										 |  |  | 	} else if ((f->frametype == AST_FRAME_VOICE) && | 
					
						
							|  |  |  | 		(ast_format_cmp(f->subclass.format, ast_format_slin) == AST_FORMAT_CMP_EQUAL)) { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		return t38_gateway_rx(&p->t38_gw_state, f->data.ptr, f->samples); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief gather data and clean up after gateway ends
 | 
					
						
							|  |  |  |  * \param s fax session*/ | 
					
						
							|  |  |  | static void spandsp_fax_gateway_cleanup(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 	t38_stats_t t38_stats; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t38_gateway_get_transfer_statistics(&p->t38_gw_state, &t38_stats); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->details->option.ecm = t38_stats.error_correcting_mode ? AST_FAX_OPTFLAG_TRUE : AST_FAX_OPTFLAG_FALSE; | 
					
						
							|  |  |  | 	s->details->pages_transferred = t38_stats.pages_transferred; | 
					
						
							|  |  |  | 	ast_string_field_build(s->details, transfer_rate, "%d", t38_stats.bit_rate); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief */ | 
					
						
							|  |  |  | static int spandsp_fax_start(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->state = AST_FAX_STATE_OPEN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		return spandsp_fax_gateway_start(s); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	if (p->ist38) { | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20080725
 | 
					
						
							|  |  |  | 		/* for spandsp shaphots 0.0.6 and higher */ | 
					
						
							|  |  |  | 		p->t30_state = &p->t38_state.t30; | 
					
						
							|  |  |  | 		p->t38_core_state = &p->t38_state.t38_fe.t38; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		/* for spandsp releases 0.0.5 */ | 
					
						
							|  |  |  | 		p->t30_state = &p->t38_state.t30_state; | 
					
						
							|  |  |  | 		p->t38_core_state = &p->t38_state.t38; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20080725
 | 
					
						
							|  |  |  | 		/* for spandsp shaphots 0.0.6 and higher */ | 
					
						
							|  |  |  | 		p->t30_state = &p->fax_state.t30; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		/* for spandsp release 0.0.5 */ | 
					
						
							|  |  |  | 		p->t30_state = &p->fax_state.t30_state; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	set_logging(&p->t30_state->logging, s->details); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* set some parameters */ | 
					
						
							|  |  |  | 	set_local_info(p->t30_state, s->details); | 
					
						
							|  |  |  | 	set_file(p->t30_state, s->details); | 
					
						
							|  |  |  | 	set_ecm(p->t30_state, s->details); | 
					
						
							| 
									
										
										
										
											2012-01-23 19:22:11 +00:00
										 |  |  | 	t30_set_supported_modems(p->t30_state, spandsp_modems(s->details)); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* perhaps set_transmit_on_idle() should be called */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t30_set_phase_e_handler(p->t30_state, t30_phase_e_handler, s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* set T.38 parameters */ | 
					
						
							|  |  |  | 	if (p->ist38) { | 
					
						
							|  |  |  | 		set_logging(&p->t38_core_state->logging, s->details); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t38_set_max_datagram_size(p->t38_core_state, s->details->their_t38_parameters.max_ifp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (s->details->their_t38_parameters.fill_bit_removal) { | 
					
						
							|  |  |  | 			t38_set_fill_bit_removal(p->t38_core_state, TRUE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (s->details->their_t38_parameters.transcoding_mmr) { | 
					
						
							|  |  |  | 			t38_set_mmr_transcoding(p->t38_core_state, TRUE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (s->details->their_t38_parameters.transcoding_jbig) { | 
					
						
							|  |  |  | 			t38_set_jbig_transcoding(p->t38_core_state, TRUE); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		/* have the fax stack generate silence if it has no data to send */ | 
					
						
							|  |  |  | 		fax_set_transmit_on_idle(&p->fax_state, 1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* start the timer */ | 
					
						
							|  |  |  | 	if (ast_timer_set_rate(p->timer, SPANDSP_FAX_TIMER_RATE)) { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_log(LOG_ERROR, "FAX session '%u' error setting rate on timing source.\n", s->id); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->state = AST_FAX_STATE_ACTIVE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief */ | 
					
						
							|  |  |  | static int spandsp_fax_cancel(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		p->isdone = 1; | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	t30_terminate(p->t30_state); | 
					
						
							|  |  |  | 	p->isdone = 1; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief */ | 
					
						
							|  |  |  | static int spandsp_fax_switch_to_t38(struct ast_fax_session *s) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* prevent the phase E handler from running, this is not a real termination */ | 
					
						
							|  |  |  | 	t30_set_phase_e_handler(p->t30_state, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	t30_terminate(p->t30_state); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s->details->option.switch_to_t38 = 1; | 
					
						
							|  |  |  | 	ast_atomic_fetchadd_int(&p->stats->switched, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	p->ist38 = 1; | 
					
						
							|  |  |  | 	p->stats = &spandsp_global_stats.t38; | 
					
						
							|  |  |  | 	spandsp_fax_start(s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief */ | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_capabilities(int fd) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	ast_cli(fd, "SEND RECEIVE T.38 G.711 GATEWAY\n\n"); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	return  CLI_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief */ | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_session(struct ast_fax_session *s, int fd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ao2_lock(s); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	if (s->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 		struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %u\n", "session", s->id); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %s\n", "operation", "Gateway"); | 
					
						
							|  |  |  | 		ast_cli(fd, "%-22s : %s\n", "state", ast_fax_state_to_str(s->state)); | 
					
						
							|  |  |  | 		if (s->state != AST_FAX_STATE_UNINITIALIZED) { | 
					
						
							|  |  |  | 			t38_stats_t stats; | 
					
						
							|  |  |  | 			t38_gateway_get_transfer_statistics(&p->t38_gw_state, &stats); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %s\n", "ECM Mode", stats.error_correcting_mode ? "Yes" : "No"); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Data Rate", stats.bit_rate); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Page Number", stats.pages_transferred + 1); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 	} else if (s->details->caps & AST_FAX_TECH_V21_DETECT) { | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %u\n", "session", s->id); | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %s\n", "operation", "V.21 Detect"); | 
					
						
							|  |  |  | 		ast_cli(fd, "%-22s : %s\n", "state", ast_fax_state_to_str(s->state)); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2011-12-28 18:59:16 +00:00
										 |  |  | 		struct spandsp_pvt *p = s->tech_pvt; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-09 22:49:26 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %u\n", "session", s->id); | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 		ast_cli(fd, "%-22s : %s\n", "operation", (s->details->caps & AST_FAX_TECH_RECEIVE) ? "Receive" : "Transmit"); | 
					
						
							|  |  |  | 		ast_cli(fd, "%-22s : %s\n", "state", ast_fax_state_to_str(s->state)); | 
					
						
							|  |  |  | 		if (s->state != AST_FAX_STATE_UNINITIALIZED) { | 
					
						
							|  |  |  | 			t30_stats_t stats; | 
					
						
							|  |  |  | 			t30_get_transfer_statistics(p->t30_state, &stats); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %s\n", "Last Status", t30_completion_code_to_str(stats.current_status)); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %s\n", "ECM Mode", stats.error_correcting_mode ? "Yes" : "No"); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Data Rate", stats.bit_rate); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %dx%d\n", "Image Resolution", stats.x_resolution, stats.y_resolution); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %d\n", "Page Number", ((s->details->caps & AST_FAX_TECH_RECEIVE) ? stats.pages_rx : stats.pages_tx) + 1); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %d\n", "Page Number", stats.pages_transferred + 1); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %s\n", "File Name", s->details->caps & AST_FAX_TECH_RECEIVE ? p->t30_state->rx_file : p->t30_state->tx_file); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "\nData Statistics:\n"); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %d\n", "Tx Pages", stats.pages_tx); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Rx Pages", stats.pages_rx); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %d\n", "Tx Pages", (s->details->caps & AST_FAX_TECH_SEND) ? stats.pages_transferred : 0); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Rx Pages", (s->details->caps & AST_FAX_TECH_RECEIVE) ? stats.pages_transferred : 0); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-06-30 18:22:28 +00:00
										 |  |  | 			ast_cli(fd, "%-22s : %d\n", "Longest Bad Line Run", stats.longest_bad_row_run); | 
					
						
							|  |  |  | 			ast_cli(fd, "%-22s : %d\n", "Total Bad Lines", stats.bad_rows); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	ao2_unlock(s); | 
					
						
							|  |  |  | 	ast_cli(fd, "\n\n"); | 
					
						
							|  |  |  | 	return CLI_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-18 15:49:46 +00:00
										 |  |  | static void spandsp_manager_fax_session(struct mansession *s, | 
					
						
							|  |  |  | 	const char *id_text, struct ast_fax_session *session) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct ast_str *message_string; | 
					
						
							|  |  |  | 	struct spandsp_pvt *span_pvt = session->tech_pvt; | 
					
						
							|  |  |  | 	int res; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	message_string = ast_str_create(128); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!message_string) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_lock(session); | 
					
						
							| 
									
										
										
										
											2014-07-18 19:07:12 +00:00
										 |  |  | 	res = ast_str_append(&message_string, 0, "SessionNumber: %u\r\n", session->id); | 
					
						
							| 
									
										
										
										
											2014-07-18 15:49:46 +00:00
										 |  |  | 	res |= ast_str_append(&message_string, 0, "Operation: %s\r\n", ast_fax_session_operation_str(session)); | 
					
						
							|  |  |  | 	res |= ast_str_append(&message_string, 0, "State: %s\r\n", ast_fax_state_to_str(session->state)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (session->details->caps & AST_FAX_TECH_GATEWAY) { | 
					
						
							|  |  |  | 		t38_stats_t stats; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (session->state == AST_FAX_STATE_UNINITIALIZED) { | 
					
						
							|  |  |  | 			goto skip_cap_additions; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t38_gateway_get_transfer_statistics(&span_pvt->t38_gw_state, &stats); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "ErrorCorrectionMode: %s\r\n", | 
					
						
							|  |  |  | 			stats.error_correcting_mode ? "yes" : "no"); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "DataRate: %d\r\n", | 
					
						
							|  |  |  | 			stats.bit_rate); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PageNumber: %d\r\n", | 
					
						
							|  |  |  | 			stats.pages_transferred + 1); | 
					
						
							|  |  |  | 	} else if (!(session->details->caps & AST_FAX_TECH_V21_DETECT)) { /* caps is SEND/RECEIVE */ | 
					
						
							|  |  |  | 		t30_stats_t stats; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (session->state == AST_FAX_STATE_UNINITIALIZED) { | 
					
						
							|  |  |  | 			goto skip_cap_additions; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t30_get_transfer_statistics(span_pvt->t30_state, &stats); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "ErrorCorrectionMode: %s\r\n", | 
					
						
							|  |  |  | 			stats.error_correcting_mode ? "Yes" : "No"); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "DataRate: %d\r\n", | 
					
						
							|  |  |  | 			stats.bit_rate); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "ImageResolution: %dx%d\r\n", | 
					
						
							|  |  |  | 			stats.x_resolution, stats.y_resolution); | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PageNumber: %d\r\n", | 
					
						
							|  |  |  | 			((session->details->caps & AST_FAX_TECH_RECEIVE) ? stats.pages_rx : stats.pages_tx) + 1); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PageNumber: %d\r\n", | 
					
						
							|  |  |  | 			stats.pages_transferred + 1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "FileName: %s\r\n", | 
					
						
							|  |  |  | 			session->details->caps & AST_FAX_TECH_RECEIVE ? span_pvt->t30_state->rx_file : | 
					
						
							|  |  |  | 			span_pvt->t30_state->tx_file); | 
					
						
							|  |  |  | #if SPANDSP_RELEASE_DATE >= 20090220
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PagesTransmitted: %d\r\n", | 
					
						
							|  |  |  | 			stats.pages_tx); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PagesReceived: %d\r\n", | 
					
						
							|  |  |  | 			stats.pages_rx); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PagesTransmitted: %d\r\n", | 
					
						
							|  |  |  | 			(session->details->caps & AST_FAX_TECH_SEND) ? stats.pages_transferred : 0); | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "PagesReceived: %d\r\n", | 
					
						
							|  |  |  | 			(session->details->caps & AST_FAX_TECH_RECEIVE) ? stats.pages_transferred : 0); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		res |= ast_str_append(&message_string, 0, "TotalBadLines: %d\r\n", | 
					
						
							|  |  |  | 			stats.bad_rows); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | skip_cap_additions: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ao2_unlock(session); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (res < 0) { | 
					
						
							|  |  |  | 		/* One or more of the ast_str_append attempts failed, cancel the message */ | 
					
						
							|  |  |  | 		ast_free(message_string); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	astman_append(s, "Event: FAXSession\r\n" | 
					
						
							|  |  |  | 		"%s" | 
					
						
							|  |  |  | 		"%s" | 
					
						
							|  |  |  | 		"\r\n", | 
					
						
							|  |  |  | 		id_text, | 
					
						
							|  |  |  | 		ast_str_buffer(message_string)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_free(message_string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief */ | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_stats(int fd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_mutex_lock(&spandsp_global_stats.lock); | 
					
						
							|  |  |  | 	ast_cli(fd, "\n%-20.20s\n", "Spandsp G.711"); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Success", spandsp_global_stats.g711.success); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Switched to T.38", spandsp_global_stats.g711.switched); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Call Dropped", spandsp_global_stats.g711.call_dropped); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "No FAX", spandsp_global_stats.g711.nofax); | 
					
						
							| 
									
										
										
										
											2011-04-22 14:49:47 +00:00
										 |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Negotiation Failed", spandsp_global_stats.g711.neg_failed); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Train Failure", spandsp_global_stats.g711.failed_to_train); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Retries Exceeded", spandsp_global_stats.g711.retries_exceeded); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Protocol Error", spandsp_global_stats.g711.protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "TX Protocol Error", spandsp_global_stats.g711.tx_protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "RX Protocol Error", spandsp_global_stats.g711.rx_protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "File Error", spandsp_global_stats.g711.file_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Memory Error", spandsp_global_stats.g711.mem_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Unknown Error", spandsp_global_stats.g711.unknown_error); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ast_cli(fd, "\n%-20.20s\n", "Spandsp T.38"); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Success", spandsp_global_stats.t38.success); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Call Dropped", spandsp_global_stats.t38.call_dropped); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "No FAX", spandsp_global_stats.t38.nofax); | 
					
						
							| 
									
										
										
										
											2011-04-22 14:49:47 +00:00
										 |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Negotiation Failed", spandsp_global_stats.t38.neg_failed); | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Train Failure", spandsp_global_stats.t38.failed_to_train); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Retries Exceeded", spandsp_global_stats.t38.retries_exceeded); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Protocol Error", spandsp_global_stats.t38.protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "TX Protocol Error", spandsp_global_stats.t38.tx_protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "RX Protocol Error", spandsp_global_stats.t38.rx_protocol_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "File Error", spandsp_global_stats.t38.file_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Memory Error", spandsp_global_stats.t38.mem_error); | 
					
						
							|  |  |  | 	ast_cli(fd, "%-20.20s : %d\n", "Unknown Error", spandsp_global_stats.t38.unknown_error); | 
					
						
							|  |  |  | 	ast_mutex_unlock(&spandsp_global_stats.lock); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return CLI_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-26 14:18:15 +00:00
										 |  |  | /*! \brief Show res_fax_spandsp settings */ | 
					
						
							|  |  |  | static char *spandsp_fax_cli_show_settings(int fd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* no settings at the moment */ | 
					
						
							|  |  |  | 	return CLI_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | /*! \brief unload res_fax_spandsp */ | 
					
						
							|  |  |  | static int unload_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_fax_tech_unregister(&spandsp_fax_tech); | 
					
						
							|  |  |  | 	ast_mutex_destroy(&spandsp_global_stats.lock); | 
					
						
							|  |  |  | 	return AST_MODULE_LOAD_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*! \brief load res_fax_spandsp */ | 
					
						
							|  |  |  | static int load_module(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ast_mutex_init(&spandsp_global_stats.lock); | 
					
						
							|  |  |  | 	spandsp_fax_tech.module = ast_module_info->self; | 
					
						
							|  |  |  | 	if (ast_fax_tech_register(&spandsp_fax_tech) < 0) { | 
					
						
							|  |  |  | 		ast_log(LOG_ERROR, "failed to register FAX technology\n"); | 
					
						
							|  |  |  | 		return AST_MODULE_LOAD_DECLINE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* prevent logging to stderr */ | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #if SPANDSP_RELEASE_DATE >= 20120902
 | 
					
						
							|  |  |  | 	/* for spandsp shaphots 3.0.0 and higher */ | 
					
						
							|  |  |  | 	span_set_message_handler(NULL, NULL); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* for spandsp release 0.0.6 */ | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 	span_set_message_handler(NULL); | 
					
						
							| 
									
										
										
										
											2021-11-08 18:30:00 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2010-03-02 23:11:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return AST_MODULE_LOAD_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "Spandsp G.711 and T.38 FAX Technologies", | 
					
						
							| 
									
										
										
										
											2015-05-05 20:49:04 -04:00
										 |  |  | 	.support_level = AST_MODULE_SUPPORT_EXTENDED, | 
					
						
							|  |  |  | 	.load = load_module, | 
					
						
							|  |  |  | 	.unload = unload_module, | 
					
						
							| 
									
										
										
										
											2017-11-19 17:30:49 -05:00
										 |  |  | 	.enhances = "res_fax", | 
					
						
							| 
									
										
										
										
											2015-05-05 20:49:04 -04:00
										 |  |  | ); |