Holger Hans Peter Freyther | ec0cb7c | 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 | 8ec4952 | 2011-08-15 15:53:00 +0200 | [diff] [blame] | 2 | * (C) 2009-2011 by On-Waves |
Holger Hans Peter Freyther | 4f448c9 | 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 | 9af6ddf | 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 | 4f448c9 | 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 | 9af6ddf | 2011-01-01 15:25:50 +0100 | [diff] [blame] | 13 | * GNU Affero General Public License for more details. |
Holger Hans Peter Freyther | 4f448c9 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 14 | * |
Harald Welte | 9af6ddf | 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 | 4f448c9 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 17 | * |
| 18 | */ |
| 19 | |
| 20 | #include <openbsc/osmo_bsc.h> |
Neels Hofmeyr | a42855f | 2017-02-23 21:49:55 +0100 | [diff] [blame] | 21 | #include <openbsc/bsc_msc_data.h> |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 22 | #include <openbsc/debug.h> |
| 23 | |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 24 | #include <openbsc/gsm_04_80.h> |
| 25 | |
Harald Welte | d36ff76 | 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 | 2d2c910 | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 28 | |
Holger Hans Peter Freyther | 9b9a171 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 29 | #include <osmocom/sccp/sccp.h> |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 30 | #include <openbsc/osmo_bsc_sigtran.h> |
Holger Hans Peter Freyther | 9b9a171 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 31 | |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 32 | #define return_when_not_connected(conn) \ |
| 33 | if (!conn->sccp_con) {\ |
| 34 | LOGP(DMSC, LOGL_ERROR, "MSC Connection not present.\n"); \ |
| 35 | return; \ |
| 36 | } |
| 37 | |
| 38 | #define return_when_not_connected_val(conn, ret) \ |
| 39 | if (!conn->sccp_con) {\ |
| 40 | LOGP(DMSC, LOGL_ERROR, "MSC Connection not present.\n"); \ |
| 41 | return ret; \ |
| 42 | } |
Holger Hans Peter Freyther | 4f448c9 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 43 | |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 44 | #define queue_msg_or_return(resp) \ |
| 45 | if (!resp) { \ |
| 46 | LOGP(DMSC, LOGL_ERROR, "Failed to allocate response.\n"); \ |
| 47 | return; \ |
| 48 | } \ |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 49 | osmo_bsc_sigtran_send(conn->sccp_con, resp); |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 50 | |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 51 | static int bsc_clear_request(struct gsm_subscriber_connection *conn, uint32_t cause); |
| 52 | static int complete_layer3(struct gsm_subscriber_connection *conn, |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 53 | struct msgb *msg, struct bsc_msc_data *msc); |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 54 | |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 55 | static uint16_t get_network_code_for_msc(struct bsc_msc_data *msc) |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 56 | { |
Holger Hans Peter Freyther | 1e36547 | 2015-07-13 11:06:10 +0200 | [diff] [blame] | 57 | if (msc->core_mnc != -1) |
| 58 | return msc->core_mnc; |
Holger Hans Peter Freyther | 4cdb050 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 59 | return msc->network->network_code; |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 60 | } |
| 61 | |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 62 | static uint16_t get_country_code_for_msc(struct bsc_msc_data *msc) |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 63 | { |
Holger Hans Peter Freyther | 4cdb050 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 64 | if (msc->core_mcc != -1) |
| 65 | return msc->core_mcc; |
| 66 | return msc->network->country_code; |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 67 | } |
| 68 | |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 69 | static uint16_t get_lac_for_msc(struct bsc_msc_data *msc, struct gsm_bts *bts) |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 70 | { |
| 71 | if (msc->core_lac != -1) |
| 72 | return msc->core_lac; |
| 73 | return bts->location_area_code; |
| 74 | } |
| 75 | |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 76 | static uint16_t get_ci_for_msc(struct bsc_msc_data *msc, struct gsm_bts *bts) |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 77 | { |
| 78 | if (msc->core_ci != -1) |
| 79 | return msc->core_ci; |
| 80 | return bts->cell_identity; |
| 81 | } |
| 82 | |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 83 | static void bsc_maybe_lu_reject(struct gsm_subscriber_connection *conn, int con_type, int cause) |
| 84 | { |
| 85 | struct msgb *msg; |
| 86 | |
| 87 | /* ignore cm service request or such */ |
| 88 | if (con_type != FLT_CON_TYPE_LU) |
| 89 | return; |
| 90 | |
| 91 | msg = gsm48_create_loc_upd_rej(cause); |
| 92 | if (!msg) { |
| 93 | LOGP(DMM, LOGL_ERROR, "Failed to create msg for LOCATION UPDATING REJECT.\n"); |
| 94 | return; |
| 95 | } |
| 96 | |
| 97 | msg->lchan = conn->lchan; |
| 98 | gsm0808_submit_dtap(conn, msg, 0, 0); |
| 99 | } |
| 100 | |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 101 | static int bsc_filter_initial(struct osmo_bsc_data *bsc, |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 102 | struct bsc_msc_data *msc, |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 103 | struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 104 | struct msgb *msg, char **imsi, int *con_type, |
| 105 | int *lu_cause) |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 106 | { |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 107 | struct bsc_filter_request req; |
| 108 | struct bsc_filter_reject_cause cause; |
| 109 | struct gsm48_hdr *gh = msgb_l3(msg); |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 110 | int rc; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 111 | |
| 112 | req.ctx = conn; |
| 113 | req.black_list = NULL; |
| 114 | req.access_lists = bsc_access_lists(); |
| 115 | req.local_lst_name = msc->acc_lst_name; |
| 116 | req.global_lst_name = conn->bts->network->bsc_data->acc_lst_name; |
| 117 | req.bsc_nr = 0; |
| 118 | |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 119 | rc = bsc_msg_filter_initial(gh, msgb_l3len(msg), &req, |
| 120 | con_type, imsi, &cause); |
| 121 | *lu_cause = cause.lu_reject_cause; |
| 122 | return rc; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | static int bsc_filter_data(struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 126 | struct msgb *msg, int *lu_cause) |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 127 | { |
| 128 | struct bsc_filter_request req; |
| 129 | struct gsm48_hdr *gh = msgb_l3(msg); |
| 130 | struct bsc_filter_reject_cause cause; |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 131 | int rc; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 132 | |
| 133 | req.ctx = conn; |
| 134 | req.black_list = NULL; |
| 135 | req.access_lists = bsc_access_lists(); |
| 136 | req.local_lst_name = conn->sccp_con->msc->acc_lst_name; |
| 137 | req.global_lst_name = conn->bts->network->bsc_data->acc_lst_name; |
| 138 | req.bsc_nr = 0; |
| 139 | |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 140 | rc = bsc_msg_filter_data(gh, msgb_l3len(msg), &req, |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 141 | &conn->sccp_con->filter_state, |
| 142 | &cause); |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 143 | *lu_cause = cause.lu_reject_cause; |
| 144 | return rc; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 145 | } |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 146 | |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 147 | static void bsc_sapi_n_reject(struct gsm_subscriber_connection *conn, int dlci) |
| 148 | { |
Holger Hans Peter Freyther | c8166f3 | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 149 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 150 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | c8166f3 | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 151 | |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 152 | LOGP(DMSC, LOGL_NOTICE, "Tx MSC SAPI N REJECT DLCI=0x%02x\n", dlci); |
| 153 | |
Holger Hans Peter Freyther | c8166f3 | 2010-11-04 12:28:32 +0100 | [diff] [blame] | 154 | resp = gsm0808_create_sapi_reject(dlci); |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 155 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 156 | } |
| 157 | |
| 158 | static void bsc_cipher_mode_compl(struct gsm_subscriber_connection *conn, |
| 159 | struct msgb *msg, uint8_t chosen_encr) |
| 160 | { |
Holger Hans Peter Freyther | 2d2c910 | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 161 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 162 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | 2d2c910 | 2010-11-04 11:59:41 +0100 | [diff] [blame] | 163 | |
| 164 | LOGP(DMSC, LOGL_DEBUG, "CIPHER MODE COMPLETE from MS, forwarding to MSC\n"); |
| 165 | resp = gsm0808_create_cipher_complete(msg, chosen_encr); |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 166 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 167 | } |
| 168 | |
Jacob Erlbeck | aff2d62 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 169 | static void bsc_send_ussd_no_srv(struct gsm_subscriber_connection *conn, |
| 170 | struct msgb *msg, const char *text) |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 171 | { |
| 172 | struct gsm48_hdr *gh; |
| 173 | int8_t pdisc; |
| 174 | uint8_t mtype; |
| 175 | int drop_message = 1; |
| 176 | |
| 177 | if (!text) |
| 178 | return; |
| 179 | |
| 180 | if (!msg || msgb_l3len(msg) < sizeof(*gh)) |
| 181 | return; |
| 182 | |
| 183 | gh = msgb_l3(msg); |
Neels Hofmeyr | 531734a | 2016-03-14 16:13:24 +0100 | [diff] [blame] | 184 | pdisc = gsm48_hdr_pdisc(gh); |
| 185 | mtype = gsm48_hdr_msg_type(gh); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 186 | |
| 187 | /* Is CM service request? */ |
| 188 | if (pdisc == GSM48_PDISC_MM && mtype == GSM48_MT_MM_CM_SERV_REQ) { |
| 189 | struct gsm48_service_request *cm; |
| 190 | |
| 191 | cm = (struct gsm48_service_request *) &gh->data[0]; |
| 192 | |
| 193 | /* Is type SMS or call? */ |
| 194 | if (cm->cm_service_type == GSM48_CMSERV_SMS) |
| 195 | drop_message = 0; |
| 196 | else if (cm->cm_service_type == GSM48_CMSERV_MO_CALL_PACKET) |
| 197 | drop_message = 0; |
| 198 | } |
| 199 | |
| 200 | if (drop_message) { |
Holger Hans Peter Freyther | 28e183f | 2013-10-31 13:35:28 +0100 | [diff] [blame] | 201 | LOGP(DMSC, LOGL_DEBUG, "Skipping (not sending) USSD message: '%s'\n", text); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 202 | return; |
| 203 | } |
| 204 | |
Jacob Erlbeck | b125031 | 2013-10-31 15:36:42 +0100 | [diff] [blame] | 205 | LOGP(DMSC, LOGL_INFO, "Sending CM Service Accept\n"); |
| 206 | gsm48_tx_mm_serv_ack(conn); |
| 207 | |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 208 | LOGP(DMSC, LOGL_INFO, "Sending USSD message: '%s'\n", text); |
Neels Hofmeyr | 43273c6 | 2016-05-10 12:50:31 +0200 | [diff] [blame] | 209 | bsc_send_ussd_notify(conn, 1, text); |
| 210 | bsc_send_ussd_release_complete(conn); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 211 | } |
| 212 | |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 213 | /* |
| 214 | * Instruct to reserve data for a new connectiom, create the complete |
| 215 | * layer three message, send it to open the connection. |
| 216 | */ |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 217 | static int bsc_compl_l3(struct gsm_subscriber_connection *conn, struct msgb *msg, |
| 218 | uint16_t chosen_channel) |
| 219 | { |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 220 | struct bsc_msc_data *msc; |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 221 | |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 222 | LOGP(DMSC, LOGL_INFO, "Tx MSC COMPL L3\n"); |
| 223 | |
Holger Hans Peter Freyther | 354c87c | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 224 | /* find the MSC link we want to use */ |
Holger Hans Peter Freyther | 076af1c | 2011-06-07 19:57:02 +0200 | [diff] [blame] | 225 | msc = bsc_find_msc(conn, msg); |
Holger Hans Peter Freyther | 354c87c | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 226 | if (!msc) { |
| 227 | LOGP(DMSC, LOGL_ERROR, "Failed to find a MSC for a connection.\n"); |
Jacob Erlbeck | aff2d62 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 228 | bsc_send_ussd_no_srv(conn, msg, |
| 229 | conn->bts->network->bsc_data->ussd_no_msc_txt); |
Holger Hans Peter Freyther | 354c87c | 2011-06-07 11:40:20 +0200 | [diff] [blame] | 230 | return -1; |
| 231 | } |
| 232 | |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 233 | return complete_layer3(conn, msg, msc); |
| 234 | } |
| 235 | |
| 236 | static int complete_layer3(struct gsm_subscriber_connection *conn, |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 237 | struct msgb *msg, struct bsc_msc_data *msc) |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 238 | { |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 239 | int con_type, rc, lu_cause; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 240 | char *imsi = NULL; |
Holger Hans Peter Freyther | 56cb729 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 241 | struct timeval tv; |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 242 | struct msgb *resp; |
| 243 | uint16_t network_code; |
| 244 | uint16_t country_code; |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 245 | uint16_t lac; |
| 246 | uint16_t ci; |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 247 | enum bsc_con ret; |
Holger Hans Peter Freyther | 56cb729 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 248 | int send_ping = msc->advanced_ping; |
| 249 | |
| 250 | /* Advanced ping/pong handling */ |
| 251 | if (osmo_timer_pending(&msc->pong_timer)) |
| 252 | send_ping = 0; |
Holger Hans Peter Freyther | db64f2e | 2014-10-29 10:06:15 +0100 | [diff] [blame] | 253 | if (msc->ping_timeout <= 0) |
Holger Hans Peter Freyther | 56cb729 | 2014-09-02 17:28:40 +0200 | [diff] [blame] | 254 | send_ping = 0; |
| 255 | if (send_ping && osmo_timer_remaining(&msc->ping_timer, NULL, &tv) == -1) |
| 256 | send_ping = 0; |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 257 | |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 258 | /* Check the filter */ |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 259 | rc = bsc_filter_initial(msc->network->bsc_data, msc, conn, msg, |
| 260 | &imsi, &con_type, &lu_cause); |
| 261 | if (rc < 0) { |
| 262 | bsc_maybe_lu_reject(conn, con_type, lu_cause); |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 263 | return BSC_API_CONN_POL_REJECT; |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 264 | } |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 265 | |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 266 | /* allocate resource for a new connection */ |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 267 | //ret = bsc_create_new_connection(conn, msc, send_ping); |
| 268 | ret = osmo_bsc_sigtran_new_conn(conn, msc); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 269 | |
| 270 | if (ret != BSC_CON_SUCCESS) { |
| 271 | /* allocation has failed */ |
| 272 | if (ret == BSC_CON_REJECT_NO_LINK) |
Jacob Erlbeck | aff2d62 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 273 | bsc_send_ussd_no_srv(conn, msg, msc->ussd_msc_lost_txt); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 274 | else if (ret == BSC_CON_REJECT_RF_GRACE) |
Jacob Erlbeck | aff2d62 | 2013-10-31 15:36:43 +0100 | [diff] [blame] | 275 | bsc_send_ussd_no_srv(conn, msg, msc->ussd_grace_txt); |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 276 | |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 277 | return BSC_API_CONN_POL_REJECT; |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 278 | } |
| 279 | |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 280 | if (imsi) |
| 281 | conn->sccp_con->filter_state.imsi = talloc_steal(conn, imsi); |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 282 | conn->sccp_con->filter_state.con_type = con_type; |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 283 | |
Jacob Erlbeck | 56595f8 | 2013-09-11 10:46:55 +0200 | [diff] [blame] | 284 | /* check return value, if failed check msg for and send USSD */ |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 285 | |
Holger Hans Peter Freyther | 4cdb050 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 286 | network_code = get_network_code_for_msc(conn->sccp_con->msc); |
| 287 | country_code = get_country_code_for_msc(conn->sccp_con->msc); |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 288 | lac = get_lac_for_msc(conn->sccp_con->msc, conn->bts); |
| 289 | ci = get_ci_for_msc(conn->sccp_con->msc, conn->bts); |
Holger Hans Peter Freyther | 4cdb050 | 2011-06-04 14:51:51 +0200 | [diff] [blame] | 290 | |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 291 | bsc_scan_bts_msg(conn, msg); |
Holger Hans Peter Freyther | c652913 | 2015-04-05 21:03:49 +0200 | [diff] [blame] | 292 | |
Holger Hans Peter Freyther | 32dd2f3 | 2015-04-01 18:15:48 +0200 | [diff] [blame] | 293 | resp = gsm0808_create_layer3(msg, network_code, country_code, lac, ci); |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 294 | if (!resp) { |
| 295 | LOGP(DMSC, LOGL_DEBUG, "Failed to create layer3 message.\n"); |
Holger Hans Peter Freyther | 9b9a171 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 296 | sccp_connection_free(conn->sccp_con->sccp); |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 297 | osmo_bsc_sigtran_del_conn(conn->sccp_con); |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 298 | return BSC_API_CONN_POL_REJECT; |
Holger Hans Peter Freyther | 5bb874d | 2010-11-05 11:21:18 +0100 | [diff] [blame] | 299 | } |
| 300 | |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 301 | if (osmo_bsc_sigtran_open_conn(conn->sccp_con, resp) != 0) { |
Holger Hans Peter Freyther | 9b9a171 | 2011-06-09 14:44:47 +0200 | [diff] [blame] | 302 | sccp_connection_free(conn->sccp_con->sccp); |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 303 | osmo_bsc_sigtran_del_conn(conn->sccp_con); |
Holger Hans Peter Freyther | 9b17c76 | 2010-11-05 19:48:47 +0100 | [diff] [blame] | 304 | msgb_free(resp); |
| 305 | return BSC_API_CONN_POL_REJECT; |
| 306 | } |
| 307 | |
| 308 | return BSC_API_CONN_POL_ACCEPT; |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 309 | } |
| 310 | |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 311 | /* |
| 312 | * Plastic surgery... we want to give up the current connection |
| 313 | */ |
| 314 | static int move_to_msc(struct gsm_subscriber_connection *_conn, |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 315 | struct msgb *msg, struct bsc_msc_data *msc) |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 316 | { |
| 317 | struct osmo_bsc_sccp_con *old_con = _conn->sccp_con; |
| 318 | |
| 319 | /* |
| 320 | * 1. Give up the old connection. |
| 321 | * This happens by sending a clear request to the MSC, |
| 322 | * it should end with the MSC releasing the connection. |
| 323 | */ |
| 324 | old_con->conn = NULL; |
| 325 | bsc_clear_request(_conn, 0); |
| 326 | |
| 327 | /* |
| 328 | * 2. Attempt to create a new connection to the local |
| 329 | * MSC. If it fails the caller will need to handle this |
| 330 | * properly. |
| 331 | */ |
| 332 | _conn->sccp_con = NULL; |
| 333 | if (complete_layer3(_conn, msg, msc) != BSC_API_CONN_POL_ACCEPT) { |
| 334 | gsm0808_clear(_conn); |
Neels Hofmeyr | 42eb014 | 2016-05-20 17:15:44 +0200 | [diff] [blame] | 335 | bsc_subscr_con_free(_conn); |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 336 | return 1; |
| 337 | } |
| 338 | |
| 339 | return 2; |
| 340 | } |
| 341 | |
| 342 | static int handle_cc_setup(struct gsm_subscriber_connection *conn, |
| 343 | struct msgb *msg) |
| 344 | { |
| 345 | struct gsm48_hdr *gh = msgb_l3(msg); |
Neels Hofmeyr | 531734a | 2016-03-14 16:13:24 +0100 | [diff] [blame] | 346 | uint8_t pdisc = gsm48_hdr_pdisc(gh); |
| 347 | uint8_t mtype = gsm48_hdr_msg_type(gh); |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 348 | |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 349 | struct bsc_msc_data *msc; |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 350 | struct gsm_mncc_number called; |
| 351 | struct tlv_parsed tp; |
| 352 | unsigned payload_len; |
| 353 | |
| 354 | char _dest_nr[35]; |
| 355 | |
| 356 | /* |
| 357 | * Do we have a setup message here? if not return fast. |
| 358 | */ |
| 359 | if (pdisc != GSM48_PDISC_CC || mtype != GSM48_MT_CC_SETUP) |
| 360 | return 0; |
| 361 | |
| 362 | payload_len = msgb_l3len(msg) - sizeof(*gh); |
| 363 | |
| 364 | tlv_parse(&tp, &gsm48_att_tlvdef, gh->data, payload_len, 0, 0); |
| 365 | if (!TLVP_PRESENT(&tp, GSM48_IE_CALLED_BCD)) { |
| 366 | LOGP(DMSC, LOGL_ERROR, "Called BCD not present in setup.\n"); |
| 367 | return -1; |
| 368 | } |
| 369 | |
| 370 | memset(&called, 0, sizeof(called)); |
| 371 | gsm48_decode_called(&called, |
| 372 | TLVP_VAL(&tp, GSM48_IE_CALLED_BCD) - 1); |
| 373 | |
Holger Hans Peter Freyther | 0b19d55 | 2011-09-30 16:24:27 +0200 | [diff] [blame] | 374 | if (called.plan != 1 && called.plan != 0) |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 375 | return 0; |
| 376 | |
Holger Hans Peter Freyther | 0b19d55 | 2011-09-30 16:24:27 +0200 | [diff] [blame] | 377 | if (called.plan == 1 && called.type == 1) { |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 378 | _dest_nr[0] = _dest_nr[1] = '0'; |
| 379 | memcpy(_dest_nr + 2, called.number, sizeof(called.number)); |
| 380 | } else |
| 381 | memcpy(_dest_nr, called.number, sizeof(called.number)); |
| 382 | |
| 383 | /* |
| 384 | * Check if the connection should be moved... |
| 385 | */ |
| 386 | llist_for_each_entry(msc, &conn->bts->network->bsc_data->mscs, entry) { |
| 387 | if (msc->type != MSC_CON_TYPE_LOCAL) |
| 388 | continue; |
| 389 | if (!msc->local_pref) |
| 390 | continue; |
| 391 | if (regexec(&msc->local_pref_reg, _dest_nr, 0, NULL, 0) != 0) |
| 392 | continue; |
| 393 | |
| 394 | return move_to_msc(conn, msg, msc); |
| 395 | } |
| 396 | |
| 397 | return 0; |
| 398 | } |
| 399 | |
| 400 | |
Holger Hans Peter Freyther | 46caa30 | 2010-11-04 12:18:00 +0100 | [diff] [blame] | 401 | static void bsc_dtap(struct gsm_subscriber_connection *conn, uint8_t link_id, struct msgb *msg) |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 402 | { |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 403 | int lu_cause; |
Holger Hans Peter Freyther | 58f2cf6 | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 404 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 405 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | 58f2cf6 | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 406 | |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 407 | LOGP(DMSC, LOGL_INFO, "Tx MSC DTAP LINK_ID=0x%02x\n", link_id); |
| 408 | |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 409 | /* |
| 410 | * We might want to move this connection to a new MSC. Ask someone |
| 411 | * to handle it. If it was handled we will return. |
| 412 | */ |
Holger Hans Peter Freyther | c7db4dc | 2011-06-27 21:29:03 +0200 | [diff] [blame] | 413 | if (handle_cc_setup(conn, msg) >= 1) |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 414 | return; |
| 415 | |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 416 | /* Check the filter */ |
Holger Hans Peter Freyther | d26b8fc | 2015-04-05 22:45:32 +0200 | [diff] [blame] | 417 | if (bsc_filter_data(conn, msg, &lu_cause) < 0) { |
| 418 | bsc_maybe_lu_reject(conn, |
| 419 | conn->sccp_con->filter_state.con_type, |
| 420 | lu_cause); |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 421 | bsc_clear_request(conn, 0); |
| 422 | return; |
| 423 | } |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 424 | |
Holger Hans Peter Freyther | ec0cb7c | 2015-04-05 20:53:42 +0200 | [diff] [blame] | 425 | bsc_scan_bts_msg(conn, msg); |
Holger Hans Peter Freyther | 83d2d38 | 2011-06-09 17:37:04 +0200 | [diff] [blame] | 426 | |
Holger Hans Peter Freyther | 58f2cf6 | 2010-11-04 12:24:05 +0100 | [diff] [blame] | 427 | resp = gsm0808_create_dtap(msg, link_id); |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 428 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 429 | } |
| 430 | |
Holger Hans Peter Freyther | b094171 | 2010-11-04 12:06:57 +0100 | [diff] [blame] | 431 | static void bsc_assign_compl(struct gsm_subscriber_connection *conn, uint8_t rr_cause, |
| 432 | uint8_t chosen_channel, uint8_t encr_alg_id, |
| 433 | uint8_t speech_model) |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 434 | { |
Holger Hans Peter Freyther | da488ae | 2010-11-04 12:09:45 +0100 | [diff] [blame] | 435 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 436 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | da488ae | 2010-11-04 12:09:45 +0100 | [diff] [blame] | 437 | |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 438 | if (is_ipaccess_bts(conn->bts) && conn->sccp_con->rtp_ip) { |
| 439 | /* NOTE: In a network that makes use of an IPA base station |
| 440 | * and AoIP, we have to wait until the BTS reports its RTP |
| 441 | * IP/Port combination back to BSC via RSL. Unfortunately, the |
| 442 | * IPA protocol sends its Abis assignment complete message |
| 443 | * before it sends its RTP IP/Port via IPACC. So we will now |
| 444 | * postpone the AoIP assignment completed message until we |
| 445 | * know the RTP IP/Port combination. */ |
| 446 | LOGP(DMSC, LOGL_INFO, "POSTPONE MSC ASSIGN COMPL\n"); |
| 447 | conn->lchan->abis_ip.ass_compl.rr_cause = rr_cause; |
| 448 | conn->lchan->abis_ip.ass_compl.chosen_channel = chosen_channel; |
| 449 | conn->lchan->abis_ip.ass_compl.encr_alg_id = encr_alg_id; |
| 450 | conn->lchan->abis_ip.ass_compl.speech_mode = speech_model; |
| 451 | conn->lchan->abis_ip.ass_compl.valid = true; |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 452 | |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 453 | } else { |
| 454 | /* NOTE: Send the A assignment complete message immediately. */ |
| 455 | LOGP(DMSC, LOGL_INFO, "Tx MSC ASSIGN COMPL\n"); |
| 456 | resp = gsm0808_create_assignment_completed(rr_cause, chosen_channel, |
| 457 | encr_alg_id, speech_model); |
| 458 | queue_msg_or_return(resp); |
| 459 | } |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 460 | } |
| 461 | |
Holger Hans Peter Freyther | 45b575f | 2010-11-05 10:37:17 +0100 | [diff] [blame] | 462 | static void bsc_assign_fail(struct gsm_subscriber_connection *conn, |
| 463 | uint8_t cause, uint8_t *rr_cause) |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 464 | { |
Holger Hans Peter Freyther | e46f1d6 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 465 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 466 | return_when_not_connected(conn); |
Holger Hans Peter Freyther | e46f1d6 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 467 | |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 468 | LOGP(DMSC, LOGL_INFO, "Tx MSC ASSIGN FAIL\n"); |
| 469 | |
Holger Hans Peter Freyther | e46f1d6 | 2010-11-05 10:59:45 +0100 | [diff] [blame] | 470 | resp = gsm0808_create_assignment_failure(cause, rr_cause); |
Holger Hans Peter Freyther | 46c5ab3 | 2010-11-05 11:02:28 +0100 | [diff] [blame] | 471 | queue_msg_or_return(resp); |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 472 | } |
| 473 | |
Holger Hans Peter Freyther | 05c6884 | 2010-11-03 19:01:58 +0100 | [diff] [blame] | 474 | static int bsc_clear_request(struct gsm_subscriber_connection *conn, uint32_t cause) |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 475 | { |
Holger Hans Peter Freyther | 80c37bd | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 476 | struct osmo_bsc_sccp_con *sccp; |
Holger Hans Peter Freyther | c2b3270 | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 477 | struct msgb *resp; |
Holger Hans Peter Freyther | 7225fd1 | 2010-11-03 19:03:35 +0100 | [diff] [blame] | 478 | return_when_not_connected_val(conn, 1); |
Holger Hans Peter Freyther | c2b3270 | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 479 | |
Harald Welte | 7d33bdf | 2011-07-12 00:05:11 +0200 | [diff] [blame] | 480 | LOGP(DMSC, LOGL_INFO, "Tx MSC CLEAR REQUEST\n"); |
| 481 | |
Holger Hans Peter Freyther | 80c37bd | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 482 | /* |
| 483 | * Remove the connection from BSC<->SCCP part, the SCCP part |
| 484 | * will either be cleared by channel release or MSC disconnect |
| 485 | */ |
| 486 | sccp = conn->sccp_con; |
| 487 | sccp->conn = NULL; |
| 488 | conn->sccp_con = NULL; |
| 489 | |
Holger Hans Peter Freyther | c2b3270 | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 490 | resp = gsm0808_create_clear_rqst(GSM0808_CAUSE_RADIO_INTERFACE_FAILURE); |
| 491 | if (!resp) { |
| 492 | LOGP(DMSC, LOGL_ERROR, "Failed to allocate response.\n"); |
Holger Hans Peter Freyther | 80c37bd | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 493 | return 1; |
Holger Hans Peter Freyther | c2b3270 | 2010-11-04 12:47:06 +0100 | [diff] [blame] | 494 | } |
| 495 | |
Philipp Maier | fbf6610 | 2017-04-09 12:32:51 +0200 | [diff] [blame^] | 496 | osmo_bsc_sigtran_send(sccp, resp); |
Holger Hans Peter Freyther | 80c37bd | 2011-08-06 14:52:56 +0200 | [diff] [blame] | 497 | return 1; |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 498 | } |
| 499 | |
Harald Welte | 95e862c | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 500 | static void bsc_cm_update(struct gsm_subscriber_connection *conn, |
| 501 | const uint8_t *cm2, uint8_t cm2_len, |
| 502 | const uint8_t *cm3, uint8_t cm3_len) |
| 503 | { |
Harald Welte | 95e862c | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 504 | struct msgb *resp; |
Holger Hans Peter Freyther | dc3389f | 2012-03-16 11:49:46 +0100 | [diff] [blame] | 505 | return_when_not_connected(conn); |
Harald Welte | 95e862c | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 506 | |
| 507 | resp = gsm0808_create_classmark_update(cm2, cm2_len, cm3, cm3_len); |
| 508 | |
| 509 | queue_msg_or_return(resp); |
| 510 | } |
| 511 | |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 512 | static void bsc_mr_config(struct gsm_subscriber_connection *conn, |
Holger Hans Peter Freyther | 7a77d01 | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 513 | struct gsm_lchan *lchan, int full_rate) |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 514 | { |
Neels Hofmeyr | a369e24 | 2017-02-23 21:57:23 +0100 | [diff] [blame] | 515 | struct bsc_msc_data *msc; |
Holger Hans Peter Freyther | 7a77d01 | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 516 | struct gsm48_multi_rate_conf *ms_conf, *bts_conf; |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 517 | |
| 518 | if (!conn->sccp_con) { |
| 519 | LOGP(DMSC, LOGL_ERROR, |
| 520 | "No msc data available on conn %p. Audio will be broken.\n", |
| 521 | conn); |
| 522 | return; |
| 523 | } |
| 524 | |
| 525 | msc = conn->sccp_con->msc; |
| 526 | |
Holger Hans Peter Freyther | 7a77d01 | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 527 | /* initialize the data structure */ |
| 528 | lchan->mr_ms_lv[0] = sizeof(*ms_conf); |
| 529 | lchan->mr_bts_lv[0] = sizeof(*bts_conf); |
| 530 | ms_conf = (struct gsm48_multi_rate_conf *) &lchan->mr_ms_lv[1]; |
| 531 | bts_conf = (struct gsm48_multi_rate_conf *) &lchan->mr_bts_lv[1]; |
| 532 | memset(ms_conf, 0, sizeof(*ms_conf)); |
| 533 | memset(bts_conf, 0, sizeof(*bts_conf)); |
| 534 | |
| 535 | bts_conf->ver = ms_conf->ver = 1; |
| 536 | bts_conf->icmi = ms_conf->icmi = 1; |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 537 | |
| 538 | /* maybe gcc see's it is copy of _one_ byte */ |
Holger Hans Peter Freyther | 7a77d01 | 2015-09-24 16:26:01 +0200 | [diff] [blame] | 539 | bts_conf->m4_75 = ms_conf->m4_75 = msc->amr_conf.m4_75; |
| 540 | bts_conf->m5_15 = ms_conf->m5_15 = msc->amr_conf.m5_15; |
| 541 | bts_conf->m5_90 = ms_conf->m5_90 = msc->amr_conf.m5_90; |
| 542 | bts_conf->m6_70 = ms_conf->m6_70 = msc->amr_conf.m6_70; |
| 543 | bts_conf->m7_40 = ms_conf->m7_40 = msc->amr_conf.m7_40; |
| 544 | bts_conf->m7_95 = ms_conf->m7_95 = msc->amr_conf.m7_95; |
| 545 | if (full_rate) { |
| 546 | bts_conf->m10_2 = ms_conf->m10_2 = msc->amr_conf.m10_2; |
| 547 | bts_conf->m12_2 = ms_conf->m12_2 = msc->amr_conf.m12_2; |
| 548 | } |
| 549 | |
| 550 | /* now copy this into the bts structure */ |
| 551 | memcpy(lchan->mr_bts_lv, lchan->mr_ms_lv, sizeof(lchan->mr_ms_lv)); |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 552 | } |
| 553 | |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 554 | static struct bsc_api bsc_handler = { |
| 555 | .sapi_n_reject = bsc_sapi_n_reject, |
| 556 | .cipher_mode_compl = bsc_cipher_mode_compl, |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 557 | .compl_l3 = bsc_compl_l3, |
Neels Hofmeyr | 84da6b1 | 2016-05-20 21:59:55 +0200 | [diff] [blame] | 558 | .dtap = bsc_dtap, |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 559 | .assign_compl = bsc_assign_compl, |
| 560 | .assign_fail = bsc_assign_fail, |
| 561 | .clear_request = bsc_clear_request, |
Harald Welte | 95e862c | 2012-01-23 10:28:35 +0100 | [diff] [blame] | 562 | .classmark_chg = bsc_cm_update, |
Holger Hans Peter Freyther | 25aa749 | 2011-08-06 07:00:52 +0200 | [diff] [blame] | 563 | .mr_config = bsc_mr_config, |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 564 | }; |
| 565 | |
Holger Hans Peter Freyther | 4f448c9 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 566 | struct bsc_api *osmo_bsc_api() |
| 567 | { |
Holger Hans Peter Freyther | 32aaef6 | 2010-09-16 17:26:35 +0800 | [diff] [blame] | 568 | return &bsc_handler; |
Holger Hans Peter Freyther | 4f448c9 | 2010-07-05 16:02:04 +0800 | [diff] [blame] | 569 | } |