blob: 48241a642f0ff61dcd3381ef0fd3419edf995519 [file] [log] [blame]
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +02001/* A Media Gateway Control Protocol Media Gateway: RFC 3435 */
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +01002/* The main method to drive it as a standalone process */
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +02003
4/*
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +02005 * (C) 2009-2011 by Holger Hans Peter Freyther <zecke@selfish.org>
6 * (C) 2009-2011 by On-Waves
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +02007 * All Rights Reserved
8 *
9 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +010010 * it under the terms of the GNU Affero General Public License as published by
11 * the Free Software Foundation; either version 3 of the License, or
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020012 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte9af6ddf2011-01-01 15:25:50 +010017 * GNU Affero General Public License for more details.
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020018 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010019 * You should have received a copy of the GNU Affero General Public License
20 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020021 *
22 */
23
24#include <ctype.h>
25#include <stdio.h>
26#include <stdlib.h>
27#include <string.h>
28#include <time.h>
29#include <limits.h>
30#include <unistd.h>
Neels Hofmeyre9920f22017-07-10 15:07:22 +020031#include <errno.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020032
33#include <sys/socket.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020034
Neels Hofmeyre9920f22017-07-10 15:07:22 +020035#include <osmocom/legacy_mgcp/mgcp.h>
36#include <osmocom/legacy_mgcp/mgcp_internal.h>
37#include <osmocom/legacy_mgcp/vty.h>
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +020038
39#include <osmocom/core/application.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010040#include <osmocom/core/msgb.h>
41#include <osmocom/core/talloc.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010042#include <osmocom/core/select.h>
Holger Hans Peter Freyther292769e2015-11-02 16:07:57 +010043#include <osmocom/core/stats.h>
44#include <osmocom/core/rate_ctr.h>
Neels Hofmeyre9920f22017-07-10 15:07:22 +020045#include <osmocom/core/logging.h>
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +020046
Harald Welte4d54d0b2011-02-19 16:48:17 +010047#include <osmocom/vty/telnet_interface.h>
48#include <osmocom/vty/logging.h>
Harald Welte55dc31e2014-08-24 17:54:49 +020049#include <osmocom/vty/ports.h>
Harald Welte4b037e42010-05-19 19:45:32 +020050#include <osmocom/vty/command.h>
Holger Hans Peter Freyther292769e2015-11-02 16:07:57 +010051#include <osmocom/vty/stats.h>
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +020052
Harald Welte5a29c7f2010-03-23 00:09:32 +080053#include "../../bscconfig.h"
54
Jacob Erlbeck239a8532014-03-13 14:25:51 +010055#ifdef BUILD_MGCP_TRANSCODING
Neels Hofmeyre9920f22017-07-10 15:07:22 +020056#include <osmocom/legacy_mgcp/mgcp_transcode.h>
Jacob Erlbeck239a8532014-03-13 14:25:51 +010057#endif
58
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020059#define _GNU_SOURCE
60#include <getopt.h>
61
62#warning "Make use of the rtp proxy code"
63
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010064static struct mgcp_config *cfg;
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +020065static struct mgcp_trunk_config *reset_trunk;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020066static int reset_endpoints = 0;
Harald Welte2c869ef2010-08-25 19:43:54 +020067static int daemonize = 0;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020068
Holger Hans Peter Freytheracf8a0c2010-03-29 08:47:44 +020069const char *openbsc_copyright =
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080070 "Copyright (C) 2009-2010 Holger Freyther and On-Waves\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010071 "Contributions by Daniel Willmann, Jan Lübbe, Stefan Schmidt\r\n"
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080072 "Dieter Spaar, Andreas Eversberg, Harald Welte\r\n\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010073 "License AGPLv3+: GNU AGPL version 3 or later <http://gnu.org/licenses/agpl-3.0.html>\r\n"
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080074 "This is free software: you are free to change and redistribute it.\r\n"
75 "There is NO WARRANTY, to the extent permitted by law.\r\n";
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020076
77static char *config_file = "mgcp.cfg";
78
79/* used by msgb and mgcp */
80void *tall_bsc_ctx = NULL;
81
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020082static void print_help()
83{
84 printf("Some useful help...\n");
85 printf(" -h --help is printing this text.\n");
86 printf(" -c --config-file filename The config file to use.\n");
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +020087 printf(" -s --disable-color\n");
Harald Welte6b36b902012-01-27 13:21:46 +010088 printf(" -D --daemonize Fork the process into a background daemon\n");
89 printf(" -V --version Print the version number\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020090}
91
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +080092static void handle_options(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020093{
94 while (1) {
95 int option_index = 0, c;
96 static struct option long_options[] = {
97 {"help", 0, 0, 'h'},
98 {"config-file", 1, 0, 'c'},
Harald Welte2c869ef2010-08-25 19:43:54 +020099 {"daemonize", 0, 0, 'D'},
Harald Welte5a29c7f2010-03-23 00:09:32 +0800100 {"version", 0, 0, 'V'},
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +0200101 {"disable-color", 0, 0, 's'},
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200102 {0, 0, 0, 0},
103 };
104
Harald Welte2c869ef2010-08-25 19:43:54 +0200105 c = getopt_long(argc, argv, "hc:VD", long_options, &option_index);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200106
107 if (c == -1)
108 break;
109
110 switch(c) {
111 case 'h':
112 print_help();
113 exit(0);
114 break;
115 case 'c':
116 config_file = talloc_strdup(tall_bsc_ctx, optarg);
117 break;
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +0200118 case 's':
119 log_set_use_color(osmo_stderr_target, 0);
120 break;
Harald Welte5a29c7f2010-03-23 00:09:32 +0800121 case 'V':
Harald Welte1353f962010-05-16 19:20:24 +0200122 print_version(1);
Harald Welte5a29c7f2010-03-23 00:09:32 +0800123 exit(0);
124 break;
Harald Welte2c869ef2010-08-25 19:43:54 +0200125 case 'D':
126 daemonize = 1;
127 break;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200128 default:
129 /* ignore */
130 break;
131 };
132 }
133}
134
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200135/* simply remember this */
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200136static int mgcp_rsip_cb(struct mgcp_trunk_config *tcfg)
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200137{
138 reset_endpoints = 1;
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200139 reset_trunk = tcfg;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200140
141 return 0;
142}
143
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200144static int read_call_agent(struct osmo_fd *fd, unsigned int what)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200145{
146 struct sockaddr_in addr;
147 socklen_t slen = sizeof(addr);
148 struct msgb *msg;
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100149 struct msgb *resp;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200150 int i;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200151
152 msg = (struct msgb *) fd->data;
153
154 /* read one less so we can use it as a \0 */
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800155 int rc = recvfrom(cfg->gw_fd.bfd.fd, msg->data, msg->data_len - 1, 0,
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200156 (struct sockaddr *) &addr, &slen);
157 if (rc < 0) {
158 perror("Gateway failed to read");
159 return -1;
160 } else if (slen > sizeof(addr)) {
Holger Hans Peter Freyther2f2be572012-03-16 12:18:39 +0100161 fprintf(stderr, "Gateway received message from outerspace: %zu %zu\n",
Holger Hans Peter Freytherdaaea0c2015-08-03 09:28:41 +0200162 (size_t) slen, sizeof(addr));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200163 return -1;
164 }
165
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200166 /* handle message now */
167 msg->l2h = msgb_put(msg, rc);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100168 resp = mgcp_handle_message(cfg, msg);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200169 msgb_reset(msg);
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100170
171 if (resp) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800172 sendto(cfg->gw_fd.bfd.fd, resp->l2h, msgb_l2len(resp), 0, (struct sockaddr *) &addr, sizeof(addr));
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100173 msgb_free(resp);
174 }
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200175
176 if (reset_endpoints) {
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200177 LOGP(DLMGCP, LOGL_NOTICE,
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200178 "Asked to reset endpoints: %d/%d\n",
179 reset_trunk->trunk_nr, reset_trunk->trunk_type);
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200180 reset_endpoints = 0;
181
182 /* is checking in_addr.s_addr == INADDR_LOOPBACK making it more secure? */
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200183 for (i = 1; i < reset_trunk->number_endpoints; ++i)
Holger Hans Peter Freythercb6ad702014-07-22 15:00:52 +0200184 mgcp_release_endp(&reset_trunk->endpoints[i]);
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200185 }
186
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200187 return 0;
188}
189
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200190int mgcp_vty_is_config_node(struct vty *vty, int node)
191{
192 switch (node) {
193 case CONFIG_NODE:
194 return 0;
195
196 default:
197 return 1;
198 }
199}
200
201int mgcp_vty_go_parent(struct vty *vty)
202{
203 switch (vty->node) {
204 case TRUNK_NODE:
205 vty->node = MGCP_NODE;
206 vty->index = NULL;
207 break;
208 case MGCP_NODE:
209 default:
210 if (mgcp_vty_is_config_node(vty, vty->node))
211 vty->node = CONFIG_NODE;
212 else
213 vty->node = ENABLE_NODE;
214
215 vty->index = NULL;
216 }
217
218 return vty->node;
219}
220
Harald Weltec31f4802010-05-25 23:31:39 +0200221
222static struct vty_app_info vty_info = {
223 .name = "OpenBSC MGCP",
224 .version = PACKAGE_VERSION,
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200225 .go_parent_cb = mgcp_vty_go_parent,
226 .is_config_node = mgcp_vty_is_config_node,
227};
228
229static const struct log_info_cat log_categories[] = {
230 /* DLMGCP is provided by the MGCP library */
231};
232
233const struct log_info log_info = {
234 .cat = log_categories,
235 .num_cat = ARRAY_SIZE(log_categories),
Harald Weltec31f4802010-05-25 23:31:39 +0200236};
Holger Hans Peter Freytherbd5130d2010-01-13 23:36:53 +0100237
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +0800238int main(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200239{
240 struct sockaddr_in addr;
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100241 int on = 1, rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200242
243 tall_bsc_ctx = talloc_named_const(NULL, 1, "mgcp-callagent");
Neels Hofmeyr4c2d4ab2016-09-16 02:31:17 +0200244 msgb_talloc_ctx_init(tall_bsc_ctx, 0);
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100245
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +0200246 osmo_init_ignore_signals();
247 osmo_init_logging(&log_info);
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100248
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100249 cfg = mgcp_config_alloc();
250 if (!cfg)
251 return -1;
252
Jacob Erlbeck239a8532014-03-13 14:25:51 +0100253#ifdef BUILD_MGCP_TRANSCODING
254 cfg->setup_rtp_processing_cb = &mgcp_transcoding_setup;
255 cfg->rtp_processing_cb = &mgcp_transcoding_process_rtp;
256 cfg->get_net_downlink_format_cb = &mgcp_transcoding_net_downlink_format;
257#endif
258
Philipp Maierdd9f6cd2017-08-08 12:04:30 +0200259 cfg->trunk.force_realloc = 1;
260
Harald Weltec31f4802010-05-25 23:31:39 +0200261 vty_info.copyright = openbsc_copyright;
262 vty_init(&vty_info);
Maxdb0e3802017-01-12 19:35:11 +0100263 logging_vty_add_cmds(NULL);
Holger Hans Peter Freyther292769e2015-11-02 16:07:57 +0100264 osmo_stats_vty_add_cmds(&log_info);
Harald Weltedcccb182010-05-16 20:52:23 +0200265 mgcp_vty_init();
Harald Welte1353f962010-05-16 19:20:24 +0200266
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200267 handle_options(argc, argv);
268
Holger Hans Peter Freyther292769e2015-11-02 16:07:57 +0100269 rate_ctr_init(tall_bsc_ctx);
270 osmo_stats_init(tall_bsc_ctx);
271
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200272 rc = mgcp_parse_config(config_file, cfg, MGCP_BSC);
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100273 if (rc < 0)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200274 return rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200275
Neels Hofmeyrfa0f7152016-02-23 14:09:38 +0100276 /* start telnet after reading config for vty_get_bind_addr() */
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200277 rc = telnet_init_dynif(tall_bsc_ctx, NULL,
Neels Hofmeyrfa0f7152016-02-23 14:09:38 +0100278 vty_get_bind_addr(), OSMO_VTY_PORT_BSC_MGCP);
Harald Weltedcccb182010-05-16 20:52:23 +0200279 if (rc < 0)
280 return rc;
281
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200282 /* set some callbacks */
283 cfg->reset_cb = mgcp_rsip_cb;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200284
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100285 /* we need to bind a socket */
286 if (rc == 0) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800287 cfg->gw_fd.bfd.when = BSC_FD_READ;
288 cfg->gw_fd.bfd.cb = read_call_agent;
289 cfg->gw_fd.bfd.fd = socket(AF_INET, SOCK_DGRAM, 0);
290 if (cfg->gw_fd.bfd.fd < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100291 perror("Gateway failed to listen");
292 return -1;
293 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200294
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800295 setsockopt(cfg->gw_fd.bfd.fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200296
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100297 memset(&addr, 0, sizeof(addr));
298 addr.sin_family = AF_INET;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100299 addr.sin_port = htons(cfg->source_port);
300 inet_aton(cfg->source_addr, &addr.sin_addr);
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100301
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800302 if (bind(cfg->gw_fd.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100303 perror("Gateway failed to bind");
304 return -1;
305 }
306
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800307 cfg->gw_fd.bfd.data = msgb_alloc(4096, "mgcp-msg");
308 if (!cfg->gw_fd.bfd.data) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100309 fprintf(stderr, "Gateway memory error.\n");
310 return -1;
311 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200312
Harald Welte50f93a42012-01-27 00:38:59 +0100313 if (cfg->call_agent_addr) {
314 addr.sin_port = htons(2727);
315 inet_aton(cfg->call_agent_addr, &addr.sin_addr);
316 if (connect(cfg->gw_fd.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200317 LOGP(DLMGCP, LOGL_ERROR, "Failed to connect to: '%s'. errno: %d\n",
Harald Welte50f93a42012-01-27 00:38:59 +0100318 cfg->call_agent_addr, errno);
319 close(cfg->gw_fd.bfd.fd);
320 cfg->gw_fd.bfd.fd = -1;
321 return -1;
322 }
323 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200324
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200325 if (osmo_fd_register(&cfg->gw_fd.bfd) != 0) {
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200326 LOGP(DLMGCP, LOGL_FATAL, "Failed to register the fd\n");
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100327 return -1;
328 }
Holger Hans Peter Freytherf986cfc2010-01-05 12:25:25 +0100329
Neels Hofmeyre9920f22017-07-10 15:07:22 +0200330 LOGP(DLMGCP, LOGL_NOTICE, "Configured for MGCP.\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200331 }
332
333 /* initialisation */
334 srand(time(NULL));
335
Harald Welte2c869ef2010-08-25 19:43:54 +0200336 if (daemonize) {
337 rc = osmo_daemonize();
338 if (rc < 0) {
339 perror("Error during daemonize");
340 exit(1);
341 }
342 }
343
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200344 /* main loop */
345 while (1) {
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200346 osmo_select_main(0);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200347 }
348
349
350 return 0;
351}