blob: 4736aaaaeb118a8354a917b053c8bd465d0c6cec [file] [log] [blame]
Harald Welte4029e8c2017-11-23 22:00:42 +01001module MGCP_Templates {
2
Harald Welte35bb7162018-01-03 21:07:52 +01003/* MGCP Templates, building on top of MGCP_Types (Osmocom) and SDP_Types from Ericsson.
4 *
5 * (C) 2017 by Harald Welte <laforge@gnumonks.org>
6 * All rights reserved.
7 *
8 * Released under the terms of GNU General Public License, Version 2 or
9 * (at your option) any later version.
10 */
11
12
Harald Welte4029e8c2017-11-23 22:00:42 +010013 import from MGCP_Types all;
14 import from SDP_Types all;
15
16 function f_mgcp_par_append(inout template MgcpParameterList list, template MgcpParameter par) {
17 var integer len := lengthof(list);
18 list[len] := par;
19 }
20
21 /* 3.2.2.6 Connection Mode (sendonly, recvonly, sendrecv, confrnce, inactive, loopback,
22 * conttest, netwloop, netwtest) */
23 template MgcpParameter t_MgcpParConnMode(template MgcpConnectionMode mode) := { "M", mode };
24
25 /* 3.2.2.2 CallId: maximum 32 hex chars */
26 template MgcpParameter ts_MgcpParCallId(MgcpCallId cid) := {
27 code := "C",
28 val := hex2str(cid)
29 };
30
31 /* 3.2.2.18 RequestIdentifier: Maximum 32 hex chars */
32 template MgcpParameter ts_MgcpParReqId(MgcpRequestId rid) := {
33 code := "X",
34 val := hex2str(rid)
35 };
36
Harald Welte363cb0a2018-01-30 19:35:53 +010037 /* 3.2.1.3 SpecificEndpointId */
38 template MgcpParameter ts_MgcpParSpecEP(MgcpEndpoint ep) := {
39 code := "Z",
40 val := ep
41 };
42
Harald Welte4029e8c2017-11-23 22:00:42 +010043 /* 3.2.2.10: LocalConnectionOptions (codec, packetization, bandwidth, ToS, eco, gain, silence, ...) */
44 template MgcpParameter t_MgcpParLocConnOpt(template charstring lco) := { "L", lco };
45
46 /* 3.2.2.5: ConnectionId: maximum 32 hex chars */
47 template MgcpParameter ts_MgcpParConnectionId(MgcpConnectionId cid) := {
48 code := "I",
49 val := hex2str(cid)
50 };
51
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +020052 /* Osmocom extension: X-Osmux: {*,%u} */
53 template MgcpParameter ts_MgcpParOsmuxCID(MgcpOsmuxCID osmux_cid) := {
54 code := "X-OSMUX",
55 val := f_mgcp_osmux_cid_encode(osmux_cid)
56 };
57
Harald Welte4029e8c2017-11-23 22:00:42 +010058 /* osmo-bsc_mgcp implements L/C/M/X only, osmo-mgw adds 'I' */
59 /* SDP: osmo-bsc_mgcp implements Tx of v,o,s,c,t,m,a */
60
61 template MgcpResponse tr_MgcpResp_Err(template MgcpResponseCode code) := {
62 line := {
63 code := code,
64 trans_id := ?,
65 string := ?
66 },
67 params := {},
68 sdp := omit
69 }
70
71 template MgcpCommandLine t_MgcpCmdLine(template charstring verb, template MgcpTransId trans_id, template charstring ep) := {
72 verb := verb,
73 trans_id := trans_id,
74 ep := ep,
75 ver := "1.0"
76 };
77
78 template MgcpCommand ts_CRCX(MgcpTransId trans_id, charstring ep, MgcpConnectionMode mode, MgcpCallId call_id, template SDP_Message sdp := omit) := {
79 line := t_MgcpCmdLine("CRCX", trans_id, ep),
80 params := {
81 t_MgcpParConnMode(mode),
82 ts_MgcpParCallId(call_id),
83 //t_MgcpParReqId(omit),
Daniel Willmannfbef7142017-11-30 13:16:14 +010084 t_MgcpParLocConnOpt("p:20, a:AMR")
Harald Welte4029e8c2017-11-23 22:00:42 +010085 },
86 sdp := sdp
87 }
88
Philipp Maier45635f42018-06-05 17:28:02 +020089 template MgcpCommand ts_CRCX_no_lco(MgcpTransId trans_id, charstring ep, MgcpConnectionMode mode, MgcpCallId call_id, template SDP_Message sdp := omit) := {
90 line := t_MgcpCmdLine("CRCX", trans_id, ep),
91 params := {
92 t_MgcpParConnMode(mode),
93 ts_MgcpParCallId(call_id)
94 },
95 sdp := sdp
96 }
97
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +020098 template MgcpCommand ts_CRCX_osmux(MgcpTransId trans_id, charstring ep, MgcpConnectionMode mode, MgcpCallId call_id, MgcpOsmuxCID osmux_cid, template SDP_Message sdp := omit) := {
99 line := t_MgcpCmdLine("CRCX", trans_id, ep),
100 params := {
101 t_MgcpParConnMode(mode),
102 ts_MgcpParCallId(call_id),
103 //t_MgcpParReqId(omit),
104 t_MgcpParLocConnOpt("p:20, a:AMR"),
105 ts_MgcpParOsmuxCID(osmux_cid)
106 },
107 sdp := sdp
108 }
109
Harald Welte4017d552018-01-26 21:40:05 +0100110 template MgcpCommand tr_CRCX(template MgcpEndpoint ep := ?) := {
111 line := t_MgcpCmdLine("CRCX", ?, ep),
Harald Welte90029572017-11-24 23:39:50 +0100112 params := *,
113 sdp := *
114 }
115
Harald Welte4029e8c2017-11-23 22:00:42 +0100116 template MgcpResponse tr_CRCX_ACK := {
117 line := {
118 code := "200",
Harald Welte51f34ad2017-11-24 20:40:43 +0100119 trans_id := ?,
Harald Welte4029e8c2017-11-23 22:00:42 +0100120 string := "OK"
121 },
122 params:= { { "I", ? }, *},
123 sdp := ?
124 }
125
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +0200126 template MgcpResponse tr_CRCX_ACK_osmux := {
127 line := {
128 code := "200",
129 trans_id := ?,
130 string := "OK"
131 },
132 params:= { { "I", ? }, {"X-OSMUX", ?}, *},
133 sdp := ?
134 }
135
Harald Welte90029572017-11-24 23:39:50 +0100136 template MgcpResponse ts_CRCX_ACK(MgcpTransId trans_id, MgcpConnectionId conn_id, template SDP_Message sdp := omit) := {
137 line := {
138 code := "200",
139 trans_id := trans_id,
140 string := "OK"
141 },
142 params:= { ts_MgcpParConnectionId(conn_id) },
143 sdp := sdp
144 }
145
Pau Espin Pedrolc6a53db2019-05-20 19:31:47 +0200146 template MgcpResponse ts_CRCX_ACK_osmux(MgcpTransId trans_id, MgcpConnectionId conn_id, MgcpOsmuxCID osmux_cid, template SDP_Message sdp := omit) := {
147 line := {
148 code := "200",
149 trans_id := trans_id,
150 string := "OK"
151 },
152 params:= {
153 ts_MgcpParConnectionId(conn_id),
154 ts_MgcpParOsmuxCID(osmux_cid)
155 },
156 sdp := sdp
157 }
158
Harald Welte4029e8c2017-11-23 22:00:42 +0100159 template MgcpCommand ts_MDCX(MgcpTransId trans_id, charstring ep, MgcpConnectionMode mode, MgcpCallId call_id, MgcpConnectionId conn_id, template SDP_Message sdp := omit) := {
160 line := t_MgcpCmdLine("MDCX", trans_id, ep),
161 params := {
162 t_MgcpParConnMode(mode),
163 ts_MgcpParCallId(call_id),
164 ts_MgcpParConnectionId(conn_id),
165 //t_MgcpParReqId(omit),
Daniel Willmannfbef7142017-11-30 13:16:14 +0100166 t_MgcpParLocConnOpt("p:20, a:AMR")
Harald Welte4029e8c2017-11-23 22:00:42 +0100167 },
168 sdp := sdp
169 }
170
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +0200171 template MgcpCommand ts_MDCX_osmux(MgcpTransId trans_id, charstring ep, MgcpConnectionMode mode, MgcpCallId call_id, MgcpConnectionId conn_id, MgcpOsmuxCID osmux_cid, template SDP_Message sdp := omit) := {
172 line := t_MgcpCmdLine("MDCX", trans_id, ep),
173 params := {
174 t_MgcpParConnMode(mode),
175 ts_MgcpParCallId(call_id),
176 ts_MgcpParConnectionId(conn_id),
177 //t_MgcpParReqId(omit),
178 t_MgcpParLocConnOpt("p:20, a:AMR"),
179 ts_MgcpParOsmuxCID(osmux_cid)
180 },
181 sdp := sdp
182 }
183
Harald Welte90029572017-11-24 23:39:50 +0100184 template MgcpCommand tr_MDCX := {
185 line := t_MgcpCmdLine("MDCX", ?, ?),
186 params := *,
187 sdp := *
188 }
189
Harald Weltebb7523b2018-03-29 08:52:01 +0200190 template MgcpResponse tr_MDCX_ACK := {
191 line := {
192 code := "200",
193 trans_id := ?,
194 string := "OK"
195 },
196 params := *,
197 sdp := ?
198 }
199
Harald Welte90029572017-11-24 23:39:50 +0100200 template MgcpResponse ts_MDCX_ACK(MgcpTransId trans_id, MgcpConnectionId conn_id, template SDP_Message sdp := omit) := ts_CRCX_ACK(trans_id, conn_id, sdp);
201
Harald Welte4029e8c2017-11-23 22:00:42 +0100202 /* have a function that generates a template, rather than a template in order to handle
203 * optional parameters */
204 function ts_DLCX(MgcpTransId trans_id, charstring ep, template MgcpCallId call_id := omit,
205 template MgcpConnectionId conn_id := omit) return template MgcpCommand {
206 var template MgcpCommand cmd;
207 cmd.line := t_MgcpCmdLine("DLCX", trans_id, ep);
208 cmd.params := {};
209 cmd.sdp := omit;
210 if (isvalue(call_id)) {
211 f_mgcp_par_append(cmd.params, ts_MgcpParCallId(valueof(call_id)));
212 if (isvalue(conn_id)) {
213 f_mgcp_par_append(cmd.params, ts_MgcpParConnectionId(valueof(conn_id)));
214 }
215 }
216 return cmd;
217 }
218
Harald Welteb71901a2018-01-26 19:16:05 +0100219 template MgcpCommand tr_DLCX(template MgcpEndpoint ep := ?) := {
220 line := t_MgcpCmdLine("DLCX", ?, ep),
Harald Welte90029572017-11-24 23:39:50 +0100221 params := *,
222 sdp := *
223 }
224
Harald Weltec82eef42017-11-24 20:40:12 +0100225 template MgcpResponse tr_DLCX_ACK := {
226 line := {
Daniel Willmann961e5c92017-11-30 16:37:40 +0100227 code := ("200", "250"),
Harald Weltec82eef42017-11-24 20:40:12 +0100228 trans_id := ?,
229 string := "OK"
230 },
231 params:= *,
232 sdp := *
233 }
234
Harald Welteb71901a2018-01-26 19:16:05 +0100235 template MgcpResponse ts_DLCX_ACK2(MgcpTransId trans_id) := {
236 line := {
237 code := "250",
238 trans_id := trans_id,
239 string := "OK"
240 },
241 params:= { /* list of ConnectionIDs */ },
242 sdp := omit
243 }
244
245
246
Harald Welte90029572017-11-24 23:39:50 +0100247 template MgcpResponse ts_DLCX_ACK(MgcpTransId trans_id, MgcpConnectionId conn_id, template SDP_Message sdp := omit) := ts_CRCX_ACK(trans_id, conn_id, sdp);
248
Harald Weltee98bb2e2017-11-29 12:09:48 +0100249 template MgcpCommand tr_RSIP := {
250 line := t_MgcpCmdLine("RSIP", ?, ?),
251 params := *,
252 sdp := *
253 }
254
Harald Welte4029e8c2017-11-23 22:00:42 +0100255 /* SDP Templates */
256 template SDP_Origin ts_SDP_origin(charstring addr, charstring session_id,
257 charstring session_version := "1",
258 charstring addr_type := "IP4",
259 charstring user_name := "-") := {
260 user_name := user_name,
261 session_id := session_id,
262 session_version := session_version,
263 net_type := "IN",
264 addr_type := addr_type,
265 addr := addr
266 }
267
268 template SDP_connection ts_SDP_connection_IP(charstring addr, charstring addr_type := "IP4",
269 template integer ttl := omit,
270 template integer num_of_addr := omit) :={
271 net_type := "IN",
272 addr_type := addr_type,
273 conn_addr := {
274 addr := addr,
275 ttl := ttl,
276 num_of_addr := num_of_addr
277 }
278 }
279
280 template SDP_time ts_SDP_time(charstring beg, charstring end) := {
281 time_field := {
282 start_time := beg,
283 stop_time := end
284 },
285 time_repeat := omit
286 }
287
288 template SDP_media_desc ts_SDP_media_desc(integer port_number, SDP_fmt_list fmts,
289 SDP_attribute_list attributes) := {
290 media_field := {
291 media := "audio",
292 ports := {
293 port_number := port_number,
294 num_of_ports := omit
295 },
296 transport := "RTP/AVP",
297 fmts := fmts
298 },
299 information := omit,
300 connections := omit,
301 bandwidth := omit,
302 key := omit,
303 attributes := attributes
304 }
305
306 /* master template for generating SDP based in template arguments */
307 template SDP_Message ts_SDP(charstring local_addr, charstring remote_addr,
308 charstring session_id, charstring session_version,
309 integer rtp_port, SDP_fmt_list fmts,
310 SDP_attribute_list attributes) := {
311 protocol_version := 0,
312 origin := ts_SDP_origin(local_addr, session_id, session_version),
313 session_name := "-",
314 information := omit,
315 uri := omit,
316 emails := omit,
317 phone_numbers := omit,
318 connection := ts_SDP_connection_IP(remote_addr),
319 bandwidth := omit,
320 times := { ts_SDP_time("0","0") },
321 timezone_adjustments := omit,
322 key := omit,
323 attributes := omit,
324 media_list := { ts_SDP_media_desc(rtp_port, fmts, attributes) }
325 }
326
327 template SDP_attribute ts_SDP_rtpmap(integer fmt, charstring val) := {
328 rtpmap := {
329 attr_value := int2str(fmt) & " " & val
330 }
331 }
332 template SDP_attribute ts_SDP_ptime(integer p) := {
333 ptime := {
334 attr_value := int2str(p)
335 }
336 }
Philipp Maierc8c0b402019-03-07 10:48:45 +0100337 template SDP_attribute ts_SDP_fmtp(integer fmt, charstring val) := {
338 fmtp := {
339 attr_value := int2str(fmt) & " " & val
340 }
341 }
Harald Welte4029e8c2017-11-23 22:00:42 +0100342
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +0200343 /* -1 is wildcard, positive is translated as string */
344 function f_mgcp_osmux_cid_encode(MgcpOsmuxCID osmux_cid) return charstring {
345 if (osmux_cid == -1) {
346 return "*";
347 }
348 return int2str(osmux_cid);
349 }
350
351 function f_mgcp_osmux_cid_decode(charstring osmux_cid) return MgcpOsmuxCID {
352 if (osmux_cid == "*") {
353 return -1;
354 }
355 return str2int(osmux_cid);
356 }
357
Neels Hofmeyr2ca1ab42019-03-08 03:45:43 +0100358 function f_mgcp_contains_par(MgcpMessage msg, MgcpInfoCode code) return boolean {
359 var MgcpParameterList pars;
360 if (ischosen(msg.command)) {
361 pars := msg.command.params;
362 } else {
363 pars := msg.response.params;
364 }
365 for (var integer i := 0; i < lengthof(pars); i := i + 1) {
366 var MgcpParameter par := pars[i];
367 if (par.code == code) {
368 return true;
369 }
370 }
371 return false;
372 }
373
Harald Welteb71901a2018-01-26 19:16:05 +0100374 function f_mgcp_extract_par(MgcpMessage msg, MgcpInfoCode code) return charstring {
375 var MgcpParameterList pars;
376 if (ischosen(msg.command)) {
377 pars := msg.command.params;
378 } else {
379 pars := msg.response.params;
380 }
381 for (var integer i := 0; i < lengthof(pars); i := i + 1) {
382 var MgcpParameter par := pars[i];
383 if (par.code == code) {
384 return par.val;
Harald Welte4c11d562017-11-24 23:39:00 +0100385 }
386 }
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200387 setverdict(fail, "Could not extract parameters for code ", code);
Harald Welteb71901a2018-01-26 19:16:05 +0100388 return "";
389 }
390
Harald Welte1fe74812018-01-29 21:57:26 +0100391 function f_MgcpResp_extract_par(MgcpResponse resp, MgcpInfoCode code) return charstring {
Harald Welteb71901a2018-01-26 19:16:05 +0100392 var MgcpMessage msg := {
393 response := resp
394 }
Harald Welte1fe74812018-01-29 21:57:26 +0100395 return f_mgcp_extract_par(msg, code);
Harald Welteb71901a2018-01-26 19:16:05 +0100396 }
397
Harald Welte1fe74812018-01-29 21:57:26 +0100398 function f_MgcpCmd_extract_par(MgcpCommand cmd, MgcpInfoCode code) return charstring {
Harald Welteb71901a2018-01-26 19:16:05 +0100399 var MgcpMessage msg := {
400 command := cmd
401 }
Harald Welte1fe74812018-01-29 21:57:26 +0100402 return f_mgcp_extract_par(msg, code);
Harald Welte4c11d562017-11-24 23:39:00 +0100403 }
404
Neels Hofmeyr2ca1ab42019-03-08 03:45:43 +0100405 function f_MgcpCmd_contains_par(MgcpCommand cmd, MgcpInfoCode code) return boolean {
406 var MgcpMessage msg := {
407 command := cmd
408 }
409 return f_mgcp_contains_par(msg, code);
410 }
411
Harald Welte1fe74812018-01-29 21:57:26 +0100412 function f_MgcpResp_extract_conn_id(MgcpResponse resp) return MgcpConnectionId {
413 return str2hex(f_MgcpResp_extract_par(resp, "I"));
414 }
415
416 function f_MgcpCmd_extract_call_id(MgcpCommand cmd) return MgcpCallId {
417 return str2hex(f_MgcpCmd_extract_par(cmd, "C"));
418 }
419
420 function f_MgcpCmd_extract_conn_id(MgcpCommand cmd) return MgcpConnectionId {
421 return str2hex(f_MgcpCmd_extract_par(cmd, "I"));
422 }
423
Pau Espin Pedrolb2c6b382019-05-14 13:40:49 +0200424 function f_MgcpCmd_extract_osmux_cid(MgcpCommand cmd) return MgcpOsmuxCID {
425 return f_mgcp_osmux_cid_decode(f_MgcpCmd_extract_par(cmd, "X-OSMUX"));
426 }
427
Harald Welte1fe74812018-01-29 21:57:26 +0100428
Harald Welte4c11d562017-11-24 23:39:00 +0100429 function f_mgcp_alloc_tid() return MgcpTransId {
430 return int2str(float2int(rnd()*2147483647.0));
431 }
432
433 function f_mgcp_alloc_call_id() return MgcpCallId {
434 return int2hex(float2int(rnd()*2147483647.0), 8);
435 }
436
437 function f_mgcp_alloc_conn_id() return MgcpConnectionId {
438 return int2hex(float2int(rnd()*2147483647.0), 8);
439 }
440
Harald Weltebb5a1212018-01-26 10:34:44 +0100441 /* those verbs that related to a connection (and hence have ConnectionId) */
442 template MgcpVerb tr_MgcpVerb_ConnectionOriented := ("CRCX", "MDCX", "DLCX", "AUCX");
443 /* entire command template matching only connection oriented verbs */
444 template MgcpCommand tr_MgcpCommand_CO := {
445 line := {
446 verb := tr_MgcpVerb_ConnectionOriented,
447 trans_id := ?,
448 ep := ?,
449 ver := ?
450 },
451 params := *,
452 sdp := *
453 }
454
Harald Welte363cb0a2018-01-30 19:35:53 +0100455 function f_mgcp_find_param(MgcpMessage msg, MgcpInfoCode code, out charstring ret)
456 return boolean {
457 var MgcpParameterList pars;
458 if (ischosen(msg.command)) {
459 pars := msg.command.params;
460 } else {
461 pars := msg.response.params;
462 }
463 for (var integer i := 0; i < sizeof(pars); i := i+1) {
464 if (pars[i].code == code) {
465 ret := pars[i].val;
466 return true;
467 }
468 }
469 return false;
470 }
471
472 /* template to determine if a MGCP endpoint is a wildcard endpoint */
473 template charstring t_MGCP_EP_wildcard := (pattern "\*@*", pattern "rtpbridge/\*@*");
474
Harald Welteb71901a2018-01-26 19:16:05 +0100475
Harald Welte4029e8c2017-11-23 22:00:42 +0100476}