blob: 232fc2812aea7900d199812b16ff00d94986feeb [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 Erlbeck91fb6802014-05-28 10:59:10 +020017enum gbproxy_patch_mode {
18 GBPROX_PATCH_DEFAULT,
19 GBPROX_PATCH_BSSGP, /*!< BSGGP messages only */
20 GBPROX_PATCH_LLC_ATTACH_REQ, /*!< BSSGP and Attach Request */
21 GBPROX_PATCH_LLC_ATTACH, /*!< BSSGP and Attach Request/Response */
22 GBPROX_PATCH_LLC_GMM, /*!< BSSGP and all GMM msgs */
Jacob Erlbeck73685282014-05-23 20:48:07 +020023 GBPROX_PATCH_LLC_GSM, /*!< BSSGP and all GMM and GSM msgs */
Jacob Erlbeck91fb6802014-05-28 10:59:10 +020024 GBPROX_PATCH_LLC, /*!< BSSGP and all supported LLC msgs */
25};
26
Jacob Erlbeck9114bee2014-08-19 12:21:01 +020027enum gbproxy_global_ctr {
28 GBPROX_GLOB_CTR_INV_BVCI,
29 GBPROX_GLOB_CTR_INV_LAI,
30 GBPROX_GLOB_CTR_INV_RAI,
31 GBPROX_GLOB_CTR_INV_NSEI,
32 GBPROX_GLOB_CTR_PROTO_ERR_BSS,
33 GBPROX_GLOB_CTR_PROTO_ERR_SGSN,
34 GBPROX_GLOB_CTR_NOT_SUPPORTED_BSS,
35 GBPROX_GLOB_CTR_NOT_SUPPORTED_SGSN,
36 GBPROX_GLOB_CTR_RESTART_RESET_SGSN,
37 GBPROX_GLOB_CTR_TX_ERR_SGSN,
38 GBPROX_GLOB_CTR_OTHER_ERR,
39 GBPROX_GLOB_CTR_PATCH_PEER_ERR,
40};
41
42enum gbproxy_peer_ctr {
43 GBPROX_PEER_CTR_BLOCKED,
44 GBPROX_PEER_CTR_UNBLOCKED,
45 GBPROX_PEER_CTR_DROPPED,
46 GBPROX_PEER_CTR_INV_NSEI,
47 GBPROX_PEER_CTR_TX_ERR,
48 GBPROX_PEER_CTR_RAID_PATCHED_BSS,
49 GBPROX_PEER_CTR_RAID_PATCHED_SGSN,
50 GBPROX_PEER_CTR_APN_PATCHED,
51 GBPROX_PEER_CTR_TLLI_PATCHED_BSS,
52 GBPROX_PEER_CTR_TLLI_PATCHED_SGSN,
53 GBPROX_PEER_CTR_PTMSI_PATCHED_BSS,
54 GBPROX_PEER_CTR_PTMSI_PATCHED_SGSN,
55 GBPROX_PEER_CTR_PATCH_CRYPT_ERR,
56 GBPROX_PEER_CTR_PATCH_ERR,
57 GBPROX_PEER_CTR_ATTACH_REQS,
58 GBPROX_PEER_CTR_ATTACH_REJS,
59 GBPROX_PEER_CTR_TLLI_UNKNOWN,
60 GBPROX_PEER_CTR_TLLI_CACHE_SIZE,
61};
62
Harald Welteb77c6972010-05-01 11:28:43 +020063struct gbproxy_config {
64 /* parsed from config file */
Holger Hans Peter Freytherc42ad8b2011-04-18 17:04:00 +020065 uint16_t nsip_sgsn_nsei;
Harald Welteb77c6972010-05-01 11:28:43 +020066
67 /* misc */
68 struct gprs_ns_inst *nsi;
Jacob Erlbeck67a44452014-05-19 10:14:58 +020069
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +020070 /* Linked list of all Gb peers (except SGSN) */
71 struct llist_head bts_peers;
72
Holger Hans Peter Freythera7027a02014-08-04 11:19:56 +020073 /* Counter */
74 struct rate_ctr_group *ctrg;
75
Jacob Erlbeck67a44452014-05-19 10:14:58 +020076 /* force mcc/mnc */
77 int core_mnc;
78 int core_mcc;
Jacob Erlbeck73685282014-05-23 20:48:07 +020079 uint8_t* core_apn;
80 size_t core_apn_size;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +020081 char * match_re;
Jacob Erlbeck91fb6802014-05-28 10:59:10 +020082 enum gbproxy_patch_mode patch_mode;
Jacob Erlbeck7c101d92014-06-06 18:49:23 +020083 int tlli_max_age;
84 int tlli_max_len;
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +020085
86 /* Experimental config */
Jacob Erlbeck9057bc32014-08-12 16:30:30 +020087 int patch_ptmsi;
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +020088 int acquire_imsi;
Jacob Erlbeckf4d60c82014-08-26 14:47:15 +020089 int route_to_sgsn2;
90 uint16_t nsip_sgsn2_nsei;
Holger Hans Peter Freyther3fa26442014-08-04 16:27:11 +020091
92 /* IMSI checking/matching */
93 int check_imsi;
94 regex_t imsi_re_comp;
Jacob Erlbeck0d376712014-08-11 19:12:24 +020095
96 /* Used to generate identifiers */
97 unsigned bss_ptmsi_state;
98 unsigned sgsn_tlli_state;
Harald Welteb77c6972010-05-01 11:28:43 +020099};
100
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200101struct gbproxy_patch_state {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200102 int local_mnc;
103 int local_mcc;
104
105 /* List of TLLIs for which patching is enabled */
106 struct llist_head enabled_tllis;
107 int enabled_tllis_count;
108};
109
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200110struct gbproxy_peer {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200111 struct llist_head list;
112
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200113 /* point back to the config */
114 struct gbproxy_config *cfg;
115
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200116 /* NSEI of the peer entity */
117 uint16_t nsei;
118
119 /* BVCI used for Point-to-Point to this peer */
120 uint16_t bvci;
121 int blocked;
122
123 /* Routeing Area that this peer is part of (raw 04.08 encoding) */
124 uint8_t ra[6];
125
126 /* Counter */
127 struct rate_ctr_group *ctrg;
128
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200129 struct gbproxy_patch_state patch_state;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200130};
131
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200132struct gbproxy_tlli_state {
133 uint32_t current;
134 uint32_t assigned;
135 int bss_validated;
136 int net_validated;
137
138 uint32_t ptmsi;
139};
140
Holger Hans Peter Freyther1ddd9e52014-08-04 11:35:32 +0200141struct gbproxy_tlli_info {
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200142 struct llist_head list;
143
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200144 struct gbproxy_tlli_state tlli;
145 struct gbproxy_tlli_state sgsn_tlli;
146
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200147 time_t timestamp;
148 uint8_t *mi_data;
149 size_t mi_data_len;
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200150
Jacob Erlbeck5f4ef322014-08-22 17:10:01 +0200151 int imsi_acq_pending;
152 struct llist_head stored_msgs;
Jacob Erlbeck31591142014-09-03 11:59:48 +0200153 int imsi_acq_retries;
Jacob Erlbeck9057bc32014-08-12 16:30:30 +0200154
Jacob Erlbeck89d3d342014-08-06 18:55:15 +0200155 int enable_patching;
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200156};
157
158
Harald Welteb77c6972010-05-01 11:28:43 +0200159/* gb_proxy_vty .c */
160
161int gbproxy_vty_init(void);
162int gbproxy_parse_config(const char *config_file, struct gbproxy_config *cfg);
163
164
165/* gb_proxy.c */
Holger Hans Peter Freyther18739ea2014-08-04 11:10:09 +0200166int gbproxy_init_config(struct gbproxy_config *cfg);
Harald Welteb77c6972010-05-01 11:28:43 +0200167
168/* Main input function for Gb proxy */
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200169int 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 +0200170
Harald Weltec1c1dd22010-05-11 06:34:24 +0200171int gbprox_signal(unsigned int subsys, unsigned int signal,
172 void *handler_data, void *signal_data);
Harald Welte1ccbf442010-05-14 11:53:08 +0000173
174/* Reset all persistent NS-VC's */
175int gbprox_reset_persistent_nsvcs(struct gprs_ns_inst *nsi);
176
Holger Hans Peter Freythereece6272014-08-04 15:42:36 +0200177void gbprox_reset(struct gbproxy_config *cfg);
Jacob Erlbeck73685282014-05-23 20:48:07 +0200178
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200179/* TLLI state handling */
180void gbproxy_delete_tllis(struct gbproxy_peer *peer);
Jacob Erlbeck18a37872014-09-08 09:59:16 +0200181int gbproxy_check_tlli(
182 struct gbproxy_peer *peer,
183 struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200184struct gbproxy_tlli_info *gbprox_find_tlli_by_ptmsi(
185 struct gbproxy_peer *peer,
186 uint32_t ptmsi);
187uint32_t gbproxy_map_tlli(
188 uint32_t other_tlli, struct gbproxy_tlli_info *tlli_info, int to_bss);
189struct gbproxy_tlli_info *gbproxy_update_tlli_state_ul(
190 struct gbproxy_peer *peer, time_t now,
191 struct gprs_gb_parse_context *parse_ctx);
192struct gbproxy_tlli_info *gbproxy_update_tlli_state_dl(
193 struct gbproxy_peer *peer, time_t now,
194 struct gprs_gb_parse_context *parse_ctx);
195void gbproxy_update_tlli_state_after(
196 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
197 time_t now, struct gprs_gb_parse_context *parse_ctx);
198int gbproxy_remove_stale_tllis(struct gbproxy_peer *peer, time_t now);
199void gbproxy_delete_tlli(struct gbproxy_peer *peer,
200 struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck31591142014-09-03 11:59:48 +0200201void gbproxy_tlli_info_discard_messages(struct gbproxy_tlli_info *tlli_info);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200202
203struct gbproxy_tlli_info *gbproxy_register_tlli(
204 struct gbproxy_peer *peer, uint32_t tlli,
205 const uint8_t *imsi, size_t imsi_len, time_t now);
206
207struct gbproxy_tlli_info *gbproxy_find_tlli(
208 struct gbproxy_peer *peer, uint32_t tlli);
209struct gbproxy_tlli_info *gbproxy_find_tlli_by_mi(
210 struct gbproxy_peer *peer, const uint8_t *mi_data, size_t mi_data_len);
211struct gbproxy_tlli_info *gbproxy_find_tlli_by_sgsn_tlli(
212 struct gbproxy_peer *peer,
213 uint32_t tlli);
214struct gbproxy_tlli_info *gbproxy_find_tlli_by_ptmsi(
215 struct gbproxy_peer *peer,
216 uint32_t ptmsi);
217
218/* needed by gb_proxy_tlli.h */
219uint32_t gbproxy_make_bss_ptmsi(struct gbproxy_peer *peer, uint32_t sgsn_ptmsi);
220uint32_t gbproxy_make_sgsn_tlli(
221 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
222 uint32_t bss_tlli);
223int gbproxy_check_imsi(
224 struct gbproxy_peer *peer, const uint8_t *imsi, size_t imsi_len);
225
226/* Message patching */
227void gbproxy_patch_bssgp(
228 struct msgb *msg, uint8_t *bssgp, size_t bssgp_len,
229 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200230 int *len_change, struct gprs_gb_parse_context *parse_ctx);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200231
232int gbproxy_patch_llc(
233 struct msgb *msg, uint8_t *llc, size_t llc_len,
234 struct gbproxy_peer *peer, struct gbproxy_tlli_info *tlli_info,
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200235 int *len_change, struct gprs_gb_parse_context *parse_ctx);
Jacob Erlbeck9114bee2014-08-19 12:21:01 +0200236
237int gbproxy_set_patch_filter(
238 struct gbproxy_config *cfg, const char *filter, const char **err_msg);
239void gbproxy_clear_patch_filter(struct gbproxy_config *cfg);
240int gbproxy_check_imsi(
241 struct gbproxy_peer *peer, const uint8_t *imsi, size_t imsi_len);
242
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200243/* Peer handling */
244struct gbproxy_peer *gbproxy_peer_by_bvci(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200245 struct gbproxy_config *cfg, uint16_t bvci);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200246struct gbproxy_peer *gbproxy_peer_by_nsei(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200247 struct gbproxy_config *cfg, uint16_t nsei);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200248struct gbproxy_peer *gbproxy_peer_by_rai(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200249 struct gbproxy_config *cfg, const uint8_t *ra);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200250struct gbproxy_peer *gbproxy_peer_by_lai(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200251 struct gbproxy_config *cfg, const uint8_t *la);
Jacob Erlbeck7fb26c22014-09-04 11:08:50 +0200252struct gbproxy_peer *gbproxy_peer_by_lac(
253 struct gbproxy_config *cfg, const uint8_t *la);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200254struct gbproxy_peer *gbproxy_peer_by_bssgp_tlv(
Jacob Erlbeck48bb3a32014-09-01 11:55:11 +0200255 struct gbproxy_config *cfg, struct tlv_parsed *tp);
256struct gbproxy_peer *gbproxy_peer_alloc(struct gbproxy_config *cfg, uint16_t bvci);
257void gbproxy_peer_free(struct gbproxy_peer *peer);
258int gbproxy_cleanup_peers(struct gbproxy_config *cfg, uint16_t nsei, uint16_t bvci);
Jacob Erlbeck5f1faa32014-08-21 10:01:30 +0200259
Harald Welteb77c6972010-05-01 11:28:43 +0200260#endif