Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 1 | /* (C) 2009-2015 by Holger Hans Peter Freyther <zecke@selfish.org> |
Holger Hans Peter Freyther | 4d61417 | 2011-08-15 15:53:00 +0200 | [diff] [blame] | 2 | * (C) 2009-2011 by On-Waves |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 3 | * All Rights Reserved |
| 4 | * |
| 5 | * This program is free software; you can redistribute it and/or modify |
Harald Welte | 0e3e88e | 2011-01-01 15:25:50 +0100 | [diff] [blame] | 6 | * it under the terms of the GNU Affero General Public License as published by |
| 7 | * the Free Software Foundation; either version 3 of the License, or |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 8 | * (at your option) any later version. |
| 9 | * |
| 10 | * This program is distributed in the hope that it will be useful, |
| 11 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 12 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
Harald Welte | 0e3e88e | 2011-01-01 15:25:50 +0100 | [diff] [blame] | 13 | * GNU Affero General Public License for more details. |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 14 | * |
Harald Welte | 0e3e88e | 2011-01-01 15:25:50 +0100 | [diff] [blame] | 15 | * You should have received a copy of the GNU Affero General Public License |
| 16 | * along with this program. If not, see <http://www.gnu.org/licenses/>. |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 17 | * |
| 18 | */ |
| 19 | |
| 20 | #include <openbsc/osmo_bsc.h> |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 21 | #include <openbsc/osmo_msc_data.h> |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 22 | #include <openbsc/debug.h> |
| 23 | |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 24 | #include <openbsc/gsm_04_80.h> |
| 25 | |
Harald Welte | 8b1713a | 2011-03-23 18:26:56 +0100 | [diff] [blame] | 26 | #include <osmocom/gsm/protocol/gsm_08_08.h> |
| 27 | #include <osmocom/gsm/gsm0808.h> |
Holger Hans Peter Freyther | a62919e | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 28 | |
Holger Hans Peter Freyther | 326a124 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 29 | #include <osmocom/sccp/sccp.h> |
| 30 | |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 31 | #define return_when_not_connected(conn) \ |
| 32 | if (!conn->sccp_con) {\ |
| 33 | LOGP(DMSC, LOGL_ERROR, "MSC Connection not present.\n"); \ |
| 34 | return; \ |
| 35 | } |
| 36 | |
| 37 | #define return_when_not_connected_val(conn, ret) \ |
| 38 | if (!conn->sccp_con) {\ |
| 39 | LOGP(DMSC, LOGL_ERROR, "MSC Connection not present.\n"); \ |
| 40 | return ret; \ |
| 41 | } |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 42 | |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 43 | #define queue_msg_or_return(resp) \ |
| 44 | if (!resp) { \ |
| 45 | LOGP(DMSC, LOGL_ERROR, "Failed to allocate response.\n"); \ |
| 46 | return; \ |
| 47 | } \ |
Holger Hans Peter Freyther | 1f334cc | 2010-11-10 09:31:41 +0100 | [diff] [blame] | 48 | bsc_queue_for_msc(conn->sccp_con, resp); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 49 | |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 50 | static int bsc_clear_request(struct gsm_subscriber_connection *conn, uint32_t cause); |
| 51 | static int complete_layer3(struct gsm_subscriber_connection *conn, |
| 52 | struct msgb *msg, struct osmo_msc_data *msc); |
| 53 | |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 54 | static uint16_t get_network_code_for_msc(struct osmo_msc_data *msc) |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 55 | { |
Holger Hans Peter Freyther | 658f964 | 2015-07-13 11:06:10 +0200 | [diff] [blame] | 56 | if (msc->core_mnc != -1) |
| 57 | return msc->core_mnc; |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 58 | return msc->network->network_code; |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 59 | } |
| 60 | |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 61 | static uint16_t get_country_code_for_msc(struct osmo_msc_data *msc) |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 62 | { |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 63 | if (msc->core_mcc != -1) |
| 64 | return msc->core_mcc; |
| 65 | return msc->network->country_code; |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 66 | } |
| 67 | |
Holger Hans Peter Freyther | 05e2770 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 68 | static uint16_t get_lac_for_msc(struct osmo_msc_data *msc, struct gsm_bts *bts) |
| 69 | { |
| 70 | if (msc->core_lac != -1) |
| 71 | return msc->core_lac; |
| 72 | return bts->location_area_code; |
| 73 | } |
| 74 | |
| 75 | static uint16_t get_ci_for_msc(struct osmo_msc_data *msc, struct gsm_bts *bts) |
| 76 | { |
| 77 | if (msc->core_ci != -1) |
| 78 | return msc->core_ci; |
| 79 | return bts->cell_identity; |
| 80 | } |
| 81 | |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 82 | static void bsc_maybe_lu_reject(struct gsm_subscriber_connection *conn, int con_type, int cause) |
| 83 | { |
| 84 | struct msgb *msg; |
| 85 | |
| 86 | /* ignore cm service request or such */ |
| 87 | if (con_type != FLT_CON_TYPE_LU) |
| 88 | return; |
| 89 | |
| 90 | msg = gsm48_create_loc_upd_rej(cause); |
| 91 | if (!msg) { |
| 92 | LOGP(DMM, LOGL_ERROR, "Failed to create msg for LOCATION UPDATING REJECT.\n"); |
| 93 | return; |
| 94 | } |
| 95 | |
| 96 | msg->lchan = conn->lchan; |
| 97 | gsm0808_submit_dtap(conn, msg, 0, 0); |
| 98 | } |
| 99 | |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 100 | static int bsc_filter_initial(struct osmo_bsc_data *bsc, |
| 101 | struct osmo_msc_data *msc, |
| 102 | struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 103 | struct msgb *msg, char **imsi, int *con_type, |
| 104 | int *lu_cause) |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 105 | { |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 106 | struct bsc_filter_request req; |
| 107 | struct bsc_filter_reject_cause cause; |
| 108 | struct gsm48_hdr *gh = msgb_l3(msg); |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 109 | int rc; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 110 | |
| 111 | req.ctx = conn; |
| 112 | req.black_list = NULL; |
| 113 | req.access_lists = bsc_access_lists(); |
| 114 | req.local_lst_name = msc->acc_lst_name; |
| 115 | req.global_lst_name = conn->bts->network->bsc_data->acc_lst_name; |
| 116 | req.bsc_nr = 0; |
| 117 | |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 118 | rc = bsc_msg_filter_initial(gh, msgb_l3len(msg), &req, |
| 119 | con_type, imsi, &cause); |
| 120 | *lu_cause = cause.lu_reject_cause; |
| 121 | return rc; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 122 | } |
| 123 | |
| 124 | static int bsc_filter_data(struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 125 | struct msgb *msg, int *lu_cause) |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 126 | { |
| 127 | struct bsc_filter_request req; |
| 128 | struct gsm48_hdr *gh = msgb_l3(msg); |
| 129 | struct bsc_filter_reject_cause cause; |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 130 | int rc; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 131 | |
| 132 | req.ctx = conn; |
| 133 | req.black_list = NULL; |
| 134 | req.access_lists = bsc_access_lists(); |
| 135 | req.local_lst_name = conn->sccp_con->msc->acc_lst_name; |
| 136 | req.global_lst_name = conn->bts->network->bsc_data->acc_lst_name; |
| 137 | req.bsc_nr = 0; |
| 138 | |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 139 | rc = bsc_msg_filter_data(gh, msgb_l3len(msg), &req, |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 140 | &conn->sccp_con->filter_state, |
| 141 | &cause); |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 142 | *lu_cause = cause.lu_reject_cause; |
| 143 | return rc; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 144 | } |
Holger Hans Peter Freyther | 05e2770 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 145 | |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 146 | static void bsc_sapi_n_reject(struct gsm_subscriber_connection *conn, int dlci) |
| 147 | { |
Holger Hans Peter Freyther | 313324e | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 148 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 149 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | 313324e | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 150 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 151 | LOGP(DMSC, LOGL_NOTICE, "Tx MSC SAPI N REJECT DLCI=0x%02x\n", dlci); |
| 152 | |
Holger Hans Peter Freyther | 313324e | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 153 | resp = gsm0808_create_sapi_reject(dlci); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 154 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 155 | } |
| 156 | |
| 157 | static void bsc_cipher_mode_compl(struct gsm_subscriber_connection *conn, |
| 158 | struct msgb *msg, uint8_t chosen_encr) |
| 159 | { |
Holger Hans Peter Freyther | a62919e | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 160 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 161 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | a62919e | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 162 | |
| 163 | LOGP(DMSC, LOGL_DEBUG, "CIPHER MODE COMPLETE from MS, forwarding to MSC\n"); |
| 164 | resp = gsm0808_create_cipher_complete(msg, chosen_encr); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 165 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 166 | } |
| 167 | |
Jacob Erlbeck | e4cca68 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 168 | static void bsc_send_ussd_no_srv(struct gsm_subscriber_connection *conn, |
| 169 | struct msgb *msg, const char *text) |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 170 | { |
| 171 | struct gsm48_hdr *gh; |
| 172 | int8_t pdisc; |
| 173 | uint8_t mtype; |
| 174 | int drop_message = 1; |
| 175 | |
| 176 | if (!text) |
| 177 | return; |
| 178 | |
| 179 | if (!msg || msgb_l3len(msg) < sizeof(*gh)) |
| 180 | return; |
| 181 | |
| 182 | gh = msgb_l3(msg); |
| 183 | pdisc = gh->proto_discr & 0x0f; |
| 184 | mtype = gh->msg_type & 0xbf; |
| 185 | |
| 186 | /* Is CM service request? */ |
| 187 | if (pdisc == GSM48_PDISC_MM && mtype == GSM48_MT_MM_CM_SERV_REQ) { |
| 188 | struct gsm48_service_request *cm; |
| 189 | |
| 190 | cm = (struct gsm48_service_request *) &gh->data[0]; |
| 191 | |
| 192 | /* Is type SMS or call? */ |
| 193 | if (cm->cm_service_type == GSM48_CMSERV_SMS) |
| 194 | drop_message = 0; |
| 195 | else if (cm->cm_service_type == GSM48_CMSERV_MO_CALL_PACKET) |
| 196 | drop_message = 0; |
| 197 | } |
| 198 | |
| 199 | if (drop_message) { |
Holger Hans Peter Freyther | 8bdc5ee | 2013-10-31 13:35:28 +0100 | [diff] [blame] | 200 | LOGP(DMSC, LOGL_DEBUG, "Skipping (not sending) USSD message: '%s'\n", text); |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 201 | return; |
| 202 | } |
| 203 | |
Jacob Erlbeck | 51f814a | 2013-10-31 15:36:42 +0100 | [diff] [blame] | 204 | LOGP(DMSC, LOGL_INFO, "Sending CM Service Accept\n"); |
| 205 | gsm48_tx_mm_serv_ack(conn); |
| 206 | |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 207 | LOGP(DMSC, LOGL_INFO, "Sending USSD message: '%s'\n", text); |
| 208 | gsm0480_send_ussdNotify(conn, 1, text); |
| 209 | gsm0480_send_releaseComplete(conn); |
| 210 | } |
| 211 | |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 212 | /* |
| 213 | * Instruct to reserve data for a new connectiom, create the complete |
| 214 | * layer three message, send it to open the connection. |
| 215 | */ |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 216 | static int bsc_compl_l3(struct gsm_subscriber_connection *conn, struct msgb *msg, |
| 217 | uint16_t chosen_channel) |
| 218 | { |
Holger Hans Peter Freyther | 508e662 | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 219 | struct osmo_msc_data *msc; |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 220 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 221 | LOGP(DMSC, LOGL_INFO, "Tx MSC COMPL L3\n"); |
| 222 | |
Holger Hans Peter Freyther | 508e662 | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 223 | /* find the MSC link we want to use */ |
Holger Hans Peter Freyther | e592cb1 | 2011-06-07 19:57:02 +0200 | [diff] [blame] | 224 | msc = bsc_find_msc(conn, msg); |
Holger Hans Peter Freyther | 508e662 | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 225 | if (!msc) { |
| 226 | LOGP(DMSC, LOGL_ERROR, "Failed to find a MSC for a connection.\n"); |
Jacob Erlbeck | e4cca68 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 227 | bsc_send_ussd_no_srv(conn, msg, |
| 228 | conn->bts->network->bsc_data->ussd_no_msc_txt); |
Holger Hans Peter Freyther | 508e662 | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 229 | return -1; |
| 230 | } |
| 231 | |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 232 | return complete_layer3(conn, msg, msc); |
| 233 | } |
| 234 | |
| 235 | static int complete_layer3(struct gsm_subscriber_connection *conn, |
| 236 | struct msgb *msg, struct osmo_msc_data *msc) |
| 237 | { |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 238 | int con_type, rc, lu_cause; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 239 | char *imsi = NULL; |
Holger Hans Peter Freyther | 903dff7 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 240 | struct timeval tv; |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 241 | struct msgb *resp; |
| 242 | uint16_t network_code; |
| 243 | uint16_t country_code; |
Holger Hans Peter Freyther | 05e2770 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 244 | uint16_t lac; |
| 245 | uint16_t ci; |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 246 | enum bsc_con ret; |
Holger Hans Peter Freyther | 903dff7 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 247 | int send_ping = msc->advanced_ping; |
| 248 | |
| 249 | /* Advanced ping/pong handling */ |
| 250 | if (osmo_timer_pending(&msc->pong_timer)) |
| 251 | send_ping = 0; |
Holger Hans Peter Freyther | d2b37c5 | 2014-10-29 10:06:15 +0100 | [diff] [blame] | 252 | if (msc->ping_timeout <= 0) |
Holger Hans Peter Freyther | 903dff7 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 253 | send_ping = 0; |
| 254 | if (send_ping && osmo_timer_remaining(&msc->ping_timer, NULL, &tv) == -1) |
| 255 | send_ping = 0; |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 256 | |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 257 | /* Check the filter */ |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 258 | rc = bsc_filter_initial(msc->network->bsc_data, msc, conn, msg, |
| 259 | &imsi, &con_type, &lu_cause); |
| 260 | if (rc < 0) { |
| 261 | bsc_maybe_lu_reject(conn, con_type, lu_cause); |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 262 | return BSC_API_CONN_POL_REJECT; |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 263 | } |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 264 | |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 265 | /* allocate resource for a new connection */ |
Holger Hans Peter Freyther | 903dff7 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 266 | ret = bsc_create_new_connection(conn, msc, send_ping); |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 267 | |
| 268 | if (ret != BSC_CON_SUCCESS) { |
| 269 | /* allocation has failed */ |
| 270 | if (ret == BSC_CON_REJECT_NO_LINK) |
Jacob Erlbeck | e4cca68 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 271 | bsc_send_ussd_no_srv(conn, msg, msc->ussd_msc_lost_txt); |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 272 | else if (ret == BSC_CON_REJECT_RF_GRACE) |
Jacob Erlbeck | e4cca68 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 273 | bsc_send_ussd_no_srv(conn, msg, msc->ussd_grace_txt); |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 274 | |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 275 | return BSC_API_CONN_POL_REJECT; |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 276 | } |
| 277 | |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 278 | if (imsi) |
| 279 | conn->sccp_con->filter_state.imsi = talloc_steal(conn, imsi); |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 280 | conn->sccp_con->filter_state.con_type = con_type; |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 281 | |
Jacob Erlbeck | 3ccb86b | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 282 | /* check return value, if failed check msg for and send USSD */ |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 283 | |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 284 | network_code = get_network_code_for_msc(conn->sccp_con->msc); |
| 285 | country_code = get_country_code_for_msc(conn->sccp_con->msc); |
Holger Hans Peter Freyther | 05e2770 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 286 | lac = get_lac_for_msc(conn->sccp_con->msc, conn->bts); |
| 287 | ci = get_ci_for_msc(conn->sccp_con->msc, conn->bts); |
Holger Hans Peter Freyther | 5e84584 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 288 | |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 289 | bsc_scan_bts_msg(conn, msg); |
Holger Hans Peter Freyther | e192c7b | 2015-04-05 21:03:49 +0200 | [diff] [blame] | 290 | |
Holger Hans Peter Freyther | 05e2770 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 291 | resp = gsm0808_create_layer3(msg, network_code, country_code, lac, ci); |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 292 | if (!resp) { |
| 293 | LOGP(DMSC, LOGL_DEBUG, "Failed to create layer3 message.\n"); |
Holger Hans Peter Freyther | 326a124 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 294 | sccp_connection_free(conn->sccp_con->sccp); |
Holger Hans Peter Freyther | 288b80a | 2010-11-06 20:15:17 +0100 | [diff] [blame] | 295 | bsc_delete_connection(conn->sccp_con); |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 296 | return BSC_API_CONN_POL_REJECT; |
Holger Hans Peter Freyther | b408d79 | 2010-11-05 11:21:18 +0100 | [diff] [blame] | 297 | } |
| 298 | |
Holger Hans Peter Freyther | 288b80a | 2010-11-06 20:15:17 +0100 | [diff] [blame] | 299 | if (bsc_open_connection(conn->sccp_con, resp) != 0) { |
Holger Hans Peter Freyther | 326a124 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 300 | sccp_connection_free(conn->sccp_con->sccp); |
Holger Hans Peter Freyther | 288b80a | 2010-11-06 20:15:17 +0100 | [diff] [blame] | 301 | bsc_delete_connection(conn->sccp_con); |
Holger Hans Peter Freyther | 9c61b2a | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 302 | msgb_free(resp); |
| 303 | return BSC_API_CONN_POL_REJECT; |
| 304 | } |
| 305 | |
| 306 | return BSC_API_CONN_POL_ACCEPT; |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 307 | } |
| 308 | |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 309 | /* |
| 310 | * Plastic surgery... we want to give up the current connection |
| 311 | */ |
| 312 | static int move_to_msc(struct gsm_subscriber_connection *_conn, |
| 313 | struct msgb *msg, struct osmo_msc_data *msc) |
| 314 | { |
| 315 | struct osmo_bsc_sccp_con *old_con = _conn->sccp_con; |
| 316 | |
| 317 | /* |
| 318 | * 1. Give up the old connection. |
| 319 | * This happens by sending a clear request to the MSC, |
| 320 | * it should end with the MSC releasing the connection. |
| 321 | */ |
| 322 | old_con->conn = NULL; |
| 323 | bsc_clear_request(_conn, 0); |
| 324 | |
| 325 | /* |
| 326 | * 2. Attempt to create a new connection to the local |
| 327 | * MSC. If it fails the caller will need to handle this |
| 328 | * properly. |
| 329 | */ |
| 330 | _conn->sccp_con = NULL; |
| 331 | if (complete_layer3(_conn, msg, msc) != BSC_API_CONN_POL_ACCEPT) { |
| 332 | gsm0808_clear(_conn); |
| 333 | subscr_con_free(_conn); |
| 334 | return 1; |
| 335 | } |
| 336 | |
| 337 | return 2; |
| 338 | } |
| 339 | |
| 340 | static int handle_cc_setup(struct gsm_subscriber_connection *conn, |
| 341 | struct msgb *msg) |
| 342 | { |
| 343 | struct gsm48_hdr *gh = msgb_l3(msg); |
| 344 | uint8_t pdisc = gh->proto_discr & 0x0f; |
| 345 | uint8_t mtype = gh->msg_type & 0xbf; |
| 346 | |
| 347 | struct osmo_msc_data *msc; |
| 348 | struct gsm_mncc_number called; |
| 349 | struct tlv_parsed tp; |
| 350 | unsigned payload_len; |
| 351 | |
| 352 | char _dest_nr[35]; |
| 353 | |
| 354 | /* |
| 355 | * Do we have a setup message here? if not return fast. |
| 356 | */ |
| 357 | if (pdisc != GSM48_PDISC_CC || mtype != GSM48_MT_CC_SETUP) |
| 358 | return 0; |
| 359 | |
| 360 | payload_len = msgb_l3len(msg) - sizeof(*gh); |
| 361 | |
| 362 | tlv_parse(&tp, &gsm48_att_tlvdef, gh->data, payload_len, 0, 0); |
| 363 | if (!TLVP_PRESENT(&tp, GSM48_IE_CALLED_BCD)) { |
| 364 | LOGP(DMSC, LOGL_ERROR, "Called BCD not present in setup.\n"); |
| 365 | return -1; |
| 366 | } |
| 367 | |
| 368 | memset(&called, 0, sizeof(called)); |
| 369 | gsm48_decode_called(&called, |
| 370 | TLVP_VAL(&tp, GSM48_IE_CALLED_BCD) - 1); |
| 371 | |
Holger Hans Peter Freyther | cba5609 | 2011-09-30 16:24:27 +0200 | [diff] [blame] | 372 | if (called.plan != 1 && called.plan != 0) |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 373 | return 0; |
| 374 | |
Holger Hans Peter Freyther | cba5609 | 2011-09-30 16:24:27 +0200 | [diff] [blame] | 375 | if (called.plan == 1 && called.type == 1) { |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 376 | _dest_nr[0] = _dest_nr[1] = '0'; |
| 377 | memcpy(_dest_nr + 2, called.number, sizeof(called.number)); |
| 378 | } else |
| 379 | memcpy(_dest_nr, called.number, sizeof(called.number)); |
| 380 | |
| 381 | /* |
| 382 | * Check if the connection should be moved... |
| 383 | */ |
| 384 | llist_for_each_entry(msc, &conn->bts->network->bsc_data->mscs, entry) { |
| 385 | if (msc->type != MSC_CON_TYPE_LOCAL) |
| 386 | continue; |
| 387 | if (!msc->local_pref) |
| 388 | continue; |
| 389 | if (regexec(&msc->local_pref_reg, _dest_nr, 0, NULL, 0) != 0) |
| 390 | continue; |
| 391 | |
| 392 | return move_to_msc(conn, msg, msc); |
| 393 | } |
| 394 | |
| 395 | return 0; |
| 396 | } |
| 397 | |
| 398 | |
Holger Hans Peter Freyther | 767ff3c | 2010-11-04 12:18:00 +0100 | [diff] [blame] | 399 | static void bsc_dtap(struct gsm_subscriber_connection *conn, uint8_t link_id, struct msgb *msg) |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 400 | { |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 401 | int lu_cause; |
Holger Hans Peter Freyther | 3623d6b | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 402 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 403 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | 3623d6b | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 404 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 405 | LOGP(DMSC, LOGL_INFO, "Tx MSC DTAP LINK_ID=0x%02x\n", link_id); |
| 406 | |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 407 | /* |
| 408 | * We might want to move this connection to a new MSC. Ask someone |
| 409 | * to handle it. If it was handled we will return. |
| 410 | */ |
Holger Hans Peter Freyther | 54f4ad9 | 2011-06-27 21:29:03 +0200 | [diff] [blame] | 411 | if (handle_cc_setup(conn, msg) >= 1) |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 412 | return; |
| 413 | |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 414 | /* Check the filter */ |
Holger Hans Peter Freyther | 7fe7027 | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 415 | if (bsc_filter_data(conn, msg, &lu_cause) < 0) { |
| 416 | bsc_maybe_lu_reject(conn, |
| 417 | conn->sccp_con->filter_state.con_type, |
| 418 | lu_cause); |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 419 | bsc_clear_request(conn, 0); |
| 420 | return; |
| 421 | } |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 422 | |
Holger Hans Peter Freyther | 48f9a4e | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 423 | bsc_scan_bts_msg(conn, msg); |
Holger Hans Peter Freyther | 260440b | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 424 | |
Holger Hans Peter Freyther | 3623d6b | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 425 | resp = gsm0808_create_dtap(msg, link_id); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 426 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 427 | } |
| 428 | |
Holger Hans Peter Freyther | 8dd01ef | 2010-11-04 12:06:57 +0100 | [diff] [blame] | 429 | static void bsc_assign_compl(struct gsm_subscriber_connection *conn, uint8_t rr_cause, |
| 430 | uint8_t chosen_channel, uint8_t encr_alg_id, |
| 431 | uint8_t speech_model) |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 432 | { |
Holger Hans Peter Freyther | 03a61d1 | 2010-11-04 12:09:45 +0100 | [diff] [blame] | 433 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 434 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | 03a61d1 | 2010-11-04 12:09:45 +0100 | [diff] [blame] | 435 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 436 | LOGP(DMSC, LOGL_INFO, "Tx MSC ASSIGN COMPL\n"); |
| 437 | |
Holger Hans Peter Freyther | 03a61d1 | 2010-11-04 12:09:45 +0100 | [diff] [blame] | 438 | resp = gsm0808_create_assignment_completed(rr_cause, chosen_channel, |
| 439 | encr_alg_id, speech_model); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 440 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 441 | } |
| 442 | |
Holger Hans Peter Freyther | 9ff6568 | 2010-11-05 10:37:17 +0100 | [diff] [blame] | 443 | static void bsc_assign_fail(struct gsm_subscriber_connection *conn, |
| 444 | uint8_t cause, uint8_t *rr_cause) |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 445 | { |
Holger Hans Peter Freyther | cfa3747 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 446 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 447 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | cfa3747 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 448 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 449 | LOGP(DMSC, LOGL_INFO, "Tx MSC ASSIGN FAIL\n"); |
| 450 | |
Holger Hans Peter Freyther | cfa3747 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 451 | resp = gsm0808_create_assignment_failure(cause, rr_cause); |
Holger Hans Peter Freyther | 5f2184c | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 452 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 453 | } |
| 454 | |
Holger Hans Peter Freyther | 0aa55a3 | 2010-11-03 19:01:58 +0100 | [diff] [blame] | 455 | static int bsc_clear_request(struct gsm_subscriber_connection *conn, uint32_t cause) |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 456 | { |
Holger Hans Peter Freyther | 97f5a1a | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 457 | struct osmo_bsc_sccp_con *sccp; |
Holger Hans Peter Freyther | ef2d1bf | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 458 | struct msgb *resp; |
Holger Hans Peter Freyther | b6d7014 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 459 | return_when_not_connected_val(conn, 1); |
Holger Hans Peter Freyther | ef2d1bf | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 460 | |
Harald Welte | 98833af | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 461 | LOGP(DMSC, LOGL_INFO, "Tx MSC CLEAR REQUEST\n"); |
| 462 | |
Holger Hans Peter Freyther | 97f5a1a | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 463 | /* |
| 464 | * Remove the connection from BSC<->SCCP part, the SCCP part |
| 465 | * will either be cleared by channel release or MSC disconnect |
| 466 | */ |
| 467 | sccp = conn->sccp_con; |
| 468 | sccp->conn = NULL; |
| 469 | conn->sccp_con = NULL; |
| 470 | |
Holger Hans Peter Freyther | ef2d1bf | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 471 | resp = gsm0808_create_clear_rqst(GSM0808_CAUSE_RADIO_INTERFACE_FAILURE); |
| 472 | if (!resp) { |
| 473 | LOGP(DMSC, LOGL_ERROR, "Failed to allocate response.\n"); |
Holger Hans Peter Freyther | 97f5a1a | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 474 | return 1; |
Holger Hans Peter Freyther | ef2d1bf | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 475 | } |
| 476 | |
Holger Hans Peter Freyther | 97f5a1a | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 477 | bsc_queue_for_msc(sccp, resp); |
| 478 | return 1; |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 479 | } |
| 480 | |
Harald Welte | cd4acaf | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 481 | static void bsc_cm_update(struct gsm_subscriber_connection *conn, |
| 482 | const uint8_t *cm2, uint8_t cm2_len, |
| 483 | const uint8_t *cm3, uint8_t cm3_len) |
| 484 | { |
Harald Welte | cd4acaf | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 485 | struct msgb *resp; |
Holger Hans Peter Freyther | 93b0a14 | 2012-03-16 11:49:46 +0100 | [diff] [blame] | 486 | return_when_not_connected(conn); |
Harald Welte | cd4acaf | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 487 | |
| 488 | resp = gsm0808_create_classmark_update(cm2, cm2_len, cm3, cm3_len); |
| 489 | |
| 490 | queue_msg_or_return(resp); |
| 491 | } |
| 492 | |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 493 | static void bsc_mr_config(struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | 38eb3dc | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 494 | struct gsm_lchan *lchan, int full_rate) |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 495 | { |
| 496 | struct osmo_msc_data *msc; |
Holger Hans Peter Freyther | 38eb3dc | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 497 | struct gsm48_multi_rate_conf *ms_conf, *bts_conf; |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 498 | |
| 499 | if (!conn->sccp_con) { |
| 500 | LOGP(DMSC, LOGL_ERROR, |
| 501 | "No msc data available on conn %p. Audio will be broken.\n", |
| 502 | conn); |
| 503 | return; |
| 504 | } |
| 505 | |
| 506 | msc = conn->sccp_con->msc; |
| 507 | |
Holger Hans Peter Freyther | 38eb3dc | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 508 | /* initialize the data structure */ |
| 509 | lchan->mr_ms_lv[0] = sizeof(*ms_conf); |
| 510 | lchan->mr_bts_lv[0] = sizeof(*bts_conf); |
| 511 | ms_conf = (struct gsm48_multi_rate_conf *) &lchan->mr_ms_lv[1]; |
| 512 | bts_conf = (struct gsm48_multi_rate_conf *) &lchan->mr_bts_lv[1]; |
| 513 | memset(ms_conf, 0, sizeof(*ms_conf)); |
| 514 | memset(bts_conf, 0, sizeof(*bts_conf)); |
| 515 | |
| 516 | bts_conf->ver = ms_conf->ver = 1; |
| 517 | bts_conf->icmi = ms_conf->icmi = 1; |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 518 | |
| 519 | /* maybe gcc see's it is copy of _one_ byte */ |
Holger Hans Peter Freyther | 38eb3dc | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 520 | bts_conf->m4_75 = ms_conf->m4_75 = msc->amr_conf.m4_75; |
| 521 | bts_conf->m5_15 = ms_conf->m5_15 = msc->amr_conf.m5_15; |
| 522 | bts_conf->m5_90 = ms_conf->m5_90 = msc->amr_conf.m5_90; |
| 523 | bts_conf->m6_70 = ms_conf->m6_70 = msc->amr_conf.m6_70; |
| 524 | bts_conf->m7_40 = ms_conf->m7_40 = msc->amr_conf.m7_40; |
| 525 | bts_conf->m7_95 = ms_conf->m7_95 = msc->amr_conf.m7_95; |
| 526 | if (full_rate) { |
| 527 | bts_conf->m10_2 = ms_conf->m10_2 = msc->amr_conf.m10_2; |
| 528 | bts_conf->m12_2 = ms_conf->m12_2 = msc->amr_conf.m12_2; |
| 529 | } |
| 530 | |
| 531 | /* now copy this into the bts structure */ |
| 532 | memcpy(lchan->mr_bts_lv, lchan->mr_ms_lv, sizeof(lchan->mr_ms_lv)); |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 533 | } |
| 534 | |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 535 | static struct bsc_api bsc_handler = { |
| 536 | .sapi_n_reject = bsc_sapi_n_reject, |
| 537 | .cipher_mode_compl = bsc_cipher_mode_compl, |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 538 | .compl_l3 = bsc_compl_l3, |
| 539 | .dtap = bsc_dtap, |
| 540 | .assign_compl = bsc_assign_compl, |
| 541 | .assign_fail = bsc_assign_fail, |
| 542 | .clear_request = bsc_clear_request, |
Harald Welte | cd4acaf | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 543 | .classmark_chg = bsc_cm_update, |
Holger Hans Peter Freyther | ce6d3d7 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 544 | .mr_config = bsc_mr_config, |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 545 | }; |
| 546 | |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 547 | struct bsc_api *osmo_bsc_api() |
| 548 | { |
Holger Hans Peter Freyther | 7d23130 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 549 | return &bsc_handler; |
Holger Hans Peter Freyther | 22e9ac3 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 550 | } |