blob: c4769710ae03786ab00e4259decae5c9c0870a8a [file] [log] [blame]
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +01001/*
Holger Hans Peter Freytherfaa67932012-10-22 18:09:35 +02002 * (C) 2011-2012 by Holger Hans Peter Freyther <zecke@selfish.org>
3 * (C) 2011-2012 by On-Waves
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +01004 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Affero General Public License for more details.
15 *
16 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include <openbsc/mgcp.h>
Holger Hans Peter Freyther49e170b2011-02-28 14:37:03 +010021#include <openbsc/mgcp_internal.h>
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +010022
Holger Hans Peter Freyther2d59cc62011-05-12 16:02:07 +020023#include <osmocom/core/application.h>
Pablo Neira Ayusodd5fff42011-03-22 16:47:59 +010024#include <osmocom/core/talloc.h>
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +010025#include <string.h>
Holger Hans Peter Freytherfaa67932012-10-22 18:09:35 +020026#include <limits.h>
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +010027
Harald Welte834f3f12012-01-27 00:00:44 +010028#define AUEP1 "AUEP 158663169 ds/e1-1/2@172.16.6.66 MGCP 1.0\r\n"
Harald Weltea7c9a922012-01-27 00:36:56 +010029#define AUEP1_RET "200 158663169 OK\r\n"
Harald Welte834f3f12012-01-27 00:00:44 +010030#define AUEP2 "AUEP 18983213 ds/e1-2/1@172.16.6.66 MGCP 1.0\r\n"
Harald Weltea7c9a922012-01-27 00:36:56 +010031#define AUEP2_RET "500 18983213 FAIL\r\n"
Holger Hans Peter Freyther0a6455c2012-03-16 09:53:49 +010032#define EMPTY "\r\n"
33#define EMPTY_RET NULL
34#define SHORT "CRCX \r\n"
35#define SHORT_RET "510 000000 FAIL\r\n"
36
37#define SHORT2 "CRCX 1"
38#define SHORT2_RET "510 000000 FAIL\r\n"
39#define SHORT3 "CRCX 1 1@mgw"
40#define SHORT4 "CRCX 1 1@mgw MGCP"
41#define SHORT5 "CRCX 1 1@mgw MGCP 1.0"
Harald Weltea7c9a922012-01-27 00:36:56 +010042
Harald Welteca765cd2012-01-27 00:37:23 +010043#define CRCX "CRCX 2 1@mgw MGCP 1.0\r\n" \
44 "M: sendrecv\r\n" \
45 "C: 2\r\n" \
46 "\r\n" \
47 "v=0\r\n" \
48 "c=IN IP4 123.12.12.123\r\n" \
49 "m=audio 5904 RTP/AVP 97\r\n" \
50 "a=rtpmap:97 GSM-EFR/8000\r\n"
51
52#define CRCX_RET "200 2 OK\r\n" \
53 "I: 1\n" \
54 "\n" \
55 "v=0\r\n" \
56 "o=- 1 23 IN IP4 0.0.0.0\r\n" \
57 "c=IN IP4 0.0.0.0\r\n" \
58 "t=0 0\r\n" \
59 "m=audio 0 RTP/AVP 126\r\n" \
60 "a=rtpmap:126 AMR/8000\r\n"
61
62
63#define CRCX_ZYN "CRCX 2 1@mgw MGCP 1.0\r" \
64 "M: sendrecv\r" \
65 "C: 2\r\r" \
66 "v=0\r" \
67 "c=IN IP4 123.12.12.123\r" \
68 "m=audio 5904 RTP/AVP 97\r" \
69 "a=rtpmap:97 GSM-EFR/8000\r"
Harald Weltea7c9a922012-01-27 00:36:56 +010070
Holger Hans Peter Freyther0e8e18e2012-12-07 14:42:03 +010071#define CRCX_ZYN_RET "200 2 OK\r\n" \
72 "I: 2\n" \
73 "\n" \
74 "v=0\r\n" \
75 "o=- 2 23 IN IP4 0.0.0.0\r\n" \
76 "c=IN IP4 0.0.0.0\r\n" \
77 "t=0 0\r\n" \
78 "m=audio 0 RTP/AVP 126\r\n" \
79 "a=rtpmap:126 AMR/8000\r\n"
80
Holger Hans Peter Freyther8fc6d082012-09-14 17:18:12 +020081#define DLCX "DLCX 7 1@mgw MGCP 1.0\r\n" \
82 "C: 2\r\n"
83
84#define DLCX_RET "250 7 OK\r\n" \
Holger Hans Peter Freyther8c87a482012-10-24 21:22:47 +020085 "P: PS=0, OS=0, PR=0, OR=0, PL=0, JI=0\r\n"
Holger Hans Peter Freyther8fc6d082012-09-14 17:18:12 +020086
Holger Hans Peter Freyther46880c22012-11-29 12:54:22 +010087#define RQNT "RQNT 186908780 1@mgw MGCP 1.0\r\n" \
88 "X: B244F267488\r\n" \
89 "S: D/9\r\n"
90
91#define RQNT2 "RQNT 186908780 1@mgw MGCP 1.0\r\n" \
92 "X: ADD4F26746F\r\n" \
93 "R: D/[0-9#*](N), G/ft, fxr/t38\r\n"
94
95#define RQNT_RET "200 186908780 OK\r\n"
Holger Hans Peter Freyther8fc6d082012-09-14 17:18:12 +020096
Harald Weltea7c9a922012-01-27 00:36:56 +010097struct mgcp_test {
98 const char *name;
99 const char *req;
100 const char *exp_resp;
101};
102
Holger Hans Peter Freyther0e8e18e2012-12-07 14:42:03 +0100103static const struct mgcp_test tests[] = {
Harald Weltea7c9a922012-01-27 00:36:56 +0100104 { "AUEP1", AUEP1, AUEP1_RET },
105 { "AUEP2", AUEP2, AUEP2_RET },
Harald Welteca765cd2012-01-27 00:37:23 +0100106 { "CRCX", CRCX, CRCX_RET },
Holger Hans Peter Freyther0e8e18e2012-12-07 14:42:03 +0100107 { "DLCX", DLCX, DLCX_RET },
108 { "CRCX_ZYN", CRCX_ZYN, CRCX_ZYN_RET },
Holger Hans Peter Freyther0a6455c2012-03-16 09:53:49 +0100109 { "EMPTY", EMPTY, EMPTY_RET },
110 { "SHORT1", SHORT, SHORT_RET },
111 { "SHORT2", SHORT2, SHORT2_RET },
112 { "SHORT3", SHORT3, SHORT2_RET },
113 { "SHORT4", SHORT4, SHORT2_RET },
Holger Hans Peter Freyther46880c22012-11-29 12:54:22 +0100114 { "RQNT1", RQNT, RQNT_RET },
115 { "RQNT2", RQNT2, RQNT_RET },
Holger Hans Peter Freyther8fc6d082012-09-14 17:18:12 +0200116 { "DLCX", DLCX, DLCX_RET },
Harald Weltea7c9a922012-01-27 00:36:56 +0100117};
Harald Welte834f3f12012-01-27 00:00:44 +0100118
119static struct msgb *create_msg(const char *str)
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100120{
121 struct msgb *msg;
122
123 msg = msgb_alloc_headroom(4096, 128, "MGCP msg");
Harald Welte834f3f12012-01-27 00:00:44 +0100124 int len = sprintf((char *)msg->data, str);
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100125 msg->l2h = msgb_put(msg, len);
126 return msg;
127}
128
Holger Hans Peter Freyther06ce8392012-03-16 09:45:13 +0100129static void test_messages(void)
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100130{
Holger Hans Peter Freyther38484aa2012-01-06 15:04:43 +0100131 struct mgcp_config *cfg;
Harald Weltea7c9a922012-01-27 00:36:56 +0100132 int i;
Holger Hans Peter Freyther38484aa2012-01-06 15:04:43 +0100133
134 cfg = mgcp_config_alloc();
Harald Weltea7c9a922012-01-27 00:36:56 +0100135
Holger Hans Peter Freytherab56ce12011-02-28 00:56:17 +0100136 cfg->trunk.number_endpoints = 64;
Holger Hans Peter Freyther49e170b2011-02-28 14:37:03 +0100137 mgcp_endpoints_allocate(&cfg->trunk);
138
139 mgcp_endpoints_allocate(mgcp_trunk_alloc(cfg, 1));
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100140
Harald Weltea7c9a922012-01-27 00:36:56 +0100141 for (i = 0; i < ARRAY_SIZE(tests); i++) {
142 const struct mgcp_test *t = &tests[i];
143 struct msgb *inp;
144 struct msgb *msg;
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100145
Harald Weltea7c9a922012-01-27 00:36:56 +0100146 printf("Testing %s\n", t->name);
147
148 inp = create_msg(t->req);
149 msg = mgcp_handle_message(cfg, inp);
150 msgb_free(inp);
Holger Hans Peter Freyther0a6455c2012-03-16 09:53:49 +0100151 if (!t->exp_resp) {
152 if (msg)
153 printf("%s failed '%s'\n", t->name, (char *) msg->data);
154 } else if (strcmp((char *) msg->data, t->exp_resp) != 0)
Harald Weltea7c9a922012-01-27 00:36:56 +0100155 printf("%s failed '%s'\n", t->name, (char *) msg->data);
156 msgb_free(msg);
157 }
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100158
159 talloc_free(cfg);
160}
161
Holger Hans Peter Freyther46880c22012-11-29 12:54:22 +0100162static int rqnt_cb(struct mgcp_endpoint *endp, char _tone, const char *data)
163{
164 ptrdiff_t tone = _tone;
165 endp->cfg->data = (void *) tone;
166 return 0;
167}
168
169static void test_rqnt_cb(void)
170{
171 struct mgcp_config *cfg;
172 struct msgb *inp, *msg;
173
174 cfg = mgcp_config_alloc();
175 cfg->rqnt_cb = rqnt_cb;
176
177 cfg->trunk.number_endpoints = 64;
178 mgcp_endpoints_allocate(&cfg->trunk);
179
180 mgcp_endpoints_allocate(mgcp_trunk_alloc(cfg, 1));
181
182 inp = create_msg(CRCX);
183 msgb_free(mgcp_handle_message(cfg, inp));
184 msgb_free(inp);
185
186 /* send the RQNT and check for the CB */
187 inp = create_msg(RQNT);
188 msg = mgcp_handle_message(cfg, inp);
189 if (strncmp((const char *) msg->l2h, "200", 3) != 0) {
190 printf("FAILED: message is not 200. '%s'\n", msg->l2h);
191 abort();
192 }
193
194 if (cfg->data != (void *) '9') {
195 printf("FAILED: callback not called: %p\n", cfg->data);
196 abort();
197 }
198
199 msgb_free(msg);
200 msgb_free(inp);
201
202 inp = create_msg(DLCX);
203 msgb_free(mgcp_handle_message(cfg, inp));
204 msgb_free(inp);
205 talloc_free(cfg);
206}
207
Holger Hans Peter Freytherfaa67932012-10-22 18:09:35 +0200208struct pl_test {
209 int cycles;
210 uint16_t base_seq;
211 uint16_t max_seq;
212 uint32_t packets;
213
214 uint32_t expected;
215 int loss;
216};
217
218static const struct pl_test pl_test_dat[] = {
219 /* basic.. just one package */
220 { .cycles = 0, .base_seq = 0, .max_seq = 0, .packets = 1, .expected = 1, .loss = 0},
221 /* some packages and a bit of loss */
222 { .cycles = 0, .base_seq = 0, .max_seq = 100, .packets = 100, .expected = 101, .loss = 1},
223 /* wrap around */
224 { .cycles = 1<<16, .base_seq = 0xffff, .max_seq = 2, .packets = 4, .expected = 4, .loss = 0},
225 /* min loss */
226 { .cycles = 0, .base_seq = 0, .max_seq = 0, .packets = UINT_MAX, .expected = 1, .loss = INT_MIN },
227 /* max loss, with wrap around on expected max */
228 { .cycles = INT_MAX, .base_seq = 0, .max_seq = UINT16_MAX, .packets = 0, .expected = ((uint32_t)(INT_MAX) + UINT16_MAX + 1), .loss = INT_MAX },
229};
230
231static void test_packet_loss_calc(void)
232{
233 int i;
234 printf("Testing packet loss calculation.\n");
235
236 for (i = 0; i < ARRAY_SIZE(pl_test_dat); ++i) {
237 uint32_t expected;
238 int loss;
239 struct mgcp_rtp_state state;
240 struct mgcp_rtp_end rtp;
241 memset(&state, 0, sizeof(state));
242 memset(&rtp, 0, sizeof(rtp));
243
244 state.initialized = 1;
245 state.base_seq = pl_test_dat[i].base_seq;
246 state.max_seq = pl_test_dat[i].max_seq;
247 state.cycles = pl_test_dat[i].cycles;
248
249 rtp.packets = pl_test_dat[i].packets;
250 mgcp_state_calc_loss(&state, &rtp, &expected, &loss);
251
252 if (loss != pl_test_dat[i].loss || expected != pl_test_dat[i].expected) {
253 printf("FAIL: Wrong exp/loss at idx(%d) Loss(%d vs. %d) Exp(%u vs. %u)\n",
254 i, loss, pl_test_dat[i].loss,
255 expected, pl_test_dat[i].expected);
256 }
257 }
258}
259
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100260int main(int argc, char **argv)
261{
Holger Hans Peter Freyther2d59cc62011-05-12 16:02:07 +0200262 osmo_init_logging(&log_info);
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100263
Holger Hans Peter Freyther06ce8392012-03-16 09:45:13 +0100264 test_messages();
Holger Hans Peter Freytherfaa67932012-10-22 18:09:35 +0200265 test_packet_loss_calc();
Holger Hans Peter Freyther46880c22012-11-29 12:54:22 +0100266 test_rqnt_cb();
Holger Hans Peter Freyther38484aa2012-01-06 15:04:43 +0100267
268 printf("Done\n");
269 return EXIT_SUCCESS;
Holger Hans Peter Freytherf0bc0752011-01-06 19:32:52 +0100270}