blob: 975ddcb67b6f47e1977494d49727ccd6ef1d14ac [file] [log] [blame]
Alexander Couzens6a161492020-07-12 13:45:50 +02001/*! \file gprs_ns2.h */
2
3
4#pragma once
5
6#include <stdint.h>
7#include <netinet/in.h>
8
9#include <osmocom/core/prim.h>
10
11struct osmo_sockaddr;
Alexander Couzens1fac6f72020-10-01 19:08:38 +020012struct osmo_sockaddr_str;
Alexander Couzens6a161492020-07-12 13:45:50 +020013
14struct gprs_ns2_inst;
15struct gprs_ns2_nse;
16struct gprs_ns2_vc;
17struct gprs_ns2_vc_bind;
18struct gprs_ns2_vc_driver;
19struct gprs_ns_ie_ip4_elem;
20struct gprs_ns_ie_ip6_elem;
21
22enum gprs_ns2_vc_mode {
Harald Weltefa2d66c2020-10-17 13:09:34 +020023 /*! The VC will use RESET/BLOCK/UNBLOCK to start the connection and do ALIVE/ACK.
24 * This is what is needed for Frame Relay transport, and if you use a R97/R99 Gb
25 * interface over an IP transport (never standardized by 3GPP) */
26 NS2_VC_MODE_BLOCKRESET,
27 /*! The VC will only use ALIVE/ACK (no RESET/BLOCK/UNBLOCK), which is for Gb-IP
28 * interface compliant to 3GPP Rel=4 or later. */
29 NS2_VC_MODE_ALIVE,
Alexander Couzens6a161492020-07-12 13:45:50 +020030};
31
32/*! Osmocom NS primitives according to 48.016 5.2 Service primitves */
33enum gprs_ns2_prim {
34 PRIM_NS_UNIT_DATA,
35 PRIM_NS_CONGESTION,
36 PRIM_NS_STATUS,
37};
38
Alexander Couzens2498f1d2020-10-27 01:09:01 +010039extern const struct value_string ns2_prim_strs[];
40
41/*! Obtain a human-readable string for NS primitives */
42static inline const char *gprs_ns2_prim_str(enum gprs_ns2_prim val)
43{ return get_value_string(ns2_prim_strs, val); }
44
Alexander Couzens6a161492020-07-12 13:45:50 +020045/*! Osmocom NS primitives according to 48.016 5.2.2.4 Service primitves */
46enum gprs_ns2_congestion_cause {
47 NS_CONG_CAUSE_BACKWARD_BEGIN,
48 NS_CONG_CAUSE_BACKWARD_END,
49 NS_CONG_CAUSE_FORWARD_BEGIN,
50 NS_CONG_CAUSE_FORWARD_END,
51};
52
53/*! Osmocom NS primitives according to 48.016 5.2.2.6 Service primitves */
54enum gprs_ns2_affecting_cause {
55 NS_AFF_CAUSE_VC_FAILURE,
56 NS_AFF_CAUSE_VC_RECOVERY,
57 NS_AFF_CAUSE_FAILURE,
58 NS_AFF_CAUSE_RECOVERY,
59 /* osmocom own causes */
60 NS_AFF_CAUSE_SNS_CONFIGURED,
61 NS_AFF_CAUSE_SNS_FAILURE,
62};
63
Alexander Couzens2498f1d2020-10-27 01:09:01 +010064extern const struct value_string gprs_ns2_aff_cause_prim_strs[];
65
66/*! Obtain a human-readable string for NS affective cause in primitives */
67static inline const char *gprs_ns2_aff_cause_prim_str(enum gprs_ns2_affecting_cause val)
68{ return get_value_string(gprs_ns2_aff_cause_prim_strs, val); }
69
Alexander Couzens6a161492020-07-12 13:45:50 +020070/*! Osmocom NS primitives according to 48.016 5.2.2.7 Service primitves */
71enum gprs_ns2_change_ip_endpoint {
72 NS_ENDPOINT_NO_CHANGE,
73 NS_ENDPOINT_REQUEST_CHANGE,
74 NS_ENDPOINT_CONFIRM_CHANGE,
75};
76
77struct osmo_gprs_ns2_prim {
78 struct osmo_prim_hdr oph;
79
80 uint16_t nsei;
81 uint16_t bvci;
82
83 union {
84 struct {
85 enum gprs_ns2_change_ip_endpoint change;
86 /* TODO: implement resource distribution
87 * add place holder for the link selector */
88 long long _resource_distribution_placeholder1;
89 long long _resource_distribution_placeholder2;
90 long long _resource_distribution_placeholder3;
91 } unitdata;
92 struct {
93 enum gprs_ns2_congestion_cause cause;
94 } congestion;
95 struct {
96 enum gprs_ns2_affecting_cause cause;
97 /* 48.016 5.2.2.6 transfer capability */
98 int transfer;
Alexander Couzensda0a2852020-10-01 23:24:07 +020099 /* osmocom specific */
100 /* Persistent NSE/NSVC are configured by vty */
101 bool persistent;
102 /* Only true on the first time it's available.
103 * Allow the BSSGP layer to reset persistent NSE */
104 bool first;
Alexander Couzens6a161492020-07-12 13:45:50 +0200105 } status;
106 } u;
107};
108
109/* instance */
110struct gprs_ns2_inst *gprs_ns2_instantiate(void *ctx, osmo_prim_cb cb, void *cb_data);
111void gprs_ns2_free(struct gprs_ns2_inst *inst);
112int gprs_ns2_dynamic_create_nse(struct gprs_ns2_inst *nsi, bool create_nse);
113
114/* Entrypoint for primitives from the NS USER */
115int gprs_ns2_recv_prim(struct gprs_ns2_inst *nsi, struct osmo_prim_hdr *oph);
116
Alexander Couzens6cb5d5f2020-10-11 23:23:31 +0200117/*! a callback to iterate over all NSVC */
118typedef int (*gprs_ns2_foreach_nsvc_cb)(struct gprs_ns2_vc *nsvc, void *ctx);
119
120int gprs_ns2_nse_foreach_nsvc(struct gprs_ns2_nse *nse,
121 gprs_ns2_foreach_nsvc_cb cb, void *cb_data);
Alexander Couzens6a161492020-07-12 13:45:50 +0200122struct gprs_ns2_nse *gprs_ns2_nse_by_nsei(struct gprs_ns2_inst *nsi, uint16_t nsei);
123struct gprs_ns2_nse *gprs_ns2_create_nse(struct gprs_ns2_inst *nsi, uint16_t nsei);
Alexander Couzens05e7f7d2020-10-11 19:51:46 +0200124uint16_t gprs_ns2_nse_nsei(struct gprs_ns2_nse *nse);
Alexander Couzens6a161492020-07-12 13:45:50 +0200125void gprs_ns2_free_nse(struct gprs_ns2_nse *nse);
Alexander Couzens4b6c8af2020-10-11 20:15:25 +0200126void gprs_ns2_free_nses(struct gprs_ns2_inst *nsi);
Alexander Couzens6a161492020-07-12 13:45:50 +0200127
128/* create vc */
129void gprs_ns2_free_nsvc(struct gprs_ns2_vc *nsvc);
130struct gprs_ns2_vc *gprs_ns2_nsvc_by_nsvci(struct gprs_ns2_inst *nsi, uint16_t nsvci);
131
132/* IP VL driver */
133int gprs_ns2_ip_bind(struct gprs_ns2_inst *nsi,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700134 const struct osmo_sockaddr *local,
Alexander Couzens6a161492020-07-12 13:45:50 +0200135 int dscp,
136 struct gprs_ns2_vc_bind **result);
Alexander Couzens4f608452020-10-11 18:41:24 +0200137struct gprs_ns2_vc_bind *gprs_ns2_ip_bind_by_sockaddr(struct gprs_ns2_inst *nsi,
138 const struct osmo_sockaddr *sockaddr);
Alexander Couzens6a161492020-07-12 13:45:50 +0200139void gprs_ns2_bind_set_mode(struct gprs_ns2_vc_bind *bind, enum gprs_ns2_vc_mode mode);
140
141/* create a VC connection */
142struct gprs_ns2_vc *gprs_ns2_ip_connect(struct gprs_ns2_vc_bind *bind,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700143 const struct osmo_sockaddr *remote,
Alexander Couzens6a161492020-07-12 13:45:50 +0200144 struct gprs_ns2_nse *nse,
145 uint16_t nsvci);
146
147struct gprs_ns2_vc *gprs_ns2_ip_connect2(struct gprs_ns2_vc_bind *bind,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700148 const struct osmo_sockaddr *remote,
Alexander Couzens6a161492020-07-12 13:45:50 +0200149 uint16_t nsei,
150 uint16_t nsvci);
151struct gprs_ns2_vc *gprs_ns2_ip_connect_inactive(struct gprs_ns2_vc_bind *bind,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700152 const struct osmo_sockaddr *remote,
Alexander Couzens6a161492020-07-12 13:45:50 +0200153 struct gprs_ns2_nse *nse,
154 uint16_t nsvci);
155
156void gprs_ns2_free_bind(struct gprs_ns2_vc_bind *bind);
Alexander Couzens896fcd52020-10-11 19:52:36 +0200157void gprs_ns2_free_binds(struct gprs_ns2_inst *nsi);
Alexander Couzens6a161492020-07-12 13:45:50 +0200158
159/* create a VC SNS connection */
160int gprs_ns2_ip_connect_sns(struct gprs_ns2_vc_bind *bind,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700161 const struct osmo_sockaddr *remote,
Alexander Couzens6a161492020-07-12 13:45:50 +0200162 uint16_t nsei);
Alexander Couzens125298f2020-10-11 21:22:42 +0200163const struct osmo_sockaddr *gprs_ns2_nse_sns_remote(struct gprs_ns2_nse *nse);
Alexander Couzens6a161492020-07-12 13:45:50 +0200164
Alexander Couzensd33512b2020-10-11 21:42:11 +0200165const struct osmo_sockaddr *gprs_ns2_ip_vc_remote(const struct gprs_ns2_vc *nsvc);
Alexander Couzens979f5f52020-10-11 21:01:48 +0200166const struct osmo_sockaddr *gprs_ns2_ip_vc_local(const struct gprs_ns2_vc *nsvc);
Alexander Couzensd420ea92020-10-12 01:11:05 +0200167bool gprs_ns2_ip_vc_equal(const struct gprs_ns2_vc *nsvc,
168 const struct osmo_sockaddr *local,
169 const struct osmo_sockaddr *remote,
170 uint16_t nsvci);
Alexander Couzens9a4cf272020-10-11 20:48:04 +0200171const struct osmo_sockaddr *gprs_ns2_ip_bind_sockaddr(struct gprs_ns2_vc_bind *bind);
Alexander Couzens6a161492020-07-12 13:45:50 +0200172int gprs_ns2_is_ip_bind(struct gprs_ns2_vc_bind *bind);
173int gprs_ns2_ip_bind_set_dscp(struct gprs_ns2_vc_bind *bind, int dscp);
Alexander Couzens38b19e82020-09-23 23:56:37 +0200174struct gprs_ns2_vc *gprs_ns2_nsvc_by_sockaddr_bind(
Alexander Couzens6a161492020-07-12 13:45:50 +0200175 struct gprs_ns2_vc_bind *bind,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700176 const struct osmo_sockaddr *saddr);
Alexander Couzens6a161492020-07-12 13:45:50 +0200177
178int gprs_ns2_frgre_bind(struct gprs_ns2_inst *nsi,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700179 const struct osmo_sockaddr *local,
Alexander Couzens6a161492020-07-12 13:45:50 +0200180 int dscp,
181 struct gprs_ns2_vc_bind **result);
182int gprs_ns2_is_frgre_bind(struct gprs_ns2_vc_bind *bind);
183
Alexander Couzens38b19e82020-09-23 23:56:37 +0200184struct gprs_ns2_vc *gprs_ns2_nsvc_by_sockaddr_nse(
185 struct gprs_ns2_nse *nse,
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700186 const struct osmo_sockaddr *sockaddr);
Alexander Couzens6a161492020-07-12 13:45:50 +0200187void gprs_ns2_start_alive_all_nsvcs(struct gprs_ns2_nse *nse);
188const char *gprs_ns2_cause_str(int cause);
189const char *gprs_ns2_ll_str(struct gprs_ns2_vc *nsvc);
190char *gprs_ns2_ll_str_buf(char *buf, size_t buf_len, struct gprs_ns2_vc *nsvc);
191char *gprs_ns2_ll_str_c(const void *ctx, struct gprs_ns2_vc *nsvc);
192
193/* vty */
Vadim Yanitskiya07f25e2020-10-09 21:47:01 +0700194int gprs_ns2_vty_init(struct gprs_ns2_inst *nsi,
195 const struct osmo_sockaddr_str *default_bind);
Alexander Couzens6a161492020-07-12 13:45:50 +0200196int gprs_ns2_vty_create();
Daniel Willmann4fb27a82020-09-25 15:39:46 +0200197void gprs_ns2_vty_force_vc_mode(bool force, enum gprs_ns2_vc_mode mode, const char *reason);
Alexander Couzens6a161492020-07-12 13:45:50 +0200198
199
200/*! @} */