mirror of
				https://github.com/asterisk/asterisk.git
				synced 2025-10-25 22:18:07 +00:00 
			
		
		
		
	Some providers require a multiple of 20 for the maxptime or fail to complete calls,
e.g. Vivo in Brazil. To increase compatibility, only multiples of 20 are now used.
Resolves: #260
(cherry picked from commit ca1ed84820)
		
	
		
			
				
	
	
		
			1009 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1009 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Asterisk -- An open source telephony toolkit.
 | |
|  *
 | |
|  * Copyright (C) 2014, Digium, Inc.
 | |
|  *
 | |
|  * Joshua Colp <jcolp@digium.com>
 | |
|  *
 | |
|  * See http://www.asterisk.org for more information about
 | |
|  * the Asterisk project. Please do not directly contact
 | |
|  * any of the maintainers of this project for assistance;
 | |
|  * the project provides a web site, mailing lists and IRC
 | |
|  * channels for your use.
 | |
|  *
 | |
|  * This program is free software, distributed under the terms of
 | |
|  * the GNU General Public License Version 2. See the LICENSE file
 | |
|  * at the top of the source tree.
 | |
|  */
 | |
| 
 | |
| /*! \file
 | |
|  *
 | |
|  * \brief Built-in supported codecs
 | |
|  *
 | |
|  * \author Joshua Colp <jcolp@digium.com>
 | |
|  */
 | |
| 
 | |
| /*** MODULEINFO
 | |
| 	<support_level>core</support_level>
 | |
|  ***/
 | |
| 
 | |
| #include "asterisk.h"
 | |
| 
 | |
| #include "asterisk/ilbc.h"
 | |
| #include "asterisk/logger.h"
 | |
| #include "asterisk/astobj2.h"
 | |
| #include "asterisk/codec.h"
 | |
| #include "asterisk/format.h"
 | |
| #include "asterisk/format_cache.h"
 | |
| #include "asterisk/frame.h"
 | |
| #include "asterisk/smoother.h"
 | |
| 
 | |
| int __ast_codec_register_with_format(struct ast_codec *codec, const char *format_name,
 | |
| 	struct ast_module *mod);
 | |
| 
 | |
| enum frame_type {
 | |
| 	TYPE_HIGH,     /* 0x0 */
 | |
| 	TYPE_LOW,      /* 0x1 */
 | |
| 	TYPE_SILENCE,  /* 0x2 */
 | |
| 	TYPE_DONTSEND  /* 0x3 */
 | |
| };
 | |
| 
 | |
| #define TYPE_MASK 0x3
 | |
| 
 | |
| static int g723_len(unsigned char buf)
 | |
| {
 | |
| 	enum frame_type type = buf & TYPE_MASK;
 | |
| 
 | |
| 	switch(type) {
 | |
| 	case TYPE_DONTSEND:
 | |
| 		return 0;
 | |
| 		break;
 | |
| 	case TYPE_SILENCE:
 | |
| 		return 4;
 | |
| 		break;
 | |
| 	case TYPE_HIGH:
 | |
| 		return 24;
 | |
| 		break;
 | |
| 	case TYPE_LOW:
 | |
| 		return 20;
 | |
| 		break;
 | |
| 	default:
 | |
| 		ast_log(LOG_WARNING, "Badly encoded frame (%u)\n", type);
 | |
| 	}
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| static int g723_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	unsigned char *buf = frame->data.ptr;
 | |
| 	int pos = 0, samples = 0, res;
 | |
| 
 | |
| 	while(pos < frame->datalen) {
 | |
| 		res = g723_len(buf[pos]);
 | |
| 		if (res <= 0)
 | |
| 			break;
 | |
| 		samples += 240;
 | |
| 		pos += res;
 | |
| 	}
 | |
| 
 | |
| 	return samples;
 | |
| }
 | |
| 
 | |
| static int g723_length(unsigned int samples)
 | |
| {
 | |
| 	return (samples / 240) * 20;
 | |
| }
 | |
| 
 | |
| static struct ast_codec g723 = {
 | |
| 	.name = "g723",
 | |
| 	.description = "G.723.1",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 30,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 30,
 | |
| 	.minimum_bytes = 20,
 | |
| 	.samples_count = g723_samples,
 | |
| 	.get_length = g723_length,
 | |
| 	.quality = 20,
 | |
| };
 | |
| 
 | |
| static int codec2_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return 160 * (frame->datalen / 6);
 | |
| }
 | |
| 
 | |
| static int codec2_length(unsigned int samples)
 | |
| {
 | |
| 	return (samples / 160) * 6;
 | |
| }
 | |
| 
 | |
| static struct ast_codec codec2 = {
 | |
| 	.name = "codec2",
 | |
| 	.description = "Codec 2",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 6,
 | |
| 	.samples_count = codec2_samples,
 | |
| 	.get_length = codec2_length,
 | |
| 	.smooth = 1,
 | |
| };
 | |
| 
 | |
| static int none_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen;
 | |
| }
 | |
| 
 | |
| static int none_length(unsigned int samples) {
 | |
| 	return samples;
 | |
| }
 | |
| 
 | |
| static struct ast_codec none = {
 | |
| 	.name = "none",
 | |
| 	.description = "<Null> codec",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000, /* This must have some sample rate to prevent divide by 0 */
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 140,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 20,
 | |
| 	.samples_count = none_samples,
 | |
| 	.get_length = none_length,
 | |
| };
 | |
| 
 | |
| static int ulaw_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen;
 | |
| }
 | |
| 
 | |
| static int ulaw_length(unsigned int samples)
 | |
| {
 | |
| 	return samples;
 | |
| }
 | |
| 
 | |
| static struct ast_codec ulaw = {
 | |
| 	.name = "ulaw",
 | |
| 	.description = "G.711 u-law",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 140,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 80,
 | |
| 	.samples_count = ulaw_samples,
 | |
| 	.get_length = ulaw_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 100, /* We are the gold standard. */
 | |
| };
 | |
| 
 | |
| static struct ast_codec alaw = {
 | |
| 	.name = "alaw",
 | |
| 	.description = "G.711 a-law",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 140,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 80,
 | |
| 	.samples_count = ulaw_samples,
 | |
| 	.get_length = ulaw_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 100, /* Just as good as ulaw */
 | |
| };
 | |
| 
 | |
| static int gsm_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return 160 * (frame->datalen / 33);
 | |
| }
 | |
| 
 | |
| static int gsm_length(unsigned int samples)
 | |
| {
 | |
| 	return (samples / 160) * 33;
 | |
| }
 | |
| 
 | |
| static struct ast_codec gsm = {
 | |
| 	.name = "gsm",
 | |
| 	.description = "GSM",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 33,
 | |
| 	.samples_count = gsm_samples,
 | |
| 	.get_length = gsm_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 60,
 | |
| };
 | |
| 
 | |
| static int g726_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen * 2;
 | |
| }
 | |
| 
 | |
| static int g726_length(unsigned int samples)
 | |
| {
 | |
| 	return samples / 2;
 | |
| }
 | |
| 
 | |
| static struct ast_codec g726rfc3551 = {
 | |
| 	.name = "g726",
 | |
| 	.description = "G.726 RFC3551",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 40,
 | |
| 	.samples_count = g726_samples,
 | |
| 	.get_length = g726_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 85,
 | |
| };
 | |
| 
 | |
| static struct ast_codec g726aal2 = {
 | |
| 	.name = "g726aal2",
 | |
| 	.description = "G.726 AAL2",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 40,
 | |
| 	.samples_count = g726_samples,
 | |
| 	.get_length = g726_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 85,
 | |
| };
 | |
| 
 | |
| static struct ast_codec adpcm = {
 | |
| 	.name = "adpcm",
 | |
| 	.description = "Dialogic ADPCM",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 40,
 | |
| 	.samples_count = g726_samples,
 | |
| 	.get_length = g726_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 80,
 | |
| };
 | |
| 
 | |
| static int slin_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen / 2;
 | |
| }
 | |
| 
 | |
| static int slin_length(unsigned int samples)
 | |
| {
 | |
| 	return samples * 2;
 | |
| }
 | |
| 
 | |
| static struct ast_codec slin8 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 160,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 115, /* Better than ulaw */
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin12 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (12kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 12000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 240,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 116,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin16 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (16kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 16000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 320,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 117,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin24 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (24kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 24000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 480,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 118,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin32 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (32kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 32000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 640,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 119,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin44 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (44kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 44100,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 882,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 120,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin48 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (48kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 48000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 960,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 121,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin96 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (96kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 96000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 1920,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 122,
 | |
| };
 | |
| 
 | |
| static struct ast_codec slin192 = {
 | |
| 	.name = "slin",
 | |
| 	.description = "16 bit Signed Linear PCM (192kHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 192000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 3840,
 | |
| 	.samples_count = slin_samples,
 | |
| 	.get_length = slin_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_BE | AST_SMOOTHER_FLAG_FORCED,
 | |
| 	.quality = 123,
 | |
| };
 | |
| 
 | |
| static int lpc10_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	int samples = 22 * 8;
 | |
| 
 | |
| 	/* assumes that the RTP packet contains one LPC10 frame */
 | |
| 	samples += (((char *)(frame->data.ptr))[7] & 0x1) * 8;
 | |
| 
 | |
| 	return samples;
 | |
| }
 | |
| 
 | |
| static struct ast_codec lpc10 = {
 | |
| 	.name = "lpc10",
 | |
| 	.description = "LPC10",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 20,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 7,
 | |
| 	.samples_count = lpc10_samples,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 25,
 | |
| };
 | |
| 
 | |
| static int g729_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen * 8;
 | |
| }
 | |
| 
 | |
| static int g729_length(unsigned int samples)
 | |
| {
 | |
| 	return samples / 8;
 | |
| }
 | |
| 
 | |
| static struct ast_codec g729a = {
 | |
| 	.name = "g729",
 | |
| 	.description = "G.729A",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 220,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 10,
 | |
| 	.samples_count = g729_samples,
 | |
| 	.get_length = g729_length,
 | |
| 	.smooth = 1,
 | |
| 	.smoother_flags = AST_SMOOTHER_FLAG_G729,
 | |
| 	.quality = 20,
 | |
| };
 | |
| 
 | |
| static unsigned char get_n_bits_at(unsigned char *data, int n, int bit)
 | |
| {
 | |
| 	int byte = bit / 8;       /* byte containing first bit */
 | |
| 	int rem = 8 - (bit % 8);  /* remaining bits in first byte */
 | |
| 	unsigned char ret = 0;
 | |
| 
 | |
| 	if (n <= 0 || n > 8)
 | |
| 		return 0;
 | |
| 
 | |
| 	if (rem < n) {
 | |
| 		ret = (data[byte] << (n - rem));
 | |
| 		ret |= (data[byte + 1] >> (8 - n + rem));
 | |
| 	} else {
 | |
| 		ret = (data[byte] >> (rem - n));
 | |
| 	}
 | |
| 
 | |
| 	return (ret & (0xff >> (8 - n)));
 | |
| }
 | |
| 
 | |
| static int speex_get_wb_sz_at(unsigned char *data, int len, int bit)
 | |
| {
 | |
| 	static const int SpeexWBSubModeSz[] = {
 | |
| 		4, 36, 112, 192,
 | |
| 		352, 0, 0, 0 };
 | |
| 	int off = bit;
 | |
| 	unsigned char c;
 | |
| 
 | |
| 	/* skip up to two wideband frames */
 | |
| 	if (((len * 8 - off) >= 5) &&
 | |
| 		get_n_bits_at(data, 1, off)) {
 | |
| 		c = get_n_bits_at(data, 3, off + 1);
 | |
| 		off += SpeexWBSubModeSz[c];
 | |
| 
 | |
| 		if (((len * 8 - off) >= 5) &&
 | |
| 			get_n_bits_at(data, 1, off)) {
 | |
| 			c = get_n_bits_at(data, 3, off + 1);
 | |
| 			off += SpeexWBSubModeSz[c];
 | |
| 
 | |
| 			if (((len * 8 - off) >= 5) &&
 | |
| 				get_n_bits_at(data, 1, off)) {
 | |
| 				ast_log(LOG_WARNING, "Encountered corrupt speex frame; too many wideband frames in a row.\n");
 | |
| 				return -1;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	}
 | |
| 	return off - bit;
 | |
| }
 | |
| 
 | |
| static int speex_samples(unsigned char *data, int len)
 | |
| {
 | |
| 	static const int SpeexSubModeSz[] = {
 | |
| 		5, 43, 119, 160,
 | |
| 		220, 300, 364, 492,
 | |
| 		79, 0, 0, 0,
 | |
| 		0, 0, 0, 0 };
 | |
| 	static const int SpeexInBandSz[] = {
 | |
| 		1, 1, 4, 4,
 | |
| 		4, 4, 4, 4,
 | |
| 		8, 8, 16, 16,
 | |
| 		32, 32, 64, 64 };
 | |
| 	int bit = 0;
 | |
| 	int cnt = 0;
 | |
| 	int off;
 | |
| 	unsigned char c;
 | |
| 
 | |
| 	while ((len * 8 - bit) >= 5) {
 | |
| 		/* skip wideband frames */
 | |
| 		off = speex_get_wb_sz_at(data, len, bit);
 | |
| 		if (off < 0)  {
 | |
| 			ast_log(LOG_WARNING, "Had error while reading wideband frames for speex samples\n");
 | |
| 			break;
 | |
| 		}
 | |
| 		bit += off;
 | |
| 
 | |
| 		if ((len * 8 - bit) < 5)
 | |
| 			break;
 | |
| 
 | |
| 		/* get control bits */
 | |
| 		c = get_n_bits_at(data, 5, bit);
 | |
| 		bit += 5;
 | |
| 
 | |
| 		if (c == 15) {
 | |
| 			/* terminator */
 | |
| 			break;
 | |
| 		} else if (c == 14) {
 | |
| 			/* in-band signal; next 4 bits contain signal id */
 | |
| 			c = get_n_bits_at(data, 4, bit);
 | |
| 			bit += 4;
 | |
| 			bit += SpeexInBandSz[c];
 | |
| 		} else if (c == 13) {
 | |
| 			/* user in-band; next 4 bits contain msg len */
 | |
| 			c = get_n_bits_at(data, 4, bit);
 | |
| 			bit += 4;
 | |
| 			/* after which it's 5-bit signal id + c bytes of data */
 | |
| 			bit += 5 + c * 8;
 | |
| 		} else if (c > 8) {
 | |
| 			/* unknown */
 | |
| 			ast_log(LOG_WARNING, "Unknown speex control frame %d\n", c);
 | |
| 			break;
 | |
| 		} else {
 | |
| 			/* skip number bits for submode (less the 5 control bits) */
 | |
| 			bit += SpeexSubModeSz[c] - 5;
 | |
| 			cnt += 160; /* new frame */
 | |
| 		}
 | |
| 	}
 | |
| 	return cnt;
 | |
| }
 | |
| 
 | |
| static int speex8_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return speex_samples(frame->data.ptr, frame->datalen);
 | |
| }
 | |
| 
 | |
| static struct ast_codec speex8 = {
 | |
| 	.name = "speex",
 | |
| 	.description = "SpeeX",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 10,
 | |
| 	.samples_count = speex8_samples,
 | |
| 	.quality = 40,
 | |
| };
 | |
| 
 | |
| static int speex16_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return 2 * speex_samples(frame->data.ptr, frame->datalen);
 | |
| }
 | |
| 
 | |
| static struct ast_codec speex16 = {
 | |
| 	.name = "speex",
 | |
| 	.description = "SpeeX 16khz",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 16000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 10,
 | |
| 	.samples_count = speex16_samples,
 | |
| 	.quality = 40,
 | |
| };
 | |
| 
 | |
| static int speex32_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return 4 * speex_samples(frame->data.ptr, frame->datalen);
 | |
| }
 | |
| 
 | |
| static struct ast_codec speex32 = {
 | |
| 	.name = "speex",
 | |
| 	.description = "SpeeX 32khz",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 32000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 10,
 | |
| 	.samples_count = speex32_samples,
 | |
| 	.quality = 40,
 | |
| };
 | |
| 
 | |
| static int ilbc_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	struct ilbc_attr *attr = ast_format_get_attribute_data(frame->subclass.format);
 | |
| 	const unsigned int mode = attr ? attr->mode : 30;
 | |
| 	const unsigned int samples_per_frame = mode * ast_format_get_sample_rate(frame->subclass.format) / 1000;
 | |
| 	const unsigned int octets_per_frame = (mode == 20) ? 38 : 50;
 | |
| 
 | |
| 	return samples_per_frame * frame->datalen / octets_per_frame;
 | |
| }
 | |
| 
 | |
| static struct ast_codec ilbc = {
 | |
| 	.name = "ilbc",
 | |
| 	.description = "iLBC",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 300,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 38,
 | |
| 	.samples_count = ilbc_samples,
 | |
| 	.smooth = 0,
 | |
| 	.quality = 45,
 | |
| };
 | |
| 
 | |
| static struct ast_codec g722 = {
 | |
| 	.name = "g722",
 | |
| 	.description = "G722",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 16000,
 | |
| 	.minimum_ms = 10,
 | |
| 	.maximum_ms = 140,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 80,
 | |
| 	.samples_count = g726_samples,
 | |
| 	.get_length = g726_length,
 | |
| 	.smooth = 1,
 | |
| 	.quality = 110, /* In theory, better than ulaw */
 | |
| };
 | |
| 
 | |
| static int siren7_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return frame->datalen * (16000 / 4000);
 | |
| }
 | |
| 
 | |
| static int siren7_length(unsigned int samples)
 | |
| {
 | |
| 	return samples / (16000 / 4000);
 | |
| }
 | |
| 
 | |
| static struct ast_codec siren7 = {
 | |
| 	.name = "siren7",
 | |
| 	.description = "ITU G.722.1 (Siren7, licensed from Polycom)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 16000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 80,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 80,
 | |
| 	.samples_count = siren7_samples,
 | |
| 	.get_length = siren7_length,
 | |
| 	.quality = 85,
 | |
| };
 | |
| 
 | |
| static int siren14_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return (int) frame->datalen * ((float) 32000 / 6000);
 | |
| }
 | |
| 
 | |
| static int siren14_length(unsigned int samples)
 | |
| {
 | |
| 	return (int) samples / ((float) 32000 / 6000);;
 | |
| }
 | |
| 
 | |
| static struct ast_codec siren14 = {
 | |
| 	.name = "siren14",
 | |
| 	.description = "ITU G.722.1 Annex C, (Siren14, licensed from Polycom)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 32000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 80,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 120,
 | |
| 	.samples_count = siren14_samples,
 | |
| 	.get_length = siren14_length,
 | |
| 	.quality = 90,
 | |
| };
 | |
| 
 | |
| static int g719_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	return (int) frame->datalen * ((float) 48000 / 8000);
 | |
| }
 | |
| 
 | |
| static int g719_length(unsigned int samples)
 | |
| {
 | |
| 	return (int) samples / ((float) 48000 / 8000);
 | |
| }
 | |
| 
 | |
| static struct ast_codec g719 = {
 | |
| 	.name = "g719",
 | |
| 	.description = "ITU G.719",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 48000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 80,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 160,
 | |
| 	.samples_count = g719_samples,
 | |
| 	.get_length = g719_length,
 | |
| 	.quality = 95,
 | |
| };
 | |
| 
 | |
| static int opus_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	/*
 | |
| 	 * XXX This is likely not at all what's intended from this
 | |
| 	 * callback.  If you have codec_opus.so loaded then this
 | |
| 	 * function is overridden anyway.  However, since opus is
 | |
| 	 * variable bit rate and I cannot extract the calculation code
 | |
| 	 * from the opus library, I am going to punt and assume 20ms
 | |
| 	 * worth of samples.  In testing, this has worked just fine.
 | |
| 	 * Pass through support doesn't seem to care about the value
 | |
| 	 * returned anyway.
 | |
| 	 */
 | |
| 	return ast_format_get_sample_rate(frame->subclass.format) / 50;
 | |
| }
 | |
| 
 | |
| static struct ast_codec opus = {
 | |
| 	.name = "opus",
 | |
| 	.description = "Opus Codec",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 48000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 60,
 | |
| 	.default_ms = 20,
 | |
| 	.samples_count = opus_samples,
 | |
| 	.minimum_bytes = 10,
 | |
| 	.quality = 50,
 | |
| };
 | |
| 
 | |
| static struct ast_codec jpeg = {
 | |
| 	.name = "jpeg",
 | |
| 	.description = "JPEG image",
 | |
| 	.type = AST_MEDIA_TYPE_IMAGE,
 | |
| };
 | |
| 
 | |
| static struct ast_codec png = {
 | |
| 	.name = "png",
 | |
| 	.description = "PNG Image",
 | |
| 	.type = AST_MEDIA_TYPE_IMAGE,
 | |
| };
 | |
| 
 | |
| static struct ast_codec h261 = {
 | |
| 	.name = "h261",
 | |
| 	.description = "H.261 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec h263 = {
 | |
| 	.name = "h263",
 | |
| 	.description = "H.263 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec h263p = {
 | |
| 	.name = "h263p",
 | |
| 	.description = "H.263+ video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec h264 = {
 | |
| 	.name = "h264",
 | |
| 	.description = "H.264 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec h265 = {
 | |
| 	.name = "h265",
 | |
| 	.description = "H.265 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec mpeg4 = {
 | |
| 	.name = "mpeg4",
 | |
| 	.description = "MPEG4 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec vp8 = {
 | |
| 	.name = "vp8",
 | |
| 	.description = "VP8 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec vp9 = {
 | |
| 	.name = "vp9",
 | |
| 	.description = "VP9 video",
 | |
| 	.type = AST_MEDIA_TYPE_VIDEO,
 | |
| 	.sample_rate = 1000,
 | |
| };
 | |
| 
 | |
| static struct ast_codec t140red = {
 | |
| 	.name = "red",
 | |
| 	.description = "T.140 Realtime Text with redundancy",
 | |
| 	.type = AST_MEDIA_TYPE_TEXT,
 | |
| };
 | |
| 
 | |
| static struct ast_codec t140 = {
 | |
| 	.name = "t140",
 | |
| 	.description = "Passthrough T.140 Realtime Text",
 | |
| 	.type = AST_MEDIA_TYPE_TEXT,
 | |
| };
 | |
| 
 | |
| static struct ast_codec t38 = {
 | |
| 	.name = "t38",
 | |
| 	.description = "T.38 UDPTL Fax",
 | |
| 	.type = AST_MEDIA_TYPE_IMAGE,
 | |
| };
 | |
| 
 | |
| static int silk_samples(struct ast_frame *frame)
 | |
| {
 | |
| 	/* XXX This is likely not at all what's intended from this callback. However,
 | |
| 	 * since SILK is variable bit rate, I have no idea how to take a frame of data
 | |
| 	 * and determine the number of samples present. Instead, we base this on the
 | |
| 	 * sample rate of the codec and the expected number of samples to receive in 20ms.
 | |
| 	 * In testing, this has worked just fine.
 | |
| 	 */
 | |
| 	return ast_format_get_sample_rate(frame->subclass.format) / 50;
 | |
| }
 | |
| 
 | |
| static struct ast_codec silk8 = {
 | |
| 	.name = "silk",
 | |
| 	.description = "SILK Codec (8 KHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 8000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 100,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 160,
 | |
| 	.samples_count = silk_samples,
 | |
| };
 | |
| 
 | |
| static struct ast_codec silk12 = {
 | |
| 	.name = "silk",
 | |
| 	.description = "SILK Codec (12 KHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 12000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 100,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 240,
 | |
| 	.samples_count = silk_samples
 | |
| };
 | |
| 
 | |
| static struct ast_codec silk16 = {
 | |
| 	.name = "silk",
 | |
| 	.description = "SILK Codec (16 KHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 16000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 100,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 320,
 | |
| 	.samples_count = silk_samples
 | |
| };
 | |
| 
 | |
| static struct ast_codec silk24 = {
 | |
| 	.name = "silk",
 | |
| 	.description = "SILK Codec (24 KHz)",
 | |
| 	.type = AST_MEDIA_TYPE_AUDIO,
 | |
| 	.sample_rate = 24000,
 | |
| 	.minimum_ms = 20,
 | |
| 	.maximum_ms = 100,
 | |
| 	.default_ms = 20,
 | |
| 	.minimum_bytes = 480,
 | |
| 	.samples_count = silk_samples
 | |
| };
 | |
| 
 | |
| #define CODEC_REGISTER_AND_CACHE(codec) \
 | |
| 	({ \
 | |
| 		int __res_ ## __LINE__ = 0; \
 | |
| 		struct ast_format *__fmt_ ## __LINE__; \
 | |
| 		struct ast_codec *__codec_ ## __LINE__; \
 | |
| 		res |= __ast_codec_register_with_format(&(codec), (codec).name, NULL); \
 | |
| 		__codec_ ## __LINE__ = ast_codec_get((codec).name, (codec).type, (codec).sample_rate); \
 | |
| 		__fmt_ ## __LINE__ = __codec_ ## __LINE__ ? ast_format_create(__codec_ ## __LINE__) : NULL; \
 | |
| 		res |= ast_format_cache_set(__fmt_ ## __LINE__); \
 | |
| 		ao2_ref(__fmt_ ## __LINE__, -1); \
 | |
| 		ao2_ref(__codec_ ## __LINE__, -1); \
 | |
| 		__res_ ## __LINE__; \
 | |
| 	})
 | |
| 
 | |
| #define CODEC_REGISTER_AND_CACHE_NAMED(fmt_name, codec) \
 | |
| 	({ \
 | |
| 		int __res_ ## __LINE__ = 0; \
 | |
| 		struct ast_format *__fmt_ ## __LINE__; \
 | |
| 		struct ast_codec *__codec_ ## __LINE__; \
 | |
| 		res |= __ast_codec_register_with_format(&(codec), fmt_name, NULL); \
 | |
| 		__codec_ ## __LINE__ = ast_codec_get((codec).name, (codec).type, (codec).sample_rate); \
 | |
| 		__fmt_ ## __LINE__ = ast_format_create_named((fmt_name), __codec_ ## __LINE__); \
 | |
| 		res |= ast_format_cache_set(__fmt_ ## __LINE__); \
 | |
| 		ao2_ref(__fmt_ ## __LINE__, -1); \
 | |
| 		ao2_ref(__codec_ ## __LINE__, -1); \
 | |
| 		__res_ ## __LINE__; \
 | |
| 	})
 | |
| 
 | |
| int ast_codec_builtin_init(void)
 | |
| {
 | |
| 	int res = 0;
 | |
| 
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(codec2);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g723);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(ulaw);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(alaw);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(gsm);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g726rfc3551);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g726aal2);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(adpcm);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(slin8);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin12", slin12);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin16", slin16);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin24", slin24);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin32", slin32);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin44", slin44);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin48", slin48);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin96", slin96);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("slin192", slin192);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(lpc10);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g729a);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(speex8);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("speex16", speex16);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("speex32", speex32);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(ilbc);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g722);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(siren7);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(siren14);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(g719);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(opus);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(jpeg);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(png);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(h261);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(h263);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(h263p);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(h264);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(h265);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(mpeg4);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(vp8);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(vp9);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(t140red);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(t140);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(t38);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE(none);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("silk8", silk8);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("silk12", silk12);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("silk16", silk16);
 | |
| 	res |= CODEC_REGISTER_AND_CACHE_NAMED("silk24", silk24);
 | |
| 
 | |
| 	return res;
 | |
| }
 |