blob: 957ceaf95f031044c12da567d494133c71362a83 [file] [log] [blame]
Holger Hans Peter Freyther8ec49522011-08-15 15:53:00 +02001/* (C) 2009-2011 by Holger Hans Peter Freyther <zecke@selfish.org>
2 * (C) 2009-2011 by On-Waves
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +01003 * All Rights Reserved
4 *
5 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01006 * 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 Freyther5bb874d2010-11-05 11:21:18 +01008 * (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 Welte9af6ddf2011-01-01 15:25:50 +010013 * GNU Affero General Public License for more details.
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +010014 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010015 * 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 Freyther5bb874d2010-11-05 11:21:18 +010017 *
18 */
19
20#include <openbsc/osmo_bsc.h>
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +010021#include <openbsc/osmo_msc_data.h>
22#include <openbsc/gsm_04_80.h>
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010023#include <openbsc/gsm_subscriber.h>
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +010024#include <openbsc/debug.h>
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010025#include <openbsc/paging.h>
26
27#include <stdlib.h>
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +010028
29static void handle_lu_request(struct gsm_subscriber_connection *conn,
30 struct msgb *msg)
31{
32 struct gsm48_hdr *gh;
33 struct gsm48_loc_upd_req *lu;
34 struct gsm48_loc_area_id lai;
35 struct gsm_network *net;
36
37 if (msgb_l3len(msg) < sizeof(*gh) + sizeof(*lu)) {
38 LOGP(DMSC, LOGL_ERROR, "LU too small to look at: %u\n", msgb_l3len(msg));
39 return;
40 }
41
42 net = conn->bts->network;
43
44 gh = msgb_l3(msg);
45 lu = (struct gsm48_loc_upd_req *) gh->data;
46
47 gsm48_generate_lai(&lai, net->country_code, net->network_code,
48 conn->bts->location_area_code);
49
50 if (memcmp(&lai, &lu->lai, sizeof(lai)) != 0) {
51 LOGP(DMSC, LOGL_DEBUG, "Marking con for welcome USSD.\n");
52 conn->sccp_con->new_subscriber = 1;
53 }
54}
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +010055
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +020056/* extract a subscriber from the paging response */
57static struct gsm_subscriber *extract_sub(struct gsm_subscriber_connection *conn,
58 struct msgb *msg)
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010059{
60 uint8_t mi_type;
61 char mi_string[GSM48_MI_SIZE];
62 struct gsm48_hdr *gh;
63 struct gsm48_pag_resp *resp;
64 struct gsm_subscriber *subscr;
65
66 if (msgb_l3len(msg) < sizeof(*gh) + sizeof(*resp)) {
67 LOGP(DMSC, LOGL_ERROR, "PagingResponse too small: %u\n", msgb_l3len(msg));
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +020068 return NULL;
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010069 }
70
71 gh = msgb_l3(msg);
72 resp = (struct gsm48_pag_resp *) &gh->data[0];
73
74 gsm48_paging_extract_mi(resp, msgb_l3len(msg) - sizeof(*gh),
75 mi_string, &mi_type);
76 DEBUGP(DRR, "PAGING RESPONSE: mi_type=0x%02x MI(%s)\n",
77 mi_type, mi_string);
78
79 switch (mi_type) {
80 case GSM_MI_TYPE_TMSI:
81 subscr = subscr_active_by_tmsi(conn->bts->network,
82 tmsi_from_string(mi_string));
83 break;
84 case GSM_MI_TYPE_IMSI:
85 subscr = subscr_active_by_imsi(conn->bts->network, mi_string);
86 break;
Holger Hans Peter Freyther3fbd2442011-01-16 18:18:56 +010087 default:
88 subscr = NULL;
89 break;
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010090 }
91
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +020092 return subscr;
93}
94
95/* we will need to stop the paging request */
96static int handle_page_resp(struct gsm_subscriber_connection *conn, struct msgb *msg)
97{
98 struct gsm_subscriber *subscr = extract_sub(conn, msg);
99
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +0100100 if (!subscr) {
101 LOGP(DMSC, LOGL_ERROR, "Non active subscriber got paged.\n");
102 return -1;
103 }
104
Harald Weltef604bba2010-12-15 15:33:08 +0100105 paging_request_stop(conn->bts, subscr, conn, msg);
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +0100106 subscr_put(subscr);
107 return 0;
108}
Holger Hans Peter Freyther18fa70a2011-06-08 19:25:38 +0200109
110static int is_cm_service_for_emerg(struct msgb *msg)
111{
112 struct gsm48_service_request *cm;
113 struct gsm48_hdr *gh = msgb_l3(msg);
114
115 if (msgb_l3len(msg) < sizeof(*gh) + sizeof(*cm)) {
116 LOGP(DMSC, LOGL_ERROR, "CM ServiceRequest does not fit.\n");
117 return 0;
118 }
119
120 cm = (struct gsm48_service_request *) &gh->data[0];
121 return cm->cm_service_type == GSM48_CMSERV_EMERGENCY;
122}
123
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200124struct osmo_msc_data *bsc_find_msc(struct gsm_subscriber_connection *conn,
125 struct msgb *msg)
126{
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +0200127 struct gsm48_hdr *gh;
128 int8_t pdisc;
129 uint8_t mtype;
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200130 struct osmo_bsc_data *bsc;
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +0200131 struct osmo_msc_data *msc, *pag_msc;
132 struct gsm_subscriber *subscr;
Holger Hans Peter Freyther18fa70a2011-06-08 19:25:38 +0200133 int is_emerg = 0;
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200134
135 bsc = conn->bts->network->bsc_data;
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +0200136
137 if (msgb_l3len(msg) < sizeof(*gh)) {
138 LOGP(DMSC, LOGL_ERROR, "There is no GSM48 header here.\n");
139 return NULL;
140 }
141
142 gh = msgb_l3(msg);
143 pdisc = gh->proto_discr & 0x0f;
144 mtype = gh->msg_type & 0xbf;
145
146 /*
147 * We are asked to select a MSC here but they are not equal. We
148 * want to respond to a paging request on the MSC where we got the
149 * request from. This is where we need to decide where this connection
150 * will go.
151 */
152 if (pdisc == GSM48_PDISC_RR && mtype == GSM48_MT_RR_PAG_RESP)
153 goto paging;
Holger Hans Peter Freyther18fa70a2011-06-08 19:25:38 +0200154 else if (pdisc == GSM48_PDISC_MM && mtype == GSM48_MT_MM_CM_SERV_REQ) {
155 is_emerg = is_cm_service_for_emerg(msg);
156 goto round_robin;
157 } else
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +0200158 goto round_robin;
159
160round_robin:
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200161 llist_for_each_entry(msc, &bsc->mscs, entry) {
162 if (!msc->msc_con->is_authenticated)
163 continue;
Holger Hans Peter Freyther18fa70a2011-06-08 19:25:38 +0200164 if (!is_emerg && msc->type != MSC_CON_TYPE_NORMAL)
165 continue;
166 if (is_emerg && !msc->allow_emerg)
Holger Hans Peter Freyther210565e2011-06-07 20:56:18 +0200167 continue;
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200168
169 /* force round robin by moving it to the end */
170 llist_move_tail(&msc->entry, &bsc->mscs);
171 return msc;
172 }
173
174 return NULL;
Holger Hans Peter Freytherf67d9a92011-06-07 20:12:33 +0200175
176paging:
177 subscr = extract_sub(conn, msg);
178
179 if (!subscr) {
180 LOGP(DMSC, LOGL_ERROR, "Got paged but no subscriber found.\n");
181 return NULL;
182 }
183
184 pag_msc = paging_get_data(conn->bts, subscr);
185 subscr_put(subscr);
186
187 llist_for_each_entry(msc, &bsc->mscs, entry) {
188 if (msc != pag_msc)
189 continue;
190
191 /*
192 * We don't check if the MSC is connected. In case it
193 * is not the connection will be dropped.
194 */
195
196 /* force round robin by moving it to the end */
197 llist_move_tail(&msc->entry, &bsc->mscs);
198 return msc;
199 }
200
201 LOGP(DMSC, LOGL_ERROR, "Got paged but no request found.\n");
202 return NULL;
Holger Hans Peter Freyther076af1c2011-06-07 19:57:02 +0200203}
204
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +0100205
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +0100206/**
207 * This is used to scan a message for extra functionality of the BSC. This
208 * includes scanning for location updating requests/acceptd and then send
209 * a welcome USSD message to the subscriber.
210 */
211int bsc_scan_bts_msg(struct gsm_subscriber_connection *conn, struct msgb *msg)
212{
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +0100213 struct gsm48_hdr *gh = msgb_l3(msg);
214 uint8_t pdisc = gh->proto_discr & 0x0f;
215 uint8_t mtype = gh->msg_type & 0xbf;
216
217 if (pdisc == GSM48_PDISC_MM) {
218 if (mtype == GSM48_MT_MM_LOC_UPD_REQUEST)
219 handle_lu_request(conn, msg);
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +0100220 } else if (pdisc == GSM48_PDISC_RR) {
221 if (mtype == GSM48_MT_RR_PAG_RESP)
222 handle_page_resp(conn, msg);
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +0100223 }
224
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +0100225 return 0;
226}
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100227
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100228static int send_welcome_ussd(struct gsm_subscriber_connection *conn)
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100229{
Holger Hans Peter Freyther383d3c32012-07-19 14:22:42 +0200230 struct osmo_bsc_sccp_con *bsc_con;
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100231
Holger Hans Peter Freyther383d3c32012-07-19 14:22:42 +0200232 bsc_con = conn->sccp_con;
Holger Hans Peter Freytherf5892212012-08-30 13:05:15 +0200233 if (!bsc_con) {
234 LOGP(DMSC, LOGL_DEBUG, "No SCCP connection associated.\n");
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100235 return 0;
Holger Hans Peter Freytherf5892212012-08-30 13:05:15 +0200236 }
237
238 if (!bsc_con->msc->ussd_welcome_txt) {
239 LOGP(DMSC, LOGL_DEBUG, "No USSD Welcome text defined.\n");
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100240 return 0;
Holger Hans Peter Freytherf5892212012-08-30 13:05:15 +0200241 }
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100242
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100243 return BSS_SEND_USSD;
244}
245
246int bsc_send_welcome_ussd(struct gsm_subscriber_connection *conn)
247{
248 gsm0480_send_ussdNotify(conn, 1, conn->sccp_con->msc->ussd_welcome_txt);
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100249 gsm0480_send_releaseComplete(conn);
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100250
251 return 0;
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100252}
253
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100254/**
255 * Messages coming back from the MSC.
256 */
257int bsc_scan_msc_msg(struct gsm_subscriber_connection *conn, struct msgb *msg)
258{
Holger Hans Peter Freytherf936fb42011-06-04 15:12:57 +0200259 struct osmo_msc_data *msc;
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100260 struct gsm_network *net;
261 struct gsm48_loc_area_id *lai;
262 struct gsm48_hdr *gh;
263 uint8_t mtype;
264
265 if (msgb_l3len(msg) < sizeof(*gh)) {
266 LOGP(DMSC, LOGL_ERROR, "GSM48 header does not fit.\n");
267 return -1;
268 }
269
270 gh = (struct gsm48_hdr *) msgb_l3(msg);
271 mtype = gh->msg_type & 0xbf;
272 net = conn->bts->network;
Holger Hans Peter Freytherf936fb42011-06-04 15:12:57 +0200273 msc = conn->sccp_con->msc;
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100274
275 if (mtype == GSM48_MT_MM_LOC_UPD_ACCEPT) {
Holger Hans Peter Freytherf936fb42011-06-04 15:12:57 +0200276 if (msc->core_ncc != -1 || msc->core_mcc != -1) {
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100277 if (msgb_l3len(msg) >= sizeof(*gh) + sizeof(*lai)) {
278 lai = (struct gsm48_loc_area_id *) &gh->data[0];
279 gsm48_generate_lai(lai, net->country_code,
280 net->network_code,
281 conn->bts->location_area_code);
282 }
283 }
284
285 if (conn->sccp_con->new_subscriber)
Holger Hans Peter Freytherd2361d92012-12-03 15:32:54 +0100286 return send_welcome_ussd(conn);
287 return 0;
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100288 }
289
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100290 return 0;
291}