blob: 43eb0beaab70f57b26d939234492f2d6c1aa3a9e [file] [log] [blame]
Harald Welteb77c6972010-05-01 11:28:43 +02001#ifndef _GB_PROXY_H
2#define _GB_PROXY_H
3
Harald Welteb77c6972010-05-01 11:28:43 +02004
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +01005#include <osmocom/core/msgb.h>
Harald Welteb77c6972010-05-01 11:28:43 +02006
Harald Welteea34a4e2012-06-16 14:59:56 +08007#include <osmocom/gprs/gprs_ns.h>
Harald Welte4b037e42010-05-19 19:45:32 +02008#include <osmocom/vty/command.h>
Harald Welteb77c6972010-05-01 11:28:43 +02009
Holger Hans Peter Freyther3fa26442014-08-04 16:27:11 +020010#include <sys/types.h>
11#include <regex.h>
12
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020013struct rate_ctr_group;
Jacob Erlbeck9114bee2014-08-19 12:21:01 +020014struct gprs_gb_parse_context;
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +020015struct tlv_parsed;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020016
Jacob Erlbeck9114bee2014-08-19 12:21:01 +020017enum gbproxy_global_ctr {
18 GBPROX_GLOB_CTR_INV_BVCI,
19 GBPROX_GLOB_CTR_INV_LAI,
20 GBPROX_GLOB_CTR_INV_RAI,
21 GBPROX_GLOB_CTR_INV_NSEI,
22 GBPROX_GLOB_CTR_PROTO_ERR_BSS,
23 GBPROX_GLOB_CTR_PROTO_ERR_SGSN,
24 GBPROX_GLOB_CTR_NOT_SUPPORTED_BSS,
25 GBPROX_GLOB_CTR_NOT_SUPPORTED_SGSN,
26 GBPROX_GLOB_CTR_RESTART_RESET_SGSN,
27 GBPROX_GLOB_CTR_TX_ERR_SGSN,
28 GBPROX_GLOB_CTR_OTHER_ERR,
29 GBPROX_GLOB_CTR_PATCH_PEER_ERR,
30};
31
32enum gbproxy_peer_ctr {
33 GBPROX_PEER_CTR_BLOCKED,
34 GBPROX_PEER_CTR_UNBLOCKED,
35 GBPROX_PEER_CTR_DROPPED,
36 GBPROX_PEER_CTR_INV_NSEI,
37 GBPROX_PEER_CTR_TX_ERR,
38 GBPROX_PEER_CTR_RAID_PATCHED_BSS,
39 GBPROX_PEER_CTR_RAID_PATCHED_SGSN,
40 GBPROX_PEER_CTR_APN_PATCHED,
41 GBPROX_PEER_CTR_TLLI_PATCHED_BSS,
42 GBPROX_PEER_CTR_TLLI_PATCHED_SGSN,
43 GBPROX_PEER_CTR_PTMSI_PATCHED_BSS,
44 GBPROX_PEER_CTR_PTMSI_PATCHED_SGSN,
45 GBPROX_PEER_CTR_PATCH_CRYPT_ERR,
46 GBPROX_PEER_CTR_PATCH_ERR,
47 GBPROX_PEER_CTR_ATTACH_REQS,
48 GBPROX_PEER_CTR_ATTACH_REJS,
49 GBPROX_PEER_CTR_TLLI_UNKNOWN,
50 GBPROX_PEER_CTR_TLLI_CACHE_SIZE,
51};
52
Harald Welteb77c6972010-05-01 11:28:43 +020053struct gbproxy_config {
54 /* parsed from config file */
Holger Hans Peter Freytherc42ad8b2011-04-18 17:04:00 +020055 uint16_t nsip_sgsn_nsei;
Harald Welteb77c6972010-05-01 11:28:43 +020056
57 /* misc */
58 struct gprs_ns_inst *nsi;
Jacob Erlbeck67a44452014-05-19 10:14:58 +020059
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020060 /* Linked list of all Gb peers (except SGSN) */
61 struct llist_head bts_peers;
62
Holger Hans Peter Freythera7027a02014-08-04 11:19:56 +020063 /* Counter */
64 struct rate_ctr_group *ctrg;
65
Jacob Erlbeck67a44452014-05-19 10:14:58 +020066 /* force mcc/mnc */
67 int core_mnc;
68 int core_mcc;
Jacob Erlbeck73685282014-05-23 20:48:07 +020069 uint8_t* core_apn;
70 size_t core_apn_size;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +020071 char * match_re;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +020072 int tlli_max_age;
73 int tlli_max_len;
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +020074
75 /* Experimental config */
Jacob Erlbeck9057bc32014-08-12 16:30:30 +020076 int patch_ptmsi;
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +020077 int acquire_imsi;
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +020078 int route_to_sgsn2;
79 uint16_t nsip_sgsn2_nsei;
Holger Hans Peter Freyther3fa26442014-08-04 16:27:11 +020080
81 /* IMSI checking/matching */
82 int check_imsi;
83 regex_t imsi_re_comp;
Jacob Erlbeck0d376712014-08-11 19:12:24 +020084
85 /* Used to generate identifiers */
86 unsigned bss_ptmsi_state;
87 unsigned sgsn_tlli_state;
Harald Welteb77c6972010-05-01 11:28:43 +020088};
89
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +020090struct gbproxy_patch_state {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020091 int local_mnc;
92 int local_mcc;
93
94 /* List of TLLIs for which patching is enabled */
95 struct llist_head enabled_tllis;
96 int enabled_tllis_count;
97};
98
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +020099struct gbproxy_peer {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200100 struct llist_head list;
101
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200102 /* point back to the config */
103 struct gbproxy_config *cfg;
104
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200105 /* NSEI of the peer entity */
106 uint16_t nsei;
107
108 /* BVCI used for Point-to-Point to this peer */
109 uint16_t bvci;
110 int blocked;
111
112 /* Routeing Area that this peer is part of (raw 04.08 encoding) */
113 uint8_t ra[6];
114
115 /* Counter */
116 struct rate_ctr_group *ctrg;
117
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200118 struct gbproxy_patch_state patch_state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200119};
120
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200121struct gbproxy_tlli_state {
122 uint32_t current;
123 uint32_t assigned;
124 int bss_validated;
125 int net_validated;
126
127 uint32_t ptmsi;
128};
129
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200130struct gbproxy_tlli_info {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200131 struct llist_head list;
132
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200133 struct gbproxy_tlli_state tlli;
134 struct gbproxy_tlli_state sgsn_tlli;
135
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200136 time_t timestamp;
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200137 uint8_t *imsi;
138 size_t imsi_len;
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200139
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200140 int imsi_acq_pending;
141 struct llist_head stored_msgs;
Jacob Erlbeck31591142014-09-03 11:59:48 +0200142 int imsi_acq_retries;
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200143
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200144 int enable_patching;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200145};
146
147
Harald Welteb77c6972010-05-01 11:28:43 +0200148/* gb_proxy_vty .c */
149
150int gbproxy_vty_init(void);
151int gbproxy_parse_config(const char *config_file, struct gbproxy_config *cfg);
152
153
154/* gb_proxy.c */
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200155int gbproxy_init_config(struct gbproxy_config *cfg);
Harald Welteb77c6972010-05-01 11:28:43 +0200156
157/* Main input function for Gb proxy */
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200158int gbprox_rcvmsg(struct gbproxy_config *cfg, struct msgb *msg, uint16_t nsei, uint16_t ns_bvci, uint16_t nsvci);
Harald Welteb77c6972010-05-01 11:28:43 +0200159
Harald Weltec1c1dd22010-05-11 06:34:24 +0200160int gbprox_signal(unsigned int subsys, unsigned int signal,
161 void *handler_data, void *signal_data);
Harald Welte1ccbf442010-05-14 11:53:08 +0000162
163/* Reset all persistent NS-VC's */
164int gbprox_reset_persistent_nsvcs(struct gprs_ns_inst *nsi);
165
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200166void gbprox_reset(struct gbproxy_config *cfg);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200167
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200168/* TLLI state handling */
169void gbproxy_delete_tllis(struct gbproxy_peer *peer);
Jacob Erlbeck18a37872014-09-08 09:59:16 +0200170int gbproxy_check_tlli(
171 struct gbproxy_peer *peer,
172 struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200173struct gbproxy_tlli_info *gbprox_find_tlli_by_ptmsi(
174 struct gbproxy_peer *peer,
175 uint32_t ptmsi);
176uint32_t gbproxy_map_tlli(
177 uint32_t other_tlli, struct gbproxy_tlli_info *tlli_info, int to_bss);
178struct gbproxy_tlli_info *gbproxy_update_tlli_state_ul(
179 struct gbproxy_peer *peer, time_t now,
180 struct gprs_gb_parse_context *parse_ctx);
181struct gbproxy_tlli_info *gbproxy_update_tlli_state_dl(
182 struct gbproxy_peer *peer, time_t now,
183 struct gprs_gb_parse_context *parse_ctx);
184void gbproxy_update_tlli_state_after(
185 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
186 time_t now, struct gprs_gb_parse_context *parse_ctx);
187int gbproxy_remove_stale_tllis(struct gbproxy_peer *peer, time_t now);
188void gbproxy_delete_tlli(struct gbproxy_peer *peer,
189 struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck31591142014-09-03 11:59:48 +0200190void gbproxy_tlli_info_discard_messages(struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200191
192struct gbproxy_tlli_info *gbproxy_register_tlli(
193 struct gbproxy_peer *peer, uint32_t tlli,
194 const uint8_t *imsi, size_t imsi_len, time_t now);
195
196struct gbproxy_tlli_info *gbproxy_find_tlli(
197 struct gbproxy_peer *peer, uint32_t tlli);
Jacob Erlbeck2fd1ba42014-09-11 14:57:03 +0200198struct gbproxy_tlli_info *gbproxy_find_tlli_by_imsi(
199 struct gbproxy_peer *peer, const uint8_t *imsi, size_t imsi_len);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200200struct gbproxy_tlli_info *gbproxy_find_tlli_by_sgsn_tlli(
201 struct gbproxy_peer *peer,
202 uint32_t tlli);
203struct gbproxy_tlli_info *gbproxy_find_tlli_by_ptmsi(
204 struct gbproxy_peer *peer,
205 uint32_t ptmsi);
206
207/* needed by gb_proxy_tlli.h */
208uint32_t gbproxy_make_bss_ptmsi(struct gbproxy_peer *peer, uint32_t sgsn_ptmsi);
209uint32_t gbproxy_make_sgsn_tlli(
210 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
211 uint32_t bss_tlli);
212int gbproxy_check_imsi(
213 struct gbproxy_peer *peer, const uint8_t *imsi, size_t imsi_len);
214
215/* Message patching */
216void gbproxy_patch_bssgp(
217 struct msgb *msg, uint8_t *bssgp, size_t bssgp_len,
218 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200219 int *len_change, struct gprs_gb_parse_context *parse_ctx);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200220
221int gbproxy_patch_llc(
222 struct msgb *msg, uint8_t *llc, size_t llc_len,
223 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200224 int *len_change, struct gprs_gb_parse_context *parse_ctx);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200225
226int gbproxy_set_patch_filter(
227 struct gbproxy_config *cfg, const char *filter, const char **err_msg);
228void gbproxy_clear_patch_filter(struct gbproxy_config *cfg);
229int gbproxy_check_imsi(
230 struct gbproxy_peer *peer, const uint8_t *imsi, size_t imsi_len);
231
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200232/* Peer handling */
233struct gbproxy_peer *gbproxy_peer_by_bvci(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200234 struct gbproxy_config *cfg, uint16_t bvci);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200235struct gbproxy_peer *gbproxy_peer_by_nsei(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200236 struct gbproxy_config *cfg, uint16_t nsei);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200237struct gbproxy_peer *gbproxy_peer_by_rai(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200238 struct gbproxy_config *cfg, const uint8_t *ra);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200239struct gbproxy_peer *gbproxy_peer_by_lai(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200240 struct gbproxy_config *cfg, const uint8_t *la);
Jacob Erlbeck7fb26c22014-09-04 11:08:50 +0200241struct gbproxy_peer *gbproxy_peer_by_lac(
242 struct gbproxy_config *cfg, const uint8_t *la);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200243struct gbproxy_peer *gbproxy_peer_by_bssgp_tlv(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200244 struct gbproxy_config *cfg, struct tlv_parsed *tp);
245struct gbproxy_peer *gbproxy_peer_alloc(struct gbproxy_config *cfg, uint16_t bvci);
246void gbproxy_peer_free(struct gbproxy_peer *peer);
247int gbproxy_cleanup_peers(struct gbproxy_config *cfg, uint16_t nsei, uint16_t bvci);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200248
Harald Welteb77c6972010-05-01 11:28:43 +0200249#endif