blob: fb16cb40f270323d3b68a9720dcf7716802daa2c [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>
Holger Hans Peter Freyther46720592015-08-01 23:35:19 +000033#include <sys/types.h>
34#include <sys/socket.h>
35#include <netinet/in.h>
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010036
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010037#define RTP_PORT_DEFAULT 4000
Holger Hans Peter Freyther314584a2010-08-05 04:10:21 +080038#define RTP_PORT_NET_DEFAULT 16000
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020039
40/**
41 * Calculate the RTP audio port for the given multiplex
42 * and the direction. This allows a semi static endpoint
43 * to port calculation removing the need for the BSC
44 * and the MediaGateway to communicate.
45 *
46 * Port usage explained:
47 * base + (multiplex * 2) + 0 == local port to wait for network packets
48 * base + (multiplex * 2) + 1 == local port for rtcp
49 *
50 * The above port will receive packets from the BTS that need
51 * to be patched and forwarded to the network.
52 * The above port will receive packets from the network that
53 * need to be patched and forwarded to the BTS.
54 *
55 * We assume to have a static BTS IP address so we can differentiate
56 * network and BTS.
57 *
58 */
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010059static inline int rtp_calculate_port(int multiplex, int base)
Holger Hans Peter Freytherf67945f2009-10-09 07:08:11 +020060{
61 return base + (multiplex * 2);
62}
Holger Hans Peter Freythere0955022010-02-03 08:50:33 +010063
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +010064
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010065/*
66 * Handling of MGCP Endpoints and the MGCP Config
67 */
68struct mgcp_endpoint;
69struct mgcp_config;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +010070struct mgcp_trunk_config;
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +010071struct mgcp_rtp_end;
Holger Hans Peter Freyther77f7afe2010-02-03 09:54:43 +010072
Holger Hans Peter Freyther77f7afe2010-02-03 09:54:43 +010073#define MGCP_ENDP_CRCX 1
74#define MGCP_ENDP_DLCX 2
75#define MGCP_ENDP_MDCX 3
76
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +010077/*
78 * what to do with the msg?
79 * - continue as usual?
80 * - reject and send a failure code?
81 * - defer? do not send anything
82 */
83#define MGCP_POLICY_CONT 4
84#define MGCP_POLICY_REJECT 5
85#define MGCP_POLICY_DEFER 6
86
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +010087typedef int (*mgcp_realloc)(struct mgcp_trunk_config *cfg, int endpoint);
88typedef int (*mgcp_change)(struct mgcp_trunk_config *cfg, int endpoint, int state);
89typedef 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 +020090typedef int (*mgcp_reset)(struct mgcp_trunk_config *cfg);
Holger Hans Peter Freytherce553612012-12-07 15:04:07 +010091typedef int (*mgcp_rqnt)(struct mgcp_endpoint *endp, char tone);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010092
Holger Hans Peter Freytherbd4109b2014-06-27 19:27:38 +020093/**
94 * Return:
95 * < 0 in case no audio was processed
96 * >= 0 in case audio was processed. The remaining payload
97 * length will be returned.
98 */
Jacob Erlbeck42a833e2014-04-14 10:31:47 +020099typedef int (*mgcp_processing)(struct mgcp_endpoint *endp,
100 struct mgcp_rtp_end *dst_end,
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100101 char *data, int *len, int buf_size);
102typedef int (*mgcp_processing_setup)(struct mgcp_endpoint *endp,
103 struct mgcp_rtp_end *dst_end,
104 struct mgcp_rtp_end *src_end);
Jacob Erlbeck168ca002014-03-17 12:40:07 +0100105
106typedef void (*mgcp_get_format)(struct mgcp_endpoint *endp,
107 int *payload_type,
108 const char**subtype_name,
109 const char**fmtp_extra);
110
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800111#define PORT_ALLOC_STATIC 0
112#define PORT_ALLOC_DYNAMIC 1
113
114/**
115 * This holds information on how to allocate ports
116 */
117struct mgcp_port_range {
118 int mode;
119
120 /* pre-allocated from a base? */
121 int base_port;
Holger Hans Peter Freyther1be9f2f2010-08-05 07:20:09 +0800122
123 /* dynamically allocated */
124 int range_start;
125 int range_end;
126 int last_port;
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800127};
128
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100129#define MGCP_KEEPALIVE_ONCE (-1)
130
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100131struct mgcp_trunk_config {
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100132 struct llist_head entry;
133
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100134 struct mgcp_config *cfg;
135
136 int trunk_nr;
137 int trunk_type;
138
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +0200139 char *audio_fmtp_extra;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100140 char *audio_name;
141 int audio_payload;
Jacob Erlbeck0a1bc562013-12-10 13:09:37 +0100142 int audio_send_ptime;
Holger Hans Peter Freyther619b0142014-11-19 16:04:45 +0100143 int audio_send_name;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100144 int audio_loop;
145
Holger Hans Peter Freythercb43a9a2015-04-24 16:03:55 -0400146 int no_audio_transcoding;
147
Holger Hans Peter Freythera8090d52012-05-11 13:00:45 +0200148 int omit_rtcp;
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100149 int keepalive_interval;
Holger Hans Peter Freythera8090d52012-05-11 13:00:45 +0200150
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100151 /* RTP patching */
152 int force_constant_ssrc; /* 0: don't, 1: once */
Jacob Erlbeck4bbddc62013-12-18 12:54:51 +0100153 int force_aligned_timing;
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100154
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100155 /* spec handling */
156 int force_realloc;
157
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100158 /* timer */
159 struct osmo_timer_list keepalive_timer;
160
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100161 unsigned int number_endpoints;
162 struct mgcp_endpoint *endpoints;
163};
164
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200165enum mgcp_role {
166 MGCP_BSC = 0,
167 MGCP_BSC_NAT,
168};
169
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100170struct mgcp_config {
171 int source_port;
172 char *local_ip;
173 char *source_addr;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100174 char *bts_ip;
Holger Hans Peter Freytherb79994c2010-03-31 11:46:41 +0200175 char *call_agent_addr;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100176
177 struct in_addr bts_in;
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800178
Holger Hans Peter Freytherb98ba722010-09-19 04:21:39 +0800179 /* transcoder handling */
180 char *transcoder_ip;
181 struct in_addr transcoder_in;
182 int transcoder_remote_base;
183
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100184 /* RTP processing */
185 mgcp_processing rtp_processing_cb;
186 mgcp_processing_setup setup_rtp_processing_cb;
187
Jacob Erlbeck168ca002014-03-17 12:40:07 +0100188 mgcp_get_format get_net_downlink_format_cb;
189
Pablo Neira Ayusoe1273b12011-05-06 12:09:47 +0200190 struct osmo_wqueue gw_fd;
Holger Hans Peter Freyther6f680102010-09-18 20:40:22 +0800191
Holger Hans Peter Freyther15e73892010-08-05 07:10:56 +0800192 struct mgcp_port_range bts_ports;
193 struct mgcp_port_range net_ports;
Holger Hans Peter Freyther54aaa0f2010-09-17 23:35:53 +0800194 struct mgcp_port_range transcoder_ports;
Holger Hans Peter Freytherd0c32292010-07-27 20:34:45 +0800195 int endp_dscp;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100196
Jacob Erlbeck42a833e2014-04-14 10:31:47 +0200197 int bts_force_ptime;
198
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100199 mgcp_change change_cb;
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +0100200 mgcp_policy policy_cb;
Holger Hans Peter Freyther9bdcc9c2010-03-31 06:39:35 +0200201 mgcp_reset reset_cb;
Holger Hans Peter Freyther869e38e2010-08-06 17:54:27 +0800202 mgcp_realloc realloc_cb;
Holger Hans Peter Freyther8d0be252012-11-29 12:54:22 +0100203 mgcp_rqnt rqnt_cb;
Holger Hans Peter Freytherfe86d3c2010-02-26 13:37:05 +0100204 void *data;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100205
Holger Hans Peter Freyther46340132010-08-06 08:26:54 +0800206 uint32_t last_call_id;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100207
208 /* trunk handling */
209 struct mgcp_trunk_config trunk;
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100210 struct llist_head trunks;
Holger Hans Peter Freyther44016fe2011-02-28 14:46:01 +0100211
212 /* only used for start with a static configuration */
213 int last_net_port;
214 int last_bts_port;
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200215
216 enum mgcp_role role;
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100217
218 /* osmux translator: 0 means disabled, 1 means enabled */
219 int osmux;
220 /* The BSC-NAT may ask for enabling osmux on demand. This tells us if
221 * the osmux socket is already initialized.
222 */
223 int osmux_init;
224 /* osmux batch factor: from 1 to 4 maximum */
225 int osmux_batch;
Pablo Neira Ayuso03ab79a2014-08-29 12:30:38 +0200226 /* osmux batch size (in bytes) */
227 int osmux_batch_size;
Pablo Neira Ayuso0fe78d32014-08-28 16:43:38 +0200228 /* osmux port */
229 uint16_t osmux_port;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100230};
231
232/* config management */
233struct mgcp_config *mgcp_config_alloc(void);
Pablo Neira Ayuso326b5d82013-08-02 21:14:14 +0200234int mgcp_parse_config(const char *config_file, struct mgcp_config *cfg,
235 enum mgcp_role role);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100236int mgcp_vty_init(void);
Holger Hans Peter Freyther1f0c5b42011-02-28 14:37:03 +0100237int mgcp_endpoints_allocate(struct mgcp_trunk_config *cfg);
Holger Hans Peter Freythercb6ad702014-07-22 15:00:52 +0200238void mgcp_release_endp(struct mgcp_endpoint *endp);
239void mgcp_initialize_endp(struct mgcp_endpoint *endp);
Holger Hans Peter Freytherf2eedff2010-09-19 04:36:07 +0800240int mgcp_reset_transcoder(struct mgcp_config *cfg);
Holger Hans Peter Freyther0bf15a82012-09-14 17:18:12 +0200241void mgcp_format_stats(struct mgcp_endpoint *endp, char *stats, size_t size);
Holger Hans Peter Freyther462b7d72012-10-24 21:53:40 +0200242int 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 +0100243
Jacob Erlbeck075a9eb2013-12-19 18:53:07 +0100244void mgcp_trunk_set_keepalive(struct mgcp_trunk_config *tcfg, int interval);
245
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100246/*
247 * format helper functions
248 */
249struct msgb *mgcp_handle_message(struct mgcp_config *cfg, struct msgb *msg);
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100250
Holger Hans Peter Freyther616d2222010-03-31 09:27:04 +0200251/* adc helper */
252static inline int mgcp_timeslot_to_endpoint(int multiplex, int timeslot)
253{
Holger Hans Peter Freythere3946f42010-09-20 01:07:23 +0800254 if (timeslot == 0) {
255 LOGP(DMGCP, LOGL_ERROR, "Timeslot should not be 0\n");
256 timeslot = 255;
257 }
258
Holger Hans Peter Freythercb8c35c2010-08-10 20:24:24 +0800259 return timeslot + (32 * multiplex);
Holger Hans Peter Freyther616d2222010-03-31 09:27:04 +0200260}
261
Holger Hans Peter Freyther82049d82010-08-28 17:59:15 +0800262static inline void mgcp_endpoint_to_timeslot(int endpoint, int *multiplex, int *timeslot)
263{
264 *multiplex = endpoint / 32;
265 *timeslot = endpoint % 32;
Holger Hans Peter Freyther82049d82010-08-28 17:59:15 +0800266}
267
Harald Welte94b499c2012-01-27 00:46:49 +0100268int mgcp_send_reset_ep(struct mgcp_endpoint *endp, int endpoint);
269int mgcp_send_reset_all(struct mgcp_config *cfg);
270
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100271
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100272int mgcp_create_bind(const char *source_addr, struct osmo_fd *fd, int port);
273int mgcp_send(struct mgcp_endpoint *endp, int dest, int is_rtp, struct sockaddr_in *addr, char *buf, int rc);
274int mgcp_udp_send(int fd, struct in_addr *addr, int port, char *buf, int len);
275
Holger Hans Peter Freyther62e836c2010-02-03 11:03:45 +0100276#endif