blob: 747fd9af4c0eae1c3942468a64769005fdb5c804 [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>
Holger Hans Peter Freyther91eeeae2014-07-04 20:55:20 +020014#include <openbsc/rtp.h>
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +020015
16#include "bscconfig.h"
17#ifndef BUILD_MGCP_TRANSCODING
18#error "Requires MGCP transcoding enabled (see --enable-mgcp-transcoding)"
19#endif
20
Jacob Erlbeck909fac62014-05-08 14:08:37 +020021#include "openbsc/mgcp_transcode.h"
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +020022
23uint8_t *audio_frame_l16[] = {
24};
25
26struct rtp_packets {
27 float t;
28 int len;
29 char *data;
30};
31
32struct rtp_packets audio_packets_l16[] = {
33 /* RTP: SeqNo=1, TS=160 */
34 {0.020000, 332,
35 "\x80\x0B\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
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 "\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
56 },
57};
58
59struct rtp_packets audio_packets_gsm[] = {
60 /* RTP: SeqNo=1, TS=160 */
61 {0.020000, 45,
62 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
63 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
64 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
65 "\xDE"
66 },
67};
68
69struct rtp_packets audio_packets_gsm_invalid_size[] = {
70 /* RTP: SeqNo=1, TS=160 */
71 {0.020000, 41,
72 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
73 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
74 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
75 "\xDE"
76 },
77};
78
79struct rtp_packets audio_packets_gsm_invalid_data[] = {
80 /* RTP: SeqNo=1, TS=160 */
81 {0.020000, 45,
82 "\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
83 "\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
84 "\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
85 "\xEE"
86 },
87};
88
89struct rtp_packets audio_packets_gsm_invalid_ptype[] = {
90 /* RTP: SeqNo=1, TS=160 */
91 {0.020000, 45,
92 "\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
93 "\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
94 "\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
95 "\xDE"
96 },
97};
98
99struct rtp_packets audio_packets_g729[] = {
100 /* RTP: SeqNo=1, TS=160 */
101 {0.020000, 32,
102 "\x80\x12\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
103 "\xAF\xC2\x81\x40\x00\xFA\xCE\xA4\x21\x7C\xC5\xC3\x4F\xA5\x98\xF5"
104 "\xB2\x95\xC4\xAD"
105 },
106};
107
108struct rtp_packets audio_packets_pcma[] = {
109 /* RTP: SeqNo=1, TS=160 */
110 {0.020000, 172,
111 "\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
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 "\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
122 },
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200123 /* RTP: SeqNo=26527, TS=232640 */
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200124 {0.020000, 92,
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200125 "\x80\x08\x67\x9f\x00\x03\x8c\xc0\x04\xaa\x67\x9f\xd5\xd5\xd5\xd5"
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200126 "\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
127 "\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
128 "\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
129 "\xd5\xd5\xd5\xd5\xd5\xd5\x55\x55\xd5\xd5\x55\x55\xd5\xd5\x55\x55"
130 "\xd5\xd5\xd5\x55\x55\xd5\xd5\xd5\x55\x55\xd5\xd5"
131 },
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200132 /* RTP: SeqNo=26528, TS=232720 */
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200133 {0.020000, 92,
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200134 "\x80\x08\x67\xa0\x00\x03\x8d\x10\x04\xaa\x67\x9f\x55\xd5\xd5\x55"
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200135 "\xd5\x55\xd5\xd5\xd5\x55\xd5\x55\xd5\xd5\x55\xd5\x55\xd5\x55\xd5"
136 "\x55\x55\xd5\x55\xd5\xd5\x55\x55\x55\x55\x55\xd5\xd5\x55\xd5\xd5"
137 "\xd5\x55\xd5\xd5\xd5\x55\x54\x55\xd5\xd5\x55\xd5\xd5\xd5\xd5\x55"
138 "\x54\x55\xd5\x55\xd5\x55\x55\x55\x55\x55\xd5\xd5\xd5\xd5\xd5\xd4"
139 "\xd5\x54\x55\xd5\xd4\xd5\x54\xd5\x55\xd5\xd5\xd5"
140 },
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200141};
142
143
144
145static int audio_name_to_type(const char *name)
146{
147 if (!strcasecmp(name, "gsm"))
148 return 3;
149#ifdef HAVE_BCG729
150 else if (!strcasecmp(name, "g729"))
151 return 18;
152#endif
153 else if (!strcasecmp(name, "pcma"))
154 return 8;
155 else if (!strcasecmp(name, "l16"))
156 return 11;
157 return -1;
158}
159
160int mgcp_get_trans_frame_size(void *state_, int nsamples, int dst);
161
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200162static int given_configured_endpoint(int in_samples, int out_samples,
163 const char *srcfmt, const char *dstfmt,
164 void **out_ctx, struct mgcp_endpoint **out_endp)
165{
166 int rc;
167 struct mgcp_rtp_end *dst_end;
168 struct mgcp_rtp_end *src_end;
169 struct mgcp_config *cfg;
170 struct mgcp_trunk_config *tcfg;
171 struct mgcp_endpoint *endp;
172
173 cfg = mgcp_config_alloc();
174 tcfg = talloc_zero(cfg, struct mgcp_trunk_config);
175 endp = talloc_zero(tcfg, struct mgcp_endpoint);
176
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200177 cfg->setup_rtp_processing_cb = mgcp_transcoding_setup;
178 cfg->rtp_processing_cb = mgcp_transcoding_process_rtp;
179 cfg->get_net_downlink_format_cb = mgcp_transcoding_net_downlink_format;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200180
181 tcfg->endpoints = endp;
182 tcfg->number_endpoints = 1;
183 tcfg->cfg = cfg;
184 endp->tcfg = tcfg;
185 endp->cfg = cfg;
Holger Hans Peter Freythercb6ad702014-07-22 15:00:52 +0200186 mgcp_initialize_endp(endp);
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200187
188 dst_end = &endp->bts_end;
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +0200189 dst_end->codec.payload_type = audio_name_to_type(dstfmt);
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200190
191 src_end = &endp->net_end;
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +0200192 src_end->codec.payload_type = audio_name_to_type(srcfmt);
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200193
194 if (out_samples) {
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +0200195 dst_end->codec.frame_duration_den = dst_end->codec.rate;
196 dst_end->codec.frame_duration_num = out_samples;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200197 dst_end->frames_per_packet = 1;
198 dst_end->force_output_ptime = 1;
199 }
200
201 rc = mgcp_transcoding_setup(endp, dst_end, src_end);
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200202 if (rc < 0) {
203 printf("setup failed: %s", strerror(-rc));
204 abort();
205 }
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200206
207 *out_ctx = cfg;
208 *out_endp = endp;
209 return 0;
210}
211
212
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200213static int transcode_test(const char *srcfmt, const char *dstfmt,
214 uint8_t *src_pkts, size_t src_pkt_size)
215{
216 char buf[4096] = {0x80, 0};
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200217 void *ctx;
218
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200219 struct mgcp_rtp_end *dst_end;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200220 struct mgcp_process_rtp_state *state;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200221 struct mgcp_endpoint *endp;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200222 int in_size;
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200223 const int in_samples = 160;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200224 int len, cont;
225
226 printf("== Transcoding test ==\n");
227 printf("converting %s -> %s\n", srcfmt, dstfmt);
228
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200229 given_configured_endpoint(in_samples, 0, srcfmt, dstfmt, &ctx, &endp);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200230
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200231 dst_end = &endp->bts_end;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200232 state = dst_end->rtp_process_data;
233 OSMO_ASSERT(state != NULL);
234
235 in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
236 OSMO_ASSERT(sizeof(buf) >= in_size + 12);
237
238 memcpy(buf, src_pkts, src_pkt_size);
239
240 len = src_pkt_size;
241
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200242 cont = mgcp_transcoding_process_rtp(endp, dst_end,
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200243 buf, &len, sizeof(buf));
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200244 if (cont < 0) {
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200245 printf("Nothing encoded due: %s\n", strerror(-cont));
246 talloc_free(ctx);
247 return -1;
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200248 }
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200249
250 if (len < 24) {
251 printf("encoded: %s\n", osmo_hexdump((unsigned char *)buf, len));
252 } else {
253 const char *str = osmo_hexdump((unsigned char *)buf, len);
254 int i = 0;
255 const int prefix = 4;
256 const int cutlen = 48;
257 int nchars = 0;
258
259 printf("encoded:\n");
260 do {
261 nchars = printf("%*s%-.*s", prefix, "", cutlen, str + i);
262 i += nchars - prefix;
263 printf("\n");
264 } while (nchars - prefix >= cutlen);
265 }
Holger Hans Peter Freyther4fb7e642014-06-28 00:10:10 +0200266 printf("counted: %d\n", cont);
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200267 talloc_free(ctx);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200268 return 0;
269}
270
Holger Hans Peter Freyther91eeeae2014-07-04 20:55:20 +0200271static void test_rtp_seq_state(void)
272{
273 char buf[4096];
274 int len;
275 int cont;
276 void *ctx;
277 struct mgcp_endpoint *endp;
278 struct mgcp_process_rtp_state *state;
279 struct rtp_hdr *hdr;
280 uint32_t ts_no;
281 uint16_t seq_no;
282
283 given_configured_endpoint(160, 0, "pcma", "l16", &ctx, &endp);
284 state = endp->bts_end.rtp_process_data;
285 OSMO_ASSERT(!state->is_running);
286 OSMO_ASSERT(state->next_seq == 0);
287 OSMO_ASSERT(state->next_time == 0);
288
289 /* initialize packet */
290 len = audio_packets_pcma[0].len;
291 memcpy(buf, audio_packets_pcma[0].data, len);
292 cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, len);
293 OSMO_ASSERT(cont >= 0);
294 OSMO_ASSERT(state->is_running);
295 OSMO_ASSERT(state->next_seq == 2);
296 OSMO_ASSERT(state->next_time = 240);
297
298 /* verify that the right timestamp was written */
299 OSMO_ASSERT(len == audio_packets_pcma[0].len);
300 hdr = (struct rtp_hdr *) &buf[0];
301
302 memcpy(&ts_no, &hdr->timestamp, sizeof(ts_no));
303 OSMO_ASSERT(htonl(ts_no) == 160);
304 memcpy(&seq_no, &hdr->sequence, sizeof(seq_no));
305 OSMO_ASSERT(htons(seq_no) == 1);
306 /* Check the right sequence number is written */
307 state->next_seq = 1234;
308 len = audio_packets_pcma[0].len;
309 memcpy(buf, audio_packets_pcma[0].data, len);
310 cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, len);
311 OSMO_ASSERT(cont >= 0);
312 OSMO_ASSERT(len == audio_packets_pcma[0].len);
313 hdr = (struct rtp_hdr *) &buf[0];
314
315 memcpy(&seq_no, &hdr->sequence, sizeof(seq_no));
316 OSMO_ASSERT(htons(seq_no) == 1234);
317
318 talloc_free(ctx);
319}
320
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200321static void test_transcode_result(void)
322{
323 char buf[4096];
324 int len, res;
325 void *ctx;
326 struct mgcp_endpoint *endp;
327 struct mgcp_process_rtp_state *state;
328
329 {
330 /* from GSM to PCMA and same ptime */
331 given_configured_endpoint(160, 0, "gsm", "pcma", &ctx, &endp);
332 state = endp->bts_end.rtp_process_data;
333
334 /* result */
335 len = audio_packets_gsm[0].len;
336 memcpy(buf, audio_packets_gsm[0].data, len);
337 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
338 OSMO_ASSERT(res == sizeof(struct rtp_hdr));
339 OSMO_ASSERT(state->sample_cnt == 0);
340
341 len = res;
342 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
343 OSMO_ASSERT(res == -ENOMSG);
344
345 talloc_free(ctx);
346 }
347
348 {
Holger Hans Peter Freyther4c18d792014-07-04 08:13:53 +0200349 /* from GSM to PCMA and same ptime */
350 given_configured_endpoint(160, 160, "gsm", "pcma", &ctx, &endp);
351 state = endp->bts_end.rtp_process_data;
352
353 /* result */
354 len = audio_packets_gsm[0].len;
355 memcpy(buf, audio_packets_gsm[0].data, len);
356 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
357 OSMO_ASSERT(res == sizeof(struct rtp_hdr));
358 OSMO_ASSERT(state->sample_cnt == 0);
359
360 len = res;
361 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
362 OSMO_ASSERT(res == -EAGAIN);
363
364 talloc_free(ctx);
365 }
366
367 {
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200368 /* from PCMA to GSM and wrong different ptime */
369 given_configured_endpoint(80, 160, "pcma", "gsm", &ctx, &endp);
370 state = endp->bts_end.rtp_process_data;
371
372 /* Add the first sample */
373 len = audio_packets_pcma[1].len;
374 memcpy(buf, audio_packets_pcma[1].data, len);
375 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
376 OSMO_ASSERT(state->sample_cnt == 80);
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200377 OSMO_ASSERT(state->next_time == 232640);
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200378 OSMO_ASSERT(res < 0);
379
380 /* Add the second sample and it should be consumable */
381 len = audio_packets_pcma[2].len;
382 memcpy(buf, audio_packets_pcma[2].data, len);
383 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
384 OSMO_ASSERT(state->sample_cnt == 0);
Holger Hans Peter Freytherc8b29082014-07-02 22:02:15 +0200385 OSMO_ASSERT(state->next_time == 232640 + 80 + 160);
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200386 OSMO_ASSERT(res == sizeof(struct rtp_hdr));
387
388 talloc_free(ctx);
389 }
Holger Hans Peter Freytherb9362782014-07-04 20:55:20 +0200390
391 {
392 /* from PCMA to GSM with a big time jump */
393 struct rtp_hdr *hdr;
394 uint32_t ts;
395
396 given_configured_endpoint(80, 160, "pcma", "gsm", &ctx, &endp);
397 state = endp->bts_end.rtp_process_data;
398
399 /* Add the first sample */
400 len = audio_packets_pcma[1].len;
401 memcpy(buf, audio_packets_pcma[1].data, len);
402 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
403 OSMO_ASSERT(state->sample_cnt == 80);
404 OSMO_ASSERT(state->next_time == 232640);
405 OSMO_ASSERT(state->next_seq == 26527);
406 OSMO_ASSERT(res < 0);
407
408 /* Add a skip to the packet to force a 'resync' */
409 len = audio_packets_pcma[2].len;
410 memcpy(buf, audio_packets_pcma[2].data, len);
411 hdr = (struct rtp_hdr *) &buf[0];
412 /* jump the time and add alignment error */
413 ts = ntohl(hdr->timestamp) + 123 * 80 + 2;
414 hdr->timestamp = htonl(ts);
415 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
416 OSMO_ASSERT(res < 0);
417 OSMO_ASSERT(state->sample_cnt == 80);
418 OSMO_ASSERT(state->next_time == ts);
419 OSMO_ASSERT(state->next_seq == 26527);
420 /* TODO: this can create alignment errors */
421
422
423 /* Now attempt to consume 160 samples */
424 len = audio_packets_pcma[2].len;
425 memcpy(buf, audio_packets_pcma[2].data, len);
426 hdr = (struct rtp_hdr *) &buf[0];
427 ts += 80;
428 hdr->timestamp = htonl(ts);
429 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
430 OSMO_ASSERT(res == 12);
431 OSMO_ASSERT(state->sample_cnt == 0);
432 OSMO_ASSERT(state->next_time == ts + 160);
433 OSMO_ASSERT(state->next_seq == 26528);
434
435 talloc_free(ctx);
436 }
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200437}
438
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200439static void test_transcode_change(void)
440{
441 char buf[4096] = {0x80, 0};
442 void *ctx;
443
444 struct mgcp_endpoint *endp;
445 struct mgcp_process_rtp_state *state;
446 struct rtp_hdr *hdr;
447
448 int len, res;
449
450 {
451 /* from GSM to PCMA and same ptime */
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200452 printf("Testing Initial L16->GSM, PCMA->GSM\n");
453 given_configured_endpoint(160, 0, "l16", "gsm", &ctx, &endp);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200454 endp->net_end.alt_codec = endp->net_end.codec;
455 endp->net_end.alt_codec.payload_type = audio_name_to_type("pcma");
456 state = endp->bts_end.rtp_process_data;
457
458 /* initial transcoding work */
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200459 OSMO_ASSERT(state->src_fmt == AF_L16);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200460 OSMO_ASSERT(state->dst_fmt == AF_GSM);
461 OSMO_ASSERT(endp->net_end.alt_codec.payload_type == 8);
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200462 OSMO_ASSERT(endp->net_end.codec.payload_type == 11);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200463
464 /* result */
465 len = audio_packets_pcma[0].len;
466 memcpy(buf, audio_packets_pcma[0].data, len);
467 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
Holger Hans Peter Freyther01699712014-09-05 12:23:36 +0200468 state = endp->bts_end.rtp_process_data;
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200469 OSMO_ASSERT(res == sizeof(struct rtp_hdr));
470 OSMO_ASSERT(state->sample_cnt == 0);
471 OSMO_ASSERT(state->src_fmt == AF_PCMA);
472 OSMO_ASSERT(state->dst_fmt == AF_GSM);
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200473 OSMO_ASSERT(endp->net_end.alt_codec.payload_type == 11);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200474 OSMO_ASSERT(endp->net_end.codec.payload_type == 8);
475
476 len = res;
477 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
478 OSMO_ASSERT(res == -ENOMSG);
Holger Hans Peter Freyther01699712014-09-05 12:23:36 +0200479 OSMO_ASSERT(state == endp->bts_end.rtp_process_data);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200480
481
482 /* now check that comfort noise doesn't change anything */
483 len = audio_packets_pcma[1].len;
484 memcpy(buf, audio_packets_pcma[1].data, len);
485 hdr = (struct rtp_hdr *) buf;
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200486 hdr->payload_type = 12;
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200487 res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
Holger Hans Peter Freyther01699712014-09-05 12:23:36 +0200488 OSMO_ASSERT(state == endp->bts_end.rtp_process_data);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200489 OSMO_ASSERT(state->sample_cnt == 80);
490 OSMO_ASSERT(state->src_fmt == AF_PCMA);
491 OSMO_ASSERT(state->dst_fmt == AF_GSM);
Holger Hans Peter Freyther0454e322014-09-02 12:15:37 +0200492 OSMO_ASSERT(endp->net_end.alt_codec.payload_type == 11);
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200493 OSMO_ASSERT(endp->net_end.codec.payload_type == 8);
494
495 talloc_free(ctx);
496 }
497}
498
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200499static int test_repacking(int in_samples, int out_samples, int no_transcode)
500{
501 char buf[4096] = {0x80, 0};
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200502 int cc;
503 struct mgcp_endpoint *endp;
504 void *ctx;
505
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200506 struct mgcp_process_rtp_state *state;
507 int in_cnt;
508 int out_size;
509 int in_size;
510 uint32_t ts = 0;
511 uint16_t seq = 0;
512 const char *srcfmt = "pcma";
513 const char *dstfmt = no_transcode ? "pcma" : "l16";
514
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200515 printf("== Transcoding test ==\n");
516 printf("converting %s -> %s\n", srcfmt, dstfmt);
517
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200518 given_configured_endpoint(in_samples, out_samples, srcfmt, dstfmt, &ctx, &endp);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200519
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200520 state = endp->bts_end.rtp_process_data;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200521 OSMO_ASSERT(state != NULL);
522
523 in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
524 OSMO_ASSERT(sizeof(buf) >= in_size + 12);
525
526 out_size = mgcp_transcoding_get_frame_size(state, -1, 1);
527 OSMO_ASSERT(sizeof(buf) >= out_size + 12);
528
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +0200529 buf[1] = endp->net_end.codec.payload_type;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200530 *(uint16_t*)(buf+2) = htons(1);
531 *(uint32_t*)(buf+4) = htonl(0);
532 *(uint32_t*)(buf+8) = htonl(0xaabbccdd);
533
534 for (in_cnt = 0; in_cnt < 16; in_cnt++) {
535 int cont;
536 int len;
537
538 /* fake PCMA data */
539 printf("generating %d %s input samples\n", in_samples, srcfmt);
540 for (cc = 0; cc < in_samples; cc++)
541 buf[12+cc] = cc;
542
543 *(uint16_t*)(buf+2) = htonl(seq);
544 *(uint32_t*)(buf+4) = htonl(ts);
545
546 seq += 1;
547 ts += in_samples;
548
549 cc += 12; /* include RTP header */
550
551 len = cc;
552
553 do {
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200554 cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end,
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200555 buf, &len, sizeof(buf));
556 if (cont == -EAGAIN) {
557 fprintf(stderr, "Got EAGAIN\n");
558 break;
559 }
560
Holger Hans Peter Freyther6041c8d2014-06-28 13:24:36 +0200561 if (cont < 0) {
562 printf("processing failed: %s", strerror(-cont));
563 abort();
564 }
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200565
566 len -= 12; /* ignore RTP header */
567
Holger Hans Peter Freyther4fb7e642014-06-28 00:10:10 +0200568 printf("got %d %s output frames (%d octets) count=%d\n",
569 len / out_size, dstfmt, len, cont);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200570
571 len = cont;
572 } while (len > 0);
573 }
Holger Hans Peter Freyther83cbac22014-06-22 21:55:50 +0200574
575 talloc_free(ctx);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200576 return 0;
577}
578
579int main(int argc, char **argv)
580{
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200581 int rc;
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200582 osmo_init_logging(&log_info);
583
584 printf("=== Transcoding Good Cases ===\n");
585
586 transcode_test("l16", "l16",
587 (uint8_t *)audio_packets_l16[0].data,
588 audio_packets_l16[0].len);
589 transcode_test("l16", "gsm",
590 (uint8_t *)audio_packets_l16[0].data,
591 audio_packets_l16[0].len);
592 transcode_test("l16", "pcma",
593 (uint8_t *)audio_packets_l16[0].data,
594 audio_packets_l16[0].len);
595 transcode_test("gsm", "l16",
596 (uint8_t *)audio_packets_gsm[0].data,
597 audio_packets_gsm[0].len);
598 transcode_test("gsm", "gsm",
599 (uint8_t *)audio_packets_gsm[0].data,
600 audio_packets_gsm[0].len);
601 transcode_test("gsm", "pcma",
602 (uint8_t *)audio_packets_gsm[0].data,
603 audio_packets_gsm[0].len);
604 transcode_test("pcma", "l16",
605 (uint8_t *)audio_packets_pcma[0].data,
606 audio_packets_pcma[0].len);
607 transcode_test("pcma", "gsm",
608 (uint8_t *)audio_packets_pcma[0].data,
609 audio_packets_pcma[0].len);
610 transcode_test("pcma", "pcma",
611 (uint8_t *)audio_packets_pcma[0].data,
612 audio_packets_pcma[0].len);
613
614 printf("=== Transcoding Bad Cases ===\n");
615
616 printf("Invalid size:\n");
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200617 rc = transcode_test("gsm", "pcma",
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200618 (uint8_t *)audio_packets_gsm_invalid_size[0].data,
619 audio_packets_gsm_invalid_size[0].len);
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200620 OSMO_ASSERT(rc < 0);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200621
622 printf("Invalid data:\n");
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200623 rc = transcode_test("gsm", "pcma",
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200624 (uint8_t *)audio_packets_gsm_invalid_data[0].data,
625 audio_packets_gsm_invalid_data[0].len);
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200626 OSMO_ASSERT(rc < 0);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200627
628 printf("Invalid payload type:\n");
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200629 rc = transcode_test("gsm", "pcma",
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200630 (uint8_t *)audio_packets_gsm_invalid_ptype[0].data,
631 audio_packets_gsm_invalid_ptype[0].len);
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200632 OSMO_ASSERT(rc == 0);
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200633
634 printf("=== Repacking ===\n");
635
636 test_repacking(160, 160, 0);
637 test_repacking(160, 160, 1);
638 test_repacking(160, 80, 0);
639 test_repacking(160, 80, 1);
640 test_repacking(160, 320, 0);
641 test_repacking(160, 320, 1);
642 test_repacking(160, 240, 0);
643 test_repacking(160, 240, 1);
644 test_repacking(160, 100, 0);
645 test_repacking(160, 100, 1);
Holger Hans Peter Freyther91eeeae2014-07-04 20:55:20 +0200646 test_rtp_seq_state();
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +0200647 test_transcode_result();
Holger Hans Peter Freyther46801212014-09-01 22:20:57 +0200648 test_transcode_change();
Jacob Erlbeck84a45cb2014-04-08 16:10:04 +0200649
650 return 0;
651}
652