blob: b2c24089d09c091588999b3fbf0f7a556c8fabb5 [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;
Neels Hofmeyr3e168c02017-03-16 16:14:34 +010067 bool patched_first_rtp_payload; /* FIXME: drop this, see OS#2459 */
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +000068};
Holger Hans Peter Freyther98a38772010-08-03 02:27:21 +080069
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020070struct mgcp_rtp_codec {
71 uint32_t rate;
72 int channels;
73 uint32_t frame_duration_num;
74 uint32_t frame_duration_den;
75
76 int payload_type;
77 char *audio_name;
78 char *subtype_name;
79};
80
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +080081struct mgcp_rtp_end {
82 /* statistics */
83 unsigned int packets;
Holger Hans Peter Freyther952f7522012-09-12 11:38:37 +020084 unsigned int octets;
Jacob Erlbeck0970bab2013-12-19 12:13:32 +010085 unsigned int dropped_packets;
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +080086 struct in_addr addr;
87
88 /* in network byte order */
89 int rtp_port, rtcp_port;
90
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020091 /* audio codec information */
92 struct mgcp_rtp_codec codec;
Holger Hans Peter Freythere46bc272014-09-01 10:50:22 +020093 struct mgcp_rtp_codec alt_codec; /* TODO/XXX: make it generic */
Holger Hans Peter Freythercac24382014-09-01 10:35:55 +020094
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +020095 /* per endpoint data */
Jacob Erlbeck72c30902013-11-25 15:23:35 +010096 int frames_per_packet;
Jacob Erlbeckf6ec0e92013-12-04 10:30:11 +010097 uint32_t packet_duration_ms;
Holger Hans Peter Freyther5ea1bc72012-09-03 00:07:39 +020098 char *fmtp_extra;
Jacob Erlbeck0970bab2013-12-19 12:13:32 +010099 int output_enabled;
Jacob Erlbeck42a833e2014-04-14 10:31:47 +0200100 int force_output_ptime;
Holger Hans Peter Freyther58ff2192010-08-05 03:08:35 +0800101
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100102 /* RTP patching */
Jacob Erlbecke2292f32013-12-03 15:13:12 +0100103 int force_constant_ssrc; /* -1: always, 0: don't, 1: once */
Jacob Erlbeck4bbddc62013-12-18 12:54:51 +0100104 int force_aligned_timing;
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100105 void *rtp_process_data;
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100106
Holger Hans Peter Freytherc4921272010-08-05 03:37:22 +0800107 /*
108 * Each end has a socket...
109 */
Pablo Neira Ayuso4db92992011-05-06 12:11:23 +0200110 struct osmo_fd rtp;
111 struct osmo_fd rtcp;
Holger Hans Peter Freytherc4921272010-08-05 03:37:22 +0800112
Holger Hans Peter Freyther58ff2192010-08-05 03:08:35 +0800113 int local_port;
Holger Hans Peter Freytherf138f912010-08-05 08:08:17 +0800114 int local_alloc;
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +0800115};
116
Holger Hans Peter Freyther260d6ed2010-08-06 01:12:21 +0800117enum {
118 MGCP_TAP_BTS_IN,
119 MGCP_TAP_BTS_OUT,
120 MGCP_TAP_NET_IN,
121 MGCP_TAP_NET_OUT,
122
123 /* last element */
124 MGCP_TAP_COUNT
125};
126
127struct mgcp_rtp_tap {
128 int enabled;
129 struct sockaddr_in forward;
130};
131
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100132struct mgcp_lco {
133 char *string;
Jacob Erlbeck452c1832014-05-09 19:42:17 +0200134 char *codec;
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100135 int pkt_period_min; /* time in ms */
136 int pkt_period_max; /* time in ms */
137};
138
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200139enum mgcp_type {
140 MGCP_RTP_DEFAULT = 0,
141 MGCP_RTP_TRANSCODED,
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100142 MGCP_OSMUX_BSC,
143 MGCP_OSMUX_BSC_NAT,
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200144};
145
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100146#include <openbsc/osmux.h>
147
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100148struct mgcp_endpoint {
Holger Hans Peter Freyther39a97e22010-08-06 18:03:11 +0800149 int allocated;
Holger Hans Peter Freyther46340132010-08-06 08:26:54 +0800150 uint32_t ci;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100151 char *callid;
Jacob Erlbeck2c2ca4d2013-12-09 14:32:03 +0100152 struct mgcp_lco local_options;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100153 int conn_mode;
Holger Hans Peter Freytherc597a4e2010-08-03 02:57:02 +0800154 int orig_mode;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100155
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100156 /* backpointer */
157 struct mgcp_config *cfg;
Holger Hans Peter Freyther88ad7722011-02-28 00:56:17 +0100158 struct mgcp_trunk_config *tcfg;
Holger Hans Peter Freytherb4b135e2010-04-07 09:37:17 +0200159
Holger Hans Peter Freythera17d7012010-08-05 01:34:51 +0800160 /* port status for bts/net */
161 struct mgcp_rtp_end bts_end;
162 struct mgcp_rtp_end net_end;
Holger Hans Peter Freyther21262332010-11-01 20:53:31 +0100163
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100164 /*
165 * For transcoding we will send from the local_port
166 * of trans_bts and it will arrive at trans_net from
167 * where we will forward it to the network.
168 */
169 struct mgcp_rtp_end trans_bts;
Holger Hans Peter Freyther21262332010-11-01 20:53:31 +0100170 struct mgcp_rtp_end trans_net;
Pablo Neira Ayuso46bd4242013-07-08 05:09:46 +0200171 enum mgcp_type type;
Holger Hans Peter Freyther380b8712010-07-29 02:38:39 +0800172
173 /* sequence bits */
Holger Hans Peter Freyther31868922010-08-03 11:59:04 +0000174 struct mgcp_rtp_state net_state;
175 struct mgcp_rtp_state bts_state;
Holger Hans Peter Freyther6357a8e2010-08-05 12:07:00 +0000176
Holger Hans Peter Freytherce553612012-12-07 15:04:07 +0100177 /* fields for re-transmission */
178 char *last_trans;
179 char *last_response;
180
Holger Hans Peter Freyther260d6ed2010-08-06 01:12:21 +0800181 /* tap for the endpoint */
182 struct mgcp_rtp_tap taps[MGCP_TAP_COUNT];
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100183
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200184 struct {
Pablo Neira Ayusob769f3c2014-08-27 17:02:52 +0200185 /* Osmux state: disabled, activating, active */
186 enum osmux_state state;
187 /* Allocated Osmux circuit ID for this endpoint */
Holger Hans Peter Freyther1afe7c72015-10-04 11:11:11 +0200188 int allocated_cid;
189 /* Used Osmux circuit ID for this endpoint */
Pablo Neira Ayusob769f3c2014-08-27 17:02:52 +0200190 uint8_t cid;
191 /* handle to batch messages */
192 struct osmux_in_handle *in;
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200193 /* handle to unbatch messages */
194 struct osmux_out_handle out;
Pablo Neira Ayusoee11bc02014-09-17 12:33:09 +0200195 /* statistics */
196 struct {
197 uint32_t chunks;
198 uint32_t octets;
199 } stats;
Pablo Neira Ayuso63650bb2014-08-26 13:31:53 +0200200 } osmux;
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100201};
202
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200203#define for_each_line(line, save) \
204 for (line = strline_r(NULL, &save); line;\
205 line = strline_r(NULL, &save))
206
207static inline char *strline_r(char *str, char **saveptr)
208{
209 char *result;
210
211 if (str)
212 *saveptr = str;
213
214 result = *saveptr;
215
216 if (*saveptr != NULL) {
217 *saveptr = strpbrk(*saveptr, "\r\n");
218
219 if (*saveptr != NULL) {
220 char *eos = *saveptr;
221
222 if ((*saveptr)[0] == '\r' && (*saveptr)[1] == '\n')
223 (*saveptr)++;
224 (*saveptr)++;
225 if ((*saveptr)[0] == '\0')
226 *saveptr = NULL;
227
228 *eos = '\0';
229 }
230 }
231
232 return result;
233}
234
235
236
Holger Hans Peter Freyther38159422015-05-02 10:02:38 +0200237#define ENDPOINT_NUMBER(endp) abs((int)(endp - endp->tcfg->endpoints))
Holger Hans Peter Freyther7bdc6372010-02-20 21:21:02 +0100238
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200239/**
240 * Internal structure while parsing a request
241 */
242struct mgcp_parse_data {
243 struct mgcp_config *cfg;
244 struct mgcp_endpoint *endp;
245 char *trans;
246 char *save;
247 int found;
Holger Hans Peter Freytherf2f15912010-04-01 03:27:04 +0200248};
249
Holger Hans Peter Freytherb844b872010-04-21 21:25:13 +0800250int mgcp_send_dummy(struct mgcp_endpoint *endp);
Holger Hans Peter Freytherbb89aa12010-08-05 03:29:33 +0800251int mgcp_bind_bts_rtp_port(struct mgcp_endpoint *endp, int rtp_port);
Holger Hans Peter Freyther314584a2010-08-05 04:10:21 +0800252int mgcp_bind_net_rtp_port(struct mgcp_endpoint *endp, int rtp_port);
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100253int mgcp_bind_trans_bts_rtp_port(struct mgcp_endpoint *enp, int rtp_port);
254int mgcp_bind_trans_net_rtp_port(struct mgcp_endpoint *enp, int rtp_port);
Holger Hans Peter Freytherf138f912010-08-05 08:08:17 +0800255int mgcp_free_rtp_port(struct mgcp_rtp_end *end);
Holger Hans Peter Freytherf2f15912010-04-01 03:27:04 +0200256
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100257/* For transcoding we need to manage an in and an output that are connected */
258static inline int endp_back_channel(int endpoint)
259{
260 return endpoint + 60;
261}
262
Holger Hans Peter Freyther0e939fe2011-02-28 12:11:02 +0100263struct mgcp_trunk_config *mgcp_trunk_alloc(struct mgcp_config *cfg, int index);
264struct mgcp_trunk_config *mgcp_trunk_num(struct mgcp_config *cfg, int index);
265
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100266void mgcp_rtp_end_config(struct mgcp_endpoint *endp, int expect_ssrc_change,
267 struct mgcp_rtp_end *rtp);
Jacob Erlbeckf6ec0e92013-12-04 10:30:11 +0100268uint32_t mgcp_rtp_packet_duration(struct mgcp_endpoint *endp,
269 struct mgcp_rtp_end *rtp);
Jacob Erlbeckdb2d4312013-12-03 14:43:34 +0100270
Holger Hans Peter Freyther38e02c52012-10-22 18:09:35 +0200271void mgcp_state_calc_loss(struct mgcp_rtp_state *s, struct mgcp_rtp_end *,
272 uint32_t *expected, int *loss);
Holger Hans Peter Freythercb306a62012-10-24 21:22:47 +0200273uint32_t mgcp_state_calc_jitter(struct mgcp_rtp_state *);
Holger Hans Peter Freyther38e02c52012-10-22 18:09:35 +0200274
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100275/* payload processing default functions */
Jacob Erlbeck42a833e2014-04-14 10:31:47 +0200276int mgcp_rtp_processing_default(struct mgcp_endpoint *endp, struct mgcp_rtp_end *dst_end,
Jacob Erlbecka0d64ce2014-03-13 14:24:52 +0100277 char *data, int *len, int buf_size);
278
279int mgcp_setup_rtp_processing_default(struct mgcp_endpoint *endp,
280 struct mgcp_rtp_end *dst_end,
281 struct mgcp_rtp_end *src_end);
282
Jacob Erlbeck168ca002014-03-17 12:40:07 +0100283void mgcp_get_net_downlink_format_default(struct mgcp_endpoint *endp,
284 int *payload_type,
285 const char**subtype_name,
286 const char**fmtp_extra);
287
Holger Hans Peter Freyther05d481a2014-10-06 21:01:26 +0200288/* internal RTP Annex A counting */
289void mgcp_rtp_annex_count(struct mgcp_endpoint *endp, struct mgcp_rtp_state *state,
290 const uint16_t seq, const int32_t transit,
291 const uint32_t ssrc);
292
Holger Hans Peter Freyther9be675e2015-01-21 11:39:47 +0100293int mgcp_set_ip_tos(int fd, int tos);
294
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100295enum {
296 MGCP_DEST_NET = 0,
297 MGCP_DEST_BTS,
298};
299
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200300
Pablo Neira Ayusocab6e752014-02-05 18:56:17 +0100301#define MGCP_DUMMY_LOAD 0x23
Holger Hans Peter Freytherbd7b3c52010-11-01 21:04:54 +0100302
Holger Hans Peter Freythera611da82015-08-14 09:24:11 +0200303
304/**
305 * SDP related information
306 */
307/* Assume audio frame length of 20ms */
308#define DEFAULT_RTP_AUDIO_FRAME_DUR_NUM 20
309#define DEFAULT_RTP_AUDIO_FRAME_DUR_DEN 1000
310#define DEFAULT_RTP_AUDIO_PACKET_DURATION_MS 20
311#define DEFAULT_RTP_AUDIO_DEFAULT_RATE 8000
312#define DEFAULT_RTP_AUDIO_DEFAULT_CHANNELS 1
313
314#define PTYPE_UNDEFINED (-1)
Holger Hans Peter Freytheraeadf262015-08-14 15:43:06 +0200315int 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 +0200316int mgcp_set_audio_info(void *ctx, struct mgcp_rtp_codec *codec,
317 int payload_type, const char *audio_name);
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200318
319
320/**
321 * Internal network related
322 */
323static inline const char *mgcp_net_src_addr(struct mgcp_endpoint *endp)
324{
Holger Hans Peter Freytherf3316e32015-10-12 09:37:35 +0200325 if (endp->cfg->net_ports.bind_addr)
326 return endp->cfg->net_ports.bind_addr;
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200327 return endp->cfg->source_addr;
328}
329
330static inline const char *mgcp_bts_src_addr(struct mgcp_endpoint *endp)
331{
Holger Hans Peter Freytherf3316e32015-10-12 09:37:35 +0200332 if (endp->cfg->bts_ports.bind_addr)
333 return endp->cfg->bts_ports.bind_addr;
Holger Hans Peter Freythere6ed0092015-08-20 14:58:19 +0200334 return endp->cfg->source_addr;
335}
Neels Hofmeyr2188a772016-05-20 21:59:55 +0200336
337int mgcp_msg_terminate_nul(struct msgb *msg);