blob: 540f7602bcc3cc8f920663befaddf9a55879f024 [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>
31
32#include <sys/socket.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020033
34#include <openbsc/debug.h>
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +020035#include <openbsc/gsm_data.h>
36#include <openbsc/mgcp.h>
37#include <openbsc/mgcp_internal.h>
38#include <openbsc/vty.h>
39
40#include <osmocom/core/application.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010041#include <osmocom/core/msgb.h>
42#include <osmocom/core/talloc.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010043#include <osmocom/core/select.h>
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +020044
Harald Welte4d54d0b2011-02-19 16:48:17 +010045#include <osmocom/vty/telnet_interface.h>
46#include <osmocom/vty/logging.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020047
Harald Welte4b037e42010-05-19 19:45:32 +020048#include <osmocom/vty/command.h>
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +020049
Harald Welte5a29c7f2010-03-23 00:09:32 +080050#include "../../bscconfig.h"
51
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020052/* this is here for the vty... it will never be called */
53void subscr_put() { abort(); }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020054
55#define _GNU_SOURCE
56#include <getopt.h>
57
58#warning "Make use of the rtp proxy code"
59
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010060static struct mgcp_config *cfg;
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +020061static struct mgcp_trunk_config *reset_trunk;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020062static int reset_endpoints = 0;
Harald Welte2c869ef2010-08-25 19:43:54 +020063static int daemonize = 0;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020064
Holger Hans Peter Freytheracf8a0c2010-03-29 08:47:44 +020065const char *openbsc_copyright =
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080066 "Copyright (C) 2009-2010 Holger Freyther and On-Waves\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010067 "Contributions by Daniel Willmann, Jan Lübbe, Stefan Schmidt\r\n"
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080068 "Dieter Spaar, Andreas Eversberg, Harald Welte\r\n\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010069 "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 +080070 "This is free software: you are free to change and redistribute it.\r\n"
71 "There is NO WARRANTY, to the extent permitted by law.\r\n";
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020072
73static char *config_file = "mgcp.cfg";
74
75/* used by msgb and mgcp */
76void *tall_bsc_ctx = NULL;
77
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020078static void print_help()
79{
80 printf("Some useful help...\n");
81 printf(" -h --help is printing this text.\n");
82 printf(" -c --config-file filename The config file to use.\n");
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +020083 printf(" -s --disable-color\n");
Harald Welte6b36b902012-01-27 13:21:46 +010084 printf(" -D --daemonize Fork the process into a background daemon\n");
85 printf(" -V --version Print the version number\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020086}
87
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +080088static void handle_options(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020089{
90 while (1) {
91 int option_index = 0, c;
92 static struct option long_options[] = {
93 {"help", 0, 0, 'h'},
94 {"config-file", 1, 0, 'c'},
Harald Welte2c869ef2010-08-25 19:43:54 +020095 {"daemonize", 0, 0, 'D'},
Harald Welte5a29c7f2010-03-23 00:09:32 +080096 {"version", 0, 0, 'V'},
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +020097 {"disable-color", 0, 0, 's'},
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020098 {0, 0, 0, 0},
99 };
100
Harald Welte2c869ef2010-08-25 19:43:54 +0200101 c = getopt_long(argc, argv, "hc:VD", long_options, &option_index);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200102
103 if (c == -1)
104 break;
105
106 switch(c) {
107 case 'h':
108 print_help();
109 exit(0);
110 break;
111 case 'c':
112 config_file = talloc_strdup(tall_bsc_ctx, optarg);
113 break;
Holger Hans Peter Freyther778695d2012-09-20 16:04:54 +0200114 case 's':
115 log_set_use_color(osmo_stderr_target, 0);
116 break;
Harald Welte5a29c7f2010-03-23 00:09:32 +0800117 case 'V':
Harald Welte1353f962010-05-16 19:20:24 +0200118 print_version(1);
Harald Welte5a29c7f2010-03-23 00:09:32 +0800119 exit(0);
120 break;
Harald Welte2c869ef2010-08-25 19:43:54 +0200121 case 'D':
122 daemonize = 1;
123 break;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200124 default:
125 /* ignore */
126 break;
127 };
128 }
129}
130
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200131/* simply remember this */
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200132static int mgcp_rsip_cb(struct mgcp_trunk_config *tcfg)
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200133{
134 reset_endpoints = 1;
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200135 reset_trunk = tcfg;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200136
137 return 0;
138}
139
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100140static int mgcp_change_cb(struct mgcp_trunk_config *cfg, int endpoint, int state)
Holger Hans Peter Freytherdc417552010-04-22 13:23:05 +0800141{
142 if (state != MGCP_ENDP_MDCX)
143 return 0;
144
145 mgcp_send_dummy(&cfg->endpoints[endpoint]);
146 return 0;
147}
148
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200149static int read_call_agent(struct osmo_fd *fd, unsigned int what)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200150{
151 struct sockaddr_in addr;
152 socklen_t slen = sizeof(addr);
153 struct msgb *msg;
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100154 struct msgb *resp;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200155 int i;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200156
157 msg = (struct msgb *) fd->data;
158
159 /* read one less so we can use it as a \0 */
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800160 int rc = recvfrom(cfg->gw_fd.bfd.fd, msg->data, msg->data_len - 1, 0,
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200161 (struct sockaddr *) &addr, &slen);
162 if (rc < 0) {
163 perror("Gateway failed to read");
164 return -1;
165 } else if (slen > sizeof(addr)) {
Holger Hans Peter Freyther2f2be572012-03-16 12:18:39 +0100166 fprintf(stderr, "Gateway received message from outerspace: %zu %zu\n",
167 slen, sizeof(addr));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200168 return -1;
169 }
170
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200171 /* handle message now */
172 msg->l2h = msgb_put(msg, rc);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100173 resp = mgcp_handle_message(cfg, msg);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200174 msgb_reset(msg);
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100175
176 if (resp) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800177 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 +0100178 msgb_free(resp);
179 }
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200180
181 if (reset_endpoints) {
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200182 LOGP(DMGCP, LOGL_NOTICE,
183 "Asked to reset endpoints: %d/%d\n",
184 reset_trunk->trunk_nr, reset_trunk->trunk_type);
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200185 reset_endpoints = 0;
186
187 /* is checking in_addr.s_addr == INADDR_LOOPBACK making it more secure? */
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +0200188 for (i = 1; i < reset_trunk->number_endpoints; ++i)
189 mgcp_free_endp(&reset_trunk->endpoints[i]);
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200190 }
191
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200192 return 0;
193}
194
Holger Hans Peter Freyther57da4472010-06-08 16:11:06 +0800195extern enum node_type bsc_vty_go_parent(struct vty *vty);
Harald Weltec31f4802010-05-25 23:31:39 +0200196
197static struct vty_app_info vty_info = {
198 .name = "OpenBSC MGCP",
199 .version = PACKAGE_VERSION,
200 .go_parent_cb = bsc_vty_go_parent,
Holger Hans Peter Freyther81506b42010-09-04 11:00:01 +0800201 .is_config_node = bsc_vty_is_config_node,
Harald Weltec31f4802010-05-25 23:31:39 +0200202};
Holger Hans Peter Freytherbd5130d2010-01-13 23:36:53 +0100203
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +0800204int main(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200205{
Holger Hans Peter Freyther338fa562009-11-19 15:03:39 +0100206 struct gsm_network dummy_network;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200207 struct sockaddr_in addr;
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100208 int on = 1, rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200209
210 tall_bsc_ctx = talloc_named_const(NULL, 1, "mgcp-callagent");
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100211
Holger Hans Peter Freyther67cd75f2011-05-12 16:02:07 +0200212 osmo_init_ignore_signals();
213 osmo_init_logging(&log_info);
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100214
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100215 cfg = mgcp_config_alloc();
216 if (!cfg)
217 return -1;
218
Harald Weltec31f4802010-05-25 23:31:39 +0200219 vty_info.copyright = openbsc_copyright;
220 vty_init(&vty_info);
Pablo Neira Ayuso739a5662011-03-09 13:36:32 +0100221 logging_vty_add_cmds(&log_info);
Harald Weltedcccb182010-05-16 20:52:23 +0200222 mgcp_vty_init();
Harald Welte1353f962010-05-16 19:20:24 +0200223
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200224 handle_options(argc, argv);
225
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100226 rc = mgcp_parse_config(config_file, cfg);
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100227 if (rc < 0)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200228 return rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200229
Harald Weltedcccb182010-05-16 20:52:23 +0200230 rc = telnet_init(tall_bsc_ctx, &dummy_network, 4243);
231 if (rc < 0)
232 return rc;
233
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200234 /* set some callbacks */
235 cfg->reset_cb = mgcp_rsip_cb;
Holger Hans Peter Freytherdc417552010-04-22 13:23:05 +0800236 cfg->change_cb = mgcp_change_cb;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200237
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100238 /* we need to bind a socket */
239 if (rc == 0) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800240 cfg->gw_fd.bfd.when = BSC_FD_READ;
241 cfg->gw_fd.bfd.cb = read_call_agent;
242 cfg->gw_fd.bfd.fd = socket(AF_INET, SOCK_DGRAM, 0);
243 if (cfg->gw_fd.bfd.fd < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100244 perror("Gateway failed to listen");
245 return -1;
246 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200247
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800248 setsockopt(cfg->gw_fd.bfd.fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200249
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100250 memset(&addr, 0, sizeof(addr));
251 addr.sin_family = AF_INET;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100252 addr.sin_port = htons(cfg->source_port);
253 inet_aton(cfg->source_addr, &addr.sin_addr);
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100254
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800255 if (bind(cfg->gw_fd.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100256 perror("Gateway failed to bind");
257 return -1;
258 }
259
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800260 cfg->gw_fd.bfd.data = msgb_alloc(4096, "mgcp-msg");
261 if (!cfg->gw_fd.bfd.data) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100262 fprintf(stderr, "Gateway memory error.\n");
263 return -1;
264 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200265
Harald Welte50f93a42012-01-27 00:38:59 +0100266 if (cfg->call_agent_addr) {
267 addr.sin_port = htons(2727);
268 inet_aton(cfg->call_agent_addr, &addr.sin_addr);
269 if (connect(cfg->gw_fd.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
270 LOGP(DMGCP, LOGL_ERROR, "Failed to connect to: '%s'. errno: %d\n",
271 cfg->call_agent_addr, errno);
272 close(cfg->gw_fd.bfd.fd);
273 cfg->gw_fd.bfd.fd = -1;
274 return -1;
275 }
276 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200277
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200278 if (osmo_fd_register(&cfg->gw_fd.bfd) != 0) {
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200279 LOGP(DMGCP, LOGL_FATAL, "Failed to register the fd\n");
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100280 return -1;
281 }
Holger Hans Peter Freytherf986cfc2010-01-05 12:25:25 +0100282
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200283 LOGP(DMGCP, LOGL_NOTICE, "Configured for MGCP.\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200284 }
285
286 /* initialisation */
287 srand(time(NULL));
288
Harald Welte2c869ef2010-08-25 19:43:54 +0200289 if (daemonize) {
290 rc = osmo_daemonize();
291 if (rc < 0) {
292 perror("Error during daemonize");
293 exit(1);
294 }
295 }
296
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200297 /* main loop */
298 while (1) {
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200299 osmo_select_main(0);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200300 }
301
302
303 return 0;
304}