blob: e222ac955ba0103f0122fcd027b2c55d4495ff19 [file] [log] [blame]
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +01001/* (C) 2009-2010 by Holger Hans Peter Freyther <zecke@selfish.org>
2 * (C) 2009-2010 by On-Waves
3 * 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 Freyther16e958d2010-11-15 13:34:03 +010056/* we will need to stop the paging request */
57static int handle_page_resp(struct gsm_subscriber_connection *conn, struct msgb *msg)
58{
59 uint8_t mi_type;
60 char mi_string[GSM48_MI_SIZE];
61 struct gsm48_hdr *gh;
62 struct gsm48_pag_resp *resp;
63 struct gsm_subscriber *subscr;
64
65 if (msgb_l3len(msg) < sizeof(*gh) + sizeof(*resp)) {
66 LOGP(DMSC, LOGL_ERROR, "PagingResponse too small: %u\n", msgb_l3len(msg));
67 return -1;
68 }
69
70 gh = msgb_l3(msg);
71 resp = (struct gsm48_pag_resp *) &gh->data[0];
72
73 gsm48_paging_extract_mi(resp, msgb_l3len(msg) - sizeof(*gh),
74 mi_string, &mi_type);
75 DEBUGP(DRR, "PAGING RESPONSE: mi_type=0x%02x MI(%s)\n",
76 mi_type, mi_string);
77
78 switch (mi_type) {
79 case GSM_MI_TYPE_TMSI:
80 subscr = subscr_active_by_tmsi(conn->bts->network,
81 tmsi_from_string(mi_string));
82 break;
83 case GSM_MI_TYPE_IMSI:
84 subscr = subscr_active_by_imsi(conn->bts->network, mi_string);
85 break;
86 }
87
88 if (!subscr) {
89 LOGP(DMSC, LOGL_ERROR, "Non active subscriber got paged.\n");
90 return -1;
91 }
92
Harald Weltef604bba2010-12-15 15:33:08 +010093 paging_request_stop(conn->bts, subscr, conn, msg);
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +010094 subscr_put(subscr);
95 return 0;
96}
97
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +010098/**
99 * This is used to scan a message for extra functionality of the BSC. This
100 * includes scanning for location updating requests/acceptd and then send
101 * a welcome USSD message to the subscriber.
102 */
103int bsc_scan_bts_msg(struct gsm_subscriber_connection *conn, struct msgb *msg)
104{
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +0100105 struct gsm48_hdr *gh = msgb_l3(msg);
106 uint8_t pdisc = gh->proto_discr & 0x0f;
107 uint8_t mtype = gh->msg_type & 0xbf;
108
109 if (pdisc == GSM48_PDISC_MM) {
110 if (mtype == GSM48_MT_MM_LOC_UPD_REQUEST)
111 handle_lu_request(conn, msg);
Holger Hans Peter Freyther16e958d2010-11-15 13:34:03 +0100112 } else if (pdisc == GSM48_PDISC_RR) {
113 if (mtype == GSM48_MT_RR_PAG_RESP)
114 handle_page_resp(conn, msg);
Holger Hans Peter Freytherd65305f2010-11-05 11:31:08 +0100115 }
116
Holger Hans Peter Freyther5bb874d2010-11-05 11:21:18 +0100117 return 0;
118}
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100119
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100120static void send_welcome_ussd(struct gsm_subscriber_connection *conn)
121{
122 struct gsm_network *net;
123 net = conn->bts->network;
124
125 if (!net->msc_data->ussd_welcome_txt)
126 return;
127
128 gsm0480_send_ussdNotify(conn, 1, net->msc_data->ussd_welcome_txt);
129 gsm0480_send_releaseComplete(conn);
130}
131
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100132/**
133 * Messages coming back from the MSC.
134 */
135int bsc_scan_msc_msg(struct gsm_subscriber_connection *conn, struct msgb *msg)
136{
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100137 struct gsm_network *net;
138 struct gsm48_loc_area_id *lai;
139 struct gsm48_hdr *gh;
140 uint8_t mtype;
141
142 if (msgb_l3len(msg) < sizeof(*gh)) {
143 LOGP(DMSC, LOGL_ERROR, "GSM48 header does not fit.\n");
144 return -1;
145 }
146
147 gh = (struct gsm48_hdr *) msgb_l3(msg);
148 mtype = gh->msg_type & 0xbf;
149 net = conn->bts->network;
150
151 if (mtype == GSM48_MT_MM_LOC_UPD_ACCEPT) {
Holger Hans Peter Freyther2a8675e2010-11-05 19:43:07 +0100152 if (net->msc_data->core_ncc != -1 ||
153 net->msc_data->core_mcc != -1) {
Holger Hans Peter Freyther52d42ab2010-11-05 18:41:04 +0100154 if (msgb_l3len(msg) >= sizeof(*gh) + sizeof(*lai)) {
155 lai = (struct gsm48_loc_area_id *) &gh->data[0];
156 gsm48_generate_lai(lai, net->country_code,
157 net->network_code,
158 conn->bts->location_area_code);
159 }
160 }
161
162 if (conn->sccp_con->new_subscriber)
163 send_welcome_ussd(conn);
164 }
165
Holger Hans Peter Freythera54732d2010-11-05 18:11:19 +0100166 return 0;
167}