blob: 88e352a10613483840cbcb233763148e1f33eabc [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
8#include <osmocom/gprs/protocol/gsm_08_16.h>
9#include <osmocom/gprs/gprs_ns2.h>
10
Harald Weltef2949742021-01-20 14:54:14 +010011#define LOGNSE(nse, lvl, fmt, args ...) \
12 LOGP(DLNS, lvl, "NSE(%05u) " fmt, (nse)->nsei, ## args)
13
14#define LOGBIND(bind, lvl, fmt, args ...) \
15 LOGP(DLNS, lvl, "BIND(%s) " fmt, (bind)->name, ## args)
16
17
18#define LOGNSVC(nsvc, lvl, fmt, args ...) \
19 do { \
20 if ((nsvc)->nsvci_is_valid) { \
21 LOGP(DLNS, lvl, "NSE(%05u)-NSVC(%05u) " fmt, \
22 (nsvc)->nse->nsei, (nsvc)->nsvci, ## args); \
23 } else { \
24 LOGP(DLNS, lvl, "NSE(%05u)-NSVC(none) " fmt, \
25 (nsvc)->nse->nsei, ## args); \
26 } \
27 } while (0)
28
Alexander Couzens6a161492020-07-12 13:45:50 +020029struct osmo_fsm_inst;
30struct tlv_parsed;
31struct vty;
32
33struct gprs_ns2_vc_driver;
34struct gprs_ns2_vc_bind;
35
Alexander Couzens90ee9632020-12-07 06:18:32 +010036#define NS_TIMERS_COUNT 10
Harald Welte3a44d172020-12-16 12:02:51 +010037#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 +020038#define NS_TIMERS_HELP \
39 "(un)blocking Timer (Tns-block) timeout\n" \
40 "(un)blocking Timer (Tns-block) number of retries\n" \
41 "Reset Timer (Tns-reset) timeout\n" \
42 "Reset Timer (Tns-reset) number of retries\n" \
43 "Test Timer (Tns-test) timeout\n" \
44 "Alive Timer (Tns-alive) timeout\n" \
45 "Alive Timer (Tns-alive) number of retries\n" \
Alexander Couzens90ee9632020-12-07 06:18:32 +010046 "SNS Provision Timer (Tsns-prov) timeout\n" \
47 "SNS Size number of retries\n" \
48 "SNS Config number of retries\n" \
Alexander Couzens6a161492020-07-12 13:45:50 +020049
50/* Educated guess - LLC user payload is 1500 bytes plus possible headers */
51#define NS_ALLOC_SIZE 3072
52#define NS_ALLOC_HEADROOM 20
53
54enum ns2_timeout {
55 NS_TOUT_TNS_BLOCK,
56 NS_TOUT_TNS_BLOCK_RETRIES,
57 NS_TOUT_TNS_RESET,
58 NS_TOUT_TNS_RESET_RETRIES,
59 NS_TOUT_TNS_TEST,
60 NS_TOUT_TNS_ALIVE,
61 NS_TOUT_TNS_ALIVE_RETRIES,
62 NS_TOUT_TSNS_PROV,
Alexander Couzens90ee9632020-12-07 06:18:32 +010063 NS_TOUT_TSNS_SIZE_RETRIES,
64 NS_TOUT_TSNS_CONFIG_RETRIES,
Alexander Couzens6a161492020-07-12 13:45:50 +020065};
66
67enum nsvc_timer_mode {
68 /* standard timers */
69 NSVC_TIMER_TNS_TEST,
70 NSVC_TIMER_TNS_ALIVE,
71 NSVC_TIMER_TNS_RESET,
72 _NSVC_TIMER_NR,
73};
74
Harald Welte76346072021-01-31 11:54:02 +010075enum ns2_vc_stat {
Alexander Couzens6a161492020-07-12 13:45:50 +020076 NS_STAT_ALIVE_DELAY,
77};
78
Harald Welte76346072021-01-31 11:54:02 +010079enum ns2_bind_stat {
80 NS2_BIND_STAT_BACKLOG_LEN,
81};
82
Alexander Couzens6a161492020-07-12 13:45:50 +020083/*! Osmocom NS2 VC create status */
Alexander Couzensba5a9922021-01-25 16:03:23 +010084enum ns2_cs {
85 NS2_CS_CREATED, /*!< A NSVC object has been created */
86 NS2_CS_FOUND, /*!< A NSVC object has been found */
87 NS2_CS_REJECTED, /*!< Rejected and answered message */
88 NS2_CS_SKIPPED, /*!< Skipped message */
89 NS2_CS_ERROR, /*!< Failed to process message */
Alexander Couzens6a161492020-07-12 13:45:50 +020090};
91
Harald Welte3d5eaee2021-01-30 21:33:02 +010092enum ns_ctr {
93 NS_CTR_PKTS_IN,
94 NS_CTR_PKTS_OUT,
Harald Weltef22ae5a2021-01-30 22:01:28 +010095 NS_CTR_PKTS_OUT_DROP,
Harald Welte3d5eaee2021-01-30 21:33:02 +010096 NS_CTR_BYTES_IN,
97 NS_CTR_BYTES_OUT,
Harald Weltef22ae5a2021-01-30 22:01:28 +010098 NS_CTR_BYTES_OUT_DROP,
Harald Welte3d5eaee2021-01-30 21:33:02 +010099 NS_CTR_BLOCKED,
Harald Welteb40bf8b2021-01-30 22:43:01 +0100100 NS_CTR_UNBLOCKED,
Harald Welte3d5eaee2021-01-30 21:33:02 +0100101 NS_CTR_DEAD,
102 NS_CTR_REPLACED,
103 NS_CTR_NSEI_CHG,
104 NS_CTR_INV_VCI,
105 NS_CTR_INV_NSEI,
106 NS_CTR_LOST_ALIVE,
107 NS_CTR_LOST_RESET,
108};
Alexander Couzens6a161492020-07-12 13:45:50 +0200109
110#define NSE_S_BLOCKED 0x0001
111#define NSE_S_ALIVE 0x0002
112#define NSE_S_RESET 0x0004
113
114#define NS_DESC_B(st) ((st) & NSE_S_BLOCKED ? "BLOCKED" : "UNBLOCKED")
115#define NS_DESC_A(st) ((st) & NSE_S_ALIVE ? "ALIVE" : "DEAD")
116#define NS_DESC_R(st) ((st) & NSE_S_RESET ? "RESET" : "UNRESET")
117
118/*! An instance of the NS protocol stack */
119struct gprs_ns2_inst {
120 /*! callback to the user for incoming UNIT DATA IND */
121 osmo_prim_cb cb;
122
123 /*! callback data */
124 void *cb_data;
125
126 /*! linked lists of all NSVC binds (e.g. IPv4 bind, but could be also E1 */
127 struct llist_head binding;
128
129 /*! linked lists of all NSVC in this instance */
130 struct llist_head nse;
131
Alexander Couzens6a161492020-07-12 13:45:50 +0200132 uint16_t timeout[NS_TIMERS_COUNT];
133
134 /*! workaround for rate counter until rate counter accepts char str as index */
Harald Welte97ccbf72021-01-31 11:49:19 +0100135 uint32_t nsvc_rate_ctr_idx;
Harald Welte76346072021-01-31 11:54:02 +0100136 uint32_t bind_rate_ctr_idx;
Harald Welte53a2fde2020-12-01 22:21:14 +0100137
138 /*! libmnl netlink socket for link state monitoring */
139 struct osmo_mnl *linkmon_mnl;
Alexander Couzens6a161492020-07-12 13:45:50 +0200140};
141
Harald Welte53a2fde2020-12-01 22:21:14 +0100142
Alexander Couzens6a161492020-07-12 13:45:50 +0200143/*! Structure repesenting a NSE. The BSS/PCU will only have a single NSE, while SGSN has one for each BSS/PCU */
144struct gprs_ns2_nse {
145 uint16_t nsei;
146
147 /*! entry back to ns2_inst */
148 struct gprs_ns2_inst *nsi;
149
150 /*! llist entry for gprs_ns2_inst */
151 struct llist_head list;
152
153 /*! llist head to hold all nsvc */
154 struct llist_head nsvc;
155
Alexander Couzens8a2a1a42020-12-26 18:00:17 +0100156 /*! count all active NSVCs */
157 int nsvc_count;
Alexander Couzensfc3dd1f2020-11-19 00:41:47 +0100158
Alexander Couzens6a161492020-07-12 13:45:50 +0200159 /*! true if this NSE was created by VTY or pcu socket) */
160 bool persistent;
161
Alexander Couzensda0a2852020-10-01 23:24:07 +0200162 /*! true if this NSE wasn't yet alive at all.
163 * Will be true after the first status ind with NS_AFF_CAUSE_RECOVERY */
164 bool first;
165
Alexander Couzens6a161492020-07-12 13:45:50 +0200166 /*! true if this NSE has at least one alive VC */
167 bool alive;
168
Alexander Couzensaac90162020-11-19 02:44:04 +0100169 /*! which link-layer are we based on? */
170 enum gprs_ns2_ll ll;
171
Alexander Couzensd923cff2020-12-01 01:03:52 +0100172 /*! which dialect does this NSE speaks? */
173 enum gprs_ns2_dialect dialect;
174
Alexander Couzens6a161492020-07-12 13:45:50 +0200175 struct osmo_fsm_inst *bss_sns_fi;
Harald Welte3221f872021-01-18 14:58:51 +0100176
177 /*! sum of all the data weight of _active_ NS-VCs */
178 uint32_t sum_data_weight;
179
180 /*! sum of all the signalling weight of _active_ NS-VCs */
181 uint32_t sum_sig_weight;
Alexander Couzens6a161492020-07-12 13:45:50 +0200182};
183
184/*! Structure representing a single NS-VC */
185struct gprs_ns2_vc {
186 /*! list of NS-VCs within NSE */
187 struct llist_head list;
188
189 /*! list of NS-VCs within bind, bind is the owner! */
190 struct llist_head blist;
191
192 /*! pointer to NS Instance */
193 struct gprs_ns2_nse *nse;
194
195 /*! pointer to NS VL bind. bind own the memory of this instance */
196 struct gprs_ns2_vc_bind *bind;
197
198 /*! true if this NS was created by VTY or pcu socket) */
199 bool persistent;
200
201 /*! uniquely identifies NS-VC if VC contains nsvci */
202 uint16_t nsvci;
203
204 /*! signalling weight. 0 = don't use for signalling (BVCI == 0)*/
205 uint8_t sig_weight;
206
Alexander Couzensfc3dd1f2020-11-19 00:41:47 +0100207 /*! signalling packet counter for the load sharing function */
208 uint8_t sig_counter;
209
210 /*! data weight. 0 = don't use for user data (BVCI != 0) */
Alexander Couzens6a161492020-07-12 13:45:50 +0200211 uint8_t data_weight;
212
213 /*! can be used by the bind/driver of the virtual circuit. e.g. ipv4/ipv6/frgre/e1 */
214 void *priv;
215
216 bool nsvci_is_valid;
217 bool sns_only;
218
219 struct rate_ctr_group *ctrg;
220 struct osmo_stat_item_group *statg;
221
Alexander Couzens6a161492020-07-12 13:45:50 +0200222 enum gprs_ns2_vc_mode mode;
223
224 struct osmo_fsm_inst *fi;
225};
226
227/*! Structure repesenting a bind instance. E.g. IPv4 listen port. */
228struct gprs_ns2_vc_bind {
Alexander Couzensaaa55a62020-12-03 06:02:03 +0100229 /*! unique name */
230 const char *name;
Alexander Couzens6a161492020-07-12 13:45:50 +0200231 /*! list entry in nsi */
232 struct llist_head list;
233 /*! list of all VC */
234 struct llist_head nsvc;
235 /*! driver private structure */
236 void *priv;
237 /*! a pointer back to the nsi */
238 struct gprs_ns2_inst *nsi;
239 struct gprs_ns2_vc_driver *driver;
240
Alexander Couzensd923cff2020-12-01 01:03:52 +0100241 bool accept_ipaccess;
242 bool accept_sns;
Alexander Couzens6a161492020-07-12 13:45:50 +0200243
Alexander Couzens1c8785d2020-12-17 06:58:53 +0100244 /*! transfer capability in mbit */
245 int transfer_capability;
246
Alexander Couzensaac90162020-11-19 02:44:04 +0100247 /*! which link-layer are we based on? */
248 enum gprs_ns2_ll ll;
249
Alexander Couzens6a161492020-07-12 13:45:50 +0200250 /*! send a msg over a VC */
251 int (*send_vc)(struct gprs_ns2_vc *nsvc, struct msgb *msg);
252
253 /*! free the vc priv data */
254 void (*free_vc)(struct gprs_ns2_vc *nsvc);
Alexander Couzens6a161492020-07-12 13:45:50 +0200255
Alexander Couzens22f34712020-10-02 02:34:39 +0200256 /*! allow to show information for the vty */
257 void (*dump_vty)(const struct gprs_ns2_vc_bind *bind,
258 struct vty *vty, bool stats);
Harald Welte76346072021-01-31 11:54:02 +0100259
Alexander Couzensc4704762021-02-08 23:13:12 +0100260 /*! the IP-SNS signalling weight when doing dynamic configuration */
261 uint8_t sns_sig_weight;
262 /*! the IP-SNS data weight when doing dynamic configuration */
263 uint8_t sns_data_weight;
264
Harald Welte76346072021-01-31 11:54:02 +0100265 struct osmo_stat_item_group *statg;
Alexander Couzens22f34712020-10-02 02:34:39 +0200266};
Alexander Couzens6a161492020-07-12 13:45:50 +0200267
268struct gprs_ns2_vc_driver {
269 const char *name;
270 void *priv;
271 void (*free_bind)(struct gprs_ns2_vc_bind *driver);
272};
273
Alexander Couzensba5a9922021-01-25 16:03:23 +0100274enum ns2_cs ns2_create_vc(struct gprs_ns2_vc_bind *bind,
Alexander Couzens6a161492020-07-12 13:45:50 +0200275 struct msgb *msg,
276 const char *logname,
277 struct msgb **reject,
278 struct gprs_ns2_vc **success);
279
Alexander Couzensffd49d02020-09-24 00:47:17 +0200280int ns2_recv_vc(struct gprs_ns2_vc *nsvc,
Alexander Couzens6a161492020-07-12 13:45:50 +0200281 struct msgb *msg);
282
283struct gprs_ns2_vc *ns2_vc_alloc(struct gprs_ns2_vc_bind *bind,
284 struct gprs_ns2_nse *nse,
Alexander Couzensd923cff2020-12-01 01:03:52 +0100285 bool initiater,
Harald Welte603f4042020-11-29 17:39:19 +0100286 enum gprs_ns2_vc_mode vc_mode,
287 const char *id);
Alexander Couzens6a161492020-07-12 13:45:50 +0200288
Harald Weltec3aa8f92021-01-31 11:41:34 +0100289int ns2_bind_alloc(struct gprs_ns2_inst *nsi, const char *name,
290 struct gprs_ns2_vc_bind **result);
291
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100292struct msgb *ns2_msgb_alloc(void);
Alexander Couzens6a161492020-07-12 13:45:50 +0200293
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100294void ns2_sns_write_vty(struct vty *vty, const struct gprs_ns2_nse *nse);
295void 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 +0200296void ns2_prim_status_ind(struct gprs_ns2_nse *nse,
Daniel Willmann15c09a82020-11-03 23:05:43 +0100297 struct gprs_ns2_vc *nsvc,
Alexander Couzensbf95f0f2020-10-01 22:56:03 +0200298 uint16_t bvci,
Alexander Couzens6a161492020-07-12 13:45:50 +0200299 enum gprs_ns2_affecting_cause cause);
300void ns2_nse_notify_alive(struct gprs_ns2_vc *nsvc, bool alive);
301
302/* message */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100303int ns2_validate(struct gprs_ns2_vc *nsvc,
304 uint8_t pdu_type,
305 struct msgb *msg,
306 struct tlv_parsed *tp,
307 uint8_t *cause);
Alexander Couzens6a161492020-07-12 13:45:50 +0200308
309/* SNS messages */
310int ns2_tx_sns_ack(struct gprs_ns2_vc *nsvc, uint8_t trans_id, uint8_t *cause,
311 const struct gprs_ns_ie_ip4_elem *ip4_elems,
312 unsigned int num_ip4_elems,
313 const struct gprs_ns_ie_ip6_elem *ip6_elems,
314 unsigned int num_ip6_elems);
315int ns2_tx_sns_config(struct gprs_ns2_vc *nsvc, bool end_flag,
316 const struct gprs_ns_ie_ip4_elem *ip4_elems,
317 unsigned int num_ip4_elems,
318 const struct gprs_ns_ie_ip6_elem *ip6_elems,
319 unsigned int num_ip6_elems);
320int ns2_tx_sns_config_ack(struct gprs_ns2_vc *nsvc, uint8_t *cause);
321int ns2_tx_sns_size(struct gprs_ns2_vc *nsvc, bool reset_flag, uint16_t max_nr_nsvc,
322 int ip4_ep_nr, int ip6_ep_nr);
323int ns2_tx_sns_size_ack(struct gprs_ns2_vc *nsvc, uint8_t *cause);
324
325/* transmit message over a VC */
326int ns2_tx_block(struct gprs_ns2_vc *nsvc, uint8_t cause);
327int ns2_tx_block_ack(struct gprs_ns2_vc *nsvc);
328
329int ns2_tx_reset(struct gprs_ns2_vc *nsvc, uint8_t cause);
330int ns2_tx_reset_ack(struct gprs_ns2_vc *nsvc);
331
332int ns2_tx_unblock(struct gprs_ns2_vc *nsvc);
333int ns2_tx_unblock_ack(struct gprs_ns2_vc *nsvc);
334
335int ns2_tx_alive(struct gprs_ns2_vc *nsvc);
336int ns2_tx_alive_ack(struct gprs_ns2_vc *nsvc);
337
338int ns2_tx_unit_data(struct gprs_ns2_vc *nsvc,
339 uint16_t bvci, uint8_t sducontrol,
340 struct msgb *msg);
341
342int ns2_tx_status(struct gprs_ns2_vc *nsvc, uint8_t cause,
343 uint16_t bvci, struct msgb *orig_msg);
344
345/* driver */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100346struct gprs_ns2_vc *ns2_ip_bind_connect(struct gprs_ns2_vc_bind *bind,
347 struct gprs_ns2_nse *nse,
348 const struct osmo_sockaddr *remote);
Alexander Couzense769f522020-12-07 07:37:07 +0100349int ns2_ip_count_bind(struct gprs_ns2_inst *nsi, struct osmo_sockaddr *remote);
350struct gprs_ns2_vc_bind *ns2_ip_get_bind_by_index(struct gprs_ns2_inst *nsi,
351 struct osmo_sockaddr *remote,
352 int index);
Alexander Couzens6a161492020-07-12 13:45:50 +0200353
354/* sns */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100355int ns2_sns_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp);
Alexander Couzens6a161492020-07-12 13:45:50 +0200356struct osmo_fsm_inst *ns2_sns_bss_fsm_alloc(struct gprs_ns2_nse *nse,
357 const char *id);
Alexander Couzensbe7cecc2021-02-03 18:25:27 +0100358void ns2_sns_replace_nsvc(struct gprs_ns2_vc *nsvc);
359void ns2_sns_notify_alive(struct gprs_ns2_nse *nse, struct gprs_ns2_vc *nsvc, bool alive);
Alexander Couzensc4704762021-02-08 23:13:12 +0100360void ns2_sns_update_weights(struct gprs_ns2_vc_bind *bind);
Alexander Couzens6a161492020-07-12 13:45:50 +0200361
362/* vc */
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100363struct osmo_fsm_inst *ns2_vc_fsm_alloc(struct gprs_ns2_vc *nsvc,
Alexander Couzens6a161492020-07-12 13:45:50 +0200364 const char *id, bool initiate);
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100365int ns2_vc_fsm_start(struct gprs_ns2_vc *nsvc);
366int ns2_vc_force_unconfigured(struct gprs_ns2_vc *nsvc);
367int ns2_vc_rx(struct gprs_ns2_vc *nsvc, struct msgb *msg, struct tlv_parsed *tp);
368int ns2_vc_is_alive(struct gprs_ns2_vc *nsvc);
369int ns2_vc_is_unblocked(struct gprs_ns2_vc *nsvc);
Alexander Couzens47afc422021-01-17 20:12:46 +0100370int ns2_vc_block(struct gprs_ns2_vc *nsvc);
371int ns2_vc_unblock(struct gprs_ns2_vc *nsvc);
Alexander Couzens6a161492020-07-12 13:45:50 +0200372
Alexander Couzens6a161492020-07-12 13:45:50 +0200373/* nse */
374void ns2_nse_notify_unblocked(struct gprs_ns2_vc *nsvc, bool unblocked);
Alexander Couzens8dfc24c2021-01-25 16:09:23 +0100375enum gprs_ns2_vc_mode ns2_dialect_to_vc_mode(enum gprs_ns2_dialect dialect);
Alexander Couzens1c8785d2020-12-17 06:58:53 +0100376int ns2_count_transfer_cap(struct gprs_ns2_nse *nse,
377 uint16_t bvci);