blob: b58eb9b583ac907079935e70f5f43e704541e7d9 [file] [log] [blame]
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +01001/* MGCP Private Data */
2
3/*
Holger Hans Peter Freyther38e02c52012-10-22 18:09:35 +02004 * (C) 2009-2012 by Holger Hans Peter Freyther <zecke@selfish.org>
5 * (C) 2009-2012 by On-Waves
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +01006 * 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 Freyther7bdc6372010-02-20 21:21:02 +010011 * (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 Freyther7bdc6372010-02-20 21:21:02 +010017 *
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 Freyther7bdc6372010-02-20 21:21:02 +010020 *
21 */
22
Holger Hans Peter Freytherf6b61e62014-05-22 17:06:14 +020023#pragma once
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010024
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +020025#include <string.h>
26
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010027#include <osmocom/core/select.h>
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +010028
29#define CI_UNUSED 0
30
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +010031enum mgcp_trunk_type {
32 MGCP_TRUNK_VIRTUAL,
33 MGCP_TRUNK_E1,
34};
35
Jacob Erlbeck50079a12013-11-25 12:53:28 +010036struct mgcp_rtp_stream_state {
37 uint32_t ssrc;
38 uint16_t last_seq;
39 uint32_t last_timestamp;
40 uint32_t err_ts_counter;
41 int32_t last_tsdelta;
Jacob Erlbeckeacc9b92014-01-30 21:01:35 +010042 uint32_t last_arrival_time;
Jacob Erlbeck50079a12013-11-25 12:53:28 +010043};
44
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +000045struct mgcp_rtp_state {
46 int initialized;
Jacob Erlbeck58340e52013-11-29 11:20:06 +010047 int patch_ssrc;
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +000048
Holger Hans Peter Freyther6aa882b2010-08-03 14:35:50 +000049 uint32_t orig_ssrc;
Holger Hans Peter Freythered3a6612012-10-22 17:08:48 +020050
Holger Hans Peter Freyther6aa882b2010-08-03 14:35:50 +000051 int seq_offset;
Holger Hans Peter Freythered3a6612012-10-22 17:08:48 +020052
Holger Hans Peter Freyther6aa882b2010-08-03 14:35:50 +000053 int32_t timestamp_offset;
Jacob Erlbeck30ce4222013-12-05 12:02:15 +010054 uint32_t packet_duration;
Jacob Erlbeck50079a12013-11-25 12:53:28 +010055
56 struct mgcp_rtp_stream_state in_stream;
57 struct mgcp_rtp_stream_state out_stream;
Holger Hans Peter Freythera5a59c92014-10-06 20:04:42 +020058
59 /* jitter and packet loss calculation */
60 int stats_initialized;
61 uint16_t stats_base_seq;
62 uint16_t stats_max_seq;
63 uint32_t stats_ssrc;
64 uint32_t stats_jitter;
65 int32_t stats_transit;
66 int stats_cycles;
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +000067};
Holger Hans Peter Freyther98a38772010-08-03 02:27:21 +080068
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020069struct mgcp_rtp_codec {
70 uint32_t rate;
71 int channels;
72 uint32_t frame_duration_num;
73 uint32_t frame_duration_den;
74
75 int payload_type;
76 char *audio_name;
77 char *subtype_name;
78};
79
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +080080struct mgcp_rtp_end {
81 /* statistics */
82 unsigned int packets;
Holger Hans Peter Freyther952f7522012-09-12 11:38:37 +020083 unsigned int octets;
Jacob Erlbeck0970bab2013-12-19 12:13:32 +010084 unsigned int dropped_packets;
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +080085 struct in_addr addr;
86
87 /* in network byte order */
88 int rtp_port, rtcp_port;
89
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020090 /* audio codec information */
91 struct mgcp_rtp_codec codec;
Holger Hans Peter Freythere46bc272014-09-01 10:50:22 +020092 struct mgcp_rtp_codec alt_codec; /* TODO/XXX: make it generic */
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020093
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +020094 /* per endpoint data */
Jacob Erlbeck72c30902013-11-25 15:23:35 +010095 int frames_per_packet;
Jacob Erlbeckf6ec0e92013-12-04 10:30:11 +010096 uint32_t packet_duration_ms;
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +020097 char *fmtp_extra;
Jacob Erlbeck0970bab2013-12-19 12:13:32 +010098 int output_enabled;
Jacob Erlbeck42a833e2014-04-14 10:31:47 +020099 int force_output_ptime;
Holger Hans Peter Freyther58ff2192010-08-05 03:08:35 +0800100
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100101 /* RTP patching */
Jacob Erlbecke2292f32013-12-03 15:13:12 +0100102 int force_constant_ssrc; /* -1: always, 0: don't, 1: once */
Jacob Erlbeck4bbddc62013-12-18 12:54:51 +0100103 int force_aligned_timing;
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100104 void *rtp_process_data;
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100105
Holger Hans Peter Freytherc4921272010-08-05 03:37:22 +0800106 /*
107 * Each end has a socket...
108 */
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200109 struct osmo_fd rtp;
110 struct osmo_fd rtcp;
Holger Hans Peter Freytherc4921272010-08-05 03:37:22 +0800111
Holger Hans Peter Freyther58ff2192010-08-05 03:08:35 +0800112 int local_port;
Holger Hans Peter Freytherf138f912010-08-05 08:08:17 +0800113 int local_alloc;
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +0800114};
115
Holger Hans Peter Freyther260d6ed2010-08-06 01:12:21 +0800116enum {
117 MGCP_TAP_BTS_IN,
118 MGCP_TAP_BTS_OUT,
119 MGCP_TAP_NET_IN,
120 MGCP_TAP_NET_OUT,
121
122 /* last element */
123 MGCP_TAP_COUNT
124};
125
126struct mgcp_rtp_tap {
127 int enabled;
128 struct sockaddr_in forward;
129};
130
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100131struct mgcp_lco {
132 char *string;
Jacob Erlbeck452c1832014-05-09 19:42:17 +0200133 char *codec;
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100134 int pkt_period_min; /* time in ms */
135 int pkt_period_max; /* time in ms */
136};
137
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200138enum mgcp_type {
139 MGCP_RTP_DEFAULT = 0,
140 MGCP_RTP_TRANSCODED,
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100141 MGCP_OSMUX_BSC,
142 MGCP_OSMUX_BSC_NAT,
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200143};
144
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100145#include <openbsc/osmux.h>
146
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100147struct mgcp_endpoint {
Holger Hans Peter Freyther39a97e22010-08-06 18:03:11 +0800148 int allocated;
Holger Hans Peter Freyther46340132010-08-06 08:26:54 +0800149 uint32_t ci;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100150 char *callid;
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100151 struct mgcp_lco local_options;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100152 int conn_mode;
Holger Hans Peter Freytherc597a4e2010-08-03 02:57:02 +0800153 int orig_mode;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100154
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100155 /* backpointer */
156 struct mgcp_config *cfg;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100157 struct mgcp_trunk_config *tcfg;
Holger Hans Peter Freytherb4b135e2010-04-07 09:37:17 +0200158
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +0800159 /* port status for bts/net */
160 struct mgcp_rtp_end bts_end;
161 struct mgcp_rtp_end net_end;
Holger Hans Peter Freyther21262332010-11-01 20:53:31 +0100162
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100163 /*
164 * For transcoding we will send from the local_port
165 * of trans_bts and it will arrive at trans_net from
166 * where we will forward it to the network.
167 */
168 struct mgcp_rtp_end trans_bts;
Holger Hans Peter Freyther21262332010-11-01 20:53:31 +0100169 struct mgcp_rtp_end trans_net;
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200170 enum mgcp_type type;
Holger Hans Peter Freyther380b8712010-07-29 02:38:39 +0800171
172 /* sequence bits */
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +0000173 struct mgcp_rtp_state net_state;
174 struct mgcp_rtp_state bts_state;
Holger Hans Peter Freyther6357a8e2010-08-05 12:07:00 +0000175
Holger Hans Peter Freytherce553612012-12-07 15:04:07 +0100176 /* fields for re-transmission */
177 char *last_trans;
178 char *last_response;
179
Holger Hans Peter Freyther260d6ed2010-08-06 01:12:21 +0800180 /* tap for the endpoint */
181 struct mgcp_rtp_tap taps[MGCP_TAP_COUNT];
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100182
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200183 struct {
Pablo Neira Ayusob769f3c2014-08-27 17:02:52 +0200184 /* Osmux state: disabled, activating, active */
185 enum osmux_state state;
186 /* Allocated Osmux circuit ID for this endpoint */
Holger Hans Peter Freyther1afe7c72015-10-04 11:11:11 +0200187 int allocated_cid;
188 /* Used Osmux circuit ID for this endpoint */
Pablo Neira Ayusob769f3c2014-08-27 17:02:52 +0200189 uint8_t cid;
190 /* handle to batch messages */
191 struct osmux_in_handle *in;
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200192 /* handle to unbatch messages */
193 struct osmux_out_handle out;
Pablo Neira Ayusoee11bc02014-09-17 12:33:09 +0200194 /* statistics */
195 struct {
196 uint32_t chunks;
197 uint32_t octets;
198 } stats;
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200199 } osmux;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100200};
201
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200202#define for_each_line(line, save) \
203 for (line = strline_r(NULL, &save); line;\
204 line = strline_r(NULL, &save))
205
206static inline char *strline_r(char *str, char **saveptr)
207{
208 char *result;
209
210 if (str)
211 *saveptr = str;
212
213 result = *saveptr;
214
215 if (*saveptr != NULL) {
216 *saveptr = strpbrk(*saveptr, "\r\n");
217
218 if (*saveptr != NULL) {
219 char *eos = *saveptr;
220
221 if ((*saveptr)[0] == '\r' && (*saveptr)[1] == '\n')
222 (*saveptr)++;
223 (*saveptr)++;
224 if ((*saveptr)[0] == '\0')
225 *saveptr = NULL;
226
227 *eos = '\0';
228 }
229 }
230
231 return result;
232}
233
234
235
Holger Hans Peter Freyther38159422015-05-02 10:02:38 +0200236#define ENDPOINT_NUMBER(endp) abs((int)(endp - endp->tcfg->endpoints))
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100237
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200238/**
239 * Internal structure while parsing a request
240 */
241struct mgcp_parse_data {
242 struct mgcp_config *cfg;
243 struct mgcp_endpoint *endp;
244 char *trans;
245 char *save;
246 int found;
Holger Hans Peter Freytherf2f15912010-04-01 03:27:04 +0200247};
248
Holger Hans Peter Freytherb844b872010-04-21 21:25:13 +0800249int mgcp_send_dummy(struct mgcp_endpoint *endp);
Holger Hans Peter Freytherbb89aa12010-08-05 03:29:33 +0800250int mgcp_bind_bts_rtp_port(struct mgcp_endpoint *endp, int rtp_port);
Holger Hans Peter Freyther314584a2010-08-05 04:10:21 +0800251int mgcp_bind_net_rtp_port(struct mgcp_endpoint *endp, int rtp_port);
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100252int mgcp_bind_trans_bts_rtp_port(struct mgcp_endpoint *enp, int rtp_port);
253int mgcp_bind_trans_net_rtp_port(struct mgcp_endpoint *enp, int rtp_port);
Holger Hans Peter Freytherf138f912010-08-05 08:08:17 +0800254int mgcp_free_rtp_port(struct mgcp_rtp_end *end);
Holger Hans Peter Freytherf2f15912010-04-01 03:27:04 +0200255
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100256/* For transcoding we need to manage an in and an output that are connected */
257static inline int endp_back_channel(int endpoint)
258{
259 return endpoint + 60;
260}
261
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100262struct mgcp_trunk_config *mgcp_trunk_alloc(struct mgcp_config *cfg, int index);
263struct mgcp_trunk_config *mgcp_trunk_num(struct mgcp_config *cfg, int index);
264
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100265void mgcp_rtp_end_config(struct mgcp_endpoint *endp, int expect_ssrc_change,
266 struct mgcp_rtp_end *rtp);
Jacob Erlbeckf6ec0e92013-12-04 10:30:11 +0100267uint32_t mgcp_rtp_packet_duration(struct mgcp_endpoint *endp,
268 struct mgcp_rtp_end *rtp);
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100269
Holger Hans Peter Freyther38e02c52012-10-22 18:09:35 +0200270void mgcp_state_calc_loss(struct mgcp_rtp_state *s, struct mgcp_rtp_end *,
271 uint32_t *expected, int *loss);
Holger Hans Peter Freythercb306a62012-10-24 21:22:47 +0200272uint32_t mgcp_state_calc_jitter(struct mgcp_rtp_state *);
Holger Hans Peter Freyther38e02c52012-10-22 18:09:35 +0200273
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100274/* payload processing default functions */
Jacob Erlbeck42a833e2014-04-14 10:31:47 +0200275int mgcp_rtp_processing_default(struct mgcp_endpoint *endp, struct mgcp_rtp_end *dst_end,
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100276 char *data, int *len, int buf_size);
277
278int mgcp_setup_rtp_processing_default(struct mgcp_endpoint *endp,
279 struct mgcp_rtp_end *dst_end,
280 struct mgcp_rtp_end *src_end);
281
Jacob Erlbeck168ca002014-03-17 12:40:07 +0100282void mgcp_get_net_downlink_format_default(struct mgcp_endpoint *endp,
283 int *payload_type,
284 const char**subtype_name,
285 const char**fmtp_extra);
286
Holger Hans Peter Freyther05d481a2014-10-06 21:01:26 +0200287/* internal RTP Annex A counting */
288void mgcp_rtp_annex_count(struct mgcp_endpoint *endp, struct mgcp_rtp_state *state,
289 const uint16_t seq, const int32_t transit,
290 const uint32_t ssrc);
291
Holger Hans Peter Freyther9be675e2015-01-21 11:39:47 +0100292int mgcp_set_ip_tos(int fd, int tos);
293
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100294enum {
295 MGCP_DEST_NET = 0,
296 MGCP_DEST_BTS,
297};
298
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200299
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100300#define MGCP_DUMMY_LOAD 0x23
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100301
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200302
303/**
304 * SDP related information
305 */
306/* Assume audio frame length of 20ms */
307#define DEFAULT_RTP_AUDIO_FRAME_DUR_NUM 20
308#define DEFAULT_RTP_AUDIO_FRAME_DUR_DEN 1000
309#define DEFAULT_RTP_AUDIO_PACKET_DURATION_MS 20
310#define DEFAULT_RTP_AUDIO_DEFAULT_RATE 8000
311#define DEFAULT_RTP_AUDIO_DEFAULT_CHANNELS 1
312
313#define PTYPE_UNDEFINED (-1)
Holger Hans Peter Freytheraeadf262015-08-14 15:43:06 +0200314int mgcp_parse_sdp_data(struct mgcp_endpoint *endp, struct mgcp_rtp_end *rtp, struct mgcp_parse_data *p);
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200315int mgcp_set_audio_info(void *ctx, struct mgcp_rtp_codec *codec,
316 int payload_type, const char *audio_name);
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200317
318
319/**
320 * Internal network related
321 */
322static inline const char *mgcp_net_src_addr(struct mgcp_endpoint *endp)
323{
Holger Hans Peter Freytherf3316e32015-10-12 09:37:35 +0200324 if (endp->cfg->net_ports.bind_addr)
325 return endp->cfg->net_ports.bind_addr;
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200326 return endp->cfg->source_addr;
327}
328
329static inline const char *mgcp_bts_src_addr(struct mgcp_endpoint *endp)
330{
Holger Hans Peter Freytherf3316e32015-10-12 09:37:35 +0200331 if (endp->cfg->bts_ports.bind_addr)
332 return endp->cfg->bts_ports.bind_addr;
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200333 return endp->cfg->source_addr;
334}
Neels Hofmeyr2188a772016-05-20 21:59:55 +0200335
336int mgcp_msg_terminate_nul(struct msgb *msg);