blob: ab5957162b8f95af92de9d27a985459c19567c2a [file] [log] [blame]
Harald Welte6eafe912009-10-16 08:32:58 +02001/* Network-specific handling of mobile-originated USSDs. */
2
3/* (C) 2008-2009 by Harald Welte <laforge@gnumonks.org>
4 * (C) 2008, 2009 by Holger Hans Peter Freyther <zecke@selfish.org>
5 * (C) 2009 by Mike Haben <michael.haben@btinternet.com>
6 *
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
Harald Welte6eafe912009-10-16 08:32:58 +020012 * (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.
Harald Welte6eafe912009-10-16 08:32:58 +020018 *
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/>.
Harald Welte6eafe912009-10-16 08:32:58 +020021 *
22 */
23
Harald Welte6307b852009-10-16 08:41:51 +020024/* This module defines the network-specific handling of mobile-originated
25 USSD messages. */
Harald Welte6eafe912009-10-16 08:32:58 +020026
27#include <stdio.h>
28#include <stdlib.h>
29#include <string.h>
30#include <errno.h>
31
Neels Hofmeyr90843962017-09-04 15:04:35 +020032#include <osmocom/msc/gsm_04_80.h>
33#include <osmocom/msc/gsm_subscriber.h>
34#include <osmocom/msc/debug.h>
35#include <osmocom/msc/osmo_msc.h>
36#include <osmocom/msc/vlr.h>
Max43b01b02017-09-15 11:22:30 +020037#include <osmocom/msc/gsm_04_08.h>
Harald Welte6eafe912009-10-16 08:32:58 +020038
39/* Declarations of USSD strings to be recognised */
40const char USSD_TEXT_OWN_NUMBER[] = "*#100#";
41
Vadim Yanitskiyb274d562018-01-10 19:47:21 +060042/* A network-specific handler function */
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070043static int send_own_number(struct gsm_subscriber_connection *conn,
44 const struct ss_request *req)
Vadim Yanitskiyb274d562018-01-10 19:47:21 +060045{
46 char *own_number = conn->vsub->msisdn;
47 char response_string[GSM_EXTENSION_LENGTH + 20];
Harald Welte6eafe912009-10-16 08:32:58 +020048
Vadim Yanitskiyb274d562018-01-10 19:47:21 +060049 DEBUGP(DMM, "%s: MSISDN = %s\n", vlr_subscr_name(conn->vsub),
50 own_number);
51
52 /* Need trailing CR as EOT character */
53 snprintf(response_string, sizeof(response_string), "Your extension is %s\r", own_number);
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070054 return gsm0480_send_ussd_response(conn, response_string, req);
Vadim Yanitskiyb274d562018-01-10 19:47:21 +060055}
Harald Welte6eafe912009-10-16 08:32:58 +020056
57/* Entrypoint - handler function common to all mobile-originated USSDs */
Holger Hans Peter Freytherd42c3f22010-06-17 17:35:57 +080058int handle_rcv_ussd(struct gsm_subscriber_connection *conn, struct msgb *msg)
Harald Welte6eafe912009-10-16 08:32:58 +020059{
Holger Hans Peter Freyther06abe9f2010-06-30 11:30:01 +080060 int rc;
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020061 struct ss_request req;
Holger Hans Peter Freyther88a5fa02010-10-11 09:31:47 +020062 struct gsm48_hdr *gh;
Harald Welte6eafe912009-10-16 08:32:58 +020063
Harald Welte2483f1b2016-06-19 18:06:02 +020064 /* TODO: Use subscriber_connection ref-counting if we ever want
65 * to keep the connection alive due ot ongoing USSD exchange.
66 * As we answer everytying synchronously so far, there's no need
67 * yet */
68
69 cm_service_request_concludes(conn, msg);
70
Holger Hans Peter Freyther06abe9f2010-06-30 11:30:01 +080071 memset(&req, 0, sizeof(req));
Holger Hans Peter Freyther88a5fa02010-10-11 09:31:47 +020072 gh = msgb_l3(msg);
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020073 rc = gsm0480_decode_ss_request(gh, msgb_l3len(msg), &req);
Tobias Engelea730322013-12-28 17:03:14 +010074 if (!rc) {
Vadim Yanitskiyf95d5be2018-04-04 06:21:00 +070075 LOGP(DMM, LOGL_ERROR, "SS/USSD message parsing error, "
76 "rejecting request...\n");
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070077 rc = gsm0480_send_ussd_reject(conn, &req);
Tobias Engelea730322013-12-28 17:03:14 +010078 return rc;
79 }
80
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020081 /* Interrogation or releaseComplete? */
82 if (req.ussd_text[0] == '\0' || req.ussd_text[0] == 0xFF) {
83 if (req.ss_code > 0) {
84 /* Assume interrogateSS or modification of it and reject */
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070085 rc = gsm0480_send_ussd_reject(conn, &req);
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020086 return rc;
87 }
88 /* Still assuming a Release-Complete and returning */
Harald Welte6307b852009-10-16 08:41:51 +020089 return 0;
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020090 }
Harald Welte6eafe912009-10-16 08:32:58 +020091
Harald Welte2483f1b2016-06-19 18:06:02 +020092 msc_subscr_conn_communicating(conn);
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020093 if (!strcmp(USSD_TEXT_OWN_NUMBER, (const char *)req.ussd_text)) {
Harald Welte6eafe912009-10-16 08:32:58 +020094 DEBUGP(DMM, "USSD: Own number requested\n");
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070095 rc = send_own_number(conn, &req);
Harald Welte6eafe912009-10-16 08:32:58 +020096 } else {
Holger Hans Peter Freyther5085e0b2016-07-12 17:53:26 +020097 DEBUGP(DMM, "Unhandled USSD %s\n", req.ussd_text);
Vadim Yanitskiy12d221f2018-04-04 05:30:16 +070098 rc = gsm0480_send_ussd_reject(conn, &req);
Harald Welte6eafe912009-10-16 08:32:58 +020099 }
Holger Hans Peter Freyther24866632010-06-30 12:15:19 +0800100
Holger Hans Peter Freyther24866632010-06-30 12:15:19 +0800101 return rc;
Harald Welte6eafe912009-10-16 08:32:58 +0200102}