blob: 80b7b543cccd5fe6335cdde6679cb005dccd6840 [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
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (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
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 *
23 */
24
25#include <ctype.h>
26#include <stdio.h>
27#include <stdlib.h>
28#include <string.h>
29#include <time.h>
30#include <limits.h>
31#include <unistd.h>
32
33#include <sys/socket.h>
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020034
35#include <openbsc/debug.h>
Harald Weltedfe6c7d2010-02-20 16:24:02 +010036#include <osmocore/msgb.h>
37#include <osmocore/talloc.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>
Holger Hans Peter Freyther338fa562009-11-19 15:03:39 +010042#include <openbsc/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
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +020045#include <vty/command.h>
46
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 Freytherf67945f2009-10-09 07:08:11 +020057static struct bsc_fd bfd;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010058static struct mgcp_config *cfg;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +020059static int reset_endpoints = 0;
60
Harald Welte5a29c7f2010-03-23 00:09:32 +080061const char *openbsc_version = "OpenBSC MGCP " PACKAGE_VERSION;
Holger Hans Peter Freytheracf8a0c2010-03-29 08:47:44 +020062const char *openbsc_copyright =
Harald Welte5a29c7f2010-03-23 00:09:32 +080063 "Copyright (C) 2009-2010 Holger Freyther and On-Waves\n"
64 "Contributions by Daniel Willmann, Jan Lübbe,Stefan Schmidt\n"
65 "Dieter Spaar, Andreas Eversberg, Harald Welte\n\n"
66 "License GPLv2+: GNU GPL version 2 or later <http://gnu.org/licenses/gpl.html>\n"
67 "This is free software: you are free to change and redistribute it.\n"
68 "There is NO WARRANTY, to the extent permitted by law.\n";
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020069
70static char *config_file = "mgcp.cfg";
71
72/* used by msgb and mgcp */
73void *tall_bsc_ctx = NULL;
74
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020075static void print_help()
76{
77 printf("Some useful help...\n");
78 printf(" -h --help is printing this text.\n");
79 printf(" -c --config-file filename The config file to use.\n");
80}
81
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +020082static void print_mgcp_version()
Harald Welte5a29c7f2010-03-23 00:09:32 +080083{
84 printf("%s\n\n", openbsc_version);
Holger Hans Peter Freytherc38b5882010-03-30 12:43:33 +020085 printf("%s", openbsc_copyright);
Harald Welte5a29c7f2010-03-23 00:09:32 +080086}
87
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020088static void handle_options(int argc, char** argv)
89{
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 Welte5a29c7f2010-03-23 00:09:32 +080095 {"version", 0, 0, 'V'},
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020096 {0, 0, 0, 0},
97 };
98
Harald Welte5a29c7f2010-03-23 00:09:32 +080099 c = getopt_long(argc, argv, "hc:V", long_options, &option_index);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200100
101 if (c == -1)
102 break;
103
104 switch(c) {
105 case 'h':
106 print_help();
107 exit(0);
108 break;
109 case 'c':
110 config_file = talloc_strdup(tall_bsc_ctx, optarg);
111 break;
Harald Welte5a29c7f2010-03-23 00:09:32 +0800112 case 'V':
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +0200113 print_mgcp_version();
Harald Welte5a29c7f2010-03-23 00:09:32 +0800114 exit(0);
115 break;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200116 default:
117 /* ignore */
118 break;
119 };
120 }
121}
122
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200123/* simply remember this */
124static int mgcp_rsip_cb(struct mgcp_config *cfg)
125{
126 reset_endpoints = 1;
127
128 return 0;
129}
130
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200131static int read_call_agent(struct bsc_fd *fd, unsigned int what)
132{
133 struct sockaddr_in addr;
134 socklen_t slen = sizeof(addr);
135 struct msgb *msg;
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100136 struct msgb *resp;
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200137 int i;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200138
139 msg = (struct msgb *) fd->data;
140
141 /* read one less so we can use it as a \0 */
142 int rc = recvfrom(bfd.fd, msg->data, msg->data_len - 1, 0,
143 (struct sockaddr *) &addr, &slen);
144 if (rc < 0) {
145 perror("Gateway failed to read");
146 return -1;
147 } else if (slen > sizeof(addr)) {
148 fprintf(stderr, "Gateway received message from outerspace: %d %d\n",
149 slen, sizeof(addr));
150 return -1;
151 }
152
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200153 /* handle message now */
154 msg->l2h = msgb_put(msg, rc);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100155 resp = mgcp_handle_message(cfg, msg);
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200156 msgb_reset(msg);
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100157
158 if (resp) {
159 sendto(bfd.fd, resp->l2h, msgb_l2len(resp), 0, (struct sockaddr *) &addr, sizeof(addr));
160 msgb_free(resp);
161 }
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200162
163 if (reset_endpoints) {
164 LOGP(DMGCP, LOGL_NOTICE, "Asked to reset endpoints.\n");
165 reset_endpoints = 0;
166
167 /* is checking in_addr.s_addr == INADDR_LOOPBACK making it more secure? */
168 for (i = 1; i < cfg->number_endpoints; ++i)
169 mgcp_free_endp(&cfg->endpoints[i]);
170 }
171
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200172 return 0;
173}
174
Holger Hans Peter Freytherbd5130d2010-01-13 23:36:53 +0100175
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200176int main(int argc, char** argv)
177{
Holger Hans Peter Freyther338fa562009-11-19 15:03:39 +0100178 struct gsm_network dummy_network;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200179 struct sockaddr_in addr;
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100180 int on = 1, rc;
Harald Weltedc5062b2010-03-26 21:28:59 +0800181 struct log_target *stderr_target;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200182
183 tall_bsc_ctx = talloc_named_const(NULL, 1, "mgcp-callagent");
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100184
Harald Weltedc5062b2010-03-26 21:28:59 +0800185 log_init(&log_info);
186 stderr_target = log_target_create_stderr();
187 log_add_target(stderr_target);
188 log_set_all_filter(stderr_target, 1);
Holger Hans Peter Freytherb61e3b22009-12-22 22:32:51 +0100189
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100190 cfg = mgcp_config_alloc();
191 if (!cfg)
192 return -1;
193
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200194 handle_options(argc, argv);
195
Holger Hans Peter Freyther338fa562009-11-19 15:03:39 +0100196 telnet_init(&dummy_network, 4243);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100197 rc = mgcp_parse_config(config_file, cfg);
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100198 if (rc < 0)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200199 return rc;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200200
Holger Hans Peter Freyther64e4e772010-03-31 07:01:35 +0200201 /* set some callbacks */
202 cfg->reset_cb = mgcp_rsip_cb;
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200203
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +0100204 /* we need to bind a socket */
205 if (rc == 0) {
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100206 bfd.when = BSC_FD_READ;
207 bfd.cb = read_call_agent;
208 bfd.fd = socket(AF_INET, SOCK_DGRAM, 0);
209 if (bfd.fd < 0) {
210 perror("Gateway failed to listen");
211 return -1;
212 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200213
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100214 setsockopt(bfd.fd, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200215
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100216 memset(&addr, 0, sizeof(addr));
217 addr.sin_family = AF_INET;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100218 addr.sin_port = htons(cfg->source_port);
219 inet_aton(cfg->source_addr, &addr.sin_addr);
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100220
221 if (bind(bfd.fd, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
222 perror("Gateway failed to bind");
223 return -1;
224 }
225
226 bfd.data = msgb_alloc(4096, "mgcp-msg");
227 if (!bfd.data) {
228 fprintf(stderr, "Gateway memory error.\n");
229 return -1;
230 }
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200231
232
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100233 if (bsc_register_fd(&bfd) != 0) {
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200234 LOGP(DMGCP, LOGL_FATAL, "Failed to register the fd\n");
Holger Hans Peter Freyther14083be2010-01-05 12:21:36 +0100235 return -1;
236 }
Holger Hans Peter Freytherf986cfc2010-01-05 12:25:25 +0100237
Holger Hans Peter Freyther52a66aa2010-03-31 06:16:45 +0200238 LOGP(DMGCP, LOGL_NOTICE, "Configured for MGCP.\n");
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200239 }
240
241 /* initialisation */
242 srand(time(NULL));
243
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +0200244 /* main loop */
245 while (1) {
246 bsc_select_main(0);
247 }
248
249
250 return 0;
251}
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +0200252
253struct gsm_network;
254int bsc_vty_init(struct gsm_network *dummy)
255{
256 cmd_init(1);
257 vty_init();
258
Holger Hans Peter Freyther44127372010-04-06 12:00:43 +0200259 openbsc_vty_add_cmds();
Holger Hans Peter Freytherf5b6aa62010-03-30 12:43:53 +0200260 mgcp_vty_init();
261 return 0;
262}
263