2014-04-08 14:10:04 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <err.h>
|
|
|
|
|
|
|
|
#include <osmocom/core/talloc.h>
|
|
|
|
#include <osmocom/core/application.h>
|
|
|
|
|
|
|
|
#include <openbsc/debug.h>
|
|
|
|
#include <openbsc/gsm_data.h>
|
|
|
|
#include <openbsc/mgcp.h>
|
|
|
|
#include <openbsc/mgcp_internal.h>
|
2014-07-04 18:55:20 +00:00
|
|
|
#include <openbsc/rtp.h>
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
#include "bscconfig.h"
|
|
|
|
#ifndef BUILD_MGCP_TRANSCODING
|
|
|
|
#error "Requires MGCP transcoding enabled (see --enable-mgcp-transcoding)"
|
|
|
|
#endif
|
|
|
|
|
2014-05-08 12:08:37 +00:00
|
|
|
#include "openbsc/mgcp_transcode.h"
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
uint8_t *audio_frame_l16[] = {
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets {
|
|
|
|
float t;
|
|
|
|
int len;
|
|
|
|
char *data;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_l16[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 332,
|
|
|
|
"\x80\x0B\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
"\x00\x00\x40\x13\x5A\x9E\x40\x13\x00\x00\xBF\xED\xA5\x62\xBF\xED"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_gsm[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 45,
|
|
|
|
"\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
|
|
|
|
"\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
|
|
|
|
"\xDE"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_gsm_invalid_size[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 41,
|
|
|
|
"\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
|
|
|
|
"\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
|
|
|
|
"\xDE"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_gsm_invalid_data[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 45,
|
|
|
|
"\x80\x03\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
|
|
|
|
"\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE\xEE"
|
|
|
|
"\xEE"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_gsm_invalid_ptype[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 45,
|
|
|
|
"\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xD4\x7C\xE3\xE9\x62\x50\x39\xF0\xF8\xB4\x68\xEA\x6C\x0E\x81\x1B"
|
|
|
|
"\x56\x2A\xD5\xBC\x69\x9C\xD1\xF0\x66\x7A\xEC\x49\x7A\x33\x3D\x0A"
|
|
|
|
"\xDE"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_g729[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 32,
|
|
|
|
"\x80\x12\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xAF\xC2\x81\x40\x00\xFA\xCE\xA4\x21\x7C\xC5\xC3\x4F\xA5\x98\xF5"
|
|
|
|
"\xB2\x95\xC4\xAD"
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtp_packets audio_packets_pcma[] = {
|
|
|
|
/* RTP: SeqNo=1, TS=160 */
|
|
|
|
{0.020000, 172,
|
|
|
|
"\x80\x08\x00\x01\x00\x00\x00\xA0\x11\x22\x33\x44"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
"\xD5\xA5\xA3\xA5\xD5\x25\x23\x25\xD5\xA5\xA3\xA5\xD5\x25\x23\x25"
|
|
|
|
},
|
2014-06-27 17:27:38 +00:00
|
|
|
/* RTP: SeqNo=26527, TS=0 */
|
|
|
|
{0.020000, 92,
|
|
|
|
"\x80\x08\x67\x9f\x00\x00\x00\x00\x04\xaa\x67\x9f\xd5\xd5\xd5\xd5"
|
|
|
|
"\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
|
|
|
|
"\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
|
|
|
|
"\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5\xd5"
|
|
|
|
"\xd5\xd5\xd5\xd5\xd5\xd5\x55\x55\xd5\xd5\x55\x55\xd5\xd5\x55\x55"
|
|
|
|
"\xd5\xd5\xd5\x55\x55\xd5\xd5\xd5\x55\x55\xd5\xd5"
|
|
|
|
},
|
|
|
|
/* RTP: SeqNo=26528, TS=80 */
|
|
|
|
{0.020000, 92,
|
|
|
|
"\x80\x08\x67\xa0\x00\x00\x00\x50\x04\xaa\x67\x9f\x55\xd5\xd5\x55"
|
|
|
|
"\xd5\x55\xd5\xd5\xd5\x55\xd5\x55\xd5\xd5\x55\xd5\x55\xd5\x55\xd5"
|
|
|
|
"\x55\x55\xd5\x55\xd5\xd5\x55\x55\x55\x55\x55\xd5\xd5\x55\xd5\xd5"
|
|
|
|
"\xd5\x55\xd5\xd5\xd5\x55\x54\x55\xd5\xd5\x55\xd5\xd5\xd5\xd5\x55"
|
|
|
|
"\x54\x55\xd5\x55\xd5\x55\x55\x55\x55\x55\xd5\xd5\xd5\xd5\xd5\xd4"
|
|
|
|
"\xd5\x54\x55\xd5\xd4\xd5\x54\xd5\x55\xd5\xd5\xd5"
|
|
|
|
},
|
2014-04-08 14:10:04 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int audio_name_to_type(const char *name)
|
|
|
|
{
|
|
|
|
if (!strcasecmp(name, "gsm"))
|
|
|
|
return 3;
|
|
|
|
#ifdef HAVE_BCG729
|
|
|
|
else if (!strcasecmp(name, "g729"))
|
|
|
|
return 18;
|
|
|
|
#endif
|
|
|
|
else if (!strcasecmp(name, "pcma"))
|
|
|
|
return 8;
|
|
|
|
else if (!strcasecmp(name, "l16"))
|
|
|
|
return 11;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int mgcp_get_trans_frame_size(void *state_, int nsamples, int dst);
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
static int given_configured_endpoint(int in_samples, int out_samples,
|
|
|
|
const char *srcfmt, const char *dstfmt,
|
|
|
|
void **out_ctx, struct mgcp_endpoint **out_endp)
|
2014-04-08 14:10:04 +00:00
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
struct mgcp_rtp_end *dst_end;
|
|
|
|
struct mgcp_rtp_end *src_end;
|
2014-06-22 19:55:50 +00:00
|
|
|
struct mgcp_config *cfg;
|
|
|
|
struct mgcp_trunk_config *tcfg;
|
|
|
|
struct mgcp_endpoint *endp;
|
2014-04-08 14:10:04 +00:00
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
cfg = mgcp_config_alloc();
|
|
|
|
tcfg = talloc_zero(cfg, struct mgcp_trunk_config);
|
|
|
|
endp = talloc_zero(tcfg, struct mgcp_endpoint);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
tcfg->endpoints = endp;
|
|
|
|
tcfg->number_endpoints = 1;
|
|
|
|
tcfg->cfg = cfg;
|
|
|
|
endp->tcfg = tcfg;
|
|
|
|
endp->cfg = cfg;
|
|
|
|
mgcp_free_endp(endp);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
dst_end = &endp->bts_end;
|
2014-04-08 14:10:04 +00:00
|
|
|
dst_end->payload_type = audio_name_to_type(dstfmt);
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
src_end = &endp->net_end;
|
|
|
|
src_end->payload_type = audio_name_to_type(srcfmt);
|
|
|
|
|
|
|
|
if (out_samples) {
|
|
|
|
dst_end->frame_duration_den = dst_end->rate;
|
|
|
|
dst_end->frame_duration_num = out_samples;
|
|
|
|
dst_end->frames_per_packet = 1;
|
|
|
|
dst_end->force_output_ptime = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = mgcp_transcoding_setup(endp, dst_end, src_end);
|
2014-06-28 11:24:36 +00:00
|
|
|
if (rc < 0) {
|
|
|
|
printf("setup failed: %s", strerror(-rc));
|
|
|
|
abort();
|
|
|
|
}
|
2014-04-08 14:10:04 +00:00
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
*out_ctx = cfg;
|
|
|
|
*out_endp = endp;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int transcode_test(const char *srcfmt, const char *dstfmt,
|
|
|
|
uint8_t *src_pkts, size_t src_pkt_size)
|
|
|
|
{
|
|
|
|
char buf[4096] = {0x80, 0};
|
|
|
|
void *ctx;
|
|
|
|
|
|
|
|
struct mgcp_rtp_end *dst_end;
|
|
|
|
struct mgcp_process_rtp_state *state;
|
|
|
|
struct mgcp_endpoint *endp;
|
|
|
|
int in_size;
|
|
|
|
const int in_samples = 160;
|
|
|
|
int len, cont;
|
|
|
|
|
|
|
|
printf("== Transcoding test ==\n");
|
|
|
|
printf("converting %s -> %s\n", srcfmt, dstfmt);
|
|
|
|
|
|
|
|
given_configured_endpoint(in_samples, 0, srcfmt, dstfmt, &ctx, &endp);
|
|
|
|
|
|
|
|
dst_end = &endp->bts_end;
|
2014-04-08 14:10:04 +00:00
|
|
|
state = dst_end->rtp_process_data;
|
|
|
|
OSMO_ASSERT(state != NULL);
|
|
|
|
|
|
|
|
in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
|
|
|
|
OSMO_ASSERT(sizeof(buf) >= in_size + 12);
|
|
|
|
|
|
|
|
memcpy(buf, src_pkts, src_pkt_size);
|
|
|
|
|
|
|
|
len = src_pkt_size;
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
cont = mgcp_transcoding_process_rtp(endp, dst_end,
|
2014-04-08 14:10:04 +00:00
|
|
|
buf, &len, sizeof(buf));
|
2014-06-28 11:24:36 +00:00
|
|
|
if (cont < 0) {
|
2014-06-27 17:27:38 +00:00
|
|
|
printf("Nothing encoded due: %s\n", strerror(-cont));
|
|
|
|
talloc_free(ctx);
|
|
|
|
return -1;
|
2014-06-28 11:24:36 +00:00
|
|
|
}
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
if (len < 24) {
|
|
|
|
printf("encoded: %s\n", osmo_hexdump((unsigned char *)buf, len));
|
|
|
|
} else {
|
|
|
|
const char *str = osmo_hexdump((unsigned char *)buf, len);
|
|
|
|
int i = 0;
|
|
|
|
const int prefix = 4;
|
|
|
|
const int cutlen = 48;
|
|
|
|
int nchars = 0;
|
|
|
|
|
|
|
|
printf("encoded:\n");
|
|
|
|
do {
|
|
|
|
nchars = printf("%*s%-.*s", prefix, "", cutlen, str + i);
|
|
|
|
i += nchars - prefix;
|
|
|
|
printf("\n");
|
|
|
|
} while (nchars - prefix >= cutlen);
|
|
|
|
}
|
2014-06-27 22:10:10 +00:00
|
|
|
printf("counted: %d\n", cont);
|
2014-06-22 19:55:50 +00:00
|
|
|
talloc_free(ctx);
|
2014-04-08 14:10:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-04 18:55:20 +00:00
|
|
|
static void test_rtp_seq_state(void)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
int len;
|
|
|
|
int cont;
|
|
|
|
void *ctx;
|
|
|
|
struct mgcp_endpoint *endp;
|
|
|
|
struct mgcp_process_rtp_state *state;
|
|
|
|
struct rtp_hdr *hdr;
|
|
|
|
uint32_t ts_no;
|
|
|
|
uint16_t seq_no;
|
|
|
|
|
|
|
|
given_configured_endpoint(160, 0, "pcma", "l16", &ctx, &endp);
|
|
|
|
state = endp->bts_end.rtp_process_data;
|
|
|
|
OSMO_ASSERT(!state->is_running);
|
|
|
|
OSMO_ASSERT(state->next_seq == 0);
|
|
|
|
OSMO_ASSERT(state->next_time == 0);
|
|
|
|
|
|
|
|
/* initialize packet */
|
|
|
|
len = audio_packets_pcma[0].len;
|
|
|
|
memcpy(buf, audio_packets_pcma[0].data, len);
|
|
|
|
cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, len);
|
|
|
|
OSMO_ASSERT(cont >= 0);
|
|
|
|
OSMO_ASSERT(state->is_running);
|
|
|
|
OSMO_ASSERT(state->next_seq == 2);
|
|
|
|
OSMO_ASSERT(state->next_time = 240);
|
|
|
|
|
|
|
|
/* verify that the right timestamp was written */
|
|
|
|
OSMO_ASSERT(len == audio_packets_pcma[0].len);
|
|
|
|
hdr = (struct rtp_hdr *) &buf[0];
|
|
|
|
|
|
|
|
memcpy(&ts_no, &hdr->timestamp, sizeof(ts_no));
|
|
|
|
OSMO_ASSERT(htonl(ts_no) == 160);
|
|
|
|
memcpy(&seq_no, &hdr->sequence, sizeof(seq_no));
|
|
|
|
OSMO_ASSERT(htons(seq_no) == 1);
|
|
|
|
/* Check the right sequence number is written */
|
|
|
|
state->next_seq = 1234;
|
|
|
|
len = audio_packets_pcma[0].len;
|
|
|
|
memcpy(buf, audio_packets_pcma[0].data, len);
|
|
|
|
cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, len);
|
|
|
|
OSMO_ASSERT(cont >= 0);
|
|
|
|
OSMO_ASSERT(len == audio_packets_pcma[0].len);
|
|
|
|
hdr = (struct rtp_hdr *) &buf[0];
|
|
|
|
|
|
|
|
memcpy(&seq_no, &hdr->sequence, sizeof(seq_no));
|
|
|
|
OSMO_ASSERT(htons(seq_no) == 1234);
|
|
|
|
|
|
|
|
talloc_free(ctx);
|
|
|
|
}
|
|
|
|
|
2014-06-27 17:27:38 +00:00
|
|
|
static void test_transcode_result(void)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
int len, res;
|
|
|
|
void *ctx;
|
|
|
|
struct mgcp_endpoint *endp;
|
|
|
|
struct mgcp_process_rtp_state *state;
|
|
|
|
|
|
|
|
{
|
|
|
|
/* from GSM to PCMA and same ptime */
|
|
|
|
given_configured_endpoint(160, 0, "gsm", "pcma", &ctx, &endp);
|
|
|
|
state = endp->bts_end.rtp_process_data;
|
|
|
|
|
|
|
|
/* result */
|
|
|
|
len = audio_packets_gsm[0].len;
|
|
|
|
memcpy(buf, audio_packets_gsm[0].data, len);
|
|
|
|
res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
|
|
|
|
OSMO_ASSERT(res == sizeof(struct rtp_hdr));
|
|
|
|
OSMO_ASSERT(state->sample_cnt == 0);
|
|
|
|
|
|
|
|
len = res;
|
|
|
|
res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
|
|
|
|
OSMO_ASSERT(res == -ENOMSG);
|
|
|
|
|
|
|
|
talloc_free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
/* from PCMA to GSM and wrong different ptime */
|
|
|
|
given_configured_endpoint(80, 160, "pcma", "gsm", &ctx, &endp);
|
|
|
|
state = endp->bts_end.rtp_process_data;
|
|
|
|
|
|
|
|
/* Add the first sample */
|
|
|
|
len = audio_packets_pcma[1].len;
|
|
|
|
memcpy(buf, audio_packets_pcma[1].data, len);
|
|
|
|
res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
|
|
|
|
OSMO_ASSERT(state->sample_cnt == 80);
|
|
|
|
OSMO_ASSERT(res < 0);
|
|
|
|
|
|
|
|
/* Add the second sample and it should be consumable */
|
|
|
|
len = audio_packets_pcma[2].len;
|
|
|
|
memcpy(buf, audio_packets_pcma[2].data, len);
|
|
|
|
res = mgcp_transcoding_process_rtp(endp, &endp->bts_end, buf, &len, ARRAY_SIZE(buf));
|
|
|
|
OSMO_ASSERT(state->sample_cnt == 0);
|
|
|
|
OSMO_ASSERT(res == sizeof(struct rtp_hdr));
|
|
|
|
|
|
|
|
talloc_free(ctx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-08 14:10:04 +00:00
|
|
|
static int test_repacking(int in_samples, int out_samples, int no_transcode)
|
|
|
|
{
|
|
|
|
char buf[4096] = {0x80, 0};
|
2014-06-22 19:55:50 +00:00
|
|
|
int cc;
|
|
|
|
struct mgcp_endpoint *endp;
|
|
|
|
void *ctx;
|
|
|
|
|
2014-04-08 14:10:04 +00:00
|
|
|
struct mgcp_process_rtp_state *state;
|
|
|
|
int in_cnt;
|
|
|
|
int out_size;
|
|
|
|
int in_size;
|
|
|
|
uint32_t ts = 0;
|
|
|
|
uint16_t seq = 0;
|
|
|
|
const char *srcfmt = "pcma";
|
|
|
|
const char *dstfmt = no_transcode ? "pcma" : "l16";
|
|
|
|
|
|
|
|
printf("== Transcoding test ==\n");
|
|
|
|
printf("converting %s -> %s\n", srcfmt, dstfmt);
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
given_configured_endpoint(in_samples, out_samples, srcfmt, dstfmt, &ctx, &endp);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
state = endp->bts_end.rtp_process_data;
|
2014-04-08 14:10:04 +00:00
|
|
|
OSMO_ASSERT(state != NULL);
|
|
|
|
|
|
|
|
in_size = mgcp_transcoding_get_frame_size(state, in_samples, 0);
|
|
|
|
OSMO_ASSERT(sizeof(buf) >= in_size + 12);
|
|
|
|
|
|
|
|
out_size = mgcp_transcoding_get_frame_size(state, -1, 1);
|
|
|
|
OSMO_ASSERT(sizeof(buf) >= out_size + 12);
|
|
|
|
|
2014-06-22 19:55:50 +00:00
|
|
|
buf[1] = endp->net_end.payload_type;
|
2014-04-08 14:10:04 +00:00
|
|
|
*(uint16_t*)(buf+2) = htons(1);
|
|
|
|
*(uint32_t*)(buf+4) = htonl(0);
|
|
|
|
*(uint32_t*)(buf+8) = htonl(0xaabbccdd);
|
|
|
|
|
|
|
|
for (in_cnt = 0; in_cnt < 16; in_cnt++) {
|
|
|
|
int cont;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
/* fake PCMA data */
|
|
|
|
printf("generating %d %s input samples\n", in_samples, srcfmt);
|
|
|
|
for (cc = 0; cc < in_samples; cc++)
|
|
|
|
buf[12+cc] = cc;
|
|
|
|
|
|
|
|
*(uint16_t*)(buf+2) = htonl(seq);
|
|
|
|
*(uint32_t*)(buf+4) = htonl(ts);
|
|
|
|
|
|
|
|
seq += 1;
|
|
|
|
ts += in_samples;
|
|
|
|
|
|
|
|
cc += 12; /* include RTP header */
|
|
|
|
|
|
|
|
len = cc;
|
|
|
|
|
|
|
|
do {
|
2014-06-22 19:55:50 +00:00
|
|
|
cont = mgcp_transcoding_process_rtp(endp, &endp->bts_end,
|
2014-04-08 14:10:04 +00:00
|
|
|
buf, &len, sizeof(buf));
|
|
|
|
if (cont == -EAGAIN) {
|
|
|
|
fprintf(stderr, "Got EAGAIN\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-06-28 11:24:36 +00:00
|
|
|
if (cont < 0) {
|
|
|
|
printf("processing failed: %s", strerror(-cont));
|
|
|
|
abort();
|
|
|
|
}
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
len -= 12; /* ignore RTP header */
|
|
|
|
|
2014-06-27 22:10:10 +00:00
|
|
|
printf("got %d %s output frames (%d octets) count=%d\n",
|
|
|
|
len / out_size, dstfmt, len, cont);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
len = cont;
|
|
|
|
} while (len > 0);
|
|
|
|
}
|
2014-06-22 19:55:50 +00:00
|
|
|
|
|
|
|
talloc_free(ctx);
|
2014-04-08 14:10:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2014-06-27 17:27:38 +00:00
|
|
|
int rc;
|
2014-04-08 14:10:04 +00:00
|
|
|
osmo_init_logging(&log_info);
|
|
|
|
|
|
|
|
printf("=== Transcoding Good Cases ===\n");
|
|
|
|
|
|
|
|
transcode_test("l16", "l16",
|
|
|
|
(uint8_t *)audio_packets_l16[0].data,
|
|
|
|
audio_packets_l16[0].len);
|
|
|
|
transcode_test("l16", "gsm",
|
|
|
|
(uint8_t *)audio_packets_l16[0].data,
|
|
|
|
audio_packets_l16[0].len);
|
|
|
|
transcode_test("l16", "pcma",
|
|
|
|
(uint8_t *)audio_packets_l16[0].data,
|
|
|
|
audio_packets_l16[0].len);
|
|
|
|
transcode_test("gsm", "l16",
|
|
|
|
(uint8_t *)audio_packets_gsm[0].data,
|
|
|
|
audio_packets_gsm[0].len);
|
|
|
|
transcode_test("gsm", "gsm",
|
|
|
|
(uint8_t *)audio_packets_gsm[0].data,
|
|
|
|
audio_packets_gsm[0].len);
|
|
|
|
transcode_test("gsm", "pcma",
|
|
|
|
(uint8_t *)audio_packets_gsm[0].data,
|
|
|
|
audio_packets_gsm[0].len);
|
|
|
|
transcode_test("pcma", "l16",
|
|
|
|
(uint8_t *)audio_packets_pcma[0].data,
|
|
|
|
audio_packets_pcma[0].len);
|
|
|
|
transcode_test("pcma", "gsm",
|
|
|
|
(uint8_t *)audio_packets_pcma[0].data,
|
|
|
|
audio_packets_pcma[0].len);
|
|
|
|
transcode_test("pcma", "pcma",
|
|
|
|
(uint8_t *)audio_packets_pcma[0].data,
|
|
|
|
audio_packets_pcma[0].len);
|
|
|
|
|
|
|
|
printf("=== Transcoding Bad Cases ===\n");
|
|
|
|
|
|
|
|
printf("Invalid size:\n");
|
2014-06-27 17:27:38 +00:00
|
|
|
rc = transcode_test("gsm", "pcma",
|
2014-04-08 14:10:04 +00:00
|
|
|
(uint8_t *)audio_packets_gsm_invalid_size[0].data,
|
|
|
|
audio_packets_gsm_invalid_size[0].len);
|
2014-06-27 17:27:38 +00:00
|
|
|
OSMO_ASSERT(rc < 0);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
printf("Invalid data:\n");
|
2014-06-27 17:27:38 +00:00
|
|
|
rc = transcode_test("gsm", "pcma",
|
2014-04-08 14:10:04 +00:00
|
|
|
(uint8_t *)audio_packets_gsm_invalid_data[0].data,
|
|
|
|
audio_packets_gsm_invalid_data[0].len);
|
2014-06-27 17:27:38 +00:00
|
|
|
OSMO_ASSERT(rc < 0);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
printf("Invalid payload type:\n");
|
2014-06-27 17:27:38 +00:00
|
|
|
rc = transcode_test("gsm", "pcma",
|
2014-04-08 14:10:04 +00:00
|
|
|
(uint8_t *)audio_packets_gsm_invalid_ptype[0].data,
|
|
|
|
audio_packets_gsm_invalid_ptype[0].len);
|
2014-06-27 17:27:38 +00:00
|
|
|
OSMO_ASSERT(rc == 0);
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
printf("=== Repacking ===\n");
|
|
|
|
|
|
|
|
test_repacking(160, 160, 0);
|
|
|
|
test_repacking(160, 160, 1);
|
|
|
|
test_repacking(160, 80, 0);
|
|
|
|
test_repacking(160, 80, 1);
|
|
|
|
test_repacking(160, 320, 0);
|
|
|
|
test_repacking(160, 320, 1);
|
|
|
|
test_repacking(160, 240, 0);
|
|
|
|
test_repacking(160, 240, 1);
|
|
|
|
test_repacking(160, 100, 0);
|
|
|
|
test_repacking(160, 100, 1);
|
2014-07-04 18:55:20 +00:00
|
|
|
test_rtp_seq_state();
|
2014-06-27 17:27:38 +00:00
|
|
|
test_transcode_result();
|
2014-04-08 14:10:04 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|