blob: 002dd7c6eb123a51914d4c59967fac392cb2183a [file] [log] [blame]
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +02001/* A Media Gateway Control Protocol Media Gateway: RFC 3435 */
2
3/*
Holger Hans Peter Freyther8d0be252012-11-29 12:54:22 +01004 * (C) 2009-2012 by Holger Hans Peter Freyther <zecke@selfish.org>
5 * (C) 2009-2012 by On-Waves
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +02006 * All Rights Reserved
7 *
8 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01009 * it under the terms of the GNU Affero General Public License as published by
10 * the Free Software Foundation; either version 3 of the License, or
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020011 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte9af6ddf2011-01-01 15:25:50 +010016 * GNU Affero General Public License for more details.
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020017 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010018 * You should have received a copy of the GNU Affero General Public License
19 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020020 *
21 */
22
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +010023#ifndef OPENBSC_MGCP_H
24#define OPENBSC_MGCP_H
25
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010026#include <osmocom/core/msgb.h>
27#include <osmocom/core/write_queue.h>
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +010028#include <osmocom/core/timer.h>
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +010029
Holger Hans Peter Freythere3946f42010-09-20 01:07:23 +080030#include "debug.h"
31
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010032#include <arpa/inet.h>
33
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010034#define RTP_PORT_DEFAULT 4000
Holger Hans Peter Freyther314584a2010-08-05 04:10:21 +080035#define RTP_PORT_NET_DEFAULT 16000
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020036
37/**
38 * Calculate the RTP audio port for the given multiplex
39 * and the direction. This allows a semi static endpoint
40 * to port calculation removing the need for the BSC
41 * and the MediaGateway to communicate.
42 *
43 * Port usage explained:
44 * base + (multiplex * 2) + 0 == local port to wait for network packets
45 * base + (multiplex * 2) + 1 == local port for rtcp
46 *
47 * The above port will receive packets from the BTS that need
48 * to be patched and forwarded to the network.
49 * The above port will receive packets from the network that
50 * need to be patched and forwarded to the BTS.
51 *
52 * We assume to have a static BTS IP address so we can differentiate
53 * network and BTS.
54 *
55 */
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010056static inline int rtp_calculate_port(int multiplex, int base)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020057{
58 return base + (multiplex * 2);
59}
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010060
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +010061
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010062/*
63 * Handling of MGCP Endpoints and the MGCP Config
64 */
65struct mgcp_endpoint;
66struct mgcp_config;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +010067struct mgcp_trunk_config;
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +010068struct mgcp_rtp_end;
Holger Hans Peter Freyther77f7afe2010-02-03 09:54:43 +010069
Holger Hans Peter Freyther77f7afe2010-02-03 09:54:43 +010070#define MGCP_ENDP_CRCX 1
71#define MGCP_ENDP_DLCX 2
72#define MGCP_ENDP_MDCX 3
73
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +010074/*
75 * what to do with the msg?
76 * - continue as usual?
77 * - reject and send a failure code?
78 * - defer? do not send anything
79 */
80#define MGCP_POLICY_CONT 4
81#define MGCP_POLICY_REJECT 5
82#define MGCP_POLICY_DEFER 6
83
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +010084typedef int (*mgcp_realloc)(struct mgcp_trunk_config *cfg, int endpoint);
85typedef int (*mgcp_change)(struct mgcp_trunk_config *cfg, int endpoint, int state);
86typedef int (*mgcp_policy)(struct mgcp_trunk_config *cfg, int endpoint, int state, const char *transactio_id);
Holger Hans Peter Freyther74db7742011-06-10 00:04:55 +020087typedef int (*mgcp_reset)(struct mgcp_trunk_config *cfg);
Holger Hans Peter Freytherce553612012-12-07 15:04:07 +010088typedef int (*mgcp_rqnt)(struct mgcp_endpoint *endp, char tone);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010089
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +010090typedef int (*mgcp_processing)(struct mgcp_rtp_end *dst_end,
91 char *data, int *len, int buf_size);
92typedef int (*mgcp_processing_setup)(struct mgcp_endpoint *endp,
93 struct mgcp_rtp_end *dst_end,
94 struct mgcp_rtp_end *src_end);
Jacob Erlbeck168ca002014-03-17 12:40:07 +010095
96typedef void (*mgcp_get_format)(struct mgcp_endpoint *endp,
97 int *payload_type,
98 const char**subtype_name,
99 const char**fmtp_extra);
100
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800101#define PORT_ALLOC_STATIC 0
102#define PORT_ALLOC_DYNAMIC 1
103
104/**
105 * This holds information on how to allocate ports
106 */
107struct mgcp_port_range {
108 int mode;
109
110 /* pre-allocated from a base? */
111 int base_port;
Holger Hans Peter Freyther1be9f2f2010-08-05 07:20:09 +0800112
113 /* dynamically allocated */
114 int range_start;
115 int range_end;
116 int last_port;
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800117};
118
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100119#define MGCP_KEEPALIVE_ONCE (-1)
120
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100121struct mgcp_trunk_config {
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100122 struct llist_head entry;
123
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100124 struct mgcp_config *cfg;
125
126 int trunk_nr;
127 int trunk_type;
128
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +0200129 char *audio_fmtp_extra;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100130 char *audio_name;
131 int audio_payload;
Jacob Erlbeck0a1bc562013-12-10 13:09:37 +0100132 int audio_send_ptime;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100133 int audio_loop;
134
Holger Hans Peter Freythera8090d52012-05-11 13:00:45 +0200135 int omit_rtcp;
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100136 int keepalive_interval;
Holger Hans Peter Freythera8090d52012-05-11 13:00:45 +0200137
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100138 /* RTP patching */
139 int force_constant_ssrc; /* 0: don't, 1: once */
Jacob Erlbeck4bbddc62013-12-18 12:54:51 +0100140 int force_aligned_timing;
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100141
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100142 /* spec handling */
143 int force_realloc;
144
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100145 /* timer */
146 struct osmo_timer_list keepalive_timer;
147
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100148 unsigned int number_endpoints;
149 struct mgcp_endpoint *endpoints;
150};
151
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200152enum mgcp_role {
153 MGCP_BSC = 0,
154 MGCP_BSC_NAT,
155};
156
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100157struct mgcp_config {
158 int source_port;
159 char *local_ip;
160 char *source_addr;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100161 char *bts_ip;
Holger Hans Peter Freytherb79994c2010-03-31 11:46:41 +0200162 char *call_agent_addr;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100163
164 struct in_addr bts_in;
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800165
Holger Hans Peter Freytherb98ba722010-09-19 04:21:39 +0800166 /* transcoder handling */
167 char *transcoder_ip;
168 struct in_addr transcoder_in;
169 int transcoder_remote_base;
170
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100171 /* RTP processing */
172 mgcp_processing rtp_processing_cb;
173 mgcp_processing_setup setup_rtp_processing_cb;
174
Jacob Erlbeck168ca002014-03-17 12:40:07 +0100175 mgcp_get_format get_net_downlink_format_cb;
176
Pablo Neira Ayusoe1273b12011-05-06 12:09:47 +0200177 struct osmo_wqueue gw_fd;
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800178
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800179 struct mgcp_port_range bts_ports;
180 struct mgcp_port_range net_ports;
Holger Hans Peter Freyther54aaa0f2010-09-17 23:35:53 +0800181 struct mgcp_port_range transcoder_ports;
Holger Hans Peter Freytherd0c32292010-07-27 20:34:45 +0800182 int endp_dscp;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100183
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100184 mgcp_change change_cb;
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +0100185 mgcp_policy policy_cb;
Holger Hans Peter Freyther9bdcc9c2010-03-31 06:39:35 +0200186 mgcp_reset reset_cb;
Holger Hans Peter Freyther869e38e2010-08-06 17:54:27 +0800187 mgcp_realloc realloc_cb;
Holger Hans Peter Freyther8d0be252012-11-29 12:54:22 +0100188 mgcp_rqnt rqnt_cb;
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +0100189 void *data;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100190
Holger Hans Peter Freyther46340132010-08-06 08:26:54 +0800191 uint32_t last_call_id;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100192
193 /* trunk handling */
194 struct mgcp_trunk_config trunk;
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100195 struct llist_head trunks;
Holger Hans Peter Freyther44016fe2011-02-28 14:46:01 +0100196
197 /* only used for start with a static configuration */
198 int last_net_port;
199 int last_bts_port;
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200200
201 enum mgcp_role role;
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100202
203 /* osmux translator: 0 means disabled, 1 means enabled */
204 int osmux;
205 /* The BSC-NAT may ask for enabling osmux on demand. This tells us if
206 * the osmux socket is already initialized.
207 */
208 int osmux_init;
209 /* osmux batch factor: from 1 to 4 maximum */
210 int osmux_batch;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100211};
212
213/* config management */
214struct mgcp_config *mgcp_config_alloc(void);
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200215int mgcp_parse_config(const char *config_file, struct mgcp_config *cfg,
216 enum mgcp_role role);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100217int mgcp_vty_init(void);
Holger Hans Peter Freyther1f0c5b42011-02-28 14:37:03 +0100218int mgcp_endpoints_allocate(struct mgcp_trunk_config *cfg);
Holger Hans Peter Freyther154b9552010-02-26 13:27:51 +0100219void mgcp_free_endp(struct mgcp_endpoint *endp);
Holger Hans Peter Freytherf2eedff2010-09-19 04:36:07 +0800220int mgcp_reset_transcoder(struct mgcp_config *cfg);
Holger Hans Peter Freyther0bf15a82012-09-14 17:18:12 +0200221void mgcp_format_stats(struct mgcp_endpoint *endp, char *stats, size_t size);
Holger Hans Peter Freyther462b7d72012-10-24 21:53:40 +0200222int mgcp_parse_stats(struct msgb *msg, uint32_t *ps, uint32_t *os, uint32_t *pr, uint32_t *_or, int *loss, uint32_t *jitter);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100223
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100224void mgcp_trunk_set_keepalive(struct mgcp_trunk_config *tcfg, int interval);
225
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100226/*
227 * format helper functions
228 */
229struct msgb *mgcp_handle_message(struct mgcp_config *cfg, struct msgb *msg);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100230
Holger Hans Peter Freyther616d2222010-03-31 09:27:04 +0200231/* adc helper */
232static inline int mgcp_timeslot_to_endpoint(int multiplex, int timeslot)
233{
Holger Hans Peter Freythere3946f42010-09-20 01:07:23 +0800234 if (timeslot == 0) {
235 LOGP(DMGCP, LOGL_ERROR, "Timeslot should not be 0\n");
236 timeslot = 255;
237 }
238
Holger Hans Peter Freythercb8c35c2010-08-10 20:24:24 +0800239 return timeslot + (32 * multiplex);
Holger Hans Peter Freyther616d2222010-03-31 09:27:04 +0200240}
241
Holger Hans Peter Freyther82049d82010-08-28 17:59:15 +0800242static inline void mgcp_endpoint_to_timeslot(int endpoint, int *multiplex, int *timeslot)
243{
244 *multiplex = endpoint / 32;
245 *timeslot = endpoint % 32;
Holger Hans Peter Freyther82049d82010-08-28 17:59:15 +0800246}
247
Harald Welte94b499c2012-01-27 00:46:49 +0100248int mgcp_send_reset_ep(struct mgcp_endpoint *endp, int endpoint);
249int mgcp_send_reset_all(struct mgcp_config *cfg);
250
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100251
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100252int mgcp_create_bind(const char *source_addr, struct osmo_fd *fd, int port);
253int mgcp_send(struct mgcp_endpoint *endp, int dest, int is_rtp, struct sockaddr_in *addr, char *buf, int rc);
254int mgcp_udp_send(int fd, struct in_addr *addr, int port, char *buf, int len);
255
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100256#endif