blob: 5cdf324fdb000a72bd0ac920463014ea6fe33bdc [file] [log] [blame]
Harald Welted86cdc62017-11-22 00:45:07 +01001module IPA_Emulation {
2
Harald Weltebdb63702017-12-09 01:15:44 +01003/* This module implements the IPA multiplex protocol on top of TCP, using the IPL4asp
4 * test-port as provider. It implements both client and server roles, as well was the CCM
Pau Espin Pedrol76ba5412019-06-10 11:00:33 +02005 * handshake for establishing the identity of the client to the server.
Harald Welte35bb7162018-01-03 21:07:52 +01006 *
7 * It already knows certain well-known sub-protocols such as A-bis RSL, MGCP and SCCP and
8 * GSUP. IT hence transcodes messages so the user can work with abstract data types rather
9 * than binary messages. It handles multiple packets inside one TCP segment.
10 *
Harald Welte34b5a952019-05-27 11:54:11 +020011 * (C) 2017-2019 by Harald Welte <laforge@gnumonks.org>
12 * contributions by sysmocom - s.f.m.c. GmbH
Harald Welte35bb7162018-01-03 21:07:52 +010013 * All rights reserved.
14 *
15 * Released under the terms of GNU General Public License, Version 2 or
16 * (at your option) any later version.
Harald Welte34b5a952019-05-27 11:54:11 +020017 *
18 * SPDX-License-Identifier: GPL-2.0-or-later
Harald Welte35bb7162018-01-03 21:07:52 +010019 */
Harald Weltebdb63702017-12-09 01:15:44 +010020
Harald Welted86cdc62017-11-22 00:45:07 +010021import from IPA_Types all;
22import from IPA_CodecPort all;
23import from IPA_CodecPort_CtrlFunct all;
24import from IPL4asp_Types all;
Harald Welte12188832017-11-29 11:47:13 +010025import from IPL4asp_PortType all;
Stefan Sperling830dc9d2018-02-12 21:08:28 +010026import from Socket_API_Definitions all;
Harald Welted86cdc62017-11-22 00:45:07 +010027
Harald Weltedf277252018-02-20 15:49:30 +010028#ifdef IPA_EMULATION_SCCP
29import from MTP3asp_Types all;
30import from MTP3asp_PortType all;
31#endif
32
33#ifdef IPA_EMULATION_RSL
34import from RSL_Types all;
35#endif
36
Harald Weltec6826662019-02-06 22:26:46 +010037#ifdef IPA_EMULATION_OML
38import from AbisOML_Types all;
39#endif
40
Harald Weltedf277252018-02-20 15:49:30 +010041#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +010042import from MGCP_Types all;
Harald Weltedf277252018-02-20 15:49:30 +010043#endif
44
45#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +010046import from GSUP_Types all;
Harald Weltedf277252018-02-20 15:49:30 +010047#endif
Harald Weltec82eef42017-11-24 20:40:12 +010048
Harald Welte7460a722018-10-10 12:28:27 +020049#ifdef IPA_EMULATION_RSPRO
50import from RSPRO all;
51import from RSPRO_Types all;
52#endif
53
Harald Welte3286eb32019-08-18 19:36:10 +020054#ifdef IPA_EMULATION_CTRL
Harald Weltea76c4bb2017-12-09 02:06:07 +010055import from Osmocom_CTRL_Types all;
Harald Welte3286eb32019-08-18 19:36:10 +020056#endif
Harald Weltea76c4bb2017-12-09 02:06:07 +010057
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +020058#ifdef IPA_EMULATION_OSMO_PCU
59import from PCUIF_Types all;
60#endif
61
Harald Welted86cdc62017-11-22 00:45:07 +010062modulepar {
Harald Welte92632e12017-11-25 02:31:20 +010063 /* Use Osmocom extended IPA mux header */
64 boolean mp_ipa_mgcp_uses_osmo_ext := true;
Harald Welted86cdc62017-11-22 00:45:07 +010065}
Harald Welted86cdc62017-11-22 00:45:07 +010066
Harald Welteb3414b22017-11-23 18:22:10 +010067type enumerated IpaMode {
68 IPA_MODE_CLIENT,
69 IPA_MODE_SERVER
70}
71
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +020072type enumerated IpaInitBehavior {
73 IPA_INIT_SEND_IPA_ID_GET,
74 IPA_INIT_SEND_IPA_ID_ACK
75}
76
Harald Weltec76f29f2017-11-22 12:46:46 +010077type record ASP_IPA_Unitdata {
78 IpaStreamId streamId,
Harald Weltec82eef42017-11-24 20:40:12 +010079 IpaExtStreamId streamIdExt optional,
Harald Weltec76f29f2017-11-22 12:46:46 +010080 octetstring payload
81}
82
Vadim Yanitskiy8701b272020-05-18 20:17:25 +070083type enumerated ASP_IPA_EventType {
Harald Welte0d846a72017-12-07 17:58:28 +010084 ASP_IPA_EVENT_DOWN,
85 ASP_IPA_EVENT_UP,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +070086 ASP_IPA_EVENT_ID_RESP,
Harald Welte0d846a72017-12-07 17:58:28 +010087 ASP_IPA_EVENT_ID_ACK
88}
89
Harald Weltebdb63702017-12-09 01:15:44 +010090/* an event indicating us whether or not a connection is physically up or down,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +070091 * and whether we have received an ID_RESP or ID_ACK */
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +070092type record ASP_IPA_Event {
93 ASP_IPA_EventType ev_type,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +070094 integer conn_id,
95
96 /* Presence of these fields depends on event type */
97 IpaCcmIdResp id_resp optional // ASP_IPA_EVENT_ID_RESP
Harald Welte0d846a72017-12-07 17:58:28 +010098}
99
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700100template (value) ASP_IPA_Event ts_ASP_IPA_EV(ASP_IPA_EventType ev_type, integer conn_id,
101 template (omit) IpaCcmIdResp id_resp := omit) := {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700102 ev_type := ev_type,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700103 conn_id := conn_id,
104 id_resp := id_resp
Vadim Yanitskiy61564be2020-05-18 20:44:14 +0700105}
106
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700107template ASP_IPA_Event tr_ASP_IPA_EV(template ASP_IPA_EventType ev_type,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700108 template integer conn_id := ?,
109 template IpaCcmIdResp id_resp := *) := {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700110 ev_type := ev_type,
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700111 conn_id := conn_id,
112 id_resp := id_resp
Harald Welte0d846a72017-12-07 17:58:28 +0100113}
114
Harald Welte1dd8f372017-11-25 02:25:27 +0100115template ASP_IPA_Unitdata t_ASP_IPA_UD(IpaStreamId sid, octetstring pl,
116 template IpaExtStreamId esid := omit) := {
117 streamId := sid,
118 streamIdExt := esid,
119 payload := pl
120}
121
Harald Weltedf277252018-02-20 15:49:30 +0100122#ifdef IPA_EMULATION_RSL
Harald Weltebdb63702017-12-09 01:15:44 +0100123/* like ASP_IPA_Unitdata, but with RSL_Message abstract type instead of octetstring */
Harald Welte0d846a72017-12-07 17:58:28 +0100124type record ASP_RSL_Unitdata {
Vadim Yanitskiyef9bcd22020-05-25 19:01:04 +0700125 integer conn_id optional,
Harald Welte0d846a72017-12-07 17:58:28 +0100126 IpaStreamId streamId,
127 RSL_Message rsl
128};
129
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700130template (value) ASP_RSL_Unitdata ts_ASP_RSL_UD(template (value) RSL_Message rsl,
Vadim Yanitskiyef9bcd22020-05-25 19:01:04 +0700131 IpaStreamId sid := IPAC_PROTO_RSL_TRX0,
132 template (omit) integer conn_id := omit) := {
133 conn_id := conn_id,
Harald Welte0d846a72017-12-07 17:58:28 +0100134 streamId := sid,
Vadim Yanitskiy6b4856d2020-05-25 21:23:46 +0700135 rsl := rsl
Harald Welte0d846a72017-12-07 17:58:28 +0100136}
137
Vadim Yanitskiy9b4e3562020-05-25 21:40:52 +0700138template ASP_RSL_Unitdata tr_ASP_RSL_UD(template (present) RSL_Message rsl,
Vadim Yanitskiyef9bcd22020-05-25 19:01:04 +0700139 template IpaStreamId sid := IPAC_PROTO_RSL_TRX0,
140 template integer conn_id := *) := {
141 conn_id := conn_id,
Harald Welte7ae019e2017-12-09 00:54:15 +0100142 streamId := sid,
143 rsl := rsl
144}
145
Harald Welte0d846a72017-12-07 17:58:28 +0100146template IpaStreamId t_IpaSidRSL := ( IPAC_PROTO_RSL_TRX0, IPAC_PROTO_RSL_TRX1,
147 IPAC_PROTO_RSL_TRX2, IPAC_PROTO_RSL_TRX3 );
Harald Weltedf277252018-02-20 15:49:30 +0100148#endif
Harald Welte0d846a72017-12-07 17:58:28 +0100149
Harald Weltec6826662019-02-06 22:26:46 +0100150
Harald Weltebdb63702017-12-09 01:15:44 +0100151/* Client port for general IPA messages, not further decoded */
Harald Weltec76f29f2017-11-22 12:46:46 +0100152type port IPA_SP_PT message {
Harald Welte0d846a72017-12-07 17:58:28 +0100153 inout ASP_IPA_Unitdata, ASP_IPA_Event;
Harald Weltec76f29f2017-11-22 12:46:46 +0100154} with { extension "internal" }
155
Harald Weltedf277252018-02-20 15:49:30 +0100156#ifdef IPA_EMULATION_MGCP
Harald Weltebdb63702017-12-09 01:15:44 +0100157/* Client port for MGCP inside IPA */
Harald Weltec82eef42017-11-24 20:40:12 +0100158type port IPA_MGCP_PT message {
159 inout MgcpCommand, MgcpResponse;
160} with { extension "internal" }
Harald Weltedf277252018-02-20 15:49:30 +0100161#endif
Harald Weltec82eef42017-11-24 20:40:12 +0100162
Harald Weltedf277252018-02-20 15:49:30 +0100163#ifdef IPA_EMULATION_RSL
Harald Weltebdb63702017-12-09 01:15:44 +0100164/* Client port for A-bis RSL inside IPA */
Harald Welte0d846a72017-12-07 17:58:28 +0100165type port IPA_RSL_PT message {
166 inout ASP_RSL_Unitdata, ASP_IPA_Event;
167} with { extension "internal" }
Harald Weltedf277252018-02-20 15:49:30 +0100168#endif
Harald Welte0d846a72017-12-07 17:58:28 +0100169
Harald Weltec6826662019-02-06 22:26:46 +0100170#ifdef IPA_EMULATION_OML
171/* Client port for A-bis OML inside IPA */
172type port IPA_OML_PT message {
173 inout OML_PDU, octetstring, ASP_IPA_Event;
174} with { extension "internal" }
175#endif
176
Harald Welte3286eb32019-08-18 19:36:10 +0200177#ifdef IPA_EMULATION_CTRL
Harald Weltea76c4bb2017-12-09 02:06:07 +0100178/* Client port for CTRL inside IPA */
179type port IPA_CTRL_PT message {
180 inout CtrlMessage, ASP_IPA_Event;
181} with { extension "internal" }
Harald Welte3286eb32019-08-18 19:36:10 +0200182#endif
Harald Weltea76c4bb2017-12-09 02:06:07 +0100183
Harald Weltedf277252018-02-20 15:49:30 +0100184#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +0100185/* Client port for CTRL inside IPA */
186type port IPA_GSUP_PT message {
187 inout GSUP_PDU, ASP_IPA_Event;
188} with { extension "internal" }
Harald Weltedf277252018-02-20 15:49:30 +0100189#endif
Harald Weltedf327232017-12-28 22:51:51 +0100190
Harald Welte7460a722018-10-10 12:28:27 +0200191#ifdef IPA_EMULATION_RSPRO
192type port IPA_RSPRO_PT message {
193 inout RsproPDU, ASP_IPA_Event;
194} with { extension "internal" }
195#endif
196
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200197#ifdef IPA_EMULATION_OSMO_PCU
198/* Client port for Osmocom PCU extension inside IPA */
199type port IPA_OSMO_PCU_PT message {
200 inout PCUIF_Message, ASP_IPA_Event;
201} with { extension "internal" }
202#endif
Harald Welte7460a722018-10-10 12:28:27 +0200203
204
Harald Weltedf327232017-12-28 22:51:51 +0100205
Harald Welted86cdc62017-11-22 00:45:07 +0100206type component IPA_Emulation_CT {
207 /* down-facing port to IPA codec port */
208 port IPA_CODEC_PT IPA_PORT;
Harald Weltedf277252018-02-20 15:49:30 +0100209#ifdef IPA_EMULATION_SCCP
Harald Welted86cdc62017-11-22 00:45:07 +0100210 /* up-facing port to SCCP */
211 port MTP3asp_SP_PT MTP3_SP_PORT;
Harald Weltedf277252018-02-20 15:49:30 +0100212#endif
213#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +0100214 /* up-facing port for MGCP */
215 port IPA_MGCP_PT IPA_MGCP_PORT;
Harald Weltedf277252018-02-20 15:49:30 +0100216#endif
217#ifdef IPA_EMULATION_RSL
Harald Welte0d846a72017-12-07 17:58:28 +0100218 /* up-facing port for RSL */
219 port IPA_RSL_PT IPA_RSL_PORT;
Harald Weltedf277252018-02-20 15:49:30 +0100220#endif
Harald Weltec6826662019-02-06 22:26:46 +0100221#ifdef IPA_EMULATION_OML
222 /* up-facing port for OML */
223 port IPA_OML_PT IPA_OML_PORT;
224#endif
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200225#ifdef IPA_EMULATION_CTRL
Harald Weltea76c4bb2017-12-09 02:06:07 +0100226 /* up-facing port for CTRL */
227 port IPA_CTRL_PT IPA_CTRL_PORT;
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200228#endif
Harald Weltedf277252018-02-20 15:49:30 +0100229#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +0100230 /* up-facing port for GSUP */
231 port IPA_GSUP_PT IPA_GSUP_PORT;
Harald Weltedf277252018-02-20 15:49:30 +0100232#endif
Harald Welte7460a722018-10-10 12:28:27 +0200233#ifdef IPA_EMULATION_RSPRO
234 /* up-facing port for RSPRO */
235 port IPA_RSPRO_PT IPA_RSPRO_PORT;
236#endif
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200237#ifdef IPA_EMULATION_OSMO_PCU
238 /* up-facing port for RSPRO */
239 port IPA_OSMO_PCU_PT IPA_OSMO_PCU_PORT;
240#endif
Harald Weltedf327232017-12-28 22:51:51 +0100241
Harald Weltec76f29f2017-11-22 12:46:46 +0100242 /* up-facing port for other streams */
243 port IPA_SP_PT IPA_SP_PORT;
Harald Welted86cdc62017-11-22 00:45:07 +0100244
245 var boolean g_initialized := false;
Harald Weltec82eef42017-11-24 20:40:12 +0100246 /* Are we a BSC/MGW (truel) or MSC (false) */
247 var boolean g_is_bsc_mgw;
Harald Welteb3414b22017-11-23 18:22:10 +0100248
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700249 /* Connection identifier of the client / server itself */
250 var IPL4asp_Types.ConnectionId g_self_conn_id := -1;
251 /* Connection identifier of the last connected client (server only) */
252 var IPL4asp_Types.ConnectionId g_last_conn_id := -1;
253
Harald Welteb3414b22017-11-23 18:22:10 +0100254 var IpaMode g_mode;
Harald Welte2d86aff2018-04-17 11:23:04 +0200255 var boolean g_ccm_enabled;
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200256 var IpaInitBehavior g_init_behavior;
Pau Espin Pedrol2dc137c2021-01-28 19:53:21 +0100257 /* Shall we stop the component once the client disconnects, or keep
258 * running to let next user in ? */
259 var boolean g_server_stop_on_ipa_ev_down := true;
Harald Weltee09921f2017-12-07 17:49:00 +0100260 var IPA_CCM_Parameters g_ccm_pars := c_IPA_default_ccm_pars;
Harald Welted86cdc62017-11-22 00:45:07 +0100261}
262
Harald Weltee21096d2017-12-04 20:45:12 +0100263type record IPA_CCM_Parameters {
264 charstring ser_nr optional,
265 charstring name optional,
266 charstring location1 optional,
267 charstring location2 optional,
268 charstring equip_version optional,
269 charstring sw_version optional,
270 charstring ip_addr optional,
271 charstring mac_addr optional,
272 charstring unit_id optional,
273 charstring osmo_rand optional
274}
275
Harald Weltee09921f2017-12-07 17:49:00 +0100276const IPA_CCM_Parameters c_IPA_default_ccm_pars := {
277 ser_nr := "",
Harald Weltee21096d2017-12-04 20:45:12 +0100278 name := "mahlzeit",
Harald Weltee09921f2017-12-07 17:49:00 +0100279 location1 := "",
280 location2 := "",
281 equip_version := "",
282 sw_version := "",
283 ip_addr := "",
284 mac_addr := "",
Harald Weltee21096d2017-12-04 20:45:12 +0100285 unit_id := "0/1/2",
Harald Weltee09921f2017-12-07 17:49:00 +0100286 osmo_rand := ""
Harald Weltee21096d2017-12-04 20:45:12 +0100287};
288
Harald Weltebdb63702017-12-09 01:15:44 +0100289/* Function to use to connect as client to a remote IPA Server */
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100290function f_connect(charstring remote_host, IPL4asp_Types.PortNumber remote_port,
291 charstring local_host, IPL4asp_Types.PortNumber local_port,
Harald Weltee09921f2017-12-07 17:49:00 +0100292 IPA_CCM_Parameters ccm_pars := c_IPA_default_ccm_pars) runs on IPA_Emulation_CT {
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100293 var IPL4asp_Types.Result res;
Harald Welted86cdc62017-11-22 00:45:07 +0100294 res := IPA_CodecPort_CtrlFunct.f_IPL4_connect(IPA_PORT, remote_host, remote_port,
295 local_host, local_port, 0, { tcp:={} });
Harald Welte9220f632018-05-23 20:27:02 +0200296 if (not ispresent(res.connId)) {
Stefan Sperling6a90be42018-08-31 15:05:39 +0200297 setverdict(fail, "Could not connect IPA socket from ", local_host, " port ", local_port,
298 " to ", remote_host, " port ", remote_port, "; check your configuration");
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200299 mtc.stop;
Harald Welte9220f632018-05-23 20:27:02 +0200300 }
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700301 g_self_conn_id := res.connId;
Harald Weltee21096d2017-12-04 20:45:12 +0100302 g_ccm_pars := ccm_pars;
Harald Weltec82eef42017-11-24 20:40:12 +0100303 g_is_bsc_mgw := true;
Harald Welted86cdc62017-11-22 00:45:07 +0100304}
305
Harald Weltebdb63702017-12-09 01:15:44 +0100306/* Function to use to bind to a local port as IPA server, accepting remote clients */
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100307function f_bind(charstring local_host, IPL4asp_Types.PortNumber local_port,
Harald Weltee09921f2017-12-07 17:49:00 +0100308 IPA_CCM_Parameters ccm_pars := c_IPA_default_ccm_pars) runs on IPA_Emulation_CT {
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100309 var IPL4asp_Types.Result res;
Pau Espin Pedrol76ba5412019-06-10 11:00:33 +0200310 res := IPA_CodecPort_CtrlFunct.f_IPL4_listen(IPA_PORT,
Harald Welteb3414b22017-11-23 18:22:10 +0100311 local_host, local_port, { tcp:={} });
Harald Welte9220f632018-05-23 20:27:02 +0200312 if (not ispresent(res.connId)) {
Maxb7aae8b2019-03-11 15:22:02 +0100313 setverdict(fail, "Could not listen IPA socket ", local_host, ":", local_port, ", check your configuration");
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200314 mtc.stop;
Harald Welte9220f632018-05-23 20:27:02 +0200315 }
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700316 g_self_conn_id := res.connId;
Harald Weltee21096d2017-12-04 20:45:12 +0100317 g_ccm_pars := ccm_pars;
Harald Weltec82eef42017-11-24 20:40:12 +0100318 g_is_bsc_mgw := false;
Harald Welteb3414b22017-11-23 18:22:10 +0100319}
320
Harald Weltedf277252018-02-20 15:49:30 +0100321#ifdef IPA_EMULATION_SCCP
Harald Welted86cdc62017-11-22 00:45:07 +0100322template ASP_MTP3_TRANSFERind ts_MTP3_XFER_ind(integer opc, octetstring data) := {
323 sio := { '10'B, '00'B, '0011'B },
324 opc := opc,
325 dpc := 0,
326 sls := 0,
327 data := data
328}
Harald Weltedf277252018-02-20 15:49:30 +0100329#endif
Harald Welted86cdc62017-11-22 00:45:07 +0100330
331
332private template IpaCcmRespPart t_IdRespPart(IpaCcmIdTag tag, charstring payload) := {
333 len := 0, /* overwritten by codec */
334 tag := tag,
Harald Welte95686e02018-08-02 15:05:43 +0200335 data := char2oct(payload) & '00'O
Harald Welted86cdc62017-11-22 00:45:07 +0100336}
337
Vadim Yanitskiy61564be2020-05-18 20:44:14 +0700338private function f_send_IPA_EVT(template (value) ASP_IPA_Event evt) runs on IPA_Emulation_CT {
Harald Welte2e32e432018-05-24 20:00:00 +0200339 if (IPA_SP_PORT.checkstate("Connected")) {
340 IPA_SP_PORT.send(evt);
341 }
Harald Weltedf277252018-02-20 15:49:30 +0100342#ifdef IPA_EMULATION_RSL
Harald Welte5819b552017-12-09 02:55:46 +0100343 if (IPA_RSL_PORT.checkstate("Connected")) {
344 IPA_RSL_PORT.send(evt);
345 }
Harald Weltedf277252018-02-20 15:49:30 +0100346#endif
Harald Weltec6826662019-02-06 22:26:46 +0100347#ifdef IPA_EMULATION_OML
348 if (IPA_OML_PORT.checkstate("Connected")) {
349 IPA_OML_PORT.send(evt);
350 }
351#endif
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200352#ifdef IPA_EMULATION_CTRL
Harald Welte5819b552017-12-09 02:55:46 +0100353 if (IPA_CTRL_PORT.checkstate("Connected")) {
354 IPA_CTRL_PORT.send(evt);
355 }
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200356#endif
Harald Weltedf277252018-02-20 15:49:30 +0100357#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +0100358 if (IPA_GSUP_PORT.checkstate("Connected")) {
359 IPA_GSUP_PORT.send(evt);
360 }
Harald Weltedf277252018-02-20 15:49:30 +0100361#endif
Harald Welte7460a722018-10-10 12:28:27 +0200362#ifdef IPA_EMULATION_RSPRO
363 if (IPA_RSPRO_PORT.checkstate("Connected")) {
364 IPA_RSPRO_PORT.send(evt);
365 }
366#endif
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200367#ifdef IPA_EMULATION_OSMO_PCU
368 if (IPA_OSMO_PCU_PORT.checkstate("Connected")) {
369 IPA_OSMO_PCU_PORT.send(evt);
370 }
371#endif
Harald Welte0d846a72017-12-07 17:58:28 +0100372 /* FIXME: to other ports */
373}
374
Harald Welted86cdc62017-11-22 00:45:07 +0100375/* build IPA CCM ID RESP response from IPA CCM GET */
Harald Weltee21096d2017-12-04 20:45:12 +0100376private function f_ccm_make_id_resp(PDU_IPA_CCM get) runs on IPA_Emulation_CT return PDU_IPA_CCM {
Harald Welted86cdc62017-11-22 00:45:07 +0100377 var integer i;
378 var PDU_IPA_CCM resp := {
379 msg_type := IPAC_MSGT_ID_RESP,
380 u := {
381 resp := {}
382 }
383 }
384
385 for (i := 0; i < sizeof(get.u.get); i := i + 1) {
386 var IpaCcmIdTag tag := get.u.get[i].tag;
387 var charstring foo;
388 select (tag) {
Harald Weltee21096d2017-12-04 20:45:12 +0100389 case (IPAC_IDTAG_SERNR) {
390 foo := g_ccm_pars.ser_nr;
Harald Welted86cdc62017-11-22 00:45:07 +0100391 }
392 case (IPAC_IDTAG_UNITNAME) {
Harald Weltee21096d2017-12-04 20:45:12 +0100393 foo := g_ccm_pars.name;
394 }
395 case (IPAC_IDTAG_LOCATION1) {
396 foo := g_ccm_pars.location1;
397 }
398 case (IPAC_IDTAG_LOCATION2) {
399 foo := g_ccm_pars.location2;
400 }
401 case (IPAC_IDTAG_EQUIPVERS) {
402 foo := g_ccm_pars.equip_version;
403 }
404 case (IPAC_IDTAG_SWVERSION) {
405 foo := g_ccm_pars.sw_version;
406 }
407 case (IPAC_IDTAG_IPADDR) {
408 foo := g_ccm_pars.ip_addr;
409 }
410 case (IPAC_IDTAG_MACADDR) {
411 foo := g_ccm_pars.mac_addr;
412 }
Vadim Yanitskiy44f0c482020-05-24 20:49:29 +0700413 case (IPAC_IDTAG_UNITID) {
Harald Weltee21096d2017-12-04 20:45:12 +0100414 foo := g_ccm_pars.unit_id;
415 }
416 case (IPAC_IDTAG_OSMO_RAND) {
417 foo := g_ccm_pars.osmo_rand;
Harald Welted86cdc62017-11-22 00:45:07 +0100418 }
419 case else {
Harald Weltee21096d2017-12-04 20:45:12 +0100420 foo := "unknown";
Harald Welted86cdc62017-11-22 00:45:07 +0100421 }
422 }
423 resp.u.resp[sizeof(resp.u.resp)] := valueof(t_IdRespPart(tag, foo));
424 }
425
426 return resp;
427}
428
429/* transmit IPA CCM message */
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700430private function f_ccm_tx(PDU_IPA_CCM ccm, IPL4asp_Types.ConnectionId conn_id)
431runs on IPA_Emulation_CT {
432 var IPA_Send ipa_tx := valueof(t_IPA_Send(conn_id, IPAC_PROTO_CCM, enc_PDU_IPA_CCM(ccm)));
Harald Welted86cdc62017-11-22 00:45:07 +0100433 log("CCM Tx:", ccm);
434 IPA_PORT.send(ipa_tx);
435}
436
437template PDU_IPA_CCM ts_IPA_PONG := {
438 msg_type := IPAC_MSGT_PONG,
439 u := omit
440}
441
442template PDU_IPA_CCM ts_IPA_ACK := {
443 msg_type := IPAC_MSGT_ID_ACK,
444 u := omit
445}
446
Harald Welteb3414b22017-11-23 18:22:10 +0100447template PDU_IPA_CCM ts_IPA_ID_GET := {
448 msg_type := IPAC_MSGT_ID_GET,
449 u := {
450 get := {
Vadim Yanitskiy40ee56c2020-05-24 20:51:24 +0700451 { 1, IPAC_IDTAG_UNITNAME },
452 { 1, IPAC_IDTAG_UNITID }
Harald Welteb3414b22017-11-23 18:22:10 +0100453 }
454 }
455}
456
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700457template PDU_IPA_CCM tr_IPA_ID_RESP := {
458 msg_type := IPAC_MSGT_ID_RESP,
459 u := {
460 resp := {
461 { ?, IPAC_IDTAG_UNITNAME, ? },
462 { ?, IPAC_IDTAG_UNITID, ? }
463 }
464 }
465}
466
Harald Welted86cdc62017-11-22 00:45:07 +0100467/* receive IPA CCM message */
Harald Welte2d86aff2018-04-17 11:23:04 +0200468private function f_ccm_rx_client(PDU_IPA_CCM ccm) runs on IPA_Emulation_CT {
Harald Welted86cdc62017-11-22 00:45:07 +0100469 select (ccm.msg_type) {
470 case (IPAC_MSGT_PING) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700471 f_ccm_tx(valueof(ts_IPA_PONG), g_self_conn_id);
Harald Welted86cdc62017-11-22 00:45:07 +0100472 }
473 case (IPAC_MSGT_ID_ACK) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700474 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_ID_ACK, g_self_conn_id));
Harald Welted86cdc62017-11-22 00:45:07 +0100475 }
476 case (IPAC_MSGT_ID_GET) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700477 f_ccm_tx(f_ccm_make_id_resp(ccm), g_self_conn_id);
Harald Welted86cdc62017-11-22 00:45:07 +0100478 }
Harald Welte2d86aff2018-04-17 11:23:04 +0200479 case else {
480 log("Unknown/unsupported IPA CCM message type", ccm);
481 }
482 }
483}
484
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700485private function f_ccm_rx_server(PDU_IPA_CCM ccm, IPL4asp_Types.ConnectionId conn_id)
486runs on IPA_Emulation_CT {
Harald Welte2d86aff2018-04-17 11:23:04 +0200487 select (ccm.msg_type) {
488 case (IPAC_MSGT_PING) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700489 f_ccm_tx(valueof(ts_IPA_PONG), conn_id);
Harald Welte2d86aff2018-04-17 11:23:04 +0200490 }
491 case (IPAC_MSGT_ID_ACK) {
492 /* the IPA server should at some point receive an ID_ACK from the client,
493 * in case of RSL/OML from nanoBTS, this is actually the first message after
494 * the TCP connection is established. Other implementations may differ.
495 * We currently ignore it completely - but actually we should make sure that
496 * one ID_ACK is received by the server at some point */
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700497 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_ID_ACK, conn_id));
Harald Welte2d86aff2018-04-17 11:23:04 +0200498 }
Harald Welte3bc387f2018-02-21 12:18:46 +0100499 case (IPAC_MSGT_ID_RESP) {
500 log("IPA ID RESP: ", ccm.u.resp);
Vadim Yanitskiy11edf3c2020-05-24 22:26:54 +0700501
502 /* make sure that the response contains all fields we requested */
503 if (not match(ccm, tr_IPA_ID_RESP)) {
504 log("IPA identity response ", ccm.u.resp, " mismatch");
505 return;
506 }
507
508 /* forward to the upper layers, so they can map conn_id with unit_id */
509 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_ID_RESP, conn_id, ccm.u.resp));
Harald Welte2d86aff2018-04-17 11:23:04 +0200510 /* acknowledge any identity that the client may have sent */
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700511 f_ccm_tx(valueof(ts_IPA_ACK), conn_id);
Harald Welte3bc387f2018-02-21 12:18:46 +0100512 }
Harald Welted86cdc62017-11-22 00:45:07 +0100513 case else {
514 log("Unknown/unsupported IPA CCM message type", ccm);
515 }
516 }
517}
518
Harald Weltec76f29f2017-11-22 12:46:46 +0100519private function f_to_asp(IPA_RecvFrom ipa_rx) return ASP_IPA_Unitdata {
520 var ASP_IPA_Unitdata ret := {
521 streamId := ipa_rx.streamId,
Harald Welte2a8f8472017-11-23 21:11:34 +0100522 streamIdExt := ipa_rx.streamIdExt,
Harald Weltec76f29f2017-11-22 12:46:46 +0100523 payload := ipa_rx.msg
524 }
525 return ret;
526}
527
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100528private function f_from_asp(IPL4asp_Types.ConnectionId connId, ASP_IPA_Unitdata ipa_tx) return IPA_Send {
Harald Weltec82eef42017-11-24 20:40:12 +0100529 var IPA_Send ret := valueof(t_IPA_Send(connId, ipa_tx.streamId, ipa_tx.payload,
530 ipa_tx.streamIdExt));
Harald Weltec76f29f2017-11-22 12:46:46 +0100531 return ret;
532}
Harald Welted86cdc62017-11-22 00:45:07 +0100533
Harald Weltedf277252018-02-20 15:49:30 +0100534#ifdef IPA_EMULATION_RSL
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100535private function f_from_rsl(IPL4asp_Types.ConnectionId connId, ASP_RSL_Unitdata rsl_tx) return IPA_Send {
Harald Welte0d846a72017-12-07 17:58:28 +0100536 var octetstring payload := enc_RSL_Message(rsl_tx.rsl);
Vadim Yanitskiyef9bcd22020-05-25 19:01:04 +0700537
538 /* ASP_RSL_Unitdata may (optionally) contain TCP/IP connection ID */
539 if (rsl_tx.conn_id != omit) {
540 connId := rsl_tx.conn_id;
541 }
542
543 return valueof(t_IPA_Send(connId, rsl_tx.streamId, payload));
Harald Welte0d846a72017-12-07 17:58:28 +0100544}
Harald Weltedf277252018-02-20 15:49:30 +0100545#endif
Harald Welte0d846a72017-12-07 17:58:28 +0100546
Harald Weltec6826662019-02-06 22:26:46 +0100547#ifdef IPA_EMULATION_OML
548private function f_from_oml(IPL4asp_Types.ConnectionId connId, OML_PDU oml_tx) return IPA_Send {
549 var octetstring payload := enc_OML_PDU(oml_tx);
550 var IPA_Send ret := valueof(t_IPA_Send(connId, IPAC_PROTO_OML, payload));
551 return ret;
552}
553#endif
554
Harald Weltebdb63702017-12-09 01:15:44 +0100555/* main function to use for a client-side IPA implementation */
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100556function main_client(charstring remote_host, IPL4asp_Types.PortNumber remote_port,
557 charstring local_host, IPL4asp_Types.PortNumber local_port,
Harald Welte2d86aff2018-04-17 11:23:04 +0200558 IPA_CCM_Parameters ccm_pars := c_IPA_default_ccm_pars,
559 boolean ccm_enabled := true) runs on IPA_Emulation_CT {
Harald Welteb3414b22017-11-23 18:22:10 +0100560 g_mode := IPA_MODE_CLIENT;
Harald Welte2d86aff2018-04-17 11:23:04 +0200561 g_ccm_enabled := ccm_enabled;
Harald Weltee09921f2017-12-07 17:49:00 +0100562 f_connect(remote_host, remote_port, local_host, local_port, ccm_pars);
Harald Welte2d86aff2018-04-17 11:23:04 +0200563 if (g_ccm_enabled) {
564 /* we're a client: Send ID_ACK immediately after connect */
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700565 f_ccm_tx(valueof(ts_IPA_ACK), g_self_conn_id);
Harald Welte2d86aff2018-04-17 11:23:04 +0200566 }
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700567 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_UP, g_self_conn_id));
Harald Welteb3414b22017-11-23 18:22:10 +0100568 ScanEvents();
569}
570
Harald Weltebdb63702017-12-09 01:15:44 +0100571/* main function to use for a server-side IPA implementation */
Harald Welte2d86aff2018-04-17 11:23:04 +0200572function main_server(charstring local_host, IPL4asp_Types.PortNumber local_port,
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200573 boolean ccm_enabled := true,
Pau Espin Pedrol2dc137c2021-01-28 19:53:21 +0100574 IpaInitBehavior init_behavior := IPA_INIT_SEND_IPA_ID_GET,
575 boolean server_stop_on_ipa_ev_down := true)
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200576runs on IPA_Emulation_CT {
Harald Welteb3414b22017-11-23 18:22:10 +0100577 g_mode := IPA_MODE_SERVER;
Harald Welte2d86aff2018-04-17 11:23:04 +0200578 g_ccm_enabled := ccm_enabled;
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200579 g_init_behavior := init_behavior;
Pau Espin Pedrol2dc137c2021-01-28 19:53:21 +0100580 g_server_stop_on_ipa_ev_down := server_stop_on_ipa_ev_down;
Harald Welteb3414b22017-11-23 18:22:10 +0100581 f_bind(local_host, local_port);
582 ScanEvents();
583}
584
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200585#ifdef IPA_EMULATION_CTRL
Harald Weltedf277252018-02-20 15:49:30 +0100586private function f_ctrl_to_user(octetstring msg) runs on IPA_Emulation_CT {
587 var charstring msg_ch := oct2char(msg);
588 IPA_CTRL_PORT.send(dec_CtrlMessage(msg_ch));
589}
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200590#endif
Harald Weltedf277252018-02-20 15:49:30 +0100591
592#ifdef IPA_EMULATION_GSUP
593private function f_gsup_to_user(octetstring msg) runs on IPA_Emulation_CT {
594 var GSUP_PDU gsup := dec_GSUP_PDU(msg);
595 f_gsup_postprocess_decoded(gsup);
596 IPA_GSUP_PORT.send(gsup);
597}
598#endif
599
Harald Welte7460a722018-10-10 12:28:27 +0200600#ifdef IPA_EMULATION_RSPRO
601private function f_rspro_to_user(octetstring msg) runs on IPA_Emulation_CT {
602 var RsproPDU rspro := dec_RsproPDU(msg);
603 IPA_RSPRO_PORT.send(rspro);
604}
605#endif
606
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200607#ifdef IPA_EMULATION_OSMO_PCU
608private function f_osmo_pcu_to_user(octetstring msg) runs on IPA_Emulation_CT {
609 var PCUIF_Message pcuif_msg := dec_PCUIF_Message(msg);
610 IPA_OSMO_PCU_PORT.send(pcuif_msg);
611}
612#endif
613
Harald Weltedf277252018-02-20 15:49:30 +0100614#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +0100615private function f_mgcp_to_user(octetstring msg) runs on IPA_Emulation_CT {
616 var charstring msg_ch := oct2char(msg);
617 if (g_is_bsc_mgw) {
618 log("============");
619 log(msg_ch);
620 IPA_MGCP_PORT.send(dec_MgcpCommand(msg_ch));
621 } else {
622 IPA_MGCP_PORT.send(dec_MgcpResponse(msg_ch));
623 }
624}
625
Harald Welte92632e12017-11-25 02:31:20 +0100626private function f_mgcp_to_ud(octetstring payload) runs on IPA_Emulation_CT return ASP_IPA_Unitdata {
627 if (mp_ipa_mgcp_uses_osmo_ext) {
628 return valueof(t_ASP_IPA_UD(IPAC_PROTO_MGCP_OLD, payload));
629 } else {
630 return valueof(t_ASP_IPA_UD(IPAC_PROTO_OSMO, payload, IPAC_PROTO_EXT_MGCP));
631 }
632}
Harald Weltedf277252018-02-20 15:49:30 +0100633#endif
Harald Welte92632e12017-11-25 02:31:20 +0100634
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700635/* Resolve TCP/IP connection identifier depending on g_mode */
636private function f_ipa_conn_id() runs on IPA_Emulation_CT
637return IPL4asp_Types.ConnectionId {
638 var IPL4asp_Types.ConnectionId conn_id;
639
640 select (g_mode) {
641 case (IPA_MODE_CLIENT) { conn_id := g_self_conn_id; }
642 case (IPA_MODE_SERVER) { conn_id := g_last_conn_id; }
643 case else {
644 setverdict(fail, "Unknown mode");
645 mtc.stop;
646 }
647 }
648
649 if (conn_id == -1) { /* Just to be sure */
650 setverdict(fail, "Connection is not established");
651 mtc.stop;
652 }
653
654 return conn_id;
655}
656
Harald Weltebdb63702017-12-09 01:15:44 +0100657/* main loop function for both client and server. 'thread' of the component */
Harald Welteb3414b22017-11-23 18:22:10 +0100658private function ScanEvents() runs on IPA_Emulation_CT {
Harald Welted86cdc62017-11-22 00:45:07 +0100659 var IPA_RecvFrom ipa_rx;
Harald Welteb3414b22017-11-23 18:22:10 +0100660 var ASP_Event asp_evt;
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100661 var Socket_API_Definitions.PortEvent port_evt;
Harald Weltedf277252018-02-20 15:49:30 +0100662 var octetstring payload;
Harald Weltedf277252018-02-20 15:49:30 +0100663 var ASP_IPA_Unitdata ipa_ud;
Harald Welte3286eb32019-08-18 19:36:10 +0200664#ifdef IPA_EMULATION_CTRL
665 var CtrlMessage ctrl_msg;
666#endif
Harald Weltedf277252018-02-20 15:49:30 +0100667#ifdef IPA_EMULATION_SCCP
668 var ASP_MTP3_TRANSFERreq mtp_req;
669#endif
670#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +0100671 var MgcpCommand mgcp_cmd;
672 var MgcpResponse mgcp_rsp;
Harald Weltedf277252018-02-20 15:49:30 +0100673#endif
674#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +0100675 var GSUP_PDU gsup_msg;
Harald Weltedf277252018-02-20 15:49:30 +0100676#endif
677#ifdef IPA_EMULATION_RSL
Harald Welte0d846a72017-12-07 17:58:28 +0100678 var ASP_RSL_Unitdata rsl;
Harald Weltedf277252018-02-20 15:49:30 +0100679#endif
Harald Weltec6826662019-02-06 22:26:46 +0100680#ifdef IPA_EMULATION_OML
681 var OML_PDU oml;
682#endif
Harald Welte7460a722018-10-10 12:28:27 +0200683#ifdef IPA_EMULATION_RSPRO
684 var RsproPDU rspro;
685#endif
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200686#ifdef IPA_EMULATION_OSMO_PCU
687 var PCUIF_Message pcu;
688#endif
Harald Welted86cdc62017-11-22 00:45:07 +0100689
Harald Welte3e6ad892017-12-12 14:39:46 +0100690 /* Set function for dissecting the binary */
691 var f_IPL4_getMsgLen vl_f := refers(f_IPL4_fixedMsgLen);
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700692 IPA_CodecPort_CtrlFunct.f_IPL4_setGetMsgLen(IPA_PORT, g_self_conn_id, vl_f, {0, 2, 3, 1, 0});
Harald Welte3e6ad892017-12-12 14:39:46 +0100693
Harald Welted86cdc62017-11-22 00:45:07 +0100694 while (true) {
695 alt {
696 /* Received IPA -> up into SCCP stack */
Harald Welte2d86aff2018-04-17 11:23:04 +0200697 [g_ccm_enabled] IPA_PORT.receive(IPA_RecvFrom:{?,IPAC_PROTO_CCM,omit,?}) -> value ipa_rx {
698 var PDU_IPA_CCM ccm := dec_PDU_IPA_CCM(ipa_rx.msg);
699 log("CCM Rx:", ccm);
700 select (g_mode) {
701 case (IPA_MODE_CLIENT) {
702 f_ccm_rx_client(ccm);
703 }
704 case (IPA_MODE_SERVER) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700705 f_ccm_rx_server(ccm, ipa_rx.connId);
Harald Welte2d86aff2018-04-17 11:23:04 +0200706 }
707 case else {
708 setverdict(fail, "Unknown mode");
Daniel Willmanne4ff5372018-07-05 17:35:03 +0200709 mtc.stop;
Harald Welte2d86aff2018-04-17 11:23:04 +0200710 }
Harald Weltedf277252018-02-20 15:49:30 +0100711 }
Harald Welte2d86aff2018-04-17 11:23:04 +0200712 }
713 [] IPA_PORT.receive(IPA_RecvFrom:?) -> value ipa_rx {
714 select (ipa_rx.streamId) {
Harald Weltedf277252018-02-20 15:49:30 +0100715#ifdef IPA_EMULATION_SCCP
716 case (IPAC_PROTO_SCCP) {
Harald Welted86cdc62017-11-22 00:45:07 +0100717 var ASP_MTP3_TRANSFERind mtp;
718 mtp := valueof(ts_MTP3_XFER_ind(0, ipa_rx.msg));
719 MTP3_SP_PORT.send(mtp);
Harald Weltedf277252018-02-20 15:49:30 +0100720 }
721#endif
722#ifdef IPA_EMULATION_MGCP
723 case (IPAC_PROTO_MGCP_OLD) {
Harald Weltec82eef42017-11-24 20:40:12 +0100724 f_mgcp_to_user(ipa_rx.msg);
Harald Weltedf277252018-02-20 15:49:30 +0100725 }
726#endif
727#ifdef IPA_EMULATION_RSL
728 case (t_IpaSidRSL) {
Vadim Yanitskiyef9bcd22020-05-25 19:01:04 +0700729 rsl := valueof(ts_ASP_RSL_UD(
730 rsl := dec_RSL_Message(ipa_rx.msg),
731 sid := ipa_rx.streamId,
732 conn_id := ipa_rx.connId
733 ));
Harald Welte0d846a72017-12-07 17:58:28 +0100734 IPA_RSL_PORT.send(rsl);
Harald Weltedf277252018-02-20 15:49:30 +0100735 }
736#endif
Harald Weltec6826662019-02-06 22:26:46 +0100737#ifdef IPA_EMULATION_OML
738 case (IPAC_PROTO_OML) {
739 oml := dec_OML_PDU(ipa_rx.msg)
740 IPA_OML_PORT.send(oml);
741 }
742#endif
Harald Weltedf277252018-02-20 15:49:30 +0100743 case (IPAC_PROTO_OSMO) {
Harald Weltec82eef42017-11-24 20:40:12 +0100744 select (ipa_rx.streamIdExt) {
Harald Weltedf277252018-02-20 15:49:30 +0100745#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +0100746 case (IPAC_PROTO_EXT_MGCP) {
747 f_mgcp_to_user(ipa_rx.msg);
Harald Weltedf277252018-02-20 15:49:30 +0100748 }
749#endif
Harald Welte3286eb32019-08-18 19:36:10 +0200750#ifdef IPA_EMULATION_CTRL
Harald Weltedf277252018-02-20 15:49:30 +0100751 case (IPAC_PROTO_EXT_CTRL) {
Harald Weltea76c4bb2017-12-09 02:06:07 +0100752 f_ctrl_to_user(ipa_rx.msg);
Harald Weltedf277252018-02-20 15:49:30 +0100753 }
Harald Welte3286eb32019-08-18 19:36:10 +0200754#endif
Harald Weltedf277252018-02-20 15:49:30 +0100755#ifdef IPA_EMULATION_GSUP
756 case (IPAC_PROTO_EXT_GSUP) {
Harald Weltedf327232017-12-28 22:51:51 +0100757 f_gsup_to_user(ipa_rx.msg);
Harald Weltedf277252018-02-20 15:49:30 +0100758 }
759#endif
Harald Welte7460a722018-10-10 12:28:27 +0200760#ifdef IPA_EMULATION_RSPRO
761 case (IPAC_PROTO_EXT_RSPRO) {
762 f_rspro_to_user(ipa_rx.msg);
763 }
764#endif
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200765#ifdef IPA_EMULATION_OSMO_PCU
766 case (IPAC_PROTO_EXT_OSMO_PCU) {
767 f_osmo_pcu_to_user(ipa_rx.msg);
768 }
769#endif
Harald Weltedf277252018-02-20 15:49:30 +0100770 case else {
Harald Weltec82eef42017-11-24 20:40:12 +0100771 IPA_SP_PORT.send(f_to_asp(ipa_rx));
772 }
Harald Welted86cdc62017-11-22 00:45:07 +0100773 }
Harald Weltedf277252018-02-20 15:49:30 +0100774 }
775 case else {
Harald Weltec76f29f2017-11-22 12:46:46 +0100776 IPA_SP_PORT.send(f_to_asp(ipa_rx));
Harald Welted86cdc62017-11-22 00:45:07 +0100777 }
778 }
779 }
780
Harald Welteb3414b22017-11-23 18:22:10 +0100781 /* server only */
782 [] IPA_PORT.receive(ASP_Event:{connOpened:=?}) -> value asp_evt {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700783 g_last_conn_id := asp_evt.connOpened.connId;
784 log("Established a new IPA connection (conn_id=", g_last_conn_id, ")");
785
786 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_UP, g_last_conn_id));
Harald Welte2d86aff2018-04-17 11:23:04 +0200787 if (g_mode == IPA_MODE_SERVER and g_ccm_enabled) {
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200788 select (g_init_behavior) {
789 case (IPA_INIT_SEND_IPA_ID_GET) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700790 f_ccm_tx(valueof(ts_IPA_ID_GET), g_last_conn_id);
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200791 }
792 case (IPA_INIT_SEND_IPA_ID_ACK) {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700793 f_ccm_tx(valueof(ts_IPA_ACK), g_last_conn_id);
Neels Hofmeyr3bf31d22018-08-24 14:44:32 +0200794 }
795 }
Harald Welteb3414b22017-11-23 18:22:10 +0100796 }
797 }
798
799 [] IPA_PORT.receive(ASP_Event:{connClosed:=?}) -> value asp_evt {
800 log("IPA: Closed");
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700801 g_self_conn_id := -1;
802 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_DOWN, asp_evt.connClosed.connId));
Pau Espin Pedrol2dc137c2021-01-28 19:53:21 +0100803 if (g_mode != IPA_MODE_SERVER or g_server_stop_on_ipa_ev_down) {
804 self.stop;
805 }
Harald Welteb3414b22017-11-23 18:22:10 +0100806 }
807
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100808 [] IPA_PORT.receive(Socket_API_Definitions.PortEvent:{result:={errorCode:=ERROR_SOCKET, connId:=?, os_error_code:=?, os_error_text:=?}}) -> value port_evt {
809 log("PortEvent: ERROR_SOCKET: ", port_evt);
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700810 g_self_conn_id := -1;
811 f_send_IPA_EVT(ts_ASP_IPA_EV(ASP_IPA_EVENT_DOWN, port_evt.result.connId));
Stefan Sperling830dc9d2018-02-12 21:08:28 +0100812 self.stop;
813 }
814
Harald Weltedf277252018-02-20 15:49:30 +0100815#ifdef IPA_EMULATION_SCCP
Harald Welted86cdc62017-11-22 00:45:07 +0100816 /* Received SCCP -> down into IPA */
817 [] MTP3_SP_PORT.receive(ASP_MTP3_TRANSFERreq: ?) -> value mtp_req {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700818 var IPA_Send ipa_tx := valueof(t_IPA_Send(f_ipa_conn_id(), IPAC_PROTO_SCCP,
Harald Weltec82eef42017-11-24 20:40:12 +0100819 mtp_req.data));
Harald Welted86cdc62017-11-22 00:45:07 +0100820 IPA_PORT.send(ipa_tx);
821 }
Harald Weltedf277252018-02-20 15:49:30 +0100822#endif
Harald Weltec76f29f2017-11-22 12:46:46 +0100823
Harald Weltedf277252018-02-20 15:49:30 +0100824#ifdef IPA_EMULATION_MGCP
Harald Weltec82eef42017-11-24 20:40:12 +0100825 /* Received MGCP -> down into IPA */
826 [] IPA_MGCP_PORT.receive(MgcpCommand:?) -> value mgcp_cmd {
Harald Welte92632e12017-11-25 02:31:20 +0100827 payload := char2oct(enc_MgcpCommand(mgcp_cmd));
828 ipa_ud := f_mgcp_to_ud(payload);
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700829 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Weltec82eef42017-11-24 20:40:12 +0100830 }
831 [] IPA_MGCP_PORT.receive(MgcpResponse:?) -> value mgcp_rsp {
Harald Welte1dd8f372017-11-25 02:25:27 +0100832 payload := char2oct(enc_MgcpResponse(mgcp_rsp));
Harald Welte92632e12017-11-25 02:31:20 +0100833 ipa_ud := f_mgcp_to_ud(payload);
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700834 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Weltec82eef42017-11-24 20:40:12 +0100835 }
Harald Weltedf277252018-02-20 15:49:30 +0100836#endif
Harald Weltec82eef42017-11-24 20:40:12 +0100837
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200838#ifdef IPA_EMULATION_CTRL
Harald Weltea76c4bb2017-12-09 02:06:07 +0100839 [] IPA_CTRL_PORT.receive(CtrlMessage:?) -> value ctrl_msg {
840 payload := char2oct(enc_CtrlMessage(ctrl_msg));
841 ipa_ud := valueof(t_ASP_IPA_UD(IPAC_PROTO_OSMO, payload, IPAC_PROTO_EXT_CTRL));
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700842 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Weltea76c4bb2017-12-09 02:06:07 +0100843 }
Pau Espin Pedrolacc51192019-06-11 15:35:19 +0200844#endif
Harald Weltea76c4bb2017-12-09 02:06:07 +0100845
Harald Weltedf277252018-02-20 15:49:30 +0100846#ifdef IPA_EMULATION_GSUP
Harald Weltedf327232017-12-28 22:51:51 +0100847 [] IPA_GSUP_PORT.receive(GSUP_PDU:?) -> value gsup_msg {
Harald Welte2f562b12018-01-24 20:52:38 +0100848 f_gsup_preprocess_encoded(gsup_msg);
Harald Weltedf327232017-12-28 22:51:51 +0100849 payload := enc_GSUP_PDU(gsup_msg);
850 ipa_ud := valueof(t_ASP_IPA_UD(IPAC_PROTO_OSMO, payload, IPAC_PROTO_EXT_GSUP));
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700851 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Weltedf327232017-12-28 22:51:51 +0100852 }
Harald Weltedf277252018-02-20 15:49:30 +0100853#endif
Harald Weltedf327232017-12-28 22:51:51 +0100854
Harald Welte7460a722018-10-10 12:28:27 +0200855#ifdef IPA_EMULATION_RSPRO
856 [] IPA_RSPRO_PORT.receive(RsproPDU:?) -> value rspro {
857 payload := enc_RsproPDU(rspro);
858 ipa_ud := valueof(t_ASP_IPA_UD(IPAC_PROTO_OSMO, payload, IPAC_PROTO_EXT_RSPRO));
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700859 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Welte7460a722018-10-10 12:28:27 +0200860 }
861#endif
862
Pau Espin Pedrol6924ef72021-06-09 16:19:26 +0200863#ifdef IPA_EMULATION_OSMO_PCU
864 [] IPA_OSMO_PCU_PORT.receive(PCUIF_Message:?) -> value pcu {
865 payload := enc_PCUIF_Message(pcu);
866 ipa_ud := valueof(t_ASP_IPA_UD(IPAC_PROTO_OSMO, payload, IPAC_PROTO_EXT_OSMO_PCU));
867 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
868 }
869#endif
870
Harald Weltedf277252018-02-20 15:49:30 +0100871#ifdef IPA_EMULATION_RSL
Harald Welte0d846a72017-12-07 17:58:28 +0100872 /* Received RSL -> down into IPA */
873 [] IPA_RSL_PORT.receive(ASP_RSL_Unitdata:?) -> value rsl {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700874 IPA_PORT.send(f_from_rsl(f_ipa_conn_id(), rsl));
Harald Welte0d846a72017-12-07 17:58:28 +0100875 }
Harald Weltedf277252018-02-20 15:49:30 +0100876#endif
Harald Weltec6826662019-02-06 22:26:46 +0100877#ifdef IPA_EMULATION_OML
878 /* Received OML -> down into IPA */
879 [] IPA_OML_PORT.receive(OML_PDU:?) -> value oml {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700880 IPA_PORT.send(f_from_oml(f_ipa_conn_id(), oml));
Harald Weltec6826662019-02-06 22:26:46 +0100881 }
882 [] IPA_OML_PORT.receive(octetstring:?) -> value payload {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700883 IPA_PORT.send(t_IPA_Send(f_ipa_conn_id(), IPAC_PROTO_OML, payload));
Harald Weltec6826662019-02-06 22:26:46 +0100884 }
885#endif
Harald Welte0d846a72017-12-07 17:58:28 +0100886 /* Received MISC (OML/CTRL) -> down into IPA */
Harald Weltec76f29f2017-11-22 12:46:46 +0100887 [] IPA_SP_PORT.receive(ASP_IPA_Unitdata: ?) -> value ipa_ud {
Vadim Yanitskiy6643e8d2020-05-18 21:50:12 +0700888 IPA_PORT.send(f_from_asp(f_ipa_conn_id(), ipa_ud));
Harald Weltec76f29f2017-11-22 12:46:46 +0100889 }
890
891
Harald Welted86cdc62017-11-22 00:45:07 +0100892 }
893 }
894}
895
Harald Welte2e32e432018-05-24 20:00:00 +0200896/***********************************************************************
897 * IPA Event waiter component. Wait for ASP_IPA_EVENT_ID_ACK
898 ***********************************************************************/
899
900type component IPA_EventWaiter_CT {
901 port IPA_SP_PT IPA_SP_PORT;
902}
903
Vadim Yanitskiy61564be2020-05-18 20:44:14 +0700904function waiter_main(template ASP_IPA_Event wait_for := tr_ASP_IPA_EV(ASP_IPA_EVENT_ID_ACK))
Harald Welte2e32e432018-05-24 20:00:00 +0200905runs on IPA_EventWaiter_CT {
906
907 alt {
908 [] IPA_SP_PORT.receive(wait_for) {
909 setverdict(pass);
910 }
911 [] IPA_SP_PORT.receive { repeat; }
912 }
913}
914
915
Harald Welted86cdc62017-11-22 00:45:07 +0100916}