blob: 9cd83c4ff25b5c8b6db4e9f0519e3f043739da7b [file] [log] [blame]
Alexander Couzens6a161492020-07-12 13:45:50 +02001/*! \file gprs_ns2_vc_fsm.c
2 * NS virtual circuit FSM implementation
3 * 3GPP TS 08.16 version 8.0.1 Release 1999 / ETSI TS 101 299 V8.0.1 (2002-05)
4 * as well as its successor 3GPP TS 48.016 */
5
6/* (C) 2020 sysmocom - s.f.m.c. GmbH
7 * Author: Alexander Couzens <lynxis@fe80.eu>
8 *
9 * All Rights Reserved
10 *
11 * SPDX-License-Identifier: GPL-2.0+
12 *
13 * This program is free software; you can redistribute it and/or modify
14 * it under the terms of the GNU General Public License as published by
15 * the Free Software Foundation; either version 2 of the License, or
16 * (at your option) any later version.
17 *
18 * This program is distributed in the hope that it will be useful,
19 * but WITHOUT ANY WARRANTY; without even the implied warranty of
20 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 * GNU General Public License for more details.
22 *
23 * You should have received a copy of the GNU General Public License
24 * along with this program. If not, see <http://www.gnu.org/licenses/>.
25 *
26 */
27
28/* The BSS NSE only has one SGSN IP address configured, and it will use the SNS procedures
29 * to communicated its local IPs/ports as well as all the SGSN side IPs/ports and
30 * associated weights. In theory, the BSS then uses this to establish a full mesh
31 * of NSVCs between all BSS-side IPs/ports and SGSN-side IPs/ports */
32
33#include <errno.h>
34
35#include <netinet/in.h>
36#include <arpa/inet.h>
37
38#include <osmocom/core/fsm.h>
39#include <osmocom/core/msgb.h>
40#include <osmocom/core/rate_ctr.h>
41#include <osmocom/core/socket.h>
42#include <osmocom/core/stat_item.h>
43#include <osmocom/gsm/prim.h>
44#include <osmocom/gsm/tlv.h>
45#include <osmocom/gprs/gprs_msgb.h>
46#include <osmocom/gprs/protocol/gsm_08_16.h>
47
48#include "gprs_ns2_internal.h"
49
50#define S(x) (1 << (x))
51
Alexander Couzens6a161492020-07-12 13:45:50 +020052struct gprs_ns2_vc_priv {
53 struct gprs_ns2_vc *nsvc;
54 /* how often the timer was triggered */
55 int N;
Daniel Willmannf5b2e282020-11-18 18:51:25 +010056 /* The initiator is responsible to UNBLOCK the VC. The BSS is usually the initiator.
57 * It can change during runtime. The side which blocks an unblocked side.*/
Alexander Couzensea01bf22021-01-18 14:01:01 +010058 bool initiator;
Daniel Willmannf5b2e282020-11-18 18:51:25 +010059 bool initiate_block;
60 bool initiate_reset;
Alexander Couzens47afc422021-01-17 20:12:46 +010061 /* if unitdata is forwarded to the user */
62 bool accept_unitdata;
Alexander Couzens6a161492020-07-12 13:45:50 +020063
64 /* the alive counter is present in all states */
65 struct {
66 struct osmo_timer_list timer;
67 enum ns2_timeout mode;
68 int N;
Alexander Couzensab0e8642021-02-12 02:50:44 +010069 struct timespec timer_started;
Alexander Couzens6a161492020-07-12 13:45:50 +020070 } alive;
71};
72
73
74/* The FSM covers both the VC with RESET/BLOCK and without RESET/BLOCK procedure..
75 *
76 * With RESET/BLOCK, the state should follow:
77 * - UNCONFIGURED -> RESET -> BLOCK -> UNBLOCKED
78 *
79 * Without RESET/BLOCK, the state should follow:
Alexander Couzensd3e31102021-02-03 11:15:07 +010080 * - UNCONFIGURED -> RECOVERY -> UNBLOCKED
Alexander Couzens6a161492020-07-12 13:45:50 +020081 *
82 * The UNBLOCKED and TEST states are used to send ALIVE PDU using the timeout Tns-test and Tns-alive.
83 * UNBLOCKED -> TEST: on expire of Tns-Test, send Alive PDU.
84 * TEST -> UNBLOCKED: on receive of Alive_Ack PDU, go into UNBLOCKED.
85 *
Alexander Couzensd3e31102021-02-03 11:15:07 +010086 * The RECOVERY state is used as intermediate, because a VC is only valid if it received an Alive ACK when
Alexander Couzens6a161492020-07-12 13:45:50 +020087 * not using RESET/BLOCK procedure.
88 */
89
90enum gprs_ns2_vc_state {
91 GPRS_NS2_ST_UNCONFIGURED,
92 GPRS_NS2_ST_RESET,
93 GPRS_NS2_ST_BLOCKED,
94 GPRS_NS2_ST_UNBLOCKED, /* allows sending NS_UNITDATA */
95
Alexander Couzensd3e31102021-02-03 11:15:07 +010096 GPRS_NS2_ST_RECOVERING, /* only used when not using RESET/BLOCK procedure */
Alexander Couzens6a161492020-07-12 13:45:50 +020097};
98
99enum gprs_ns2_vc_event {
Alexander Couzensf5775432021-01-18 13:49:00 +0100100 GPRS_NS2_EV_REQ_START,
Alexander Couzens6a161492020-07-12 13:45:50 +0200101
102 /* received messages */
Alexander Couzensf5775432021-01-18 13:49:00 +0100103 GPRS_NS2_EV_RX_RESET,
104 GPRS_NS2_EV_RX_RESET_ACK,
105 GPRS_NS2_EV_RX_UNBLOCK,
106 GPRS_NS2_EV_RX_UNBLOCK_ACK,
107 GPRS_NS2_EV_RX_BLOCK,
108 GPRS_NS2_EV_RX_BLOCK_ACK,
109 GPRS_NS2_EV_RX_ALIVE,
110 GPRS_NS2_EV_RX_ALIVE_ACK,
111 GPRS_NS2_EV_RX_STATUS,
Alexander Couzens6a161492020-07-12 13:45:50 +0200112
Alexander Couzensf5775432021-01-18 13:49:00 +0100113 GPRS_NS2_EV_RX_UNITDATA,
Daniel Willmanned0c9822020-11-18 14:08:07 +0100114
Alexander Couzensf5775432021-01-18 13:49:00 +0100115 GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, /* called via vty for tests */
Alexander Couzens27e58732021-03-21 17:12:08 +0100116 GPRS_NS2_EV_REQ_OM_RESET, /* vty cmd: reset */
Alexander Couzensf5775432021-01-18 13:49:00 +0100117 GPRS_NS2_EV_REQ_OM_BLOCK, /* vty cmd: block */
118 GPRS_NS2_EV_REQ_OM_UNBLOCK, /* vty cmd: unblock*/
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200119 GPRS_NS2_EV_RX_BLOCK_FOREIGN, /* received a BLOCK over another NSVC */
Alexander Couzens6a161492020-07-12 13:45:50 +0200120};
121
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100122static const struct value_string ns2_vc_event_names[] = {
Harald Welteb339cbb2021-02-05 14:49:21 +0100123 { GPRS_NS2_EV_REQ_START, "REQ-START" },
124 { GPRS_NS2_EV_RX_RESET, "RX-RESET" },
125 { GPRS_NS2_EV_RX_RESET_ACK, "RX-RESET_ACK" },
126 { GPRS_NS2_EV_RX_UNBLOCK, "RX-UNBLOCK" },
127 { GPRS_NS2_EV_RX_UNBLOCK_ACK, "RX-UNBLOCK_ACK" },
128 { GPRS_NS2_EV_RX_BLOCK, "RX-BLOCK" },
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200129 { GPRS_NS2_EV_RX_BLOCK_FOREIGN, "RX-BLOCK_FOREIGN" },
Harald Welteb339cbb2021-02-05 14:49:21 +0100130 { GPRS_NS2_EV_RX_BLOCK_ACK, "RX-BLOCK_ACK" },
131 { GPRS_NS2_EV_RX_ALIVE, "RX-ALIVE" },
132 { GPRS_NS2_EV_RX_ALIVE_ACK, "RX-ALIVE_ACK" },
133 { GPRS_NS2_EV_RX_STATUS, "RX-STATUS" },
134 { GPRS_NS2_EV_RX_UNITDATA, "RX-UNITDATA" },
135 { GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, "REQ-FORCE_UNCONFIGURED" },
Alexander Couzens27e58732021-03-21 17:12:08 +0100136 { GPRS_NS2_EV_REQ_OM_RESET, "REQ-O&M-RESET"},
Alexander Couzens47afc422021-01-17 20:12:46 +0100137 { GPRS_NS2_EV_REQ_OM_BLOCK, "REQ-O&M-BLOCK"},
138 { GPRS_NS2_EV_REQ_OM_UNBLOCK, "REQ-O&M-UNBLOCK"},
Alexander Couzens6a161492020-07-12 13:45:50 +0200139 { 0, NULL }
140};
141
142static inline struct gprs_ns2_inst *ns_inst_from_fi(struct osmo_fsm_inst *fi)
143{
144 struct gprs_ns2_vc_priv *priv = fi->priv;
145 return priv->nsvc->nse->nsi;
146}
147
Harald Welte5e408312021-03-23 18:16:30 +0100148/* Start the NS-TEST procedure, either with transmitting a tx_alive,
149 * (start_tx_alive==true) or with starting tns-test */
150static void start_test_procedure(struct osmo_fsm_inst *fi, bool start_tx_alive)
Alexander Couzens6a161492020-07-12 13:45:50 +0200151{
Harald Welte5e408312021-03-23 18:16:30 +0100152 struct gprs_ns2_vc_priv *priv = fi->priv;
Alexander Couzens6a161492020-07-12 13:45:50 +0200153 struct gprs_ns2_inst *nsi = priv->nsvc->nse->nsi;
Harald Welte5e408312021-03-23 18:16:30 +0100154 unsigned int tout_idx;
Alexander Couzens6a161492020-07-12 13:45:50 +0200155
Harald Welte5e408312021-03-23 18:16:30 +0100156 if (osmo_timer_pending(&priv->alive.timer)) {
157 if (start_tx_alive) {
158 if (priv->alive.mode == NS_TOUT_TNS_ALIVE)
159 return;
160 } else {
161 if (priv->alive.mode == NS_TOUT_TNS_TEST)
162 return;
163 }
164 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200165
Alexander Couzens6a161492020-07-12 13:45:50 +0200166 priv->alive.N = 0;
167
Harald Welte5e408312021-03-23 18:16:30 +0100168 if (start_tx_alive) {
169 priv->alive.mode = NS_TOUT_TNS_ALIVE;
170 osmo_clock_gettime(CLOCK_MONOTONIC, &priv->alive.timer_started);
171 ns2_tx_alive(priv->nsvc);
172 tout_idx = NS_TOUT_TNS_ALIVE;
173 } else {
174 priv->alive.mode = NS_TOUT_TNS_TEST;
175 tout_idx = NS_TOUT_TNS_TEST;
176 }
177 LOGPFSML(fi, LOGL_DEBUG, "Starting Tns-%s of %u seconds\n",
178 tout_idx == NS_TOUT_TNS_ALIVE ? "alive" : "test", nsi->timeout[tout_idx]);
179 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[tout_idx], 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200180}
181
182static void stop_test_procedure(struct gprs_ns2_vc_priv *priv)
183{
Alexander Couzens8138c532021-06-15 22:11:40 +0200184 osmo_stat_item_set(osmo_stat_item_group_get_item(priv->nsvc->statg, NS_STAT_ALIVE_DELAY), 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200185 osmo_timer_del(&priv->alive.timer);
186}
187
Harald Welte5e408312021-03-23 18:16:30 +0100188/* how many milliseconds have expired since the last alive timer start? */
Alexander Couzens6a161492020-07-12 13:45:50 +0200189static int alive_timer_elapsed_ms(struct gprs_ns2_vc_priv *priv)
190{
Alexander Couzensab0e8642021-02-12 02:50:44 +0100191 struct timespec now, elapsed;
Alexander Couzens6a161492020-07-12 13:45:50 +0200192
Alexander Couzensab0e8642021-02-12 02:50:44 +0100193 if (osmo_clock_gettime(CLOCK_MONOTONIC, &now) != 0)
194 return 0;
195
196 timespecsub(&now, &priv->alive.timer_started, &elapsed);
Alexander Couzensab0e8642021-02-12 02:50:44 +0100197 return elapsed.tv_sec * 1000 + (elapsed.tv_nsec / 1000000);
Alexander Couzens6a161492020-07-12 13:45:50 +0200198}
199
Harald Welte5e408312021-03-23 18:16:30 +0100200/* we just received a NS-ALIVE-ACK; re-schedule after Tns-test */
Alexander Couzens6a161492020-07-12 13:45:50 +0200201static void recv_test_procedure(struct osmo_fsm_inst *fi)
202{
203 struct gprs_ns2_vc_priv *priv = fi->priv;
204 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
205 struct gprs_ns2_vc *nsvc = priv->nsvc;
206
207 /* ignoring ACKs without sending an ALIVE */
208 if (priv->alive.mode != NS_TOUT_TNS_ALIVE)
209 return;
210
211 priv->alive.mode = NS_TOUT_TNS_TEST;
212 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_TEST], 0);
Pau Espin Pedrol7b894a72021-06-04 18:17:12 +0200213 osmo_stat_item_set(osmo_stat_item_group_get_item(nsvc->statg, NS_STAT_ALIVE_DELAY),
Alexander Couzens6a161492020-07-12 13:45:50 +0200214 alive_timer_elapsed_ms(priv));
215}
216
217
218static void alive_timeout_handler(void *data)
219{
220 struct osmo_fsm_inst *fi = data;
221 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
222 struct gprs_ns2_vc_priv *priv = fi->priv;
223
224 switch (priv->alive.mode) {
225 case NS_TOUT_TNS_TEST:
226 priv->alive.mode = NS_TOUT_TNS_ALIVE;
Alexander Couzens2f8f7b62021-02-03 11:04:22 +0100227 priv->alive.N = 0;
Alexander Couzensfa4121d2021-02-12 02:54:46 +0100228 osmo_clock_gettime(CLOCK_MONOTONIC, &priv->alive.timer_started);
Alexander Couzens6a161492020-07-12 13:45:50 +0200229 ns2_tx_alive(priv->nsvc);
230 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
231 break;
232 case NS_TOUT_TNS_ALIVE:
Daniel Willmannefa64f92021-07-09 20:35:00 +0200233 RATE_CTR_INC_NS(priv->nsvc, NS_CTR_LOST_ALIVE);
Alexander Couzens6a161492020-07-12 13:45:50 +0200234 priv->alive.N++;
235
236 if (priv->alive.N <= nsi->timeout[NS_TOUT_TNS_ALIVE_RETRIES]) {
237 /* retransmission */
238 ns2_tx_alive(priv->nsvc);
239 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
240 } else {
241 /* lost connection */
Alexander Couzens138b96f2021-01-25 16:23:29 +0100242 if (priv->nsvc->mode == GPRS_NS2_VC_MODE_BLOCKRESET) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200243 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
244 } else {
Alexander Couzensd3e31102021-02-03 11:15:07 +0100245 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RECOVERING, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200246 }
247 }
248 break;
249 default:
250 break;
251 }
252}
253
Harald Welte6a9ec422021-01-31 18:56:29 +0100254
255static void ns2_st_unconfigured_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
256{
Alexander Couzensa140d442021-07-12 16:42:32 +0200257 struct gprs_ns2_vc_priv *priv = fi->priv;
258
Harald Welte6a9ec422021-01-31 18:56:29 +0100259 stop_test_procedure(fi->priv);
Alexander Couzensa140d442021-07-12 16:42:32 +0200260 ns2_nse_notify_unblocked(priv->nsvc, false);
Harald Welte6a9ec422021-01-31 18:56:29 +0100261}
262
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100263static void ns2_st_unconfigured(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200264{
265 struct gprs_ns2_vc_priv *priv = fi->priv;
266 struct gprs_ns2_inst *nsi = priv->nsvc->nse->nsi;
267
Alexander Couzensea01bf22021-01-18 14:01:01 +0100268 priv->initiate_reset = priv->initiate_block = priv->initiator;
arehbein0d9b6b02022-10-27 18:28:52 +0200269 priv->nsvc->om_blocked = false;
Alexander Couzensea01bf22021-01-18 14:01:01 +0100270
Alexander Couzens6a161492020-07-12 13:45:50 +0200271 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100272 case GPRS_NS2_EV_REQ_START:
Alexander Couzens6a161492020-07-12 13:45:50 +0200273 switch (priv->nsvc->mode) {
Alexander Couzens138b96f2021-01-25 16:23:29 +0100274 case GPRS_NS2_VC_MODE_ALIVE:
Harald Weltec51ddf22021-03-05 09:48:18 +0100275 if (priv->nsvc->nse->dialect == GPRS_NS2_DIALECT_SNS) {
276 /* In IP-SNS, the NS-VC are assumed initially alive, until the alive
277 * procedure should fail at some future point */
278 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED, 0, 0);
279 } else {
280 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RECOVERING, nsi->timeout[NS_TOUT_TNS_ALIVE], NS_TOUT_TNS_ALIVE);
281 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200282 break;
Alexander Couzens138b96f2021-01-25 16:23:29 +0100283 case GPRS_NS2_VC_MODE_BLOCKRESET:
Alexander Couzens6a161492020-07-12 13:45:50 +0200284 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], NS_TOUT_TNS_RESET);
285 break;
286 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200287 break;
288 default:
289 OSMO_ASSERT(0);
290 }
291}
292
293
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100294static void ns2_st_reset_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200295{
296 struct gprs_ns2_vc_priv *priv = fi->priv;
297
298 if (old_state != GPRS_NS2_ST_RESET)
299 priv->N = 0;
300
Alexander Couzens47afc422021-01-17 20:12:46 +0100301 priv->accept_unitdata = false;
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100302 if (priv->initiate_reset)
Alexander Couzens6a161492020-07-12 13:45:50 +0200303 ns2_tx_reset(priv->nsvc, NS_CAUSE_OM_INTERVENTION);
304
305 stop_test_procedure(priv);
306 ns2_nse_notify_unblocked(priv->nsvc, false);
307}
308
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100309static void ns2_st_reset(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200310{
311 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
312 struct gprs_ns2_vc_priv *priv = fi->priv;
313
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100314 if (priv->initiate_reset) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200315 switch (event) {
Alexander Couzens273f0632021-01-20 13:11:26 +0100316 case GPRS_NS2_EV_RX_RESET:
317 ns2_tx_reset_ack(priv->nsvc);
318 /* fall-through */
Alexander Couzensf5775432021-01-18 13:49:00 +0100319 case GPRS_NS2_EV_RX_RESET_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200320 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
321 nsi->timeout[NS_TOUT_TNS_BLOCK], NS_TOUT_TNS_BLOCK);
322 break;
323 }
324 } else {
325 /* we are on the receiving end */
326 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100327 case GPRS_NS2_EV_RX_RESET:
Alexander Couzens6a161492020-07-12 13:45:50 +0200328 ns2_tx_reset_ack(priv->nsvc);
329 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
330 0, 0);
331 break;
332 }
333 }
334}
335
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100336static void ns2_st_blocked_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200337{
338 struct gprs_ns2_vc_priv *priv = fi->priv;
339
Harald Welte4c54f592021-01-30 22:42:15 +0100340 if (old_state != GPRS_NS2_ST_BLOCKED) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200341 priv->N = 0;
Daniel Willmannefa64f92021-07-09 20:35:00 +0200342 RATE_CTR_INC_NS(priv->nsvc, NS_CTR_BLOCKED);
Harald Welte4c54f592021-01-30 22:42:15 +0100343 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200344
Alexander Couzense09deb62021-06-15 21:35:45 +0200345 ns2_nse_notify_unblocked(priv->nsvc, false);
arehbein0d9b6b02022-10-27 18:28:52 +0200346 if (priv->nsvc->om_blocked) {
Alexander Couzens47afc422021-01-17 20:12:46 +0100347 /* we are already blocked after a RESET */
348 if (old_state == GPRS_NS2_ST_RESET) {
349 osmo_timer_del(&fi->timer);
350 } else {
Alexander Couzens67cfc5d2021-09-07 01:10:38 +0200351 ns2_tx_block(priv->nsvc, NS_CAUSE_OM_INTERVENTION, NULL);
Alexander Couzens47afc422021-01-17 20:12:46 +0100352 }
353 } else if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200354 ns2_tx_unblock(priv->nsvc);
Alexander Couzens47afc422021-01-17 20:12:46 +0100355 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200356
Harald Welte5e408312021-03-23 18:16:30 +0100357 start_test_procedure(fi, true);
Alexander Couzens6a161492020-07-12 13:45:50 +0200358}
359
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100360static void ns2_st_blocked(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200361{
362 struct gprs_ns2_vc_priv *priv = fi->priv;
363
arehbein0d9b6b02022-10-27 18:28:52 +0200364 if (priv->nsvc->om_blocked) {
Alexander Couzens47afc422021-01-17 20:12:46 +0100365 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100366 case GPRS_NS2_EV_RX_BLOCK_ACK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100367 priv->accept_unitdata = false;
368 osmo_timer_del(&fi->timer);
369 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100370 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens67cfc5d2021-09-07 01:10:38 +0200371 ns2_tx_block_ack(priv->nsvc, NULL);
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200372 /* fall through */
373 case GPRS_NS2_EV_RX_BLOCK_FOREIGN:
374 /* the BLOCK ACK for foreign BLOCK PDUs (rx over another nsvc) will be sent
375 * from the receiving nsvc */
376 priv->accept_unitdata = false;
Alexander Couzens47afc422021-01-17 20:12:46 +0100377 osmo_timer_del(&fi->timer);
378 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100379 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100380 priv->accept_unitdata = false;
Alexander Couzens67cfc5d2021-09-07 01:10:38 +0200381 ns2_tx_block(priv->nsvc, NS_CAUSE_OM_INTERVENTION, NULL);
Alexander Couzens47afc422021-01-17 20:12:46 +0100382 osmo_timer_add(&fi->timer);
383 break;
384 }
385 } else if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200386 switch (event) {
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200387 case GPRS_NS2_EV_RX_BLOCK_FOREIGN:
388 /* the block ack will be sent by the rx NSVC */
389 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100390 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200391 /* TODO: BLOCK is a UNBLOCK_NACK */
Alexander Couzens67cfc5d2021-09-07 01:10:38 +0200392 ns2_tx_block_ack(priv->nsvc, NULL);
Alexander Couzens6a161492020-07-12 13:45:50 +0200393 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100394 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200395 ns2_tx_unblock_ack(priv->nsvc);
396 /* fall through */
Alexander Couzensf5775432021-01-18 13:49:00 +0100397 case GPRS_NS2_EV_RX_UNBLOCK_ACK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100398 priv->accept_unitdata = true;
Alexander Couzens6a161492020-07-12 13:45:50 +0200399 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED,
400 0, NS_TOUT_TNS_TEST);
401 break;
402 }
403 } else {
404 /* we are on the receiving end. The initiator who sent RESET is responsible to UNBLOCK! */
405 switch (event) {
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200406 case GPRS_NS2_EV_RX_BLOCK_FOREIGN:
407 /* the block ack will be sent by the rx NSVC */
408 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100409 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens67cfc5d2021-09-07 01:10:38 +0200410 ns2_tx_block_ack(priv->nsvc, NULL);
Alexander Couzens856b94c2021-01-19 19:42:17 +0100411 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100412 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200413 ns2_tx_unblock_ack(priv->nsvc);
414 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED,
415 0, 0);
416 break;
417 }
418 }
419}
420
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100421static void ns2_st_unblocked_on_enter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200422{
423 struct gprs_ns2_vc_priv *priv = fi->priv;
Daniel Willmann15c09a82020-11-03 23:05:43 +0100424 struct gprs_ns2_vc *nsvc = priv->nsvc;
425 struct gprs_ns2_nse *nse = nsvc->nse;
Alexander Couzens6a161492020-07-12 13:45:50 +0200426
Alexander Couzensca5ce0d2021-09-05 23:15:56 +0200427 if (old_state != GPRS_NS2_ST_UNBLOCKED) {
Daniel Willmannefa64f92021-07-09 20:35:00 +0200428 RATE_CTR_INC_NS(nsvc, NS_CTR_UNBLOCKED);
Alexander Couzensca5ce0d2021-09-05 23:15:56 +0200429 osmo_clock_gettime(CLOCK_MONOTONIC, &nsvc->ts_alive_change);
430 }
Harald Welteb40bf8b2021-01-30 22:43:01 +0100431
Alexander Couzens47afc422021-01-17 20:12:46 +0100432 priv->accept_unitdata = true;
Daniel Willmann15c09a82020-11-03 23:05:43 +0100433 ns2_nse_notify_unblocked(nsvc, true);
Alexander Couzens138b96f2021-01-25 16:23:29 +0100434 ns2_prim_status_ind(nse, nsvc, 0, GPRS_NS2_AFF_CAUSE_VC_RECOVERY);
Harald Welte5e408312021-03-23 18:16:30 +0100435
436 /* the closest interpretation of the spec would start Tns-test here first,
437 * and only send a NS-ALIVE after Tns-test has expired (i.e. setting the
438 * second argument to 'false'. However, being quick in detecting unavailability
439 * of a NS-VC seems like a good idea */
440 start_test_procedure(fi, true);
Alexander Couzens6a161492020-07-12 13:45:50 +0200441}
442
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100443static void ns2_st_unblocked(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200444{
445 struct gprs_ns2_vc_priv *priv = fi->priv;
446
447 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100448 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzensc4b74622021-01-10 20:44:26 +0100449 ns2_tx_unblock_ack(priv->nsvc);
450 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100451 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200452 ns2_tx_block_ack(priv->nsvc, NULL);
453 /* fall through */
454 case GPRS_NS2_EV_RX_BLOCK_FOREIGN:
455 /* the BLOCK ACK for foreign BLOCK PDUs (rx over another nsvc) will be sent
456 * from the receiving nsvc */
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100457 priv->initiate_block = false;
Alexander Couzensf27fbf62021-09-06 00:19:15 +0200458 priv->accept_unitdata = false;
Alexander Couzens6a161492020-07-12 13:45:50 +0200459 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
460 0, 2);
461 break;
462 }
463}
464
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100465static void ns2_st_alive(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200466{
467 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100468 case GPRS_NS2_EV_RX_ALIVE_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200469 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED, 0, 0);
470 break;
471 }
472}
473
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100474static void ns2_st_alive_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200475{
476 struct gprs_ns2_vc_priv *priv = fi->priv;
477 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
478
479 priv->alive.mode = NS_TOUT_TNS_TEST;
480 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_TEST], 0);
481
Alexander Couzensd3e31102021-02-03 11:15:07 +0100482 if (old_state != GPRS_NS2_ST_RECOVERING)
Alexander Couzens6a161492020-07-12 13:45:50 +0200483 priv->N = 0;
484
Harald Welte5e408312021-03-23 18:16:30 +0100485 start_test_procedure(fi, true);
Alexander Couzens5b722472021-04-01 15:36:54 +0200486 ns2_nse_notify_unblocked(priv->nsvc, false);
Alexander Couzens6a161492020-07-12 13:45:50 +0200487}
488
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100489static const struct osmo_fsm_state ns2_vc_states[] = {
Alexander Couzens6a161492020-07-12 13:45:50 +0200490 [GPRS_NS2_ST_UNCONFIGURED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100491 .in_event_mask = S(GPRS_NS2_EV_REQ_START),
Harald Weltec51ddf22021-03-05 09:48:18 +0100492 .out_state_mask = S(GPRS_NS2_ST_RESET) |
493 S(GPRS_NS2_ST_RECOVERING) |
494 S(GPRS_NS2_ST_UNBLOCKED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200495 .name = "UNCONFIGURED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100496 .action = ns2_st_unconfigured,
Harald Welte6a9ec422021-01-31 18:56:29 +0100497 .onenter = ns2_st_unconfigured_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200498 },
499 [GPRS_NS2_ST_RESET] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100500 .in_event_mask = S(GPRS_NS2_EV_RX_RESET_ACK) | S(GPRS_NS2_EV_RX_RESET),
Alexander Couzens6a161492020-07-12 13:45:50 +0200501 .out_state_mask = S(GPRS_NS2_ST_RESET) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100502 S(GPRS_NS2_ST_BLOCKED) |
503 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200504 .name = "RESET",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100505 .action = ns2_st_reset,
506 .onenter = ns2_st_reset_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200507 },
508 [GPRS_NS2_ST_BLOCKED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100509 .in_event_mask = S(GPRS_NS2_EV_RX_BLOCK) | S(GPRS_NS2_EV_RX_BLOCK_ACK) |
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200510 S(GPRS_NS2_EV_RX_UNBLOCK) | S(GPRS_NS2_EV_RX_UNBLOCK_ACK) |
511 S(GPRS_NS2_EV_RX_BLOCK_FOREIGN),
Alexander Couzens6a161492020-07-12 13:45:50 +0200512 .out_state_mask = S(GPRS_NS2_ST_RESET) |
513 S(GPRS_NS2_ST_UNBLOCKED) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100514 S(GPRS_NS2_ST_BLOCKED) |
515 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200516 .name = "BLOCKED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100517 .action = ns2_st_blocked,
518 .onenter = ns2_st_blocked_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200519 },
520 [GPRS_NS2_ST_UNBLOCKED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100521 .in_event_mask = S(GPRS_NS2_EV_RX_BLOCK) | S(GPRS_NS2_EV_RX_UNBLOCK_ACK) |
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200522 S(GPRS_NS2_EV_RX_UNBLOCK) | S(GPRS_NS2_EV_RX_BLOCK_FOREIGN),
Alexander Couzensd3e31102021-02-03 11:15:07 +0100523 .out_state_mask = S(GPRS_NS2_ST_RESET) | S(GPRS_NS2_ST_RECOVERING) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100524 S(GPRS_NS2_ST_BLOCKED) |
525 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200526 .name = "UNBLOCKED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100527 .action = ns2_st_unblocked,
528 .onenter = ns2_st_unblocked_on_enter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200529 },
530
Alexander Couzensd3e31102021-02-03 11:15:07 +0100531 /* ST_RECOVERING is only used on VC without RESET/BLOCK */
532 [GPRS_NS2_ST_RECOVERING] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100533 .in_event_mask = S(GPRS_NS2_EV_RX_ALIVE_ACK),
Alexander Couzensd3e31102021-02-03 11:15:07 +0100534 .out_state_mask = S(GPRS_NS2_ST_RECOVERING) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100535 S(GPRS_NS2_ST_UNBLOCKED) |
536 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzensd3e31102021-02-03 11:15:07 +0100537 .name = "RECOVERING",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100538 .action = ns2_st_alive,
539 .onenter = ns2_st_alive_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200540 },
541};
542
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100543static int ns2_vc_fsm_timer_cb(struct osmo_fsm_inst *fi)
Alexander Couzens6a161492020-07-12 13:45:50 +0200544{
545 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
546 struct gprs_ns2_vc_priv *priv = fi->priv;
547
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100548 switch (fi->state) {
549 case GPRS_NS2_ST_RESET:
550 if (priv->initiate_reset) {
Daniel Willmannefa64f92021-07-09 20:35:00 +0200551 RATE_CTR_INC_NS(priv->nsvc, NS_CTR_LOST_RESET);
Alexander Couzens6a161492020-07-12 13:45:50 +0200552 priv->N++;
553 if (priv->N <= nsi->timeout[NS_TOUT_TNS_RESET_RETRIES]) {
554 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
555 } else {
556 priv->N = 0;
557 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
558 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100559 }
560 break;
561 case GPRS_NS2_ST_BLOCKED:
562 if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200563 priv->N++;
arehbein0d9b6b02022-10-27 18:28:52 +0200564 if (priv->nsvc->om_blocked) {
Alexander Couzens47afc422021-01-17 20:12:46 +0100565 if (priv->N <= nsi->timeout[NS_TOUT_TNS_BLOCK_RETRIES]) {
566 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
567 } else {
568 /* 7.2 stop accepting data when BLOCK PDU not responded */
569 priv->accept_unitdata = false;
570 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200571 } else {
Alexander Couzens47afc422021-01-17 20:12:46 +0100572 if (priv->N <= nsi->timeout[NS_TOUT_TNS_BLOCK_RETRIES]) {
573 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
574 } else {
575 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
576 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200577 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100578 }
579 break;
Alexander Couzensd3e31102021-02-03 11:15:07 +0100580 case GPRS_NS2_ST_RECOVERING:
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100581 if (priv->initiate_reset) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200582 priv->N++;
583 if (priv->N <= nsi->timeout[NS_TOUT_TNS_ALIVE_RETRIES]) {
Alexander Couzensd3e31102021-02-03 11:15:07 +0100584 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RECOVERING, 0, 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200585 } else {
586 priv->N = 0;
Alexander Couzensd3e31102021-02-03 11:15:07 +0100587 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RECOVERING, 0, 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200588 }
589 break;
590 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100591 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200592 }
593 return 0;
594}
595
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100596static void ns2_recv_unitdata(struct osmo_fsm_inst *fi,
Alexander Couzens6a161492020-07-12 13:45:50 +0200597 struct msgb *msg)
598{
599 struct gprs_ns2_vc_priv *priv = fi->priv;
600 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
601 struct gprs_ns_hdr *nsh = (struct gprs_ns_hdr *) msg->l2h;
602 struct osmo_gprs_ns2_prim nsp = {};
603 uint16_t bvci;
604
Alexander Couzenscce88282020-10-26 00:25:50 +0100605 if (msgb_l2len(msg) < sizeof(*nsh) + 3) {
606 msgb_free(msg);
Alexander Couzens6a161492020-07-12 13:45:50 +0200607 return;
Alexander Couzenscce88282020-10-26 00:25:50 +0100608 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200609
610 /* TODO: 7.1: For an IP sub-network, an NS-UNITDATA PDU
611 * for a PTP BVC may indicate a request to change the IP endpoint
612 * and/or a response to a change in the IP endpoint. */
613
614 /* TODO: nsh->data[0] -> C/R only valid in IP SNS */
615 bvci = nsh->data[1] << 8 | nsh->data[2];
616
Alexander Couzens89acdef2020-09-23 18:22:31 +0200617 msg->l3h = &nsh->data[3];
618 nsp.bvci = bvci;
619 nsp.nsei = priv->nsvc->nse->nsei;
Alexander Couzens6a161492020-07-12 13:45:50 +0200620
Alexander Couzensc1cd3332020-09-23 23:24:02 +0200621 /* 10.3.9 NS SDU Control Bits */
622 if (nsh->data[0] & 0x1)
Alexander Couzens138b96f2021-01-25 16:23:29 +0100623 nsp.u.unitdata.change = GPRS_NS2_ENDPOINT_REQUEST_CHANGE;
Alexander Couzens6a161492020-07-12 13:45:50 +0200624
Alexander Couzens138b96f2021-01-25 16:23:29 +0100625 osmo_prim_init(&nsp.oph, SAP_NS, GPRS_NS2_PRIM_UNIT_DATA,
Alexander Couzens6a161492020-07-12 13:45:50 +0200626 PRIM_OP_INDICATION, msg);
627 nsi->cb(&nsp.oph, nsi->cb_data);
628}
629
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100630static void ns2_vc_fsm_allstate_action(struct osmo_fsm_inst *fi,
Alexander Couzens6a161492020-07-12 13:45:50 +0200631 uint32_t event,
632 void *data)
633{
634 struct gprs_ns2_vc_priv *priv = fi->priv;
635 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200636 struct tlv_parsed *tp;
Alexander Couzenscce88282020-10-26 00:25:50 +0100637 struct msgb *msg = data;
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200638 uint8_t cause;
Alexander Couzens6a161492020-07-12 13:45:50 +0200639
640 switch (event) {
Alexander Couzens27e58732021-03-21 17:12:08 +0100641 case GPRS_NS2_EV_REQ_OM_RESET:
642 if (priv->nsvc->mode != GPRS_NS2_VC_MODE_BLOCKRESET)
643 break;
644 /* move the FSM into reset */
645 if (fi->state != GPRS_NS2_ST_RESET) {
646 priv->initiate_reset = true;
647 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], NS_TOUT_TNS_RESET);
648 }
649 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100650 case GPRS_NS2_EV_RX_RESET:
Alexander Couzens138b96f2021-01-25 16:23:29 +0100651 if (priv->nsvc->mode != GPRS_NS2_VC_MODE_BLOCKRESET)
Alexander Couzens6a161492020-07-12 13:45:50 +0200652 break;
653
654 /* move the FSM into reset */
655 if (fi->state != GPRS_NS2_ST_RESET) {
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100656 priv->initiate_reset = false;
Alexander Couzens6a161492020-07-12 13:45:50 +0200657 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], NS_TOUT_TNS_RESET);
658 }
659 /* pass the event down into FSM action */
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100660 ns2_st_reset(fi, event, data);
Alexander Couzens6a161492020-07-12 13:45:50 +0200661 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100662 case GPRS_NS2_EV_RX_ALIVE:
Alexander Couzens6a161492020-07-12 13:45:50 +0200663 switch (fi->state) {
664 case GPRS_NS2_ST_UNCONFIGURED:
665 case GPRS_NS2_ST_RESET:
666 /* ignore ALIVE */
667 break;
668 default:
669 ns2_tx_alive_ack(priv->nsvc);
670 }
671 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100672 case GPRS_NS2_EV_RX_ALIVE_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200673 /* for VCs without RESET/BLOCK/UNBLOCK, the connections comes after ALIVE_ACK unblocked */
Alexander Couzensd3e31102021-02-03 11:15:07 +0100674 if (fi->state == GPRS_NS2_ST_RECOVERING)
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100675 ns2_st_alive(fi, event, data);
Alexander Couzens6a161492020-07-12 13:45:50 +0200676 else
677 recv_test_procedure(fi);
678 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100679 case GPRS_NS2_EV_RX_UNITDATA:
Alexander Couzenscce88282020-10-26 00:25:50 +0100680 /* UNITDATA has to handle the release of msg.
681 * If send upwards (gprs_ns2_recv_unitdata) it must NOT free
682 * the msg, the upper layer has to do it.
683 * Otherwise the msg must be freed.
684 */
Alexander Couzens6cf65d92021-01-18 17:55:35 +0100685
686 LOG_NS_DATA(priv->nsvc, "Rx", NS_PDUT_UNITDATA, LOGL_INFO, "\n");
Alexander Couzens6a161492020-07-12 13:45:50 +0200687 switch (fi->state) {
688 case GPRS_NS2_ST_BLOCKED:
689 /* 7.2.1: the BLOCKED_ACK might be lost */
Alexander Couzens47afc422021-01-17 20:12:46 +0100690 if (priv->accept_unitdata) {
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100691 ns2_recv_unitdata(fi, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100692 return;
693 }
694
695 ns2_tx_status(priv->nsvc,
696 NS_CAUSE_NSVC_BLOCKED,
Alexander Couzensd802f9a2021-09-23 16:19:32 +0200697 0, msg, NULL);
Alexander Couzens6a161492020-07-12 13:45:50 +0200698 break;
699 /* ALIVE can receive UNITDATA if the ALIVE_ACK is lost */
Alexander Couzensd3e31102021-02-03 11:15:07 +0100700 case GPRS_NS2_ST_RECOVERING:
Alexander Couzens6a161492020-07-12 13:45:50 +0200701 case GPRS_NS2_ST_UNBLOCKED:
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100702 ns2_recv_unitdata(fi, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100703 return;
Alexander Couzens6a161492020-07-12 13:45:50 +0200704 }
Alexander Couzenscce88282020-10-26 00:25:50 +0100705
706 msgb_free(msg);
Alexander Couzens6a161492020-07-12 13:45:50 +0200707 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100708 case GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED:
Alexander Couzens6f3b7382020-12-21 15:57:04 +0100709 if (fi->state != GPRS_NS2_ST_UNCONFIGURED) {
710 /* Force the NSVC back to its initial state */
711 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNCONFIGURED, 0, 0);
Alexander Couzens6f3b7382020-12-21 15:57:04 +0100712 return;
713 }
Daniel Willmanned0c9822020-11-18 14:08:07 +0100714 break;
Alexander Couzens47afc422021-01-17 20:12:46 +0100715 case GPRS_NS2_EV_REQ_OM_BLOCK:
716 /* vty cmd: block */
717 priv->initiate_block = true;
arehbein0d9b6b02022-10-27 18:28:52 +0200718 priv->nsvc->om_blocked = true;
Alexander Couzens47afc422021-01-17 20:12:46 +0100719 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
720 break;
721 case GPRS_NS2_EV_REQ_OM_UNBLOCK:
722 /* vty cmd: unblock*/
arehbein0d9b6b02022-10-27 18:28:52 +0200723 if (!priv->nsvc->om_blocked)
Alexander Couzens47afc422021-01-17 20:12:46 +0100724 return;
arehbein0d9b6b02022-10-27 18:28:52 +0200725 priv->nsvc->om_blocked = false;
Alexander Couzens47afc422021-01-17 20:12:46 +0100726 if (fi->state == GPRS_NS2_ST_BLOCKED)
727 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
728 break;
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200729 case GPRS_NS2_EV_RX_STATUS:
730 tp = data;
731 cause = tlvp_val8(tp, NS_IE_CAUSE, 0);
732 switch (cause) {
733 case NS_CAUSE_NSVC_BLOCKED:
734 if (fi->state != GPRS_NS2_ST_BLOCKED) {
735 LOG_NS_SIGNAL(priv->nsvc, "Rx", NS_PDUT_STATUS, LOGL_ERROR, ": remote side reported blocked state.\n");
736 priv->initiate_block = false;
737 priv->accept_unitdata = false;
738 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
739 }
740 break;
741 case NS_CAUSE_NSVC_UNKNOWN:
742 if (fi->state != GPRS_NS2_ST_RESET && fi->state != GPRS_NS2_ST_UNCONFIGURED) {
743 LOG_NS_SIGNAL(priv->nsvc, "Rx", NS_PDUT_STATUS, LOGL_ERROR, ": remote side reported unknown nsvc.\n");
744 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
745 }
746 break;
747 }
748
749 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200750 }
751}
752
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100753static void ns2_vc_fsm_clean(struct osmo_fsm_inst *fi,
Alexander Couzens0346b642020-10-27 13:05:56 +0100754 enum osmo_fsm_term_cause cause)
755{
756 struct gprs_ns2_vc_priv *priv = fi->priv;
757
758 osmo_timer_del(&priv->alive.timer);
759}
760
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100761static struct osmo_fsm ns2_vc_fsm = {
Alexander Couzens6a161492020-07-12 13:45:50 +0200762 .name = "GPRS-NS2-VC",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100763 .states = ns2_vc_states,
764 .num_states = ARRAY_SIZE(ns2_vc_states),
Alexander Couzensf5775432021-01-18 13:49:00 +0100765 .allstate_event_mask = S(GPRS_NS2_EV_RX_UNITDATA) |
766 S(GPRS_NS2_EV_RX_RESET) |
767 S(GPRS_NS2_EV_RX_ALIVE) |
768 S(GPRS_NS2_EV_RX_ALIVE_ACK) |
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200769 S(GPRS_NS2_EV_RX_STATUS) |
Alexander Couzensf5775432021-01-18 13:49:00 +0100770 S(GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED) |
Alexander Couzens27e58732021-03-21 17:12:08 +0100771 S(GPRS_NS2_EV_REQ_OM_RESET) |
Alexander Couzens47afc422021-01-17 20:12:46 +0100772 S(GPRS_NS2_EV_REQ_OM_BLOCK) |
773 S(GPRS_NS2_EV_REQ_OM_UNBLOCK),
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100774 .allstate_action = ns2_vc_fsm_allstate_action,
775 .cleanup = ns2_vc_fsm_clean,
776 .timer_cb = ns2_vc_fsm_timer_cb,
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100777 .event_names = ns2_vc_event_names,
Alexander Couzens6a161492020-07-12 13:45:50 +0200778 .pre_term = NULL,
779 .log_subsys = DLNS,
780};
781
782/*!
783 * \brief gprs_ns2_vc_fsm_alloc
784 * \param ctx
785 * \param vc
786 * \param id a char representation of the virtual curcuit
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100787 * \param initiator initiator is the site which starts the connection. Usually the BSS.
Alexander Couzens6a161492020-07-12 13:45:50 +0200788 * \return NULL on error, otherwise the fsm
789 */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100790struct osmo_fsm_inst *ns2_vc_fsm_alloc(struct gprs_ns2_vc *nsvc,
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100791 const char *id, bool initiator)
Alexander Couzens6a161492020-07-12 13:45:50 +0200792{
793 struct osmo_fsm_inst *fi;
794 struct gprs_ns2_vc_priv *priv;
795
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100796 fi = osmo_fsm_inst_alloc(&ns2_vc_fsm, nsvc, NULL, LOGL_DEBUG, id);
Alexander Couzens6a161492020-07-12 13:45:50 +0200797 if (!fi)
798 return fi;
799
800 nsvc->fi = fi;
801 priv = fi->priv = talloc_zero(fi, struct gprs_ns2_vc_priv);
802 priv->nsvc = nsvc;
Alexander Couzensea01bf22021-01-18 14:01:01 +0100803 priv->initiator = initiator;
Alexander Couzens6a161492020-07-12 13:45:50 +0200804
805 osmo_timer_setup(&priv->alive.timer, alive_timeout_handler, fi);
806
807 return fi;
808}
809
Harald Welte5bef2cc2020-09-18 22:33:24 +0200810/*! Start a NS-VC FSM.
811 * \param nsvc the virtual circuit
812 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100813int ns2_vc_fsm_start(struct gprs_ns2_vc *nsvc)
Alexander Couzens6a161492020-07-12 13:45:50 +0200814{
815 /* allows to call this function even for started nsvc by gprs_ns2_start_alive_all_nsvcs */
816 if (nsvc->fi->state == GPRS_NS2_ST_UNCONFIGURED)
Alexander Couzensf5775432021-01-18 13:49:00 +0100817 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_START, NULL);
Alexander Couzens6a161492020-07-12 13:45:50 +0200818 return 0;
819}
820
Daniel Willmanned0c9822020-11-18 14:08:07 +0100821/*! Reset a NS-VC FSM.
822 * \param nsvc the virtual circuit
823 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100824int ns2_vc_force_unconfigured(struct gprs_ns2_vc *nsvc)
Daniel Willmanned0c9822020-11-18 14:08:07 +0100825{
Alexander Couzensf5775432021-01-18 13:49:00 +0100826 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, NULL);
Daniel Willmanned0c9822020-11-18 14:08:07 +0100827}
828
Alexander Couzens47afc422021-01-17 20:12:46 +0100829/*! Block a NS-VC.
830 * \param nsvc the virtual circuit
831 * \return 0 on success; negative on error */
832int ns2_vc_block(struct gprs_ns2_vc *nsvc)
833{
Alexander Couzense7dfeac2021-03-21 17:28:36 +0100834 struct gprs_ns2_vc_priv *priv = nsvc->fi->priv;
arehbein0d9b6b02022-10-27 18:28:52 +0200835 if (priv->nsvc->om_blocked)
Alexander Couzense7dfeac2021-03-21 17:28:36 +0100836 return -EALREADY;
837
Alexander Couzens47afc422021-01-17 20:12:46 +0100838 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_OM_BLOCK, NULL);
839}
840
841/*! Unblock a NS-VC.
842 * \param nsvc the virtual circuit
843 * \return 0 on success; negative on error */
844int ns2_vc_unblock(struct gprs_ns2_vc *nsvc)
845{
Alexander Couzense7dfeac2021-03-21 17:28:36 +0100846 struct gprs_ns2_vc_priv *priv = nsvc->fi->priv;
arehbein0d9b6b02022-10-27 18:28:52 +0200847 if (!priv->nsvc->om_blocked)
Alexander Couzense7dfeac2021-03-21 17:28:36 +0100848 return -EALREADY;
849
Alexander Couzens47afc422021-01-17 20:12:46 +0100850 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_OM_UNBLOCK, NULL);
851}
852
Alexander Couzens27e58732021-03-21 17:12:08 +0100853/*! Reset a NS-VC.
854 * \param nsvc the virtual circuit
855 * \return 0 on success; negative on error */
856int ns2_vc_reset(struct gprs_ns2_vc *nsvc)
857{
858 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_OM_RESET, NULL);
859}
860
Harald Welte5bef2cc2020-09-18 22:33:24 +0200861/*! entry point for messages from the driver/VL
862 * \param nsvc virtual circuit on which the message was received
863 * \param msg message that was received
864 * \param tp parsed TLVs of the received message
865 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100866int ns2_vc_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp)
Alexander Couzens6a161492020-07-12 13:45:50 +0200867{
868 struct gprs_ns_hdr *nsh = (struct gprs_ns_hdr *) msg->l2h;
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200869 struct gprs_ns2_vc *target_nsvc = nsvc;
Alexander Couzens6a161492020-07-12 13:45:50 +0200870 struct osmo_fsm_inst *fi = nsvc->fi;
Alexander Couzenscce88282020-10-26 00:25:50 +0100871 int rc = 0;
Alexander Couzens6a161492020-07-12 13:45:50 +0200872 uint8_t cause;
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100873 uint16_t nsei, nsvci;
Alexander Couzens6a161492020-07-12 13:45:50 +0200874
875 /* TODO: 7.2: on UNBLOCK/BLOCK: check if NS-VCI is correct,
876 * if not answer STATUS with "NS-VC unknown" */
Alexander Couzens6a161492020-07-12 13:45:50 +0200877 /* TODO: handle BLOCK/UNBLOCK/ALIVE with different VCI */
878
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100879 if (ns2_validate(nsvc, nsh->pdu_type, msg, tp, &cause)) {
Alexander Couzenscc1621e2021-09-07 15:30:26 +0200880 /* don't answer on a STATUS with a STATUS */
Alexander Couzens6a161492020-07-12 13:45:50 +0200881 if (nsh->pdu_type != NS_PDUT_STATUS) {
Alexander Couzensd802f9a2021-09-23 16:19:32 +0200882 rc = ns2_tx_status(nsvc, cause, 0, msg, NULL);
Alexander Couzenscce88282020-10-26 00:25:50 +0100883 goto out;
Alexander Couzens6a161492020-07-12 13:45:50 +0200884 }
885 }
886
Alexander Couzens43771df2021-01-20 13:03:03 +0100887 if (TLVP_PRESENT(tp, NS_IE_NSEI)) {
888 nsei = tlvp_val16be(tp, NS_IE_NSEI);
889 if (nsei != nsvc->nse->nsei) {
890 /* 48.016 ยง 7.3.1 send, RESET_ACK to wrong NSVCI + ignore */
891 if (nsh->pdu_type == NS_PDUT_RESET)
892 ns2_tx_reset_ack(nsvc);
893
Alexander Couzensf8635c72021-10-12 18:58:31 +0200894 LOG_NS_SIGNAL(nsvc, "Rx", nsh->pdu_type, LOGL_ERROR, " with wrong NSEI (exp: %05u, got %05u). Ignoring PDU.\n", nsvc->nse->nsei, nsei);
Alexander Couzens43771df2021-01-20 13:03:03 +0100895 goto out;
896 }
897 }
898
Alexander Couzense7873332021-09-06 18:17:41 +0200899 if (nsvc->nsvci_is_valid && TLVP_PRESENT(tp, NS_IE_VCI)) {
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100900 nsvci = tlvp_val16be(tp, NS_IE_VCI);
901 if (nsvci != nsvc->nsvci) {
902 /* 48.016 ยง 7.3.1 send RESET_ACK to wrong NSVCI + ignore */
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200903 if (nsh->pdu_type == NS_PDUT_RESET) {
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100904 ns2_tx_reset_ack(nsvc);
Alexander Couzens58be4272021-10-12 18:57:06 +0200905 LOG_NS_SIGNAL(nsvc, "Rx", nsh->pdu_type, LOGL_ERROR, " with wrong NSVCI (exp: %05u, got %05u). Ignoring PDU.\n", nsvc->nsvci, nsvci);
906 goto out;
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200907 } else if (nsh->pdu_type == NS_PDUT_BLOCK || nsh->pdu_type == NS_PDUT_STATUS) {
908 /* this is a PDU received over a NSVC and reports a status/block for another NSVC */
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200909 target_nsvc = gprs_ns2_nsvc_by_nsvci(nsvc->nse->nsi, nsvci);
910 if (!target_nsvc) {
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200911 LOGPFSML(fi, LOGL_ERROR, "Received a %s PDU for unknown NSVC (NSVCI %d)\n",
912 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type), nsvci);
913 if (nsh->pdu_type == NS_PDUT_BLOCK)
914 ns2_tx_status(nsvc, NS_CAUSE_NSVC_UNKNOWN, 0, msg, &nsvci);
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200915 goto out;
916 }
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100917
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200918 if (target_nsvc->nse != nsvc->nse) {
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200919 LOGPFSML(fi, LOGL_ERROR, "Received a %s PDU for a NSVC (NSVCI %d) but it belongs to a different NSE!\n",
920 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type), nsvci);
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200921 goto out;
922 }
923
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200924 /* the status/block will be passed to the nsvc/target nsvc in the switch */
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200925 } else {
926 LOG_NS_SIGNAL(nsvc, "Rx", nsh->pdu_type, LOGL_ERROR, " with wrong NSVCI=%05u. Ignoring PDU.\n", nsvci);
927 goto out;
928 }
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100929 }
930 }
931
Alexander Couzens6a161492020-07-12 13:45:50 +0200932 switch (nsh->pdu_type) {
933 case NS_PDUT_RESET:
Alexander Couzensf5775432021-01-18 13:49:00 +0100934 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_RESET, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200935 break;
936 case NS_PDUT_RESET_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100937 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_RESET_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200938 break;
939 case NS_PDUT_BLOCK:
Alexander Couzens4ef56c02021-09-07 16:33:30 +0200940 if (target_nsvc != nsvc) {
941 osmo_fsm_inst_dispatch(target_nsvc->fi, GPRS_NS2_EV_RX_BLOCK_FOREIGN, tp);
942 ns2_tx_block_ack(nsvc, &nsvci);
943 } else {
944 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_BLOCK, tp);
945 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200946 break;
947 case NS_PDUT_BLOCK_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100948 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_BLOCK_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200949 break;
950 case NS_PDUT_UNBLOCK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100951 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNBLOCK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200952 break;
953 case NS_PDUT_UNBLOCK_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100954 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNBLOCK_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200955 break;
956 case NS_PDUT_ALIVE:
Alexander Couzensf5775432021-01-18 13:49:00 +0100957 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_ALIVE, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200958 break;
959 case NS_PDUT_ALIVE_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100960 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_ALIVE_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200961 break;
962 case NS_PDUT_UNITDATA:
Alexander Couzenscce88282020-10-26 00:25:50 +0100963 /* UNITDATA have to free msg because it might send the msg layer upwards */
Alexander Couzensf5775432021-01-18 13:49:00 +0100964 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNITDATA, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100965 return 0;
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200966 case NS_PDUT_STATUS:
Alexander Couzens69cc4b62021-09-07 16:33:08 +0200967 osmo_fsm_inst_dispatch(target_nsvc->fi, GPRS_NS2_EV_RX_STATUS, tp);
Alexander Couzensa7a757f2021-09-07 01:22:53 +0200968 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200969 default:
Harald Weltef2949742021-01-20 14:54:14 +0100970 LOGPFSML(fi, LOGL_ERROR, "NSEI=%u Rx unknown NS PDU type %s\n", nsvc->nse->nsei,
971 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type));
Alexander Couzens7619ed42021-03-24 17:44:03 +0100972 rc = -EINVAL;
973 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200974 }
975
Alexander Couzenscce88282020-10-26 00:25:50 +0100976out:
977 msgb_free(msg);
978
979 return rc;
Alexander Couzens6a161492020-07-12 13:45:50 +0200980}
981
Harald Welte5bef2cc2020-09-18 22:33:24 +0200982/*! is the given NS-VC unblocked? */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100983int ns2_vc_is_unblocked(struct gprs_ns2_vc *nsvc)
Alexander Couzens6a161492020-07-12 13:45:50 +0200984{
985 return (nsvc->fi->state == GPRS_NS2_ST_UNBLOCKED);
986}
987
988/* initialize osmo_ctx on main tread */
989static __attribute__((constructor)) void on_dso_load_ctx(void)
990{
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100991 OSMO_ASSERT(osmo_fsm_register(&ns2_vc_fsm) == 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200992}