blob: 20b9836e3bb5b0e0b9aa53cb83fc20eada7c1bb7 [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/*
5 * (C) 2009 by Holger Hans Peter Freyther <zecke@selfish.org>
Holger Hans Peter Freyther92026f12010-02-03 18:10:07 +01006 * (C) 2009 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>
Harald Weltedfe6c7d2010-02-20 16:24:02 +010035#include <osmocore/msgb.h>
36#include <osmocore/talloc.h>
Harald Welte2c869ef2010-08-25 19:43:54 +020037#include <osmocore/process.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020038#include <openbsc/gsm_data.h>
Harald Weltedfe6c7d2010-02-20 16:24:02 +010039#include <osmocore/select.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020040#include <openbsc/mgcp.h>
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020041#include <openbsc/mgcp_internal.h>
Harald Welte4b037e42010-05-19 19:45:32 +020042#include <osmocom/vty//telnet_interface.h>
Holger Hans Peter Freyther44127372010-04-06 12:00:43 +020043#include <openbsc/vty.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020044
Harald Welte4b037e42010-05-19 19:45:32 +020045#include <osmocom/vty/command.h>
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +020046
Harald Welte5a29c7f2010-03-23 00:09:32 +080047#include "../../bscconfig.h"
48
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020049/* this is here for the vty... it will never be called */
50void subscr_put() { abort(); }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020051
52#define _GNU_SOURCE
53#include <getopt.h>
54
55#warning "Make use of the rtp proxy code"
56
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010057static struct mgcp_config *cfg;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020058static int reset_endpoints = 0;
Harald Welte2c869ef2010-08-25 19:43:54 +020059static int daemonize = 0;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020060
Holger Hans Peter Freytheracf8a0c2010-03-29 08:47:44 +020061const char *openbsc_copyright =
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080062 "Copyright (C) 2009-2010 Holger Freyther and On-Waves\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010063 "Contributions by Daniel Willmann, Jan Lübbe, Stefan Schmidt\r\n"
Holger Hans Peter Freyther5f540752010-09-11 13:32:30 +080064 "Dieter Spaar, Andreas Eversberg, Harald Welte\r\n\r\n"
Harald Welte9af6ddf2011-01-01 15:25:50 +010065 "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 +080066 "This is free software: you are free to change and redistribute it.\r\n"
67 "There is NO WARRANTY, to the extent permitted by law.\r\n";
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020068
69static char *config_file = "mgcp.cfg";
70
71/* used by msgb and mgcp */
72void *tall_bsc_ctx = NULL;
73
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020074static void print_help()
75{
76 printf("Some useful help...\n");
77 printf(" -h --help is printing this text.\n");
78 printf(" -c --config-file filename The config file to use.\n");
79}
80
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +080081static void handle_options(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020082{
83 while (1) {
84 int option_index = 0, c;
85 static struct option long_options[] = {
86 {"help", 0, 0, 'h'},
87 {"config-file", 1, 0, 'c'},
Harald Welte2c869ef2010-08-25 19:43:54 +020088 {"daemonize", 0, 0, 'D'},
Harald Welte5a29c7f2010-03-23 00:09:32 +080089 {"version", 0, 0, 'V'},
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020090 {0, 0, 0, 0},
91 };
92
Harald Welte2c869ef2010-08-25 19:43:54 +020093 c = getopt_long(argc, argv, "hc:VD", long_options, &option_index);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020094
95 if (c == -1)
96 break;
97
98 switch(c) {
99 case 'h':
100 print_help();
101 exit(0);
102 break;
103 case 'c':
104 config_file = talloc_strdup(tall_bsc_ctx, optarg);
105 break;
Harald Welte5a29c7f2010-03-23 00:09:32 +0800106 case 'V':
Harald Welte1353f962010-05-16 19:20:24 +0200107 print_version(1);
Harald Welte5a29c7f2010-03-23 00:09:32 +0800108 exit(0);
109 break;
Harald Welte2c869ef2010-08-25 19:43:54 +0200110 case 'D':
111 daemonize = 1;
112 break;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200113 default:
114 /* ignore */
115 break;
116 };
117 }
118}
119
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200120/* simply remember this */
121static int mgcp_rsip_cb(struct mgcp_config *cfg)
122{
123 reset_endpoints = 1;
124
125 return 0;
126}
127
Holger Hans Peter Freythera3b46c92010-08-05 07:09:14 +0800128static int mgcp_change_cb(struct mgcp_config *cfg, int endpoint, int state)
Holger Hans Peter Freytherdc417552010-04-22 13:23:05 +0800129{
130 if (state != MGCP_ENDP_MDCX)
131 return 0;
132
133 mgcp_send_dummy(&cfg->endpoints[endpoint]);
134 return 0;
135}
136
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200137static int read_call_agent(struct bsc_fd *fd, unsigned int what)
138{
139 struct sockaddr_in addr;
140 socklen_t slen = sizeof(addr);
141 struct msgb *msg;
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100142 struct msgb *resp;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200143 int i;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200144
145 msg = (struct msgb *) fd->data;
146
147 /* read one less so we can use it as a \0 */
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800148 int rc = recvfrom(cfg->gw_fd.bfd.fd, msg->data, msg->data_len - 1, 0,
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200149 (struct sockaddr *) &addr, &slen);
150 if (rc < 0) {
151 perror("Gateway failed to read");
152 return -1;
153 } else if (slen > sizeof(addr)) {
154 fprintf(stderr, "Gateway received message from outerspace: %d %d\n",
155 slen, sizeof(addr));
156 return -1;
157 }
158
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200159 /* handle message now */
160 msg->l2h = msgb_put(msg, rc);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100161 resp = mgcp_handle_message(cfg, msg);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200162 msgb_reset(msg);
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100163
164 if (resp) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800165 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 +0100166 msgb_free(resp);
167 }
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200168
169 if (reset_endpoints) {
170 LOGP(DMGCP, LOGL_NOTICE, "Asked to reset endpoints.\n");
171 reset_endpoints = 0;
172
173 /* is checking in_addr.s_addr == INADDR_LOOPBACK making it more secure? */
174 for (i = 1; i < cfg->number_endpoints; ++i)
175 mgcp_free_endp(&cfg->endpoints[i]);
176 }
177
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200178 return 0;
179}
180
Holger Hans Peter Freyther57da4472010-06-08 16:11:06 +0800181extern enum node_type bsc_vty_go_parent(struct vty *vty);
Harald Weltec31f4802010-05-25 23:31:39 +0200182
183static struct vty_app_info vty_info = {
184 .name = "OpenBSC MGCP",
185 .version = PACKAGE_VERSION,
186 .go_parent_cb = bsc_vty_go_parent,
Holger Hans Peter Freyther81506b42010-09-04 11:00:01 +0800187 .is_config_node = bsc_vty_is_config_node,
Harald Weltec31f4802010-05-25 23:31:39 +0200188};
Holger Hans Peter Freytherbd5130d2010-01-13 23:36:53 +0100189
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +0800190int main(int argc, char **argv)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200191{
Holger Hans Peter Freyther338fa562009-11-19 15:03:39 +0100192 struct gsm_network dummy_network;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200193 struct sockaddr_in addr;
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100194 int on = 1, rc;
Harald Weltedc5062b2010-03-26 21:28:59 +0800195 struct log_target *stderr_target;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200196
197 tall_bsc_ctx = talloc_named_const(NULL, 1, "mgcp-callagent");
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100198
Harald Weltedc5062b2010-03-26 21:28:59 +0800199 log_init(&log_info);
200 stderr_target = log_target_create_stderr();
201 log_add_target(stderr_target);
202 log_set_all_filter(stderr_target, 1);
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100203
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100204 cfg = mgcp_config_alloc();
205 if (!cfg)
206 return -1;
207
Harald Weltec31f4802010-05-25 23:31:39 +0200208 vty_info.copyright = openbsc_copyright;
209 vty_init(&vty_info);
Harald Welte5bc61dc2010-05-16 22:02:16 +0200210 logging_vty_add_cmds();
Harald Weltedcccb182010-05-16 20:52:23 +0200211 mgcp_vty_init();
Harald Welte1353f962010-05-16 19:20:24 +0200212
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200213 handle_options(argc, argv);
214
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100215 rc = mgcp_parse_config(config_file, cfg);
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100216 if (rc < 0)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200217 return rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200218
Harald Weltedcccb182010-05-16 20:52:23 +0200219 rc = telnet_init(tall_bsc_ctx, &dummy_network, 4243);
220 if (rc < 0)
221 return rc;
222
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200223 /* set some callbacks */
224 cfg->reset_cb = mgcp_rsip_cb;
Holger Hans Peter Freytherdc417552010-04-22 13:23:05 +0800225 cfg->change_cb = mgcp_change_cb;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200226
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100227 /* we need to bind a socket */
228 if (rc == 0) {
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800229 cfg->gw_fd.bfd.when = BSC_FD_READ;
230 cfg->gw_fd.bfd.cb = read_call_agent;
231 cfg->gw_fd.bfd.fd = socket(AF_INET, SOCK_DGRAM, 0);
232 if (cfg->gw_fd.bfd.fd < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100233 perror("Gateway failed to listen");
234 return -1;
235 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200236
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800237 setsockopt(cfg->gw_fd.bfd.fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200238
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100239 memset(&addr, 0, sizeof(addr));
240 addr.sin_family = AF_INET;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100241 addr.sin_port = htons(cfg->source_port);
242 inet_aton(cfg->source_addr, &addr.sin_addr);
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100243
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800244 if (bind(cfg->gw_fd.bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100245 perror("Gateway failed to bind");
246 return -1;
247 }
248
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800249 cfg->gw_fd.bfd.data = msgb_alloc(4096, "mgcp-msg");
250 if (!cfg->gw_fd.bfd.data) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100251 fprintf(stderr, "Gateway memory error.\n");
252 return -1;
253 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200254
255
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800256 if (bsc_register_fd(&cfg->gw_fd.bfd) != 0) {
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200257 LOGP(DMGCP, LOGL_FATAL, "Failed to register the fd\n");
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100258 return -1;
259 }
Holger Hans Peter Freytherf986cfc2010-01-05 12:25:25 +0100260
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200261 LOGP(DMGCP, LOGL_NOTICE, "Configured for MGCP.\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200262 }
263
264 /* initialisation */
265 srand(time(NULL));
266
Harald Welte2c869ef2010-08-25 19:43:54 +0200267 if (daemonize) {
268 rc = osmo_daemonize();
269 if (rc < 0) {
270 perror("Error during daemonize");
271 exit(1);
272 }
273 }
274
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200275 /* main loop */
276 while (1) {
277 bsc_select_main(0);
278 }
279
280
281 return 0;
282}