blob: 2389980d3eb228e7131d78f0e8acc893f01c0155 [file] [log] [blame]
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +08001/* main MSC management code... */
2
3/*
Holger Hans Peter Freythere7bd8632013-06-30 15:30:47 +02004 * (C) 2010,2013 by Holger Hans Peter Freyther <zecke@selfish.org>
Holger Hans Peter Freyther85531cc2010-10-06 20:37:09 +08005 * (C) 2010 by On-Waves
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +08006 *
7 * All Rights Reserved
8 *
9 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +010010 * it under the terms of the GNU Affero General Public License as published by
11 * the Free Software Foundation; either version 3 of the License, or
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080012 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte9af6ddf2011-01-01 15:25:50 +010017 * GNU Affero General Public License for more details.
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080018 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010019 * You should have received a copy of the GNU Affero General Public License
20 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080021 *
22 */
23
24#include <openbsc/bsc_api.h>
25#include <openbsc/debug.h>
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +080026#include <openbsc/transaction.h>
Harald Welte95e862c2012-01-23 10:28:35 +010027#include <openbsc/db.h>
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080028
Holger Hans Peter Freyther6a3d7652010-06-15 12:03:10 +080029#include <openbsc/gsm_04_11.h>
30
Holger Hans Peter Freytheradb6e1c2010-09-18 06:44:24 +080031static void msc_sapi_n_reject(struct gsm_subscriber_connection *conn, int dlci)
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080032{
Holger Hans Peter Freyther6a3d7652010-06-15 12:03:10 +080033 int sapi = dlci & 0x7;
34
35 if (sapi == UM_SAPI_SMS)
36 gsm411_sapi_n_reject(conn);
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +080037}
38
Holger Hans Peter Freyther05c68842010-11-03 19:01:58 +010039static int msc_clear_request(struct gsm_subscriber_connection *conn, uint32_t cause)
Holger Hans Peter Freytherf6fb3ef2010-06-15 13:16:52 +080040{
41 gsm0408_clear_request(conn, cause);
Holger Hans Peter Freyther05c68842010-11-03 19:01:58 +010042 return 1;
Holger Hans Peter Freytherf6fb3ef2010-06-15 13:16:52 +080043}
44
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080045static int msc_compl_l3(struct gsm_subscriber_connection *conn, struct msgb *msg,
46 uint16_t chosen_channel)
47{
Holger Hans Peter Freyther02d39b22010-07-05 15:34:16 +080048 gsm0408_new_conn(conn);
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080049 gsm0408_dispatch(conn, msg);
50
Holger Hans Peter Freythere9f420d2016-02-10 10:42:20 +010051 /*
52 * If this is a silent call we want the channel to remain open as long as
53 * possible and this is why we accept this connection regardless of any
54 * pending transaction or ongoing operation.
55 */
Holger Hans Peter Freyther70ae5d32012-11-23 21:33:15 +010056 if (conn->silent_call)
57 return BSC_API_CONN_POL_ACCEPT;
58 if (conn->loc_operation || conn->sec_operation || conn->anch_operation)
59 return BSC_API_CONN_POL_ACCEPT;
60 if (trans_has_conn(conn))
61 return BSC_API_CONN_POL_ACCEPT;
Jacob Erlbeck8e68b562014-01-30 21:01:12 +010062
63 LOGP(DRR, LOGL_INFO, "MSC Complete L3: Rejecting connection.\n");
Holger Hans Peter Freyther70ae5d32012-11-23 21:33:15 +010064 return BSC_API_CONN_POL_REJECT;
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080065}
66
Holger Hans Peter Freyther46caa302010-11-04 12:18:00 +010067static void msc_dtap(struct gsm_subscriber_connection *conn, uint8_t link_id, struct msgb *msg)
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080068{
69 gsm0408_dispatch(conn, msg);
70}
71
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010072static void msc_assign_compl(struct gsm_subscriber_connection *conn,
73 uint8_t rr_cause, uint8_t chosen_channel,
74 uint8_t encr_alg_id, uint8_t speec)
75{
Jacob Erlbeck8e68b562014-01-30 21:01:12 +010076 LOGP(DRR, LOGL_DEBUG, "MSC assign complete (do nothing).\n");
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010077}
78
79static void msc_assign_fail(struct gsm_subscriber_connection *conn,
80 uint8_t cause, uint8_t *rr_cause)
81{
Jacob Erlbeck8e68b562014-01-30 21:01:12 +010082 LOGP(DRR, LOGL_DEBUG, "MSC assign failure (do nothing).\n");
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010083}
84
Harald Welte95e862c2012-01-23 10:28:35 +010085static void msc_classmark_chg(struct gsm_subscriber_connection *conn,
86 const uint8_t *cm2, uint8_t cm2_len,
87 const uint8_t *cm3, uint8_t cm3_len)
88{
89 struct gsm_subscriber *subscr = conn->subscr;
90
91 if (subscr) {
92 subscr->equipment.classmark2_len = cm2_len;
93 memcpy(subscr->equipment.classmark2, cm2, cm2_len);
94 if (cm3) {
95 subscr->equipment.classmark3_len = cm3_len;
96 memcpy(subscr->equipment.classmark3, cm3, cm3_len);
97 }
98 db_sync_equipment(&subscr->equipment);
99 }
100}
101
Harald Weltecf149ee2012-01-23 16:40:24 +0100102static void msc_ciph_m_compl(struct gsm_subscriber_connection *conn,
103 struct msgb *msg, uint8_t alg_id)
104{
105 gsm_cbfn *cb;
106
107 DEBUGP(DRR, "CIPHERING MODE COMPLETE\n");
108
109 /* Safety check */
110 if (!conn->sec_operation) {
111 DEBUGP(DRR, "No authentication/cipher operation in progress !!!\n");
112 return;
113 }
114
115 /* FIXME: check for MI (if any) */
116
117 /* Call back whatever was in progress (if anything) ... */
118 cb = conn->sec_operation->cb;
119 if (cb) {
Holger Hans Peter Freythera5050b12012-09-11 11:55:03 +0200120 cb(GSM_HOOK_RR_SECURITY, GSM_SECURITY_SUCCEEDED,
Harald Weltecf149ee2012-01-23 16:40:24 +0100121 NULL, conn, conn->sec_operation->cb_data);
122
123 }
124
125 /* Complete the operation */
126 release_security_operation(conn);
127}
128
Harald Welte95e862c2012-01-23 10:28:35 +0100129
130
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +0800131static struct bsc_api msc_handler = {
132 .sapi_n_reject = msc_sapi_n_reject,
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +0800133 .compl_l3 = msc_compl_l3,
Neels Hofmeyrcc7db182016-12-18 23:52:38 +0100134 .dtap = msc_dtap,
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +0100135 .clear_request = msc_clear_request,
136 .assign_compl = msc_assign_compl,
137 .assign_fail = msc_assign_fail,
Harald Welte95e862c2012-01-23 10:28:35 +0100138 .classmark_chg = msc_classmark_chg,
Harald Weltecf149ee2012-01-23 16:40:24 +0100139 .cipher_mode_compl = msc_ciph_m_compl,
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +0800140};
141
142struct bsc_api *msc_bsc_api() {
143 return &msc_handler;
144}
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800145
146/* lchan release handling */
147void msc_release_connection(struct gsm_subscriber_connection *conn)
148{
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800149 /* skip when we are in release, e.g. due an error */
150 if (conn->in_release)
151 return;
152
153 /* skip releasing of silent calls as they have no transaction */
154 if (conn->silent_call)
155 return;
156
157 /* check if there is a pending operation */
Holger Hans Peter Freyther02d39b22010-07-05 15:34:16 +0800158 if (conn->loc_operation || conn->sec_operation || conn->anch_operation)
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800159 return;
160
Holger Hans Peter Freyther70ae5d32012-11-23 21:33:15 +0100161 if (trans_has_conn(conn))
162 return;
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800163
164 /* no more connections, asking to release the channel */
Holger Hans Peter Freythere7bd8632013-06-30 15:30:47 +0200165
166 /*
167 * We had stopped the LU expire timer T3212. Now we are about
168 * to send the MS back to the idle state and this should lead
169 * to restarting the timer. Set the new expiration time.
170 */
171 if (conn->expire_timer_stopped)
172 subscr_update_expire_lu(conn->subscr, conn->bts);
173
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800174 conn->in_release = 1;
175 gsm0808_clear(conn);
Neels Hofmeyr42eb0142016-05-20 17:15:44 +0200176 msc_subscr_con_free(conn);
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800177}