blob: 6314408a96f95b9cdd5c6e6047914780769d7422 [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
52#define DNS 10
53
54struct gprs_ns2_vc_priv {
55 struct gprs_ns2_vc *nsvc;
56 /* how often the timer was triggered */
57 int N;
Daniel Willmannf5b2e282020-11-18 18:51:25 +010058 /* The initiator is responsible to UNBLOCK the VC. The BSS is usually the initiator.
59 * It can change during runtime. The side which blocks an unblocked side.*/
Alexander Couzensea01bf22021-01-18 14:01:01 +010060 bool initiator;
Daniel Willmannf5b2e282020-11-18 18:51:25 +010061 bool initiate_block;
62 bool initiate_reset;
Alexander Couzens47afc422021-01-17 20:12:46 +010063 /* if blocked by O&M/vty */
64 bool om_blocked;
65 /* if unitdata is forwarded to the user */
66 bool accept_unitdata;
Alexander Couzens6a161492020-07-12 13:45:50 +020067
68 /* the alive counter is present in all states */
69 struct {
70 struct osmo_timer_list timer;
71 enum ns2_timeout mode;
72 int N;
73 struct timeval timer_started;
74 } alive;
75};
76
77
78/* The FSM covers both the VC with RESET/BLOCK and without RESET/BLOCK procedure..
79 *
80 * With RESET/BLOCK, the state should follow:
81 * - UNCONFIGURED -> RESET -> BLOCK -> UNBLOCKED
82 *
83 * Without RESET/BLOCK, the state should follow:
84 * - UNCONFIGURED -> ALIVE -> UNBLOCKED
85 *
86 * The UNBLOCKED and TEST states are used to send ALIVE PDU using the timeout Tns-test and Tns-alive.
87 * UNBLOCKED -> TEST: on expire of Tns-Test, send Alive PDU.
88 * TEST -> UNBLOCKED: on receive of Alive_Ack PDU, go into UNBLOCKED.
89 *
90 * The ALIVE state is used as intermediate, because a VC is only valid if it received an Alive ACK when
91 * not using RESET/BLOCK procedure.
92 */
93
94enum gprs_ns2_vc_state {
95 GPRS_NS2_ST_UNCONFIGURED,
96 GPRS_NS2_ST_RESET,
97 GPRS_NS2_ST_BLOCKED,
98 GPRS_NS2_ST_UNBLOCKED, /* allows sending NS_UNITDATA */
99
100 GPRS_NS2_ST_ALIVE, /* only used when not using RESET/BLOCK procedure */
101};
102
103enum gprs_ns2_vc_event {
Alexander Couzensf5775432021-01-18 13:49:00 +0100104 GPRS_NS2_EV_REQ_START,
Alexander Couzens6a161492020-07-12 13:45:50 +0200105
106 /* received messages */
Alexander Couzensf5775432021-01-18 13:49:00 +0100107 GPRS_NS2_EV_RX_RESET,
108 GPRS_NS2_EV_RX_RESET_ACK,
109 GPRS_NS2_EV_RX_UNBLOCK,
110 GPRS_NS2_EV_RX_UNBLOCK_ACK,
111 GPRS_NS2_EV_RX_BLOCK,
112 GPRS_NS2_EV_RX_BLOCK_ACK,
113 GPRS_NS2_EV_RX_ALIVE,
114 GPRS_NS2_EV_RX_ALIVE_ACK,
115 GPRS_NS2_EV_RX_STATUS,
Alexander Couzens6a161492020-07-12 13:45:50 +0200116
Alexander Couzensf5775432021-01-18 13:49:00 +0100117 GPRS_NS2_EV_RX_UNITDATA,
Daniel Willmanned0c9822020-11-18 14:08:07 +0100118
Alexander Couzensf5775432021-01-18 13:49:00 +0100119 GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, /* called via vty for tests */
120 GPRS_NS2_EV_REQ_OM_BLOCK, /* vty cmd: block */
121 GPRS_NS2_EV_REQ_OM_UNBLOCK, /* vty cmd: unblock*/
Alexander Couzens6a161492020-07-12 13:45:50 +0200122};
123
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100124static const struct value_string ns2_vc_event_names[] = {
Harald Welted0835762021-01-31 17:20:41 +0100125 { GPRS_NS2_EV_REQ_START, "START" },
Alexander Couzensf5775432021-01-18 13:49:00 +0100126 { GPRS_NS2_EV_RX_RESET, "RESET" },
127 { GPRS_NS2_EV_RX_RESET_ACK, "RESET_ACK" },
Harald Welted0835762021-01-31 17:20:41 +0100128 { GPRS_NS2_EV_RX_UNBLOCK, "UNBLOCK" },
Alexander Couzensf5775432021-01-18 13:49:00 +0100129 { GPRS_NS2_EV_RX_UNBLOCK_ACK, "UNBLOCK_ACK" },
130 { GPRS_NS2_EV_RX_BLOCK, "BLOCK" },
131 { GPRS_NS2_EV_RX_BLOCK_ACK, "BLOCK_ACK" },
132 { GPRS_NS2_EV_RX_ALIVE, "ALIVE" },
133 { GPRS_NS2_EV_RX_ALIVE_ACK, "ALIVE_ACK" },
Harald Welted0835762021-01-31 17:20:41 +0100134 { GPRS_NS2_EV_RX_STATUS, "STATUS" },
135 { GPRS_NS2_EV_RX_UNITDATA, "UNITDATA" },
Alexander Couzensf5775432021-01-18 13:49:00 +0100136 { GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, "FORCE_UNCONFIGURED" },
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
148static void start_test_procedure(struct gprs_ns2_vc_priv *priv)
149{
150 struct gprs_ns2_inst *nsi = priv->nsvc->nse->nsi;
151
152 if (osmo_timer_pending(&priv->alive.timer))
153 return;
154
155 priv->alive.mode = NS_TOUT_TNS_ALIVE;
156 priv->alive.N = 0;
157
158 osmo_gettimeofday(&priv->alive.timer_started, NULL);
159 ns2_tx_alive(priv->nsvc);
160 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
161}
162
163static void stop_test_procedure(struct gprs_ns2_vc_priv *priv)
164{
165 osmo_timer_del(&priv->alive.timer);
166}
167
168static int alive_timer_elapsed_ms(struct gprs_ns2_vc_priv *priv)
169{
170 struct timeval now, elapsed;
171 osmo_gettimeofday(&now, NULL);
172 timersub(&now, &priv->alive.timer_started, &elapsed);
173
174 return 1000 * elapsed.tv_sec + elapsed.tv_usec / 1000;
175}
176
177static void recv_test_procedure(struct osmo_fsm_inst *fi)
178{
179 struct gprs_ns2_vc_priv *priv = fi->priv;
180 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
181 struct gprs_ns2_vc *nsvc = priv->nsvc;
182
183 /* ignoring ACKs without sending an ALIVE */
184 if (priv->alive.mode != NS_TOUT_TNS_ALIVE)
185 return;
186
187 priv->alive.mode = NS_TOUT_TNS_TEST;
188 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_TEST], 0);
189 osmo_stat_item_set(nsvc->statg->items[NS_STAT_ALIVE_DELAY],
190 alive_timer_elapsed_ms(priv));
191}
192
193
194static void alive_timeout_handler(void *data)
195{
196 struct osmo_fsm_inst *fi = data;
197 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
198 struct gprs_ns2_vc_priv *priv = fi->priv;
199
200 switch (priv->alive.mode) {
201 case NS_TOUT_TNS_TEST:
202 priv->alive.mode = NS_TOUT_TNS_ALIVE;
Alexander Couzens2f8f7b62021-02-03 11:04:22 +0100203 priv->alive.N = 0;
Alexander Couzens6a161492020-07-12 13:45:50 +0200204 ns2_tx_alive(priv->nsvc);
205 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
206 break;
207 case NS_TOUT_TNS_ALIVE:
Harald Welte4c54f592021-01-30 22:42:15 +0100208 rate_ctr_inc(&priv->nsvc->ctrg->ctr[NS_CTR_LOST_ALIVE]);
Alexander Couzens6a161492020-07-12 13:45:50 +0200209 priv->alive.N++;
210
211 if (priv->alive.N <= nsi->timeout[NS_TOUT_TNS_ALIVE_RETRIES]) {
212 /* retransmission */
213 ns2_tx_alive(priv->nsvc);
214 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
215 } else {
216 /* lost connection */
Alexander Couzens138b96f2021-01-25 16:23:29 +0100217 if (priv->nsvc->mode == GPRS_NS2_VC_MODE_BLOCKRESET) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200218 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
219 } else {
220 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_ALIVE, nsi->timeout[NS_TOUT_TNS_ALIVE], 0);
221 }
222 }
223 break;
224 default:
225 break;
226 }
227}
228
Harald Welte6a9ec422021-01-31 18:56:29 +0100229
230static void ns2_st_unconfigured_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
231{
232 stop_test_procedure(fi->priv);
233}
234
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100235static void ns2_st_unconfigured(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200236{
237 struct gprs_ns2_vc_priv *priv = fi->priv;
238 struct gprs_ns2_inst *nsi = priv->nsvc->nse->nsi;
239
Alexander Couzensea01bf22021-01-18 14:01:01 +0100240 priv->initiate_reset = priv->initiate_block = priv->initiator;
241 priv->om_blocked = false;
242
Alexander Couzens6a161492020-07-12 13:45:50 +0200243 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100244 case GPRS_NS2_EV_REQ_START:
Alexander Couzens6a161492020-07-12 13:45:50 +0200245 switch (priv->nsvc->mode) {
Alexander Couzens138b96f2021-01-25 16:23:29 +0100246 case GPRS_NS2_VC_MODE_ALIVE:
Alexander Couzens6a161492020-07-12 13:45:50 +0200247 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_ALIVE, nsi->timeout[NS_TOUT_TNS_ALIVE], NS_TOUT_TNS_ALIVE);
248 break;
Alexander Couzens138b96f2021-01-25 16:23:29 +0100249 case GPRS_NS2_VC_MODE_BLOCKRESET:
Alexander Couzens6a161492020-07-12 13:45:50 +0200250 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], NS_TOUT_TNS_RESET);
251 break;
252 }
253
254 break;
255 default:
256 OSMO_ASSERT(0);
257 }
258}
259
260
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100261static void ns2_st_reset_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200262{
263 struct gprs_ns2_vc_priv *priv = fi->priv;
264
265 if (old_state != GPRS_NS2_ST_RESET)
266 priv->N = 0;
267
Alexander Couzens47afc422021-01-17 20:12:46 +0100268 priv->accept_unitdata = false;
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100269 if (priv->initiate_reset)
Alexander Couzens6a161492020-07-12 13:45:50 +0200270 ns2_tx_reset(priv->nsvc, NS_CAUSE_OM_INTERVENTION);
271
272 stop_test_procedure(priv);
273 ns2_nse_notify_unblocked(priv->nsvc, false);
274}
275
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100276static void ns2_st_reset(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200277{
278 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
279 struct gprs_ns2_vc_priv *priv = fi->priv;
280
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100281 if (priv->initiate_reset) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200282 switch (event) {
Alexander Couzens273f0632021-01-20 13:11:26 +0100283 case GPRS_NS2_EV_RX_RESET:
284 ns2_tx_reset_ack(priv->nsvc);
285 /* fall-through */
Alexander Couzensf5775432021-01-18 13:49:00 +0100286 case GPRS_NS2_EV_RX_RESET_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200287 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
288 nsi->timeout[NS_TOUT_TNS_BLOCK], NS_TOUT_TNS_BLOCK);
289 break;
290 }
291 } else {
292 /* we are on the receiving end */
293 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100294 case GPRS_NS2_EV_RX_RESET:
Alexander Couzens6a161492020-07-12 13:45:50 +0200295 ns2_tx_reset_ack(priv->nsvc);
296 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
297 0, 0);
298 break;
299 }
300 }
301}
302
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100303static void ns2_st_blocked_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200304{
305 struct gprs_ns2_vc_priv *priv = fi->priv;
306
Harald Welte4c54f592021-01-30 22:42:15 +0100307 if (old_state != GPRS_NS2_ST_BLOCKED) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200308 priv->N = 0;
Harald Welte4c54f592021-01-30 22:42:15 +0100309 rate_ctr_inc(&priv->nsvc->ctrg->ctr[NS_CTR_BLOCKED]);
310 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200311
Alexander Couzens47afc422021-01-17 20:12:46 +0100312 if (priv->om_blocked) {
313 /* we are already blocked after a RESET */
314 if (old_state == GPRS_NS2_ST_RESET) {
315 osmo_timer_del(&fi->timer);
316 } else {
317 ns2_tx_block(priv->nsvc, NS_CAUSE_OM_INTERVENTION);
318 }
319 } else if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200320 ns2_tx_unblock(priv->nsvc);
Alexander Couzens47afc422021-01-17 20:12:46 +0100321 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200322
323 start_test_procedure(priv);
324}
325
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100326static void ns2_st_blocked(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200327{
328 struct gprs_ns2_vc_priv *priv = fi->priv;
329
Alexander Couzens47afc422021-01-17 20:12:46 +0100330 if (priv->om_blocked) {
331 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100332 case GPRS_NS2_EV_RX_BLOCK_ACK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100333 priv->accept_unitdata = false;
334 osmo_timer_del(&fi->timer);
335 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100336 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100337 priv->accept_unitdata = false;
338 ns2_tx_block_ack(priv->nsvc);
339 osmo_timer_del(&fi->timer);
340 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100341 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100342 priv->accept_unitdata = false;
343 ns2_tx_block(priv->nsvc, NS_CAUSE_OM_INTERVENTION);
344 osmo_timer_add(&fi->timer);
345 break;
346 }
347 } else if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200348 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100349 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200350 /* TODO: BLOCK is a UNBLOCK_NACK */
351 ns2_tx_block_ack(priv->nsvc);
352 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100353 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200354 ns2_tx_unblock_ack(priv->nsvc);
355 /* fall through */
Alexander Couzensf5775432021-01-18 13:49:00 +0100356 case GPRS_NS2_EV_RX_UNBLOCK_ACK:
Alexander Couzens47afc422021-01-17 20:12:46 +0100357 priv->accept_unitdata = true;
Alexander Couzens6a161492020-07-12 13:45:50 +0200358 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED,
359 0, NS_TOUT_TNS_TEST);
360 break;
361 }
362 } else {
363 /* we are on the receiving end. The initiator who sent RESET is responsible to UNBLOCK! */
364 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100365 case GPRS_NS2_EV_RX_BLOCK:
Alexander Couzens856b94c2021-01-19 19:42:17 +0100366 ns2_tx_block_ack(priv->nsvc);
367 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100368 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200369 ns2_tx_unblock_ack(priv->nsvc);
370 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED,
371 0, 0);
372 break;
373 }
374 }
375}
376
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100377static void ns2_st_unblocked_on_enter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200378{
379 struct gprs_ns2_vc_priv *priv = fi->priv;
Daniel Willmann15c09a82020-11-03 23:05:43 +0100380 struct gprs_ns2_vc *nsvc = priv->nsvc;
381 struct gprs_ns2_nse *nse = nsvc->nse;
Alexander Couzens6a161492020-07-12 13:45:50 +0200382
Harald Welteb40bf8b2021-01-30 22:43:01 +0100383 if (old_state != GPRS_NS2_ST_UNBLOCKED)
384 rate_ctr_inc(&nsvc->ctrg->ctr[NS_CTR_UNBLOCKED]);
385
Alexander Couzens47afc422021-01-17 20:12:46 +0100386 priv->accept_unitdata = true;
Daniel Willmann15c09a82020-11-03 23:05:43 +0100387 ns2_nse_notify_unblocked(nsvc, true);
Alexander Couzens138b96f2021-01-25 16:23:29 +0100388 ns2_prim_status_ind(nse, nsvc, 0, GPRS_NS2_AFF_CAUSE_VC_RECOVERY);
Alexander Couzens6a161492020-07-12 13:45:50 +0200389}
390
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100391static void ns2_st_unblocked(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200392{
393 struct gprs_ns2_vc_priv *priv = fi->priv;
394
395 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100396 case GPRS_NS2_EV_RX_UNBLOCK:
Alexander Couzensc4b74622021-01-10 20:44:26 +0100397 ns2_tx_unblock_ack(priv->nsvc);
398 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100399 case GPRS_NS2_EV_RX_BLOCK:
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100400 priv->initiate_block = false;
Alexander Couzens6a161492020-07-12 13:45:50 +0200401 ns2_tx_block_ack(priv->nsvc);
402 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED,
403 0, 2);
404 break;
405 }
406}
407
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100408static void ns2_st_alive(struct osmo_fsm_inst *fi, uint32_t event, void *data)
Alexander Couzens6a161492020-07-12 13:45:50 +0200409{
410 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100411 case GPRS_NS2_EV_RX_ALIVE_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200412 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNBLOCKED, 0, 0);
413 break;
414 }
415}
416
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100417static void ns2_st_alive_onenter(struct osmo_fsm_inst *fi, uint32_t old_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200418{
419 struct gprs_ns2_vc_priv *priv = fi->priv;
420 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
421
422 priv->alive.mode = NS_TOUT_TNS_TEST;
423 osmo_timer_schedule(&priv->alive.timer, nsi->timeout[NS_TOUT_TNS_TEST], 0);
424
425 if (old_state != GPRS_NS2_ST_ALIVE)
426 priv->N = 0;
427
428 ns2_tx_alive(priv->nsvc);
429 ns2_nse_notify_unblocked(priv->nsvc, false);
430}
431
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100432static void ns2_st_alive_onleave(struct osmo_fsm_inst *fi, uint32_t next_state)
Alexander Couzens6a161492020-07-12 13:45:50 +0200433{
434 start_test_procedure(fi->priv);
435}
436
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100437static const struct osmo_fsm_state ns2_vc_states[] = {
Alexander Couzens6a161492020-07-12 13:45:50 +0200438 [GPRS_NS2_ST_UNCONFIGURED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100439 .in_event_mask = S(GPRS_NS2_EV_REQ_START),
Alexander Couzens6a161492020-07-12 13:45:50 +0200440 .out_state_mask = S(GPRS_NS2_ST_RESET) | S(GPRS_NS2_ST_ALIVE),
441 .name = "UNCONFIGURED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100442 .action = ns2_st_unconfigured,
Harald Welte6a9ec422021-01-31 18:56:29 +0100443 .onenter = ns2_st_unconfigured_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200444 },
445 [GPRS_NS2_ST_RESET] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100446 .in_event_mask = S(GPRS_NS2_EV_RX_RESET_ACK) | S(GPRS_NS2_EV_RX_RESET),
Alexander Couzens6a161492020-07-12 13:45:50 +0200447 .out_state_mask = S(GPRS_NS2_ST_RESET) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100448 S(GPRS_NS2_ST_BLOCKED) |
449 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200450 .name = "RESET",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100451 .action = ns2_st_reset,
452 .onenter = ns2_st_reset_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200453 },
454 [GPRS_NS2_ST_BLOCKED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100455 .in_event_mask = S(GPRS_NS2_EV_RX_BLOCK) | S(GPRS_NS2_EV_RX_BLOCK_ACK) |
456 S(GPRS_NS2_EV_RX_UNBLOCK) | S(GPRS_NS2_EV_RX_UNBLOCK_ACK),
Alexander Couzens6a161492020-07-12 13:45:50 +0200457 .out_state_mask = S(GPRS_NS2_ST_RESET) |
458 S(GPRS_NS2_ST_UNBLOCKED) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100459 S(GPRS_NS2_ST_BLOCKED) |
460 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200461 .name = "BLOCKED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100462 .action = ns2_st_blocked,
463 .onenter = ns2_st_blocked_onenter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200464 },
465 [GPRS_NS2_ST_UNBLOCKED] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100466 .in_event_mask = S(GPRS_NS2_EV_RX_BLOCK) | S(GPRS_NS2_EV_RX_UNBLOCK_ACK) |
467 S(GPRS_NS2_EV_RX_UNBLOCK),
Alexander Couzens6a161492020-07-12 13:45:50 +0200468 .out_state_mask = S(GPRS_NS2_ST_RESET) | S(GPRS_NS2_ST_ALIVE) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100469 S(GPRS_NS2_ST_BLOCKED) |
470 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200471 .name = "UNBLOCKED",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100472 .action = ns2_st_unblocked,
473 .onenter = ns2_st_unblocked_on_enter,
Alexander Couzens6a161492020-07-12 13:45:50 +0200474 },
475
476 /* ST_ALIVE is only used on VC without RESET/BLOCK */
477 [GPRS_NS2_ST_ALIVE] = {
Alexander Couzensf5775432021-01-18 13:49:00 +0100478 .in_event_mask = S(GPRS_NS2_EV_RX_ALIVE_ACK),
Alexander Couzens191c2d72021-01-19 17:59:45 +0100479 .out_state_mask = S(GPRS_NS2_ST_ALIVE) |
Daniel Willmanned0c9822020-11-18 14:08:07 +0100480 S(GPRS_NS2_ST_UNBLOCKED) |
481 S(GPRS_NS2_ST_UNCONFIGURED),
Alexander Couzens6a161492020-07-12 13:45:50 +0200482 .name = "ALIVE",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100483 .action = ns2_st_alive,
484 .onenter = ns2_st_alive_onenter,
485 .onleave = ns2_st_alive_onleave,
Alexander Couzens6a161492020-07-12 13:45:50 +0200486 },
487};
488
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100489static int ns2_vc_fsm_timer_cb(struct osmo_fsm_inst *fi)
Alexander Couzens6a161492020-07-12 13:45:50 +0200490{
491 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
492 struct gprs_ns2_vc_priv *priv = fi->priv;
493
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100494 switch (fi->state) {
495 case GPRS_NS2_ST_RESET:
496 if (priv->initiate_reset) {
Harald Welte4c54f592021-01-30 22:42:15 +0100497 rate_ctr_inc(&priv->nsvc->ctrg->ctr[NS_CTR_LOST_RESET]);
Alexander Couzens6a161492020-07-12 13:45:50 +0200498 priv->N++;
499 if (priv->N <= nsi->timeout[NS_TOUT_TNS_RESET_RETRIES]) {
500 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
501 } else {
502 priv->N = 0;
503 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
504 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100505 }
506 break;
507 case GPRS_NS2_ST_BLOCKED:
508 if (priv->initiate_block) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200509 priv->N++;
Alexander Couzens47afc422021-01-17 20:12:46 +0100510 if (priv->om_blocked) {
511 if (priv->N <= nsi->timeout[NS_TOUT_TNS_BLOCK_RETRIES]) {
512 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
513 } else {
514 /* 7.2 stop accepting data when BLOCK PDU not responded */
515 priv->accept_unitdata = false;
516 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200517 } else {
Alexander Couzens47afc422021-01-17 20:12:46 +0100518 if (priv->N <= nsi->timeout[NS_TOUT_TNS_BLOCK_RETRIES]) {
519 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
520 } else {
521 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], 0);
522 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200523 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100524 }
525 break;
526 case GPRS_NS2_ST_ALIVE:
527 if (priv->initiate_reset) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200528 priv->N++;
529 if (priv->N <= nsi->timeout[NS_TOUT_TNS_ALIVE_RETRIES]) {
530 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_ALIVE, 0, 0);
531 } else {
532 priv->N = 0;
533 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_ALIVE, 0, 0);
534 }
535 break;
536 }
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100537 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200538 }
539 return 0;
540}
541
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100542static void ns2_recv_unitdata(struct osmo_fsm_inst *fi,
Alexander Couzens6a161492020-07-12 13:45:50 +0200543 struct msgb *msg)
544{
545 struct gprs_ns2_vc_priv *priv = fi->priv;
546 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
547 struct gprs_ns_hdr *nsh = (struct gprs_ns_hdr *) msg->l2h;
548 struct osmo_gprs_ns2_prim nsp = {};
549 uint16_t bvci;
550
Alexander Couzenscce88282020-10-26 00:25:50 +0100551 if (msgb_l2len(msg) < sizeof(*nsh) + 3) {
552 msgb_free(msg);
Alexander Couzens6a161492020-07-12 13:45:50 +0200553 return;
Alexander Couzenscce88282020-10-26 00:25:50 +0100554 }
Alexander Couzens6a161492020-07-12 13:45:50 +0200555
556 /* TODO: 7.1: For an IP sub-network, an NS-UNITDATA PDU
557 * for a PTP BVC may indicate a request to change the IP endpoint
558 * and/or a response to a change in the IP endpoint. */
559
560 /* TODO: nsh->data[0] -> C/R only valid in IP SNS */
561 bvci = nsh->data[1] << 8 | nsh->data[2];
562
Alexander Couzens89acdef2020-09-23 18:22:31 +0200563 msg->l3h = &nsh->data[3];
564 nsp.bvci = bvci;
565 nsp.nsei = priv->nsvc->nse->nsei;
Alexander Couzens6a161492020-07-12 13:45:50 +0200566
Alexander Couzensc1cd3332020-09-23 23:24:02 +0200567 /* 10.3.9 NS SDU Control Bits */
568 if (nsh->data[0] & 0x1)
Alexander Couzens138b96f2021-01-25 16:23:29 +0100569 nsp.u.unitdata.change = GPRS_NS2_ENDPOINT_REQUEST_CHANGE;
Alexander Couzens6a161492020-07-12 13:45:50 +0200570
Alexander Couzens138b96f2021-01-25 16:23:29 +0100571 osmo_prim_init(&nsp.oph, SAP_NS, GPRS_NS2_PRIM_UNIT_DATA,
Alexander Couzens6a161492020-07-12 13:45:50 +0200572 PRIM_OP_INDICATION, msg);
573 nsi->cb(&nsp.oph, nsi->cb_data);
574}
575
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100576static void ns2_vc_fsm_allstate_action(struct osmo_fsm_inst *fi,
Alexander Couzens6a161492020-07-12 13:45:50 +0200577 uint32_t event,
578 void *data)
579{
580 struct gprs_ns2_vc_priv *priv = fi->priv;
581 struct gprs_ns2_inst *nsi = ns_inst_from_fi(fi);
Alexander Couzenscce88282020-10-26 00:25:50 +0100582 struct msgb *msg = data;
Alexander Couzens6a161492020-07-12 13:45:50 +0200583
584 switch (event) {
Alexander Couzensf5775432021-01-18 13:49:00 +0100585 case GPRS_NS2_EV_RX_RESET:
Alexander Couzens138b96f2021-01-25 16:23:29 +0100586 if (priv->nsvc->mode != GPRS_NS2_VC_MODE_BLOCKRESET)
Alexander Couzens6a161492020-07-12 13:45:50 +0200587 break;
588
589 /* move the FSM into reset */
590 if (fi->state != GPRS_NS2_ST_RESET) {
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100591 priv->initiate_reset = false;
Alexander Couzens6a161492020-07-12 13:45:50 +0200592 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_RESET, nsi->timeout[NS_TOUT_TNS_RESET], NS_TOUT_TNS_RESET);
593 }
594 /* pass the event down into FSM action */
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100595 ns2_st_reset(fi, event, data);
Alexander Couzens6a161492020-07-12 13:45:50 +0200596 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100597 case GPRS_NS2_EV_RX_ALIVE:
Alexander Couzens6a161492020-07-12 13:45:50 +0200598 switch (fi->state) {
599 case GPRS_NS2_ST_UNCONFIGURED:
600 case GPRS_NS2_ST_RESET:
601 /* ignore ALIVE */
602 break;
603 default:
604 ns2_tx_alive_ack(priv->nsvc);
605 }
606 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100607 case GPRS_NS2_EV_RX_ALIVE_ACK:
Alexander Couzens6a161492020-07-12 13:45:50 +0200608 /* for VCs without RESET/BLOCK/UNBLOCK, the connections comes after ALIVE_ACK unblocked */
609 if (fi->state == GPRS_NS2_ST_ALIVE)
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100610 ns2_st_alive(fi, event, data);
Alexander Couzens6a161492020-07-12 13:45:50 +0200611 else
612 recv_test_procedure(fi);
613 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100614 case GPRS_NS2_EV_RX_UNITDATA:
Alexander Couzenscce88282020-10-26 00:25:50 +0100615 /* UNITDATA has to handle the release of msg.
616 * If send upwards (gprs_ns2_recv_unitdata) it must NOT free
617 * the msg, the upper layer has to do it.
618 * Otherwise the msg must be freed.
619 */
Alexander Couzens6a161492020-07-12 13:45:50 +0200620 switch (fi->state) {
621 case GPRS_NS2_ST_BLOCKED:
622 /* 7.2.1: the BLOCKED_ACK might be lost */
Alexander Couzens47afc422021-01-17 20:12:46 +0100623 if (priv->accept_unitdata) {
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100624 ns2_recv_unitdata(fi, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100625 return;
626 }
627
628 ns2_tx_status(priv->nsvc,
629 NS_CAUSE_NSVC_BLOCKED,
630 0, msg);
Alexander Couzens6a161492020-07-12 13:45:50 +0200631 break;
632 /* ALIVE can receive UNITDATA if the ALIVE_ACK is lost */
633 case GPRS_NS2_ST_ALIVE:
634 case GPRS_NS2_ST_UNBLOCKED:
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100635 ns2_recv_unitdata(fi, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100636 return;
Alexander Couzens6a161492020-07-12 13:45:50 +0200637 }
Alexander Couzenscce88282020-10-26 00:25:50 +0100638
639 msgb_free(msg);
Alexander Couzens6a161492020-07-12 13:45:50 +0200640 break;
Alexander Couzensf5775432021-01-18 13:49:00 +0100641 case GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED:
Alexander Couzens6f3b7382020-12-21 15:57:04 +0100642 if (fi->state != GPRS_NS2_ST_UNCONFIGURED) {
643 /* Force the NSVC back to its initial state */
644 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_UNCONFIGURED, 0, 0);
Alexander Couzens6f3b7382020-12-21 15:57:04 +0100645 return;
646 }
Daniel Willmanned0c9822020-11-18 14:08:07 +0100647 break;
Alexander Couzens47afc422021-01-17 20:12:46 +0100648 case GPRS_NS2_EV_REQ_OM_BLOCK:
649 /* vty cmd: block */
650 priv->initiate_block = true;
651 priv->om_blocked = true;
652 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
653 break;
654 case GPRS_NS2_EV_REQ_OM_UNBLOCK:
655 /* vty cmd: unblock*/
656 if (!priv->om_blocked)
657 return;
658 priv->om_blocked = false;
659 if (fi->state == GPRS_NS2_ST_BLOCKED)
660 osmo_fsm_inst_state_chg(fi, GPRS_NS2_ST_BLOCKED, nsi->timeout[NS_TOUT_TNS_BLOCK], 0);
661 break;
Alexander Couzens6a161492020-07-12 13:45:50 +0200662 }
663}
664
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100665static void ns2_vc_fsm_clean(struct osmo_fsm_inst *fi,
Alexander Couzens0346b642020-10-27 13:05:56 +0100666 enum osmo_fsm_term_cause cause)
667{
668 struct gprs_ns2_vc_priv *priv = fi->priv;
669
670 osmo_timer_del(&priv->alive.timer);
671}
672
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100673static struct osmo_fsm ns2_vc_fsm = {
Alexander Couzens6a161492020-07-12 13:45:50 +0200674 .name = "GPRS-NS2-VC",
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100675 .states = ns2_vc_states,
676 .num_states = ARRAY_SIZE(ns2_vc_states),
Alexander Couzensf5775432021-01-18 13:49:00 +0100677 .allstate_event_mask = S(GPRS_NS2_EV_RX_UNITDATA) |
678 S(GPRS_NS2_EV_RX_RESET) |
679 S(GPRS_NS2_EV_RX_ALIVE) |
680 S(GPRS_NS2_EV_RX_ALIVE_ACK) |
681 S(GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED) |
Alexander Couzens47afc422021-01-17 20:12:46 +0100682 S(GPRS_NS2_EV_REQ_OM_BLOCK) |
683 S(GPRS_NS2_EV_REQ_OM_UNBLOCK),
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100684 .allstate_action = ns2_vc_fsm_allstate_action,
685 .cleanup = ns2_vc_fsm_clean,
686 .timer_cb = ns2_vc_fsm_timer_cb,
Alexander Couzens6a161492020-07-12 13:45:50 +0200687 /* .log_subsys = DNS, "is not constant" */
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100688 .event_names = ns2_vc_event_names,
Alexander Couzens6a161492020-07-12 13:45:50 +0200689 .pre_term = NULL,
690 .log_subsys = DLNS,
691};
692
693/*!
694 * \brief gprs_ns2_vc_fsm_alloc
695 * \param ctx
696 * \param vc
697 * \param id a char representation of the virtual curcuit
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100698 * \param initiator initiator is the site which starts the connection. Usually the BSS.
Alexander Couzens6a161492020-07-12 13:45:50 +0200699 * \return NULL on error, otherwise the fsm
700 */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100701struct osmo_fsm_inst *ns2_vc_fsm_alloc(struct gprs_ns2_vc *nsvc,
Daniel Willmannf5b2e282020-11-18 18:51:25 +0100702 const char *id, bool initiator)
Alexander Couzens6a161492020-07-12 13:45:50 +0200703{
704 struct osmo_fsm_inst *fi;
705 struct gprs_ns2_vc_priv *priv;
706
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100707 fi = osmo_fsm_inst_alloc(&ns2_vc_fsm, nsvc, NULL, LOGL_DEBUG, id);
Alexander Couzens6a161492020-07-12 13:45:50 +0200708 if (!fi)
709 return fi;
710
711 nsvc->fi = fi;
712 priv = fi->priv = talloc_zero(fi, struct gprs_ns2_vc_priv);
713 priv->nsvc = nsvc;
Alexander Couzensea01bf22021-01-18 14:01:01 +0100714 priv->initiator = initiator;
Alexander Couzens6a161492020-07-12 13:45:50 +0200715
716 osmo_timer_setup(&priv->alive.timer, alive_timeout_handler, fi);
717
718 return fi;
719}
720
Harald Welte5bef2cc2020-09-18 22:33:24 +0200721/*! Start a NS-VC FSM.
722 * \param nsvc the virtual circuit
723 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100724int ns2_vc_fsm_start(struct gprs_ns2_vc *nsvc)
Alexander Couzens6a161492020-07-12 13:45:50 +0200725{
726 /* allows to call this function even for started nsvc by gprs_ns2_start_alive_all_nsvcs */
727 if (nsvc->fi->state == GPRS_NS2_ST_UNCONFIGURED)
Alexander Couzensf5775432021-01-18 13:49:00 +0100728 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_START, NULL);
Alexander Couzens6a161492020-07-12 13:45:50 +0200729 return 0;
730}
731
Daniel Willmanned0c9822020-11-18 14:08:07 +0100732/*! Reset a NS-VC FSM.
733 * \param nsvc the virtual circuit
734 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100735int ns2_vc_force_unconfigured(struct gprs_ns2_vc *nsvc)
Daniel Willmanned0c9822020-11-18 14:08:07 +0100736{
Alexander Couzensf5775432021-01-18 13:49:00 +0100737 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_FORCE_UNCONFIGURED, NULL);
Daniel Willmanned0c9822020-11-18 14:08:07 +0100738}
739
Alexander Couzens47afc422021-01-17 20:12:46 +0100740/*! Block a NS-VC.
741 * \param nsvc the virtual circuit
742 * \return 0 on success; negative on error */
743int ns2_vc_block(struct gprs_ns2_vc *nsvc)
744{
745 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_OM_BLOCK, NULL);
746}
747
748/*! Unblock a NS-VC.
749 * \param nsvc the virtual circuit
750 * \return 0 on success; negative on error */
751int ns2_vc_unblock(struct gprs_ns2_vc *nsvc)
752{
753 return osmo_fsm_inst_dispatch(nsvc->fi, GPRS_NS2_EV_REQ_OM_UNBLOCK, NULL);
754}
755
Harald Welte5bef2cc2020-09-18 22:33:24 +0200756/*! entry point for messages from the driver/VL
757 * \param nsvc virtual circuit on which the message was received
758 * \param msg message that was received
759 * \param tp parsed TLVs of the received message
760 * \return 0 on success; negative on error */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100761int ns2_vc_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp)
Alexander Couzens6a161492020-07-12 13:45:50 +0200762{
763 struct gprs_ns_hdr *nsh = (struct gprs_ns_hdr *) msg->l2h;
764 struct osmo_fsm_inst *fi = nsvc->fi;
Alexander Couzenscce88282020-10-26 00:25:50 +0100765 int rc = 0;
Alexander Couzens6a161492020-07-12 13:45:50 +0200766 uint8_t cause;
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100767 uint16_t nsei, nsvci;
Alexander Couzens6a161492020-07-12 13:45:50 +0200768
769 /* TODO: 7.2: on UNBLOCK/BLOCK: check if NS-VCI is correct,
770 * if not answer STATUS with "NS-VC unknown" */
Alexander Couzens6a161492020-07-12 13:45:50 +0200771 /* TODO: handle BLOCK/UNBLOCK/ALIVE with different VCI */
772
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100773 if (ns2_validate(nsvc, nsh->pdu_type, msg, tp, &cause)) {
Alexander Couzens6a161492020-07-12 13:45:50 +0200774 if (nsh->pdu_type != NS_PDUT_STATUS) {
Alexander Couzenscce88282020-10-26 00:25:50 +0100775 rc = ns2_tx_status(nsvc, cause, 0, msg);
776 goto out;
Alexander Couzens6a161492020-07-12 13:45:50 +0200777 }
778 }
779
Alexander Couzens43771df2021-01-20 13:03:03 +0100780 if (TLVP_PRESENT(tp, NS_IE_NSEI)) {
781 nsei = tlvp_val16be(tp, NS_IE_NSEI);
782 if (nsei != nsvc->nse->nsei) {
783 /* 48.016 § 7.3.1 send, RESET_ACK to wrong NSVCI + ignore */
784 if (nsh->pdu_type == NS_PDUT_RESET)
785 ns2_tx_reset_ack(nsvc);
786
787 LOGNSVC(nsvc, LOGL_ERROR, "Rx %s with wrong NSEI=%05u. Ignoring PDU.\n",
788 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type), nsei);
789 goto out;
790 }
791 }
792
Alexander Couzens3f576ab2021-01-20 14:50:31 +0100793 if (nsvc->nsvci_is_valid && TLVP_PRESENT(tp, NS_IE_VCI)) {
794 nsvci = tlvp_val16be(tp, NS_IE_VCI);
795 if (nsvci != nsvc->nsvci) {
796 /* 48.016 § 7.3.1 send RESET_ACK to wrong NSVCI + ignore */
797 if (nsh->pdu_type == NS_PDUT_RESET)
798 ns2_tx_reset_ack(nsvc);
799
800 LOGNSVC(nsvc, LOGL_ERROR, "Rx %s with wrong NSVCI=%05u. Ignoring PDU.\n",
801 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type), nsvci);
802 goto out;
803 }
804 }
805
Alexander Couzens6a161492020-07-12 13:45:50 +0200806 switch (nsh->pdu_type) {
807 case NS_PDUT_RESET:
Alexander Couzensf5775432021-01-18 13:49:00 +0100808 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_RESET, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200809 break;
810 case NS_PDUT_RESET_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100811 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_RESET_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200812 break;
813 case NS_PDUT_BLOCK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100814 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_BLOCK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200815 break;
816 case NS_PDUT_BLOCK_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100817 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_BLOCK_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200818 break;
819 case NS_PDUT_UNBLOCK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100820 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNBLOCK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200821 break;
822 case NS_PDUT_UNBLOCK_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100823 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNBLOCK_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200824 break;
825 case NS_PDUT_ALIVE:
Alexander Couzensf5775432021-01-18 13:49:00 +0100826 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_ALIVE, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200827 break;
828 case NS_PDUT_ALIVE_ACK:
Alexander Couzensf5775432021-01-18 13:49:00 +0100829 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_ALIVE_ACK, tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200830 break;
831 case NS_PDUT_UNITDATA:
Alexander Couzenscce88282020-10-26 00:25:50 +0100832 /* UNITDATA have to free msg because it might send the msg layer upwards */
Alexander Couzensf5775432021-01-18 13:49:00 +0100833 osmo_fsm_inst_dispatch(fi, GPRS_NS2_EV_RX_UNITDATA, msg);
Alexander Couzenscce88282020-10-26 00:25:50 +0100834 return 0;
Alexander Couzens6a161492020-07-12 13:45:50 +0200835 default:
Harald Weltef2949742021-01-20 14:54:14 +0100836 LOGPFSML(fi, LOGL_ERROR, "NSEI=%u Rx unknown NS PDU type %s\n", nsvc->nse->nsei,
837 get_value_string(gprs_ns_pdu_strings, nsh->pdu_type));
Alexander Couzens6a161492020-07-12 13:45:50 +0200838 return -EINVAL;
839 }
840
Alexander Couzenscce88282020-10-26 00:25:50 +0100841out:
842 msgb_free(msg);
843
844 return rc;
Alexander Couzens6a161492020-07-12 13:45:50 +0200845}
846
Harald Welte5bef2cc2020-09-18 22:33:24 +0200847/*! is the given NS-VC unblocked? */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100848int ns2_vc_is_unblocked(struct gprs_ns2_vc *nsvc)
Alexander Couzens6a161492020-07-12 13:45:50 +0200849{
850 return (nsvc->fi->state == GPRS_NS2_ST_UNBLOCKED);
851}
852
853/* initialize osmo_ctx on main tread */
854static __attribute__((constructor)) void on_dso_load_ctx(void)
855{
Alexander Couzensf7e2cac2021-01-25 16:18:21 +0100856 OSMO_ASSERT(osmo_fsm_register(&ns2_vc_fsm) == 0);
Alexander Couzens6a161492020-07-12 13:45:50 +0200857}