Neels Hofmeyr | 17518fe | 2017-06-20 04:35:06 +0200 | [diff] [blame] | 1 | /*! \file gprs_ns.h */ |
| 2 | |
Sylvain Munaut | 12ba778 | 2014-06-16 10:13:40 +0200 | [diff] [blame] | 3 | #pragma once |
Harald Welte | 9ba5005 | 2010-03-14 15:45:01 +0800 | [diff] [blame] | 4 | |
Harald Welte | 8f9a3ee | 2010-05-02 11:26:34 +0200 | [diff] [blame] | 5 | #include <stdint.h> |
| 6 | |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 7 | /* Our Implementation */ |
| 8 | #include <netinet/in.h> |
Pablo Neira Ayuso | ff66323 | 2011-03-22 16:47:59 +0100 | [diff] [blame] | 9 | #include <osmocom/core/linuxlist.h> |
| 10 | #include <osmocom/core/msgb.h> |
| 11 | #include <osmocom/core/timer.h> |
| 12 | #include <osmocom/core/select.h> |
Harald Welte | 605ac5d | 2012-06-16 16:09:52 +0800 | [diff] [blame] | 13 | #include <osmocom/gprs/gprs_msgb.h> |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 14 | |
Harald Welte | 8648e49 | 2012-06-17 13:12:51 +0800 | [diff] [blame] | 15 | #include <osmocom/gprs/protocol/gsm_08_16.h> |
| 16 | |
Harald Welte | fe4ab90 | 2010-05-12 17:19:53 +0000 | [diff] [blame] | 17 | #define NS_TIMERS_COUNT 7 |
| 18 | #define NS_TIMERS "(tns-block|tns-block-retries|tns-reset|tns-reset-retries|tns-test|tns-alive|tns-alive-retries)" |
| 19 | #define NS_TIMERS_HELP \ |
| 20 | "(un)blocking Timer (Tns-block) timeout\n" \ |
| 21 | "(un)blocking Timer (Tns-block) number of retries\n" \ |
| 22 | "Reset Timer (Tns-reset) timeout\n" \ |
| 23 | "Reset Timer (Tns-reset) number of retries\n" \ |
| 24 | "Test Timer (Tns-test) timeout\n" \ |
Holger Hans Peter Freyther | 887934e | 2011-11-05 15:14:59 +0100 | [diff] [blame] | 25 | "Alive Timer (Tns-alive) timeout\n" \ |
| 26 | "Alive Timer (Tns-alive) number of retries\n" |
Harald Welte | fe4ab90 | 2010-05-12 17:19:53 +0000 | [diff] [blame] | 27 | |
Daniel Willmann | ba34a2f | 2016-09-02 15:44:12 +0200 | [diff] [blame] | 28 | /* Educated guess - LLC user payload is 1500 bytes plus possible headers */ |
| 29 | #define NS_ALLOC_SIZE 3072 |
Jacob Erlbeck | 8d192d7 | 2015-04-07 17:52:45 +0200 | [diff] [blame] | 30 | #define NS_ALLOC_HEADROOM 20 |
| 31 | |
Harald Welte | fe4ab90 | 2010-05-12 17:19:53 +0000 | [diff] [blame] | 32 | enum ns_timeout { |
| 33 | NS_TOUT_TNS_BLOCK, |
| 34 | NS_TOUT_TNS_BLOCK_RETRIES, |
| 35 | NS_TOUT_TNS_RESET, |
| 36 | NS_TOUT_TNS_RESET_RETRIES, |
| 37 | NS_TOUT_TNS_TEST, |
| 38 | NS_TOUT_TNS_ALIVE, |
| 39 | NS_TOUT_TNS_ALIVE_RETRIES, |
| 40 | }; |
| 41 | |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 42 | #define NSE_S_BLOCKED 0x0001 |
| 43 | #define NSE_S_ALIVE 0x0002 |
Jacob Erlbeck | 5405a10 | 2013-10-24 01:33:23 +0200 | [diff] [blame] | 44 | #define NSE_S_RESET 0x0004 |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 45 | |
Max | 4ce24c4 | 2017-10-23 15:09:23 +0200 | [diff] [blame] | 46 | #define NS_DESC_B(st) ((st) & NSE_S_BLOCKED ? "BLOCKED" : "UNBLOCKED") |
| 47 | #define NS_DESC_A(st) ((st) & NSE_S_ALIVE ? "ALIVE" : "DEAD") |
| 48 | #define NS_DESC_R(st) ((st) & NSE_S_RESET ? "RESET" : "UNRESET") |
| 49 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 50 | /*! Osmocom NS link layer types */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 51 | enum gprs_ns_ll { |
Harald Welte | a9f23c8 | 2011-11-23 15:01:31 +0100 | [diff] [blame] | 52 | GPRS_NS_LL_UDP, /*!< NS/UDP/IP */ |
| 53 | GPRS_NS_LL_E1, /*!< NS/E1 */ |
| 54 | GPRS_NS_LL_FR_GRE, /*!< NS/FR/GRE/IP */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 55 | }; |
| 56 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 57 | /*! Osmoco NS events */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 58 | enum gprs_ns_evt { |
| 59 | GPRS_NS_EVT_UNIT_DATA, |
| 60 | }; |
| 61 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 62 | /*! Osmocom NS VC create status */ |
Jacob Erlbeck | 84cdc70 | 2013-10-08 12:04:44 +0200 | [diff] [blame] | 63 | enum gprs_ns_cs { |
| 64 | GPRS_NS_CS_CREATED, /*!< A NSVC object has been created */ |
| 65 | GPRS_NS_CS_FOUND, /*!< A NSVC object has been found */ |
| 66 | GPRS_NS_CS_REJECTED, /*!< Rejected and answered message */ |
| 67 | GPRS_NS_CS_SKIPPED, /*!< Skipped message */ |
| 68 | GPRS_NS_CS_ERROR, /*!< Failed to process message */ |
| 69 | }; |
| 70 | |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 71 | struct gprs_nsvc; |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 72 | /*! Osmocom GPRS callback function type */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 73 | typedef int gprs_ns_cb_t(enum gprs_ns_evt event, struct gprs_nsvc *nsvc, |
Harald Welte | 8f9a3ee | 2010-05-02 11:26:34 +0200 | [diff] [blame] | 74 | struct msgb *msg, uint16_t bvci); |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 75 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 76 | /*! An instance of the NS protocol stack */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 77 | struct gprs_ns_inst { |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 78 | /*! callback to the user for incoming UNIT DATA IND */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 79 | gprs_ns_cb_t *cb; |
| 80 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 81 | /*! linked lists of all NSVC in this instance */ |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 82 | struct llist_head gprs_nsvcs; |
| 83 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 84 | /*! a NSVC object that's needed to deal with packets for |
Harald Welte | a9f23c8 | 2011-11-23 15:01:31 +0100 | [diff] [blame] | 85 | * unknown NSVC */ |
Harald Welte | dd1c83c | 2010-05-13 13:58:08 +0200 | [diff] [blame] | 86 | struct gprs_nsvc *unknown_nsvc; |
| 87 | |
Harald Welte | fe4ab90 | 2010-05-12 17:19:53 +0000 | [diff] [blame] | 88 | uint16_t timeout[NS_TIMERS_COUNT]; |
| 89 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 90 | /*! NS-over-IP specific bits */ |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 91 | struct { |
Pablo Neira Ayuso | bfadfb7 | 2011-05-06 12:11:23 +0200 | [diff] [blame] | 92 | struct osmo_fd fd; |
Harald Welte | 7fb0523 | 2010-05-19 15:09:09 +0200 | [diff] [blame] | 93 | uint32_t local_ip; |
| 94 | uint16_t local_port; |
Philipp Maier | 224a6dd | 2017-10-17 15:23:53 +0200 | [diff] [blame] | 95 | uint32_t remote_ip; |
| 96 | uint16_t remote_port; |
Holger Hans Peter Freyther | 2c3393d | 2013-03-25 11:59:58 +0100 | [diff] [blame] | 97 | int dscp; |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 98 | } nsip; |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 99 | /*! NS-over-FR-over-GRE-over-IP specific bits */ |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 100 | struct { |
Pablo Neira Ayuso | bfadfb7 | 2011-05-06 12:11:23 +0200 | [diff] [blame] | 101 | struct osmo_fd fd; |
Harald Welte | 7fb0523 | 2010-05-19 15:09:09 +0200 | [diff] [blame] | 102 | uint32_t local_ip; |
Holger Hans Peter Freyther | 1e57eb6 | 2012-03-02 14:14:33 +0100 | [diff] [blame] | 103 | unsigned int enabled:1; |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 104 | } frgre; |
Harald Welte | 1203de3 | 2010-05-01 11:28:43 +0200 | [diff] [blame] | 105 | }; |
| 106 | |
Harald Welte | 8040545 | 2010-05-03 20:16:13 +0200 | [diff] [blame] | 107 | enum nsvc_timer_mode { |
| 108 | /* standard timers */ |
| 109 | NSVC_TIMER_TNS_TEST, |
| 110 | NSVC_TIMER_TNS_ALIVE, |
Harald Welte | 69a4cf2 | 2010-05-03 20:55:10 +0200 | [diff] [blame] | 111 | NSVC_TIMER_TNS_RESET, |
| 112 | _NSVC_TIMER_NR, |
Harald Welte | 8040545 | 2010-05-03 20:16:13 +0200 | [diff] [blame] | 113 | }; |
| 114 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 115 | /*! Structure representing a single NS-VC */ |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 116 | struct gprs_nsvc { |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 117 | /*! list of NS-VCs within NS Instance */ |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 118 | struct llist_head list; |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 119 | /*! pointer to NS Instance */ |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 120 | struct gprs_ns_inst *nsi; |
| 121 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 122 | uint16_t nsei; /*! end-to-end significance */ |
| 123 | uint16_t nsvci; /*! uniquely identifies NS-VC at SGSN */ |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 124 | |
Harald Welte | 8f9a3ee | 2010-05-02 11:26:34 +0200 | [diff] [blame] | 125 | uint32_t state; |
| 126 | uint32_t remote_state; |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 127 | |
Pablo Neira Ayuso | 2b15952 | 2011-05-06 12:11:06 +0200 | [diff] [blame] | 128 | struct osmo_timer_list timer; |
Harald Welte | 8040545 | 2010-05-03 20:16:13 +0200 | [diff] [blame] | 129 | enum nsvc_timer_mode timer_mode; |
Jacob Erlbeck | 0a1400f | 2015-10-06 15:23:25 +0200 | [diff] [blame] | 130 | struct timeval timer_started; |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 131 | int alive_retries; |
| 132 | |
Harald Welte | 2bffac5 | 2010-05-12 15:55:23 +0000 | [diff] [blame] | 133 | unsigned int remote_end_is_sgsn:1; |
| 134 | unsigned int persistent:1; |
Jacob Erlbeck | 5e6d679 | 2013-10-14 22:06:48 +0200 | [diff] [blame] | 135 | unsigned int nsvci_is_valid:1; |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 136 | |
Harald Welte | 144e029 | 2010-05-13 11:45:07 +0200 | [diff] [blame] | 137 | struct rate_ctr_group *ctrg; |
Jacob Erlbeck | fc9533d | 2015-10-29 00:55:58 +0100 | [diff] [blame] | 138 | struct osmo_stat_item_group *statg; |
Harald Welte | 144e029 | 2010-05-13 11:45:07 +0200 | [diff] [blame] | 139 | |
Neels Hofmeyr | 87e4550 | 2017-06-20 00:17:59 +0200 | [diff] [blame] | 140 | /*! which link-layer are we based on? */ |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 141 | enum gprs_ns_ll ll; |
| 142 | |
Max | 9530859 | 2017-10-24 15:54:28 +0200 | [diff] [blame] | 143 | /*! make sure to always keep bts_addr as first struct member to not break the assumption |
| 144 | that those structs are similar */ |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 145 | union { |
| 146 | struct { |
| 147 | struct sockaddr_in bts_addr; |
| 148 | } ip; |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 149 | struct { |
| 150 | struct sockaddr_in bts_addr; |
| 151 | } frgre; |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 152 | }; |
| 153 | }; |
| 154 | |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 155 | /* Create a new NS protocol instance */ |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 156 | struct gprs_ns_inst *gprs_ns_instantiate(gprs_ns_cb_t *cb, void *ctx); |
Harald Welte | 9ba5005 | 2010-03-14 15:45:01 +0800 | [diff] [blame] | 157 | |
Holger Hans Peter Freyther | c62a1bf | 2013-07-02 09:10:11 +0200 | [diff] [blame] | 158 | /* Close a NS protocol instance */ |
| 159 | void gprs_ns_close(struct gprs_ns_inst *nsi); |
| 160 | |
| 161 | /* Close and Destroy a NS protocol instance */ |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 162 | void gprs_ns_destroy(struct gprs_ns_inst *nsi); |
| 163 | |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 164 | /* Listen for incoming GPRS packets via NS/UDP */ |
Harald Welte | 7fb0523 | 2010-05-19 15:09:09 +0200 | [diff] [blame] | 165 | int gprs_ns_nsip_listen(struct gprs_ns_inst *nsi); |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 166 | |
Harald Welte | f543036 | 2012-06-17 12:25:53 +0800 | [diff] [blame] | 167 | /* Establish a connection (from the BSS) to the SGSN */ |
| 168 | struct gprs_nsvc *gprs_ns_nsip_connect(struct gprs_ns_inst *nsi, |
| 169 | struct sockaddr_in *dest, |
| 170 | uint16_t nsei, uint16_t nsvci); |
| 171 | |
| 172 | |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 173 | struct sockaddr_in; |
| 174 | |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 175 | /* main function for higher layers (BSSGP) to send NS messages */ |
Harald Welte | 24a655f | 2010-04-30 19:54:29 +0200 | [diff] [blame] | 176 | int gprs_ns_sendmsg(struct gprs_ns_inst *nsi, struct msgb *msg); |
Harald Welte | f030b21 | 2010-04-26 19:18:54 +0200 | [diff] [blame] | 177 | |
Harald Welte | 834f26d | 2010-05-11 06:20:54 +0200 | [diff] [blame] | 178 | int gprs_ns_tx_reset(struct gprs_nsvc *nsvc, uint8_t cause); |
| 179 | int gprs_ns_tx_block(struct gprs_nsvc *nsvc, uint8_t cause); |
| 180 | int gprs_ns_tx_unblock(struct gprs_nsvc *nsvc); |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 181 | |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 182 | /* Listen for incoming GPRS packets via NS/FR/GRE */ |
Harald Welte | 7fb0523 | 2010-05-19 15:09:09 +0200 | [diff] [blame] | 183 | int gprs_ns_frgre_listen(struct gprs_ns_inst *nsi); |
Harald Welte | 3771d09 | 2010-04-30 20:26:32 +0200 | [diff] [blame] | 184 | |
Harald Welte | f543036 | 2012-06-17 12:25:53 +0800 | [diff] [blame] | 185 | struct gprs_nsvc *gprs_nsvc_create(struct gprs_ns_inst *nsi, uint16_t nsvci); |
| 186 | void gprs_nsvc_delete(struct gprs_nsvc *nsvc); |
| 187 | struct gprs_nsvc *gprs_nsvc_by_nsei(struct gprs_ns_inst *nsi, uint16_t nsei); |
| 188 | struct gprs_nsvc *gprs_nsvc_by_nsvci(struct gprs_ns_inst *nsi, uint16_t nsvci); |
Harald Welte | 144e029 | 2010-05-13 11:45:07 +0200 | [diff] [blame] | 189 | |
Harald Welte | 731d1fc | 2010-05-14 11:53:08 +0000 | [diff] [blame] | 190 | /* Initiate a RESET procedure (including timer start, ...)*/ |
Jacob Erlbeck | 0540d83 | 2014-10-08 11:47:36 +0200 | [diff] [blame] | 191 | int gprs_nsvc_reset(struct gprs_nsvc *nsvc, uint8_t cause); |
Harald Welte | 731d1fc | 2010-05-14 11:53:08 +0000 | [diff] [blame] | 192 | |
Harald Welte | 2bffac5 | 2010-05-12 15:55:23 +0000 | [diff] [blame] | 193 | /* Add NS-specific VTY stuff */ |
| 194 | int gprs_ns_vty_init(struct gprs_ns_inst *nsi); |
| 195 | |
Jacob Erlbeck | 34fc470 | 2013-10-09 11:27:04 +0200 | [diff] [blame] | 196 | /* Resturn peer info as string (NOTE: the buffer is allocated statically) */ |
Max | cbf5cdf | 2017-10-24 18:18:55 +0200 | [diff] [blame] | 197 | const char *gprs_ns_ll_str(const struct gprs_nsvc *nsvc); |
Jacob Erlbeck | 96550e0 | 2013-10-14 22:06:47 +0200 | [diff] [blame] | 198 | |
| 199 | /* Copy the link layer info from other into nsvc */ |
| 200 | void gprs_ns_ll_copy(struct gprs_nsvc *nsvc, struct gprs_nsvc *other); |
| 201 | |
| 202 | /* Clear the link layer info (will never match a real link then) */ |
| 203 | void gprs_ns_ll_clear(struct gprs_nsvc *nsvc); |
Jacob Erlbeck | 34fc470 | 2013-10-09 11:27:04 +0200 | [diff] [blame] | 204 | |
Jacob Erlbeck | 8d192d7 | 2015-04-07 17:52:45 +0200 | [diff] [blame] | 205 | struct msgb *gprs_ns_msgb_alloc(void); |
Harald Welte | b3ee265 | 2010-05-19 14:38:50 +0200 | [diff] [blame] | 206 | |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 207 | enum signal_ns { |
| 208 | S_NS_RESET, |
| 209 | S_NS_BLOCK, |
| 210 | S_NS_UNBLOCK, |
| 211 | S_NS_ALIVE_EXP, /* Tns-alive expired more than N times */ |
Jacob Erlbeck | 5e6d679 | 2013-10-14 22:06:48 +0200 | [diff] [blame] | 212 | S_NS_REPLACED, /* nsvc object is replaced (sets old_nsvc) */ |
Jacob Erlbeck | 5405a10 | 2013-10-24 01:33:23 +0200 | [diff] [blame] | 213 | S_NS_MISMATCH, /* got an unexpected IE (sets msg, pdu_type, ie_type) */ |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 214 | }; |
| 215 | |
Max | 54ca30c | 2017-05-22 14:48:27 +0200 | [diff] [blame] | 216 | extern const struct value_string gprs_ns_signal_ns_names[]; |
| 217 | |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 218 | struct ns_signal_data { |
| 219 | struct gprs_nsvc *nsvc; |
Jacob Erlbeck | 5e6d679 | 2013-10-14 22:06:48 +0200 | [diff] [blame] | 220 | struct gprs_nsvc *old_nsvc; |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 221 | uint8_t cause; |
Jacob Erlbeck | 5405a10 | 2013-10-24 01:33:23 +0200 | [diff] [blame] | 222 | uint8_t pdu_type; |
| 223 | uint8_t ie_type; |
| 224 | struct msgb *msg; |
Harald Welte | 4fcdd76 | 2012-06-16 16:40:42 +0800 | [diff] [blame] | 225 | }; |
| 226 | |
Harald Welte | cca4963 | 2012-06-16 17:45:59 +0800 | [diff] [blame] | 227 | void gprs_ns_set_log_ss(int ss); |
| 228 | |
Daniel Willmann | f1318fe | 2018-05-29 20:55:18 +0200 | [diff] [blame] | 229 | char *gprs_nsvc_state_append(char *s, struct gprs_nsvc *nsvc); |
| 230 | |
Neels Hofmeyr | b525b9e | 2017-10-16 16:46:43 +0200 | [diff] [blame] | 231 | /*! @} */ |