blob: d2407f663366b80fd903e16add737c097a59e2d9 [file] [log] [blame]
Alexander Couzens6a161492020-07-12 13:45:50 +02001/*! \file gprs_ns2_internal.h */
2
3#pragma once
4
5#include <stdbool.h>
6#include <stdint.h>
7
Alexander Couzens6cf65d92021-01-18 17:55:35 +01008#include <osmocom/core/logging.h>
Alexander Couzens6a161492020-07-12 13:45:50 +02009#include <osmocom/gprs/protocol/gsm_08_16.h>
10#include <osmocom/gprs/gprs_ns2.h>
11
Harald Weltef2949742021-01-20 14:54:14 +010012#define LOGNSE(nse, lvl, fmt, args ...) \
13 LOGP(DLNS, lvl, "NSE(%05u) " fmt, (nse)->nsei, ## args)
14
15#define LOGBIND(bind, lvl, fmt, args ...) \
16 LOGP(DLNS, lvl, "BIND(%s) " fmt, (bind)->name, ## args)
17
Alexander Couzens6cf65d92021-01-18 17:55:35 +010018#define LOGNSVC_SS(ss, nsvc, lvl, fmt, args ...) \
Harald Weltef2949742021-01-20 14:54:14 +010019 do { \
20 if ((nsvc)->nsvci_is_valid) { \
Alexander Couzens6cf65d92021-01-18 17:55:35 +010021 LOGP(ss, lvl, "NSE(%05u)-NSVC(%05u) " fmt, \
Harald Weltef2949742021-01-20 14:54:14 +010022 (nsvc)->nse->nsei, (nsvc)->nsvci, ## args); \
23 } else { \
Alexander Couzens6cf65d92021-01-18 17:55:35 +010024 LOGP(ss, lvl, "NSE(%05u)-NSVC(none) " fmt, \
Harald Weltef2949742021-01-20 14:54:14 +010025 (nsvc)->nse->nsei, ## args); \
26 } \
27 } while (0)
28
Alexander Couzens6cf65d92021-01-18 17:55:35 +010029#define LOGNSVC(nsvc, lvl, fmt, args ...) \
30 LOGNSVC_SS(DLNS, nsvc, lvl, fmt, ## args)
31
32#define LOG_NS_SIGNAL(nsvc, direction, pdu_type, lvl, fmt, args ...) \
33 LOGNSVC_SS(DLNSSIGNAL, nsvc, lvl, "%s %s" fmt, direction, get_value_string(gprs_ns_pdu_strings, pdu_type), ## args)
34
35#define LOG_NS_DATA(nsvc, direction, pdu_type, lvl, fmt, args ...) \
36 LOGNSVC_SS(DLNSDATA, nsvc, lvl, "%s %s" fmt, direction, get_value_string(gprs_ns_pdu_strings, pdu_type), ## args)
37
38#define LOG_NS_RX_SIGNAL(nsvc, pdu_type) LOG_NS_SIGNAL(nsvc, "Rx", pdu_type, LOGL_INFO, "\n")
39#define LOG_NS_TX_SIGNAL(nsvc, pdu_type) LOG_NS_SIGNAL(nsvc, "Tx", pdu_type, LOGL_INFO, "\n")
40
Daniel Willmannefa64f92021-07-09 20:35:00 +020041#define RATE_CTR_INC_NS(nsvc, ctr) \
42 do { \
43 struct gprs_ns2_vc *_nsvc = (nsvc); \
44 rate_ctr_inc(rate_ctr_group_get_ctr(_nsvc->ctrg, ctr)); \
45 rate_ctr_inc(rate_ctr_group_get_ctr(_nsvc->nse->ctrg, ctr)); \
46 } while (0)
47
48#define RATE_CTR_ADD_NS(nsvc, ctr, val) \
49 do { \
50 struct gprs_ns2_vc *_nsvc = (nsvc); \
51 rate_ctr_add(rate_ctr_group_get_ctr(_nsvc->ctrg, ctr), val); \
52 rate_ctr_add(rate_ctr_group_get_ctr(_nsvc->nse->ctrg, ctr), val); \
53 } while (0)
54
Alexander Couzens6cf65d92021-01-18 17:55:35 +010055
Alexander Couzens6a161492020-07-12 13:45:50 +020056struct osmo_fsm_inst;
57struct tlv_parsed;
58struct vty;
59
60struct gprs_ns2_vc_driver;
61struct gprs_ns2_vc_bind;
62
Alexander Couzens90ee9632020-12-07 06:18:32 +010063#define NS_TIMERS_COUNT 10
Harald Welte3a44d172020-12-16 12:02:51 +010064#define NS_TIMERS "(tns-block|tns-block-retries|tns-reset|tns-reset-retries|tns-test|tns-alive|tns-alive-retries|tsns-prov|tsns-size-retries|tsns-config-retries)"
Alexander Couzens6a161492020-07-12 13:45:50 +020065#define NS_TIMERS_HELP \
66 "(un)blocking Timer (Tns-block) timeout\n" \
67 "(un)blocking Timer (Tns-block) number of retries\n" \
68 "Reset Timer (Tns-reset) timeout\n" \
69 "Reset Timer (Tns-reset) number of retries\n" \
70 "Test Timer (Tns-test) timeout\n" \
71 "Alive Timer (Tns-alive) timeout\n" \
72 "Alive Timer (Tns-alive) number of retries\n" \
Alexander Couzens90ee9632020-12-07 06:18:32 +010073 "SNS Provision Timer (Tsns-prov) timeout\n" \
74 "SNS Size number of retries\n" \
75 "SNS Config number of retries\n" \
Alexander Couzens6a161492020-07-12 13:45:50 +020076
77/* Educated guess - LLC user payload is 1500 bytes plus possible headers */
78#define NS_ALLOC_SIZE 3072
79#define NS_ALLOC_HEADROOM 20
80
81enum ns2_timeout {
82 NS_TOUT_TNS_BLOCK,
83 NS_TOUT_TNS_BLOCK_RETRIES,
84 NS_TOUT_TNS_RESET,
85 NS_TOUT_TNS_RESET_RETRIES,
86 NS_TOUT_TNS_TEST,
87 NS_TOUT_TNS_ALIVE,
88 NS_TOUT_TNS_ALIVE_RETRIES,
89 NS_TOUT_TSNS_PROV,
Alexander Couzens90ee9632020-12-07 06:18:32 +010090 NS_TOUT_TSNS_SIZE_RETRIES,
91 NS_TOUT_TSNS_CONFIG_RETRIES,
Alexander Couzens6a161492020-07-12 13:45:50 +020092};
93
94enum nsvc_timer_mode {
95 /* standard timers */
96 NSVC_TIMER_TNS_TEST,
97 NSVC_TIMER_TNS_ALIVE,
98 NSVC_TIMER_TNS_RESET,
99 _NSVC_TIMER_NR,
100};
101
Harald Welte76346072021-01-31 11:54:02 +0100102enum ns2_vc_stat {
Alexander Couzens6a161492020-07-12 13:45:50 +0200103 NS_STAT_ALIVE_DELAY,
104};
105
Harald Welte76346072021-01-31 11:54:02 +0100106enum ns2_bind_stat {
107 NS2_BIND_STAT_BACKLOG_LEN,
108};
109
Alexander Couzens6a161492020-07-12 13:45:50 +0200110/*! Osmocom NS2 VC create status */
Alexander Couzensba5a9922021-01-25 16:03:23 +0100111enum ns2_cs {
112 NS2_CS_CREATED, /*!< A NSVC object has been created */
113 NS2_CS_FOUND, /*!< A NSVC object has been found */
114 NS2_CS_REJECTED, /*!< Rejected and answered message */
115 NS2_CS_SKIPPED, /*!< Skipped message */
116 NS2_CS_ERROR, /*!< Failed to process message */
Alexander Couzens6a161492020-07-12 13:45:50 +0200117};
118
Harald Welte3d5eaee2021-01-30 21:33:02 +0100119enum ns_ctr {
120 NS_CTR_PKTS_IN,
121 NS_CTR_PKTS_OUT,
Harald Weltef22ae5a2021-01-30 22:01:28 +0100122 NS_CTR_PKTS_OUT_DROP,
Harald Welte3d5eaee2021-01-30 21:33:02 +0100123 NS_CTR_BYTES_IN,
124 NS_CTR_BYTES_OUT,
Harald Weltef22ae5a2021-01-30 22:01:28 +0100125 NS_CTR_BYTES_OUT_DROP,
Harald Welte3d5eaee2021-01-30 21:33:02 +0100126 NS_CTR_BLOCKED,
Harald Welteb40bf8b2021-01-30 22:43:01 +0100127 NS_CTR_UNBLOCKED,
Harald Welte3d5eaee2021-01-30 21:33:02 +0100128 NS_CTR_DEAD,
129 NS_CTR_REPLACED,
130 NS_CTR_NSEI_CHG,
131 NS_CTR_INV_VCI,
132 NS_CTR_INV_NSEI,
133 NS_CTR_LOST_ALIVE,
134 NS_CTR_LOST_RESET,
135};
Alexander Couzens6a161492020-07-12 13:45:50 +0200136
137#define NSE_S_BLOCKED 0x0001
138#define NSE_S_ALIVE 0x0002
139#define NSE_S_RESET 0x0004
140
141#define NS_DESC_B(st) ((st) & NSE_S_BLOCKED ? "BLOCKED" : "UNBLOCKED")
142#define NS_DESC_A(st) ((st) & NSE_S_ALIVE ? "ALIVE" : "DEAD")
143#define NS_DESC_R(st) ((st) & NSE_S_RESET ? "RESET" : "UNRESET")
144
145/*! An instance of the NS protocol stack */
146struct gprs_ns2_inst {
147 /*! callback to the user for incoming UNIT DATA IND */
148 osmo_prim_cb cb;
149
150 /*! callback data */
151 void *cb_data;
152
153 /*! linked lists of all NSVC binds (e.g. IPv4 bind, but could be also E1 */
154 struct llist_head binding;
155
156 /*! linked lists of all NSVC in this instance */
157 struct llist_head nse;
158
Alexander Couzens6a161492020-07-12 13:45:50 +0200159 uint16_t timeout[NS_TIMERS_COUNT];
160
161 /*! workaround for rate counter until rate counter accepts char str as index */
Harald Welte97ccbf72021-01-31 11:49:19 +0100162 uint32_t nsvc_rate_ctr_idx;
Harald Welte76346072021-01-31 11:54:02 +0100163 uint32_t bind_rate_ctr_idx;
Harald Welte53a2fde2020-12-01 22:21:14 +0100164
165 /*! libmnl netlink socket for link state monitoring */
166 struct osmo_mnl *linkmon_mnl;
Alexander Couzens6a161492020-07-12 13:45:50 +0200167};
168
Harald Welte53a2fde2020-12-01 22:21:14 +0100169
Alexander Couzens6a161492020-07-12 13:45:50 +0200170/*! Structure repesenting a NSE. The BSS/PCU will only have a single NSE, while SGSN has one for each BSS/PCU */
171struct gprs_ns2_nse {
172 uint16_t nsei;
173
174 /*! entry back to ns2_inst */
175 struct gprs_ns2_inst *nsi;
176
177 /*! llist entry for gprs_ns2_inst */
178 struct llist_head list;
179
180 /*! llist head to hold all nsvc */
181 struct llist_head nsvc;
182
Alexander Couzens8a2a1a42020-12-26 18:00:17 +0100183 /*! count all active NSVCs */
184 int nsvc_count;
Alexander Couzensfc3dd1f2020-11-19 00:41:47 +0100185
Alexander Couzens6a161492020-07-12 13:45:50 +0200186 /*! true if this NSE was created by VTY or pcu socket) */
187 bool persistent;
188
Alexander Couzensda0a2852020-10-01 23:24:07 +0200189 /*! true if this NSE wasn't yet alive at all.
190 * Will be true after the first status ind with NS_AFF_CAUSE_RECOVERY */
191 bool first;
192
Alexander Couzens6a161492020-07-12 13:45:50 +0200193 /*! true if this NSE has at least one alive VC */
194 bool alive;
195
Alexander Couzensaac90162020-11-19 02:44:04 +0100196 /*! which link-layer are we based on? */
197 enum gprs_ns2_ll ll;
198
Alexander Couzensd923cff2020-12-01 01:03:52 +0100199 /*! which dialect does this NSE speaks? */
200 enum gprs_ns2_dialect dialect;
201
Alexander Couzens6a161492020-07-12 13:45:50 +0200202 struct osmo_fsm_inst *bss_sns_fi;
Harald Welte3221f872021-01-18 14:58:51 +0100203
Alexander Couzense052c412021-02-15 02:10:57 +0100204 /*! sum of all the data weight of _alive_ NS-VCs */
Harald Welte3221f872021-01-18 14:58:51 +0100205 uint32_t sum_data_weight;
206
Alexander Couzense052c412021-02-15 02:10:57 +0100207 /*! sum of all the signalling weight of _alive_ NS-VCs */
Harald Welte3221f872021-01-18 14:58:51 +0100208 uint32_t sum_sig_weight;
Alexander Couzens4f1128f2021-01-20 17:42:48 +0100209
210 /*! MTU of a NS PDU. This is the lowest MTU of all NSVCs */
211 uint16_t mtu;
Harald Welte5b034fb2021-03-04 14:16:49 +0100212
213 /*! are we implementing the SGSN role? */
214 bool ip_sns_role_sgsn;
Daniel Willmannefa64f92021-07-09 20:35:00 +0200215
216 /*! NSE-wide statistics */
217 struct rate_ctr_group *ctrg;
Alexander Couzens41589a32021-07-19 03:40:02 +0200218
219 /*! recursive anchor */
220 bool freed;
Alexander Couzens6a161492020-07-12 13:45:50 +0200221};
222
223/*! Structure representing a single NS-VC */
224struct gprs_ns2_vc {
225 /*! list of NS-VCs within NSE */
226 struct llist_head list;
227
228 /*! list of NS-VCs within bind, bind is the owner! */
229 struct llist_head blist;
230
231 /*! pointer to NS Instance */
232 struct gprs_ns2_nse *nse;
233
234 /*! pointer to NS VL bind. bind own the memory of this instance */
235 struct gprs_ns2_vc_bind *bind;
236
237 /*! true if this NS was created by VTY or pcu socket) */
238 bool persistent;
239
240 /*! uniquely identifies NS-VC if VC contains nsvci */
241 uint16_t nsvci;
242
243 /*! signalling weight. 0 = don't use for signalling (BVCI == 0)*/
244 uint8_t sig_weight;
245
Alexander Couzensfc3dd1f2020-11-19 00:41:47 +0100246 /*! signalling packet counter for the load sharing function */
247 uint8_t sig_counter;
248
249 /*! data weight. 0 = don't use for user data (BVCI != 0) */
Alexander Couzens6a161492020-07-12 13:45:50 +0200250 uint8_t data_weight;
251
252 /*! can be used by the bind/driver of the virtual circuit. e.g. ipv4/ipv6/frgre/e1 */
253 void *priv;
254
255 bool nsvci_is_valid;
Harald Weltec962a2e2021-03-05 08:09:08 +0100256 /*! should this NS-VC only be used for SNS-SIZE and SNS-CONFIG? */
Alexander Couzens6a161492020-07-12 13:45:50 +0200257 bool sns_only;
258
259 struct rate_ctr_group *ctrg;
260 struct osmo_stat_item_group *statg;
261
Alexander Couzens6a161492020-07-12 13:45:50 +0200262 enum gprs_ns2_vc_mode mode;
263
264 struct osmo_fsm_inst *fi;
Alexander Couzens41589a32021-07-19 03:40:02 +0200265
266 /*! recursive anchor */
267 bool freed;
Alexander Couzens6a161492020-07-12 13:45:50 +0200268};
269
270/*! Structure repesenting a bind instance. E.g. IPv4 listen port. */
271struct gprs_ns2_vc_bind {
Alexander Couzensaaa55a62020-12-03 06:02:03 +0100272 /*! unique name */
273 const char *name;
Alexander Couzens6a161492020-07-12 13:45:50 +0200274 /*! list entry in nsi */
275 struct llist_head list;
276 /*! list of all VC */
277 struct llist_head nsvc;
278 /*! driver private structure */
279 void *priv;
280 /*! a pointer back to the nsi */
281 struct gprs_ns2_inst *nsi;
282 struct gprs_ns2_vc_driver *driver;
283
Alexander Couzensd923cff2020-12-01 01:03:52 +0100284 bool accept_ipaccess;
285 bool accept_sns;
Alexander Couzens6a161492020-07-12 13:45:50 +0200286
Alexander Couzens1c8785d2020-12-17 06:58:53 +0100287 /*! transfer capability in mbit */
288 int transfer_capability;
289
Alexander Couzens4f1128f2021-01-20 17:42:48 +0100290 /*! MTU of a NS PDU on this bind. */
291 uint16_t mtu;
292
Alexander Couzensaac90162020-11-19 02:44:04 +0100293 /*! which link-layer are we based on? */
294 enum gprs_ns2_ll ll;
295
Alexander Couzens6a161492020-07-12 13:45:50 +0200296 /*! send a msg over a VC */
297 int (*send_vc)(struct gprs_ns2_vc *nsvc, struct msgb *msg);
298
299 /*! free the vc priv data */
300 void (*free_vc)(struct gprs_ns2_vc *nsvc);
Alexander Couzens6a161492020-07-12 13:45:50 +0200301
Alexander Couzens22f34712020-10-02 02:34:39 +0200302 /*! allow to show information for the vty */
303 void (*dump_vty)(const struct gprs_ns2_vc_bind *bind,
304 struct vty *vty, bool stats);
Harald Welte76346072021-01-31 11:54:02 +0100305
Alexander Couzensc4704762021-02-08 23:13:12 +0100306 /*! the IP-SNS signalling weight when doing dynamic configuration */
307 uint8_t sns_sig_weight;
308 /*! the IP-SNS data weight when doing dynamic configuration */
309 uint8_t sns_data_weight;
310
Harald Welte76346072021-01-31 11:54:02 +0100311 struct osmo_stat_item_group *statg;
Alexander Couzens41589a32021-07-19 03:40:02 +0200312
313 /*! recursive anchor */
314 bool freed;
Alexander Couzens22f34712020-10-02 02:34:39 +0200315};
Alexander Couzens6a161492020-07-12 13:45:50 +0200316
317struct gprs_ns2_vc_driver {
318 const char *name;
319 void *priv;
320 void (*free_bind)(struct gprs_ns2_vc_bind *driver);
321};
322
Alexander Couzens175eb7b2021-07-20 18:41:14 +0200323enum ns2_sns_event {
324 NS2_SNS_EV_REQ_SELECT_ENDPOINT, /*!< Select a SNS endpoint from the list */
325 NS2_SNS_EV_RX_SIZE,
326 NS2_SNS_EV_RX_SIZE_ACK,
327 NS2_SNS_EV_RX_CONFIG,
328 NS2_SNS_EV_RX_CONFIG_END, /*!< SNS-CONFIG with end flag received */
329 NS2_SNS_EV_RX_CONFIG_ACK,
330 NS2_SNS_EV_RX_ADD,
331 NS2_SNS_EV_RX_DELETE,
332 NS2_SNS_EV_RX_CHANGE_WEIGHT,
333 NS2_SNS_EV_RX_ACK, /*!< Rx of SNS-ACK (response to ADD/DELETE/CHG_WEIGHT */
334 NS2_SNS_EV_REQ_NO_NSVC, /*!< no more NS-VC remaining (all dead) */
Alexander Couzens83f06ce2021-08-06 19:50:09 +0200335 NS2_SNS_EV_REQ_FREE_NSVCS, /*!< free all NS-VCs */
Alexander Couzens175eb7b2021-07-20 18:41:14 +0200336 NS2_SNS_EV_REQ_NSVC_ALIVE, /*!< a NS-VC became alive */
337 NS2_SNS_EV_REQ_ADD_BIND, /*!< add a new local bind to this NSE */
338 NS2_SNS_EV_REQ_DELETE_BIND, /*!< remove a local bind from this NSE */
339};
340
Alexander Couzensba5a9922021-01-25 16:03:23 +0100341enum ns2_cs ns2_create_vc(struct gprs_ns2_vc_bind *bind,
Alexander Couzens6a161492020-07-12 13:45:50 +0200342 struct msgb *msg,
Harald Welte7d0daac2021-03-02 23:08:43 +0100343 const struct osmo_sockaddr *remote,
Alexander Couzens6a161492020-07-12 13:45:50 +0200344 const char *logname,
345 struct msgb **reject,
346 struct gprs_ns2_vc **success);
347
Alexander Couzensffd49d02020-09-24 00:47:17 +0200348int ns2_recv_vc(struct gprs_ns2_vc *nsvc,
Alexander Couzens6a161492020-07-12 13:45:50 +0200349 struct msgb *msg);
350
351struct gprs_ns2_vc *ns2_vc_alloc(struct gprs_ns2_vc_bind *bind,
352 struct gprs_ns2_nse *nse,
Alexander Couzensd923cff2020-12-01 01:03:52 +0100353 bool initiater,
Harald Welte603f4042020-11-29 17:39:19 +0100354 enum gprs_ns2_vc_mode vc_mode,
355 const char *id);
Alexander Couzens6a161492020-07-12 13:45:50 +0200356
Harald Weltec3aa8f92021-01-31 11:41:34 +0100357int ns2_bind_alloc(struct gprs_ns2_inst *nsi, const char *name,
358 struct gprs_ns2_vc_bind **result);
359
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100360struct msgb *ns2_msgb_alloc(void);
Alexander Couzens6a161492020-07-12 13:45:50 +0200361
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100362void ns2_sns_write_vty(struct vty *vty, const struct gprs_ns2_nse *nse);
363void ns2_sns_dump_vty(struct vty *vty, const char *prefix, const struct gprs_ns2_nse *nse, bool stats);
Alexander Couzensbf95f0f2020-10-01 22:56:03 +0200364void ns2_prim_status_ind(struct gprs_ns2_nse *nse,
Daniel Willmann15c09a82020-11-03 23:05:43 +0100365 struct gprs_ns2_vc *nsvc,
Alexander Couzensbf95f0f2020-10-01 22:56:03 +0200366 uint16_t bvci,
Alexander Couzens6a161492020-07-12 13:45:50 +0200367 enum gprs_ns2_affecting_cause cause);
368void ns2_nse_notify_alive(struct gprs_ns2_vc *nsvc, bool alive);
Alexander Couzens4f1128f2021-01-20 17:42:48 +0100369void ns2_nse_update_mtu(struct gprs_ns2_nse *nse);
Harald Welte06d9bf92021-03-05 10:20:11 +0100370int ns2_nse_set_dialect(struct gprs_ns2_nse *nse, enum gprs_ns2_dialect dialect);
Alexander Couzens6a161492020-07-12 13:45:50 +0200371
372/* message */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100373int ns2_validate(struct gprs_ns2_vc *nsvc,
374 uint8_t pdu_type,
375 struct msgb *msg,
376 struct tlv_parsed *tp,
377 uint8_t *cause);
Alexander Couzens6a161492020-07-12 13:45:50 +0200378
379/* SNS messages */
380int ns2_tx_sns_ack(struct gprs_ns2_vc *nsvc, uint8_t trans_id, uint8_t *cause,
381 const struct gprs_ns_ie_ip4_elem *ip4_elems,
382 unsigned int num_ip4_elems,
383 const struct gprs_ns_ie_ip6_elem *ip6_elems,
384 unsigned int num_ip6_elems);
385int ns2_tx_sns_config(struct gprs_ns2_vc *nsvc, bool end_flag,
386 const struct gprs_ns_ie_ip4_elem *ip4_elems,
387 unsigned int num_ip4_elems,
388 const struct gprs_ns_ie_ip6_elem *ip6_elems,
389 unsigned int num_ip6_elems);
390int ns2_tx_sns_config_ack(struct gprs_ns2_vc *nsvc, uint8_t *cause);
391int ns2_tx_sns_size(struct gprs_ns2_vc *nsvc, bool reset_flag, uint16_t max_nr_nsvc,
392 int ip4_ep_nr, int ip6_ep_nr);
393int ns2_tx_sns_size_ack(struct gprs_ns2_vc *nsvc, uint8_t *cause);
394
Alexander Couzens27a55922021-02-27 01:08:35 +0100395int ns2_tx_sns_add(struct gprs_ns2_vc *nsvc,
396 uint8_t trans_id,
397 const struct gprs_ns_ie_ip4_elem *ip4_elems,
398 unsigned int num_ip4_elems,
399 const struct gprs_ns_ie_ip6_elem *ip6_elems,
400 unsigned int num_ip6_elems);
401int ns2_tx_sns_change_weight(struct gprs_ns2_vc *nsvc,
402 uint8_t trans_id,
403 const struct gprs_ns_ie_ip4_elem *ip4_elems,
404 unsigned int num_ip4_elems,
405 const struct gprs_ns_ie_ip6_elem *ip6_elems,
406 unsigned int num_ip6_elems);
407int ns2_tx_sns_del(struct gprs_ns2_vc *nsvc,
408 uint8_t trans_id,
409 const struct gprs_ns_ie_ip4_elem *ip4_elems,
410 unsigned int num_ip4_elems,
411 const struct gprs_ns_ie_ip6_elem *ip6_elems,
412 unsigned int num_ip6_elems);
413
Alexander Couzens6a161492020-07-12 13:45:50 +0200414/* transmit message over a VC */
415int ns2_tx_block(struct gprs_ns2_vc *nsvc, uint8_t cause);
416int ns2_tx_block_ack(struct gprs_ns2_vc *nsvc);
417
418int ns2_tx_reset(struct gprs_ns2_vc *nsvc, uint8_t cause);
419int ns2_tx_reset_ack(struct gprs_ns2_vc *nsvc);
420
421int ns2_tx_unblock(struct gprs_ns2_vc *nsvc);
422int ns2_tx_unblock_ack(struct gprs_ns2_vc *nsvc);
423
424int ns2_tx_alive(struct gprs_ns2_vc *nsvc);
425int ns2_tx_alive_ack(struct gprs_ns2_vc *nsvc);
426
427int ns2_tx_unit_data(struct gprs_ns2_vc *nsvc,
428 uint16_t bvci, uint8_t sducontrol,
429 struct msgb *msg);
430
431int ns2_tx_status(struct gprs_ns2_vc *nsvc, uint8_t cause,
432 uint16_t bvci, struct msgb *orig_msg);
433
434/* driver */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100435struct gprs_ns2_vc *ns2_ip_bind_connect(struct gprs_ns2_vc_bind *bind,
436 struct gprs_ns2_nse *nse,
437 const struct osmo_sockaddr *remote);
Alexander Couzense769f522020-12-07 07:37:07 +0100438int ns2_ip_count_bind(struct gprs_ns2_inst *nsi, struct osmo_sockaddr *remote);
439struct gprs_ns2_vc_bind *ns2_ip_get_bind_by_index(struct gprs_ns2_inst *nsi,
440 struct osmo_sockaddr *remote,
441 int index);
Alexander Couzens6a161492020-07-12 13:45:50 +0200442
443/* sns */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100444int ns2_sns_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200445struct osmo_fsm_inst *ns2_sns_bss_fsm_alloc(struct gprs_ns2_nse *nse,
446 const char *id);
Harald Welte4f127462021-03-02 20:49:10 +0100447struct osmo_fsm_inst *ns2_sns_sgsn_fsm_alloc(struct gprs_ns2_nse *nse, const char *id);
Alexander Couzensbe7cecc2021-02-03 18:25:27 +0100448void ns2_sns_replace_nsvc(struct gprs_ns2_vc *nsvc);
449void ns2_sns_notify_alive(struct gprs_ns2_nse *nse, struct gprs_ns2_vc *nsvc, bool alive);
Alexander Couzensc4704762021-02-08 23:13:12 +0100450void ns2_sns_update_weights(struct gprs_ns2_vc_bind *bind);
Alexander Couzens6a161492020-07-12 13:45:50 +0200451
452/* vc */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100453struct osmo_fsm_inst *ns2_vc_fsm_alloc(struct gprs_ns2_vc *nsvc,
Alexander Couzens6a161492020-07-12 13:45:50 +0200454 const char *id, bool initiate);
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100455int ns2_vc_fsm_start(struct gprs_ns2_vc *nsvc);
456int ns2_vc_force_unconfigured(struct gprs_ns2_vc *nsvc);
457int ns2_vc_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp);
458int ns2_vc_is_alive(struct gprs_ns2_vc *nsvc);
459int ns2_vc_is_unblocked(struct gprs_ns2_vc *nsvc);
Alexander Couzens47afc422021-01-17 20:12:46 +0100460int ns2_vc_block(struct gprs_ns2_vc *nsvc);
Alexander Couzens27e58732021-03-21 17:12:08 +0100461int ns2_vc_reset(struct gprs_ns2_vc *nsvc);
Alexander Couzens47afc422021-01-17 20:12:46 +0100462int ns2_vc_unblock(struct gprs_ns2_vc *nsvc);
Alexander Couzens75b61882021-03-21 16:18:17 +0100463void ns2_vty_dump_nsvc(struct vty *vty, struct gprs_ns2_vc *nsvc, bool stats);
Alexander Couzens6a161492020-07-12 13:45:50 +0200464
Alexander Couzens6a161492020-07-12 13:45:50 +0200465/* nse */
466void ns2_nse_notify_unblocked(struct gprs_ns2_vc *nsvc, bool unblocked);
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100467enum gprs_ns2_vc_mode ns2_dialect_to_vc_mode(enum gprs_ns2_dialect dialect);
Alexander Couzens1c8785d2020-12-17 06:58:53 +0100468int ns2_count_transfer_cap(struct gprs_ns2_nse *nse,
469 uint16_t bvci);
Harald Welted164ef82021-03-04 22:29:17 +0100470
471/* vty */
472int ns2_sns_add_sns_default_binds(struct gprs_ns2_nse *nse);