Commit some cleanups to the format type code.
[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 static int lin16tog722_new(struct ast_trans_pvt *pvt)
78 {
79         struct g722_encoder_pvt *tmp = pvt->pvt;
80
81         g722_encode_init(&tmp->g722, 64000, 0);
82
83         return 0;
84 }
85
86 /*! \brief init a new instance of g722_encoder_pvt. */
87 static int g722tolin_new(struct ast_trans_pvt *pvt)
88 {
89         struct g722_decoder_pvt *tmp = pvt->pvt;
90
91         g722_decode_init(&tmp->g722, 64000, G722_SAMPLE_RATE_8000);
92
93         return 0;
94 }
95
96 static int g722tolin16_new(struct ast_trans_pvt *pvt)
97 {
98         struct g722_decoder_pvt *tmp = pvt->pvt;
99
100         g722_decode_init(&tmp->g722, 64000, 0);
101
102         return 0;
103 }
104
105 static int g722tolin_framein(struct ast_trans_pvt *pvt, struct ast_frame *f)
106 {
107         struct g722_decoder_pvt *tmp = pvt->pvt;
108         unsigned char *src = f->data;
109         int16_t *dst = (int16_t *) pvt->outbuf + pvt->samples;
110
111         g722_decode(&tmp->g722, dst, src, f->samples);
112         pvt->samples += f->samples;
113         pvt->datalen += 2 * f->samples;
114
115         return 0;
116 }
117
118 static int lintog722_framein(struct ast_trans_pvt *pvt, struct ast_frame *f)
119 {
120         struct g722_encoder_pvt *tmp = pvt->pvt;
121         int16_t *src = f->data;
122
123         g722_encode(&tmp->g722, (uint8_t*)(&pvt->outbuf[pvt->datalen]), src, f->samples);
124         /* Since G.722 64kbps per second is one bye per sample, all of these
125            calculations are easy */
126         pvt->samples += f->samples;
127         pvt->datalen += f->samples;
128
129         return 0;
130 }
131
132 static struct ast_frame *g722tolin_sample(void)
133 {
134         static struct ast_frame f = {
135                 .frametype = AST_FRAME_VOICE,
136                 .subclass = AST_FORMAT_G722,
137                 .datalen = sizeof(g722_slin_ex),
138                 .samples = sizeof(g722_slin_ex) / sizeof(g722_slin_ex[0]),
139                 .src = __PRETTY_FUNCTION__,
140                 .data = g722_slin_ex,
141         };
142
143         return &f;
144 }
145
146 static struct ast_frame *g722tolin16_sample(void)
147 {
148         static struct ast_frame f = {
149                 .frametype = AST_FRAME_VOICE,
150                 .subclass = AST_FORMAT_G722,
151                 .datalen = sizeof(slin_g722_ex),
152                 .samples = sizeof(slin_g722_ex) / sizeof(slin_g722_ex[0]),
153                 .src = __PRETTY_FUNCTION__,
154                 .data = slin_g722_ex,
155         };
156
157         return &f;
158 }
159
160 static struct ast_frame *lintog722_sample (void)
161 {
162         static struct ast_frame f = {
163                 .frametype = AST_FRAME_VOICE,
164                 .subclass = AST_FORMAT_SLINEAR,
165                 .datalen = sizeof(slin_g722_ex),
166                 .samples = sizeof(slin_g722_ex) / sizeof(slin_g722_ex[0]),
167                 .src = __PRETTY_FUNCTION__,
168                 .data = slin_g722_ex,
169         };
170
171         return &f;
172 }
173
174 static struct ast_frame *lin16tog722_sample (void)
175 {
176         static struct ast_frame f = {
177                 .frametype = AST_FRAME_VOICE,
178                 .subclass = AST_FORMAT_SLINEAR16,
179                 .datalen = sizeof(slin_g722_ex),
180                 .samples = sizeof(slin_g722_ex) / sizeof(slin_g722_ex[0]),
181                 .src = __PRETTY_FUNCTION__,
182                 .data = slin_g722_ex,
183         };
184
185         return &f;
186 }
187
188 static struct ast_translator g722tolin = {
189         .name = "g722tolin",
190         .srcfmt = AST_FORMAT_G722,
191         .dstfmt = AST_FORMAT_SLINEAR,
192         .newpvt = g722tolin_new,        /* same for both directions */
193         .framein = g722tolin_framein,
194         .sample = g722tolin_sample,
195         .desc_size = sizeof(struct g722_decoder_pvt),
196         .buffer_samples = BUFFER_SAMPLES,
197         .buf_size = BUFFER_SAMPLES,
198         .plc_samples = 160,
199 };
200
201 static struct ast_translator lintog722 = {
202         .name = "lintog722",
203         .srcfmt = AST_FORMAT_SLINEAR,
204         .dstfmt = AST_FORMAT_G722,
205         .newpvt = lintog722_new,        /* same for both directions */
206         .framein = lintog722_framein,
207         .sample = lintog722_sample,
208         .desc_size = sizeof(struct g722_encoder_pvt),
209         .buffer_samples = BUFFER_SAMPLES,
210         .buf_size = BUFFER_SAMPLES,
211 };
212
213 static struct ast_translator g722tolin16 = {
214         .name = "g722tolin16",
215         .srcfmt = AST_FORMAT_G722,
216         .dstfmt = AST_FORMAT_SLINEAR16,
217         .newpvt = g722tolin16_new,      /* same for both directions */
218         .framein = g722tolin_framein,
219         .sample = g722tolin16_sample,
220         .desc_size = sizeof(struct g722_decoder_pvt),
221         .buffer_samples = BUFFER_SAMPLES,
222         .buf_size = BUFFER_SAMPLES,
223         .plc_samples = 160,
224 };
225
226 static struct ast_translator lin16tog722 = {
227         .name = "lin16tog722",
228         .srcfmt = AST_FORMAT_SLINEAR16,
229         .dstfmt = AST_FORMAT_G722,
230         .newpvt = lin16tog722_new,      /* same for both directions */
231         .framein = lintog722_framein,
232         .sample = lin16tog722_sample,
233         .desc_size = sizeof(struct g722_encoder_pvt),
234         .buffer_samples = BUFFER_SAMPLES,
235         .buf_size = BUFFER_SAMPLES,
236 };
237
238 static int parse_config(int reload)
239 {
240         struct ast_variable *var;
241         struct ast_flags config_flags = { reload ? CONFIG_FLAG_FILEUNCHANGED : 0 };
242         struct ast_config *cfg = ast_config_load("codecs.conf", config_flags);
243
244         if (cfg == NULL)
245                 return -1;
246         if (cfg == CONFIG_STATUS_FILEUNCHANGED)
247                 return 0;
248         for (var = ast_variable_browse(cfg, "plc"); var; var = var->next) {
249                 if (!strcasecmp(var->name, "genericplc")) {
250                         g722tolin.useplc = ast_true(var->value) ? 1 : 0;
251                         ast_verb(3, "codec_g722: %susing generic PLC\n",
252                                         g722tolin.useplc ? "" : "not ");
253                 }
254         }
255         ast_config_destroy(cfg);
256         return 0;
257 }
258
259 static int reload(void)
260 {
261         if (parse_config(1))
262                 return AST_MODULE_LOAD_DECLINE;
263         return AST_MODULE_LOAD_SUCCESS;
264 }
265
266 static int unload_module(void)
267 {
268         int res = 0;
269
270         res |= ast_unregister_translator(&g722tolin);
271         res |= ast_unregister_translator(&lintog722);
272         res |= ast_unregister_translator(&g722tolin16);
273         res |= ast_unregister_translator(&lin16tog722);
274
275         return res;
276 }
277
278 static int load_module(void)
279 {
280         int res = 0;
281
282         if (parse_config(0))
283                 return AST_MODULE_LOAD_DECLINE;
284
285         res |= ast_register_translator(&g722tolin);
286         res |= ast_register_translator(&lintog722);
287         res |= ast_register_translator(&g722tolin16);
288         res |= ast_register_translator(&lin16tog722);
289
290         if (res) {
291                 unload_module();
292                 return AST_MODULE_LOAD_FAILURE;
293         }       
294
295         return AST_MODULE_LOAD_SUCCESS;
296 }
297
298 AST_MODULE_INFO(ASTERISK_GPL_KEY, AST_MODFLAG_DEFAULT, "ITU G.722-64kbps G722 Transcoder",
299                 .load = load_module,
300                 .unload = unload_module,
301                 .reload = reload,
302                );