blob: 6b3364516a3bd1985b5aa7b4da2bffb9e74416e6 [file] [log] [blame]
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +02001#include <stdlib.h>
2#include <unistd.h>
3#include <stdio.h>
4#include <string.h>
5#include <err.h>
6
7#include <osmocom/core/talloc.h>
8#include <osmocom/core/application.h>
9
10#include <openbsc/debug.h>
11#include <openbsc/gsm_data.h>
12#include <openbsc/mgcp.h>
13#include <openbsc/mgcp_internal.h>
14
15#include "bscconfig.h"
16#ifndef BUILD_MGCP_TRANSCODING
17#error "Requires MGCP transcoding enabled (see --enable-mgcp-transcoding)"
18#endif
19
Jacob Erlbeck909fac62014-05-08 14:08:37 +020020#include "openbsc/mgcp_transcode.h"
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +020021
22uint8_t *audio_frame_l16[] = {
23};
24
25struct rtp_packets {
26 float t;
27 int len;
28 char *data;
29};
30
31struct rtp_packets audio_packets_l16[] = {
32 /* RTP: SeqNo=1, TS=160 */
33 {0.020000, 332,
34 "\x80\x0B\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
35 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
36 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
37 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
38 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
39 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
40 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
41 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
42 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
43 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
44 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
45 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
46 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
47 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
48 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
49 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
50 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
51 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
52 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
53 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
54 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
55 },
56};
57
58struct rtp_packets audio_packets_gsm[] = {
59 /* RTP: SeqNo=1, TS=160 */
60 {0.020000, 45,
61 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
62 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
63 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
64 "\xDE"
65 },
66};
67
68struct rtp_packets audio_packets_gsm_invalid_size[] = {
69 /* RTP: SeqNo=1, TS=160 */
70 {0.020000, 41,
71 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
72 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
73 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
74 "\xDE"
75 },
76};
77
78struct rtp_packets audio_packets_gsm_invalid_data[] = {
79 /* RTP: SeqNo=1, TS=160 */
80 {0.020000, 45,
81 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
82 "\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
83 "\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
84 "\xEE"
85 },
86};
87
88struct rtp_packets audio_packets_gsm_invalid_ptype[] = {
89 /* RTP: SeqNo=1, TS=160 */
90 {0.020000, 45,
91 "\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
92 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
93 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
94 "\xDE"
95 },
96};
97
98struct rtp_packets audio_packets_g729[] = {
99 /* RTP: SeqNo=1, TS=160 */
100 {0.020000, 32,
101 "\x80\x12\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
102 "\xAF\xC2\x81\x40\x00\xFA\xCE\xA4\x21\x7C\xC5\xC3\x4F\xA5\x98\xF5"
103 "\xB2\x95\xC4\xAD"
104 },
105};
106
107struct rtp_packets audio_packets_pcma[] = {
108 /* RTP: SeqNo=1, TS=160 */
109 {0.020000, 172,
110 "\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
111 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
112 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
113 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
114 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
115 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
116 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
117 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
118 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
119 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
120 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
121 },
122};
123
124
125
126static int audio_name_to_type(const char *name)
127{
128 if (!strcasecmp(name, "gsm"))
129 return 3;
130#ifdef HAVE_BCG729
131 else if (!strcasecmp(name, "g729"))
132 return 18;
133#endif
134 else if (!strcasecmp(name, "pcma"))
135 return 8;
136 else if (!strcasecmp(name, "l16"))
137 return 11;
138 return -1;
139}
140
141int mgcp_get_trans_frame_size(void *state_, int nsamples, int dst);
142
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200143static int given_configured_endpoint(int in_samples, int out_samples,
144 const char *srcfmt, const char *dstfmt,
145 void **out_ctx, struct mgcp_endpoint **out_endp)
146{
147 int rc;
148 struct mgcp_rtp_end *dst_end;
149 struct mgcp_rtp_end *src_end;
150 struct mgcp_config *cfg;
151 struct mgcp_trunk_config *tcfg;
152 struct mgcp_endpoint *endp;
153
154 cfg = mgcp_config_alloc();
155 tcfg = talloc_zero(cfg, struct mgcp_trunk_config);
156 endp = talloc_zero(tcfg, struct mgcp_endpoint);
157
158
159 tcfg->endpoints = endp;
160 tcfg->number_endpoints = 1;
161 tcfg->cfg = cfg;
162 endp->tcfg = tcfg;
163 endp->cfg = cfg;
164 mgcp_free_endp(endp);
165
166 dst_end = &endp->bts_end;
167 dst_end->payload_type = audio_name_to_type(dstfmt);
168
169 src_end = &endp->net_end;
170 src_end->payload_type = audio_name_to_type(srcfmt);
171
172 if (out_samples) {
173 dst_end->frame_duration_den = dst_end->rate;
174 dst_end->frame_duration_num = out_samples;
175 dst_end->frames_per_packet = 1;
176 dst_end->force_output_ptime = 1;
177 }
178
179 rc = mgcp_transcoding_setup(endp, dst_end, src_end);
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200180 if (rc < 0) {
181 printf("setup failed: %s", strerror(-rc));
182 abort();
183 }
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200184
185 *out_ctx = cfg;
186 *out_endp = endp;
187 return 0;
188}
189
190
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200191static int transcode_test(const char *srcfmt, const char *dstfmt,
192 uint8_t *src_pkts, size_t src_pkt_size)
193{
194 char buf[4096] = {0x80, 0};
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200195 void *ctx;
196
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200197 struct mgcp_rtp_end *dst_end;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200198 struct mgcp_process_rtp_state *state;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200199 struct mgcp_endpoint *endp;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200200 int in_size;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200201 const int in_samples = 160;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200202 int len, cont;
203
204 printf("== Transcoding test ==\n");
205 printf("converting %s -> %s\n", srcfmt, dstfmt);
206
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200207 given_configured_endpoint(in_samples, 0, srcfmt, dstfmt, &ctx, &endp);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200208
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200209 dst_end = &endp->bts_end;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200210 state = dst_end->rtp_process_data;
211 OSMO_ASSERT(state != NULL);
212
213 in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
214 OSMO_ASSERT(sizeof(buf) >= in_size + 12);
215
216 memcpy(buf, src_pkts, src_pkt_size);
217
218 len = src_pkt_size;
219
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200220 cont = mgcp_transcoding_process_rtp(endp, dst_end,
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200221 buf, &len, sizeof(buf));
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200222 if (cont < 0) {
223 printf("processing failed: %s", strerror(-cont));
224 abort();
225 }
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200226
227 if (len < 24) {
228 printf("encoded: %s\n", osmo_hexdump((unsigned char *)buf, len));
229 } else {
230 const char *str = osmo_hexdump((unsigned char *)buf, len);
231 int i = 0;
232 const int prefix = 4;
233 const int cutlen = 48;
234 int nchars = 0;
235
236 printf("encoded:\n");
237 do {
238 nchars = printf("%*s%-.*s", prefix, "", cutlen, str + i);
239 i += nchars - prefix;
240 printf("\n");
241 } while (nchars - prefix >= cutlen);
242 }
Holger Hans Peter Freyther4fb7e642014-06-28 00:10:10 +0200243 printf("counted: %d\n", cont);
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200244 talloc_free(ctx);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200245 return 0;
246}
247
248static int test_repacking(int in_samples, int out_samples, int no_transcode)
249{
250 char buf[4096] = {0x80, 0};
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200251 int cc;
252 struct mgcp_endpoint *endp;
253 void *ctx;
254
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200255 struct mgcp_process_rtp_state *state;
256 int in_cnt;
257 int out_size;
258 int in_size;
259 uint32_t ts = 0;
260 uint16_t seq = 0;
261 const char *srcfmt = "pcma";
262 const char *dstfmt = no_transcode ? "pcma" : "l16";
263
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200264 printf("== Transcoding test ==\n");
265 printf("converting %s -> %s\n", srcfmt, dstfmt);
266
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200267 given_configured_endpoint(in_samples, out_samples, srcfmt, dstfmt, &ctx, &endp);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200268
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200269 state = endp->bts_end.rtp_process_data;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200270 OSMO_ASSERT(state != NULL);
271
272 in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
273 OSMO_ASSERT(sizeof(buf) >= in_size + 12);
274
275 out_size = mgcp_transcoding_get_frame_size(state, -1, 1);
276 OSMO_ASSERT(sizeof(buf) >= out_size + 12);
277
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200278 buf[1] = endp->net_end.payload_type;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200279 *(uint16_t*)(buf+2) = htons(1);
280 *(uint32_t*)(buf+4) = htonl(0);
281 *(uint32_t*)(buf+8) = htonl(0xaabbccdd);
282
283 for (in_cnt = 0; in_cnt < 16; in_cnt++) {
284 int cont;
285 int len;
286
287 /* fake PCMA data */
288 printf("generating %d %s input samples\n", in_samples, srcfmt);
289 for (cc = 0; cc < in_samples; cc++)
290 buf[12+cc] = cc;
291
292 *(uint16_t*)(buf+2) = htonl(seq);
293 *(uint32_t*)(buf+4) = htonl(ts);
294
295 seq += 1;
296 ts += in_samples;
297
298 cc += 12; /* include RTP header */
299
300 len = cc;
301
302 do {
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200303 cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end,
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200304 buf, &len, sizeof(buf));
305 if (cont == -EAGAIN) {
306 fprintf(stderr, "Got EAGAIN\n");
307 break;
308 }
309
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200310 if (cont < 0) {
311 printf("processing failed: %s", strerror(-cont));
312 abort();
313 }
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200314
315 len -= 12; /* ignore RTP header */
316
Holger Hans Peter Freyther4fb7e642014-06-28 00:10:10 +0200317 printf("got %d %s output frames (%d octets) count=%d\n",
318 len / out_size, dstfmt, len, cont);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200319
320 len = cont;
321 } while (len > 0);
322 }
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200323
324 talloc_free(ctx);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200325 return 0;
326}
327
328int main(int argc, char **argv)
329{
330 osmo_init_logging(&log_info);
331
332 printf("=== Transcoding Good Cases ===\n");
333
334 transcode_test("l16", "l16",
335 (uint8_t *)audio_packets_l16[0].data,
336 audio_packets_l16[0].len);
337 transcode_test("l16", "gsm",
338 (uint8_t *)audio_packets_l16[0].data,
339 audio_packets_l16[0].len);
340 transcode_test("l16", "pcma",
341 (uint8_t *)audio_packets_l16[0].data,
342 audio_packets_l16[0].len);
343 transcode_test("gsm", "l16",
344 (uint8_t *)audio_packets_gsm[0].data,
345 audio_packets_gsm[0].len);
346 transcode_test("gsm", "gsm",
347 (uint8_t *)audio_packets_gsm[0].data,
348 audio_packets_gsm[0].len);
349 transcode_test("gsm", "pcma",
350 (uint8_t *)audio_packets_gsm[0].data,
351 audio_packets_gsm[0].len);
352 transcode_test("pcma", "l16",
353 (uint8_t *)audio_packets_pcma[0].data,
354 audio_packets_pcma[0].len);
355 transcode_test("pcma", "gsm",
356 (uint8_t *)audio_packets_pcma[0].data,
357 audio_packets_pcma[0].len);
358 transcode_test("pcma", "pcma",
359 (uint8_t *)audio_packets_pcma[0].data,
360 audio_packets_pcma[0].len);
361
362 printf("=== Transcoding Bad Cases ===\n");
363
364 printf("Invalid size:\n");
365 transcode_test("gsm", "pcma",
366 (uint8_t *)audio_packets_gsm_invalid_size[0].data,
367 audio_packets_gsm_invalid_size[0].len);
368
369 printf("Invalid data:\n");
370 transcode_test("gsm", "pcma",
371 (uint8_t *)audio_packets_gsm_invalid_data[0].data,
372 audio_packets_gsm_invalid_data[0].len);
373
374 printf("Invalid payload type:\n");
375 transcode_test("gsm", "pcma",
376 (uint8_t *)audio_packets_gsm_invalid_ptype[0].data,
377 audio_packets_gsm_invalid_ptype[0].len);
378
379 printf("=== Repacking ===\n");
380
381 test_repacking(160, 160, 0);
382 test_repacking(160, 160, 1);
383 test_repacking(160, 80, 0);
384 test_repacking(160, 80, 1);
385 test_repacking(160, 320, 0);
386 test_repacking(160, 320, 1);
387 test_repacking(160, 240, 0);
388 test_repacking(160, 240, 1);
389 test_repacking(160, 100, 0);
390 test_repacking(160, 100, 1);
391
392 return 0;
393}
394