Merged revisions 80426 via svnmerge from
[asterisk/asterisk.git] / codecs / codec_g722.c
1 /*
2  * Asterisk -- An open source telephony toolkit.
3  *
4  * Copyright (C) 1999 - 2006, Digium, Inc.
5  *
6  * Matthew Fredrickson <creslin@digium.com>
7  *
8  * Special thanks to Steve Underwood for the implementation
9  * and for doing the 8khz<->g.722 direct translation code.
10  *
11  * See http://www.asterisk.org for more information about
12  * the Asterisk project. Please do not directly contact
13  * any of the maintainers of this project for assistance;
14  * the project provides a web site, mailing lists and IRC
15  * channels for your use.
16  *
17  * This program is free software, distributed under the terms of
18  * the GNU General Public License Version 2. See the LICENSE file
19  * at the top of the source tree.
20  */
21
22 /*! \file
23  *
24  * \brief codec_g722.c - translate between signed linear and ITU G.722-64kbps
25  *
26  * \ingroup codecs
27  */
28
29 #include "asterisk.h"
30
31 ASTERISK_FILE_VERSION(__FILE__, "$Revision$")
32
33 #include <fcntl.h>
34 #include <netinet/in.h>
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <unistd.h>
39
40 #include "asterisk/lock.h"
41 #include "asterisk/logger.h"
42 #include "asterisk/linkedlists.h"
43 #include "asterisk/module.h"
44 #include "asterisk/config.h"
45 #include "asterisk/options.h"
46 #include "asterisk/translate.h"
47 #include "asterisk/channel.h"
48 #include "asterisk/utils.h"
49
50 #define BUFFER_SAMPLES   8096   /* size for the translation buffers */
51 #define BUF_SHIFT       5
52
53 /* Sample frame data */
54
55 #include "g722/g722.h"
56 #include "slin_g722_ex.h"
57 #include "g722_slin_ex.h"
58
59 struct g722_encoder_pvt {
60         g722_encode_state_t g722;
61 };
62
63 struct g722_decoder_pvt {
64         g722_decode_state_t g722;
65 };
66
67 /*! \brief init a new instance of g722_encoder_pvt. */
68 static int lintog722_new(struct ast_trans_pvt *pvt)
69 {
70         struct g722_encoder_pvt *tmp = pvt->pvt;
71
72         g722_encode_init(&tmp->g722, 64000, G722_SAMPLE_RATE_8000);
73
74         return 0;
75 }
76
77 /*! \brief init a new instance of g722_encoder_pvt. */
78 static int g722tolin_new(struct ast_trans_pvt *pvt)
79 {
80         struct g722_decoder_pvt *tmp = pvt->pvt;
81
82         g722_decode_init(&tmp->g722, 64000, G722_SAMPLE_RATE_8000);
83
84         return 0;
85 }
86
87 static int g722tolin_framein(struct ast_trans_pvt *pvt, struct ast_frame *f)
88 {
89         struct g722_decoder_pvt *tmp = pvt->pvt;
90         unsigned char *src = f->data;
91         int16_t *dst = (int16_t *) pvt->outbuf + pvt->samples;
92
93         g722_decode(&tmp->g722, dst, src, f->samples);
94         pvt->samples += f->samples;
95         pvt->datalen += 2 * f->samples;
96
97         return 0;
98 }
99
100 static int lintog722_framein(struct ast_trans_pvt *pvt, struct ast_frame *f)
101 {
102         struct g722_encoder_pvt *tmp = pvt->pvt;
103         int16_t *src = f->data;
104
105         g722_encode(&tmp->g722, (uint8_t*)(&pvt->outbuf[pvt->datalen]), src, f->samples);
106         /* Since G.722 64kbps per second is one bye per sample, all of these
107            calculations are easy */
108         pvt->samples += f->samples;
109         pvt->datalen += f->samples;
110
111         return 0;
112 }
113
114 static struct ast_frame *g722tolin_sample(void)
115 {
116         static struct ast_frame f = {
117                 .frametype = AST_FRAME_VOICE,
118                 .subclass = AST_FORMAT_G722,
119                 .datalen = sizeof(g722_slin_ex),
120                 .samples = sizeof(g722_slin_ex) / sizeof(g722_slin_ex[0]),
121                 .src = __PRETTY_FUNCTION__,
122                 .data = g722_slin_ex,
123         };
124
125         return &f;
126 }
127
128 static struct ast_frame *lintog722_sample (void)
129 {
130         static struct ast_frame f = {
131                 .frametype = AST_FRAME_VOICE,
132                 .subclass = AST_FORMAT_SLINEAR,
133                 .datalen = sizeof(slin_g722_ex),
134                 .samples = sizeof(slin_g722_ex) / sizeof(slin_g722_ex[0]),
135                 .src = __PRETTY_FUNCTION__,
136                 .data = slin_g722_ex,
137         };
138
139         return &f;
140 }
141
142 static struct ast_translator g722tolin = {
143         .name = "g722tolin",
144         .srcfmt = AST_FORMAT_G722,
145         .dstfmt = AST_FORMAT_SLINEAR,
146         .newpvt = g722tolin_new,        /* same for both directions */
147         .framein = g722tolin_framein,
148         .sample = g722tolin_sample,
149         .desc_size = sizeof(struct g722_decoder_pvt),
150         .buffer_samples = BUFFER_SAMPLES,
151         .buf_size = BUFFER_SAMPLES,
152         .plc_samples = 160,
153 };
154
155 static struct ast_translator lintog722 = {
156         .name = "lintog722",
157         .srcfmt = AST_FORMAT_SLINEAR,
158         .dstfmt = AST_FORMAT_G722,
159         .newpvt = lintog722_new,        /* same for both directions */
160         .framein = lintog722_framein,
161         .sample = lintog722_sample,
162         .desc_size = sizeof(struct g722_encoder_pvt),
163         .buffer_samples = BUFFER_SAMPLES,
164         .buf_size = BUFFER_SAMPLES,
165 };
166
167 static void parse_config(int reload)
168 {
169         struct ast_variable *var;
170         struct ast_flags config_flags = { reload ? CONFIG_FLAG_FILEUNCHANGED : 0 };
171         struct ast_config *cfg = ast_config_load("codecs.conf", config_flags);
172
173         if (cfg == NULL || cfg == CONFIG_STATUS_FILEUNCHANGED)
174                 return;
175         for (var = ast_variable_browse(cfg, "plc"); var; var = var->next) {
176                 if (!strcasecmp(var->name, "genericplc")) {
177                         g722tolin.useplc = ast_true(var->value) ? 1 : 0;
178                         ast_verb(3, "codec_g722: %susing generic PLC\n",
179                                         g722tolin.useplc ? "" : "not ");
180                 }
181         }
182         ast_config_destroy(cfg);
183 }
184
185 static int reload(void)
186 {
187         parse_config(1);
188
189         return 0;
190 }
191
192 static int unload_module(void)
193 {
194         int res = 0;
195
196         res |= ast_unregister_translator(&g722tolin);
197         res |= ast_unregister_translator(&lintog722);
198
199         return res;
200 }
201
202 static int load_module(void)
203 {
204         int res = 0;
205
206
207         parse_config(0);
208
209         res |= ast_register_translator(&g722tolin);
210         res |= ast_register_translator(&lintog722);
211
212         if (res)
213                 unload_module();
214
215         return res;
216 }
217
218 AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "ITU G.722-64kbps G722 Transcoder",
219                 .load = load_module,
220                 .unload = unload_module,
221                 .reload = reload,
222                );