blob: 59d48970b22b2f4bed85e3ea21f0fd6e4a1cc0b7 [file] [log] [blame]
Harald Weltea1482332009-11-14 10:08:40 +01001/* GSM silent call feature */
2
3/*
4 * (C) 2009 by Harald Welte <laforge@gnumonks.org>
5 *
6 * All Rights Reserved
7 *
8 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01009 * it under the terms of the GNU Affero General Public License as published by
10 * the Free Software Foundation; either version 3 of the License, or
Harald Weltea1482332009-11-14 10:08:40 +010011 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte9af6ddf2011-01-01 15:25:50 +010016 * GNU Affero General Public License for more details.
Harald Weltea1482332009-11-14 10:08:40 +010017 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010018 * You should have received a copy of the GNU Affero General Public License
19 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Harald Weltea1482332009-11-14 10:08:40 +010020 *
21 */
22
23#include <stdlib.h>
24#include <unistd.h>
25#include <errno.h>
26
Harald Weltedfe6c7d2010-02-20 16:24:02 +010027#include <osmocore/msgb.h>
Harald Weltea1482332009-11-14 10:08:40 +010028#include <openbsc/signal.h>
29#include <openbsc/debug.h>
30#include <openbsc/paging.h>
31#include <openbsc/gsm_data.h>
32#include <openbsc/gsm_subscriber.h>
33#include <openbsc/abis_rsl.h>
Harald Welte986c3d72009-11-17 06:12:16 +010034#include <openbsc/chan_alloc.h>
Holger Hans Peter Freyther88519ea2010-06-30 12:44:07 +080035#include <openbsc/osmo_msc.h>
Harald Weltea1482332009-11-14 10:08:40 +010036
Harald Welte51008772009-12-29 11:49:12 +010037/* paging of the requested subscriber has completed */
Harald Weltea1482332009-11-14 10:08:40 +010038static int paging_cb_silent(unsigned int hooknum, unsigned int event,
Holger Hans Peter Freyther86481c22010-06-17 15:05:57 +080039 struct msgb *msg, void *_conn, void *_data)
Harald Weltea1482332009-11-14 10:08:40 +010040{
Holger Hans Peter Freyther86481c22010-06-17 15:05:57 +080041 struct gsm_subscriber_connection *conn = _conn;
Harald Weltea1482332009-11-14 10:08:40 +010042 struct scall_signal_data sigdata;
Holger Hans Peter Freythera97152b2010-07-23 19:34:34 +080043 int rc = 0;
Harald Weltea1482332009-11-14 10:08:40 +010044
45 if (hooknum != GSM_HOOK_RR_PAGING)
46 return -EINVAL;
47
48 DEBUGP(DSMS, "paging_cb_silent: ");
49
Holger Hans Peter Freyther86481c22010-06-17 15:05:57 +080050 sigdata.conn = conn;
Harald Weltea1482332009-11-14 10:08:40 +010051 sigdata.data = _data;
52
53 switch (event) {
54 case GSM_PAGING_SUCCEEDED:
55 DEBUGPC(DSMS, "success, using Timeslot %u on ARFCN %u\n",
Holger Hans Peter Freyther86481c22010-06-17 15:05:57 +080056 conn->lchan->ts->nr, conn->lchan->ts->trx->arfcn);
Holger Hans Peter Freyther68884aa2010-03-23 06:41:45 +010057 conn->silent_call = 1;
Harald Weltea1482332009-11-14 10:08:40 +010058 /* increment lchan reference count */
59 dispatch_signal(SS_SCALL, S_SCALL_SUCCESS, &sigdata);
Harald Weltea1482332009-11-14 10:08:40 +010060 break;
61 case GSM_PAGING_EXPIRED:
Holger Hans Peter Freytherd3baf412010-12-23 18:19:17 +010062 case GSM_PAGING_BUSY:
Harald Weltea1482332009-11-14 10:08:40 +010063 DEBUGP(DSMS, "expired\n");
64 dispatch_signal(SS_SCALL, S_SCALL_EXPIRED, &sigdata);
65 break;
66 default:
67 rc = -EINVAL;
68 break;
69 }
70
71 return rc;
72}
73
Harald Welte51008772009-12-29 11:49:12 +010074/* receive a layer 3 message from a silent call */
Holger Hans Peter Freyther758f4df2010-06-21 10:34:03 +080075int silent_call_rx(struct gsm_subscriber_connection *conn, struct msgb *msg)
Harald Welte51008772009-12-29 11:49:12 +010076{
77 /* FIXME: do something like sending it through a UDP port */
78 return 0;
79}
80
81struct msg_match {
82 u_int8_t pdisc;
83 u_int8_t msg_type;
84};
85
86/* list of messages that are handled inside OpenBSC, even in a silent call */
87static const struct msg_match silent_call_accept[] = {
88 { GSM48_PDISC_MM, GSM48_MT_MM_LOC_UPD_REQUEST },
89 { GSM48_PDISC_MM, GSM48_MT_MM_CM_SERV_REQ },
90};
91
92/* decide if we need to reroute a message as part of a silent call */
Holger Hans Peter Freyther758f4df2010-06-21 10:34:03 +080093int silent_call_reroute(struct gsm_subscriber_connection *conn, struct msgb *msg)
Harald Welte51008772009-12-29 11:49:12 +010094{
95 struct gsm48_hdr *gh = msgb_l3(msg);
96 u_int8_t pdisc = gh->proto_discr & 0x0f;
97 int i;
98
99 /* if we're not part of a silent call, never reroute */
Holger Hans Peter Freyther758f4df2010-06-21 10:34:03 +0800100 if (!conn->silent_call)
Harald Welte51008772009-12-29 11:49:12 +0100101 return 0;
102
103 /* check if we are a special message that is handled in openbsc */
104 for (i = 0; i < ARRAY_SIZE(silent_call_accept); i++) {
105 if (silent_call_accept[i].pdisc == pdisc &&
106 silent_call_accept[i].msg_type == gh->msg_type)
107 return 0;
108 }
109
110 /* otherwise, reroute */
111 return 1;
112}
113
114
115/* initiate a silent call with a given subscriber */
Sylvain Munaut50480702010-01-02 14:29:43 +0100116int gsm_silent_call_start(struct gsm_subscriber *subscr, void *data, int type)
Harald Weltea1482332009-11-14 10:08:40 +0100117{
118 int rc;
119
Sylvain Munaut50480702010-01-02 14:29:43 +0100120 rc = paging_request(subscr->net, subscr, type,
Harald Weltea1482332009-11-14 10:08:40 +0100121 paging_cb_silent, data);
122 return rc;
123}
124
Harald Welte51008772009-12-29 11:49:12 +0100125/* end a silent call with a given subscriber */
Harald Weltea1482332009-11-14 10:08:40 +0100126int gsm_silent_call_stop(struct gsm_subscriber *subscr)
127{
Holger Hans Peter Freyther68884aa2010-03-23 06:41:45 +0100128 struct gsm_subscriber_connection *conn;
Harald Weltea1482332009-11-14 10:08:40 +0100129
Holger Hans Peter Freytherb2be1952010-06-16 13:23:55 +0800130 conn = connection_for_subscr(subscr);
131 if (!conn)
Harald Weltea1482332009-11-14 10:08:40 +0100132 return -EINVAL;
133
Harald Welte83579ca2009-12-29 11:17:18 +0100134 /* did we actually establish a silent call for this guy? */
Holger Hans Peter Freyther68884aa2010-03-23 06:41:45 +0100135 if (!conn->silent_call)
Harald Welte83579ca2009-12-29 11:17:18 +0100136 return -EINVAL;
137
Holger Hans Peter Freyther40494552010-06-28 17:09:29 +0800138 conn->silent_call = 0;
139 msc_release_connection(conn);
Harald Weltea1482332009-11-14 10:08:40 +0100140
141 return 0;
142}