blob: 452de62d5352670ade0e11dd436ef0226be7f68f [file] [log] [blame]
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +08001/* main MSC management code... */
2
3/*
4 * (C) 2010 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 Freyther182c81f2010-12-29 16:28:33 +010042 if (conn->put_channel) {
43 conn->put_channel = 0;
44 subscr_put_channel(conn->subscr);
45 }
Holger Hans Peter Freyther05c68842010-11-03 19:01:58 +010046 return 1;
Holger Hans Peter Freytherf6fb3ef2010-06-15 13:16:52 +080047}
48
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080049static int msc_compl_l3(struct gsm_subscriber_connection *conn, struct msgb *msg,
50 uint16_t chosen_channel)
51{
Holger Hans Peter Freyther02d39b22010-07-05 15:34:16 +080052 gsm0408_new_conn(conn);
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080053 gsm0408_dispatch(conn, msg);
54
55 /* TODO: do better */
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;
62 return BSC_API_CONN_POL_REJECT;
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080063}
64
Holger Hans Peter Freyther46caa302010-11-04 12:18:00 +010065static void msc_dtap(struct gsm_subscriber_connection *conn, uint8_t link_id, struct msgb *msg)
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +080066{
67 gsm0408_dispatch(conn, msg);
68}
69
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010070static void msc_assign_compl(struct gsm_subscriber_connection *conn,
71 uint8_t rr_cause, uint8_t chosen_channel,
72 uint8_t encr_alg_id, uint8_t speec)
73{
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010074}
75
76static void msc_assign_fail(struct gsm_subscriber_connection *conn,
77 uint8_t cause, uint8_t *rr_cause)
78{
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +010079}
80
Harald Welte95e862c2012-01-23 10:28:35 +010081static void msc_classmark_chg(struct gsm_subscriber_connection *conn,
82 const uint8_t *cm2, uint8_t cm2_len,
83 const uint8_t *cm3, uint8_t cm3_len)
84{
85 struct gsm_subscriber *subscr = conn->subscr;
86
87 if (subscr) {
88 subscr->equipment.classmark2_len = cm2_len;
89 memcpy(subscr->equipment.classmark2, cm2, cm2_len);
90 if (cm3) {
91 subscr->equipment.classmark3_len = cm3_len;
92 memcpy(subscr->equipment.classmark3, cm3, cm3_len);
93 }
94 db_sync_equipment(&subscr->equipment);
95 }
96}
97
Harald Weltecf149ee2012-01-23 16:40:24 +010098static void msc_ciph_m_compl(struct gsm_subscriber_connection *conn,
99 struct msgb *msg, uint8_t alg_id)
100{
101 gsm_cbfn *cb;
102
103 DEBUGP(DRR, "CIPHERING MODE COMPLETE\n");
104
105 /* Safety check */
106 if (!conn->sec_operation) {
107 DEBUGP(DRR, "No authentication/cipher operation in progress !!!\n");
108 return;
109 }
110
111 /* FIXME: check for MI (if any) */
112
113 /* Call back whatever was in progress (if anything) ... */
114 cb = conn->sec_operation->cb;
115 if (cb) {
Holger Hans Peter Freythera5050b12012-09-11 11:55:03 +0200116 cb(GSM_HOOK_RR_SECURITY, GSM_SECURITY_SUCCEEDED,
Harald Weltecf149ee2012-01-23 16:40:24 +0100117 NULL, conn, conn->sec_operation->cb_data);
118
119 }
120
121 /* Complete the operation */
122 release_security_operation(conn);
123}
124
Harald Welte95e862c2012-01-23 10:28:35 +0100125
126
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +0800127static struct bsc_api msc_handler = {
128 .sapi_n_reject = msc_sapi_n_reject,
Holger Hans Peter Freyther97643312010-06-17 16:41:25 +0800129 .compl_l3 = msc_compl_l3,
130 .dtap = msc_dtap,
Holger Hans Peter Freyther40aac3f2011-12-27 12:31:02 +0100131 .clear_request = msc_clear_request,
132 .assign_compl = msc_assign_compl,
133 .assign_fail = msc_assign_fail,
Harald Welte95e862c2012-01-23 10:28:35 +0100134 .classmark_chg = msc_classmark_chg,
Harald Weltecf149ee2012-01-23 16:40:24 +0100135 .cipher_mode_compl = msc_ciph_m_compl,
Holger Hans Peter Freyther43b09092010-06-15 11:52:51 +0800136};
137
138struct bsc_api *msc_bsc_api() {
139 return &msc_handler;
140}
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800141
142/* lchan release handling */
143void msc_release_connection(struct gsm_subscriber_connection *conn)
144{
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800145 /* skip when we are in release, e.g. due an error */
146 if (conn->in_release)
147 return;
148
149 /* skip releasing of silent calls as they have no transaction */
150 if (conn->silent_call)
151 return;
152
153 /* check if there is a pending operation */
Holger Hans Peter Freyther02d39b22010-07-05 15:34:16 +0800154 if (conn->loc_operation || conn->sec_operation || conn->anch_operation)
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800155 return;
156
Holger Hans Peter Freyther70ae5d32012-11-23 21:33:15 +0100157 if (trans_has_conn(conn))
158 return;
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800159
160 /* no more connections, asking to release the channel */
161 conn->in_release = 1;
162 gsm0808_clear(conn);
Holger Hans Peter Freyther182c81f2010-12-29 16:28:33 +0100163 if (conn->put_channel) {
164 conn->put_channel = 0;
165 subscr_put_channel(conn->subscr);
166 }
Holger Hans Peter Freytheraaa40b82010-09-16 20:48:15 +0800167 subscr_con_free(conn);
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800168}