blob: 3b95a901fc88ab01095e899efd53d469f8df202f [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
Sylvain Munaut93558302019-02-14 20:13:08 +010027#include <osmocom/core/byteswap.h>
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010028#include <osmocom/core/msgb.h>
Philipp Maier92861142019-03-15 09:43:40 +010029#include <osmocom/core/utils.h>
Neels Hofmeyr90843962017-09-04 15:04:35 +020030#include <osmocom/msc/signal.h>
31#include <osmocom/msc/debug.h>
Neels Hofmeyr90843962017-09-04 15:04:35 +020032#include <osmocom/msc/gsm_data.h>
33#include <osmocom/msc/gsm_subscriber.h>
Neels Hofmeyrd656dff2018-03-09 14:59:44 +010034#include <osmocom/msc/vlr.h>
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010035#include <osmocom/msc/msc_a.h>
36#include <osmocom/msc/paging.h>
37#include <osmocom/msc/transaction.h>
38#include <osmocom/msc/silent_call.h>
Harald Weltea1482332009-11-14 10:08:40 +010039
Sylvain Munaut93558302019-02-14 20:13:08 +010040#include <osmocom/sigtran/sccp_helpers.h>
41
Harald Welte51008772009-12-29 11:49:12 +010042/* paging of the requested subscriber has completed */
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010043void paging_cb_silent(struct msc_a *msc_a, struct gsm_trans *trans)
Harald Weltea1482332009-11-14 10:08:40 +010044{
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010045 struct scall_signal_data sigdata = {
46 .msc_a = msc_a,
47 .vty = trans->silent_call.from_vty,
48 };
49 struct ran_msg assignment;
Harald Weltea1482332009-11-14 10:08:40 +010050
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010051 if (!msc_a) {
52 LOG_MSC_A(msc_a, LOGL_ERROR, "Silent call: MS not responding to Paging\n");
53 osmo_signal_dispatch(SS_SCALL, S_SCALL_FAILED, &sigdata);
54 trans_free(trans);
55 return;
Harald Weltea1482332009-11-14 10:08:40 +010056 }
57
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010058 LOG_MSC_A(msc_a, LOGL_INFO, "Silent call: MS responding to Paging\n");
Sylvain Munaut93558302019-02-14 20:13:08 +010059
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010060 trans->msc_a = msc_a;
61 msc_a_get(msc_a, MSC_A_USE_SILENT_CALL);
62
63 osmo_fsm_inst_dispatch(msc_a->c.fi, MSC_A_EV_TRANSACTION_ACCEPTED, trans);
64
65 assignment = (struct ran_msg){
66 .msg_type = RAN_MSG_ASSIGNMENT_COMMAND,
67 .assignment_command = {
68 .channel_type = &trans->silent_call.ct,
69 .cn_rtp = &trans->silent_call.rtp_cn,
70 },
71 };
72 if (msc_a_ran_down(msc_a, MSC_ROLE_I, &assignment)) {
73 LOG_MSC_A(msc_a, LOGL_ERROR, "Silent call failed\n");
74 osmo_signal_dispatch(SS_SCALL, S_SCALL_FAILED, &sigdata);
75 trans_free(trans);
76 } else {
77 osmo_signal_dispatch(SS_SCALL, S_SCALL_SUCCESS, &sigdata);
78 }
79}
80
81void trans_silent_call_free(struct gsm_trans *trans)
82{
83 struct scall_signal_data sigdata = {
84 .msc_a = trans->msc_a,
85 .vty = trans->silent_call.from_vty,
86 };
87 osmo_signal_dispatch(SS_SCALL, S_SCALL_DETACHED, &sigdata);
Harald Weltea1482332009-11-14 10:08:40 +010088}
89
Philipp Maiere0d5caa2017-02-27 16:56:59 +010090#if 0
Harald Welte51008772009-12-29 11:49:12 +010091/* receive a layer 3 message from a silent call */
Neels Hofmeyrc036b792018-11-29 22:37:51 +010092int silent_call_rx(struct ran_conn *conn, struct msgb *msg)
Harald Welte51008772009-12-29 11:49:12 +010093{
94 /* FIXME: do something like sending it through a UDP port */
Jacob Erlbeck8e68b562014-01-30 21:01:12 +010095 LOGP(DLSMS, LOGL_NOTICE, "Discarding L3 message from a silent call.\n");
Harald Welte51008772009-12-29 11:49:12 +010096 return 0;
97}
98
99struct msg_match {
Holger Hans Peter Freytherc42ad8b2011-04-18 17:04:00 +0200100 uint8_t pdisc;
101 uint8_t msg_type;
Harald Welte51008772009-12-29 11:49:12 +0100102};
103
104/* list of messages that are handled inside OpenBSC, even in a silent call */
105static const struct msg_match silent_call_accept[] = {
106 { GSM48_PDISC_MM, GSM48_MT_MM_LOC_UPD_REQUEST },
107 { GSM48_PDISC_MM, GSM48_MT_MM_CM_SERV_REQ },
108};
109
110/* decide if we need to reroute a message as part of a silent call */
Neels Hofmeyrc036b792018-11-29 22:37:51 +0100111int silent_call_reroute(struct ran_conn *conn, struct msgb *msg)
Harald Welte51008772009-12-29 11:49:12 +0100112{
113 struct gsm48_hdr *gh = msgb_l3(msg);
Neels Hofmeyr531734a2016-03-14 16:13:24 +0100114 uint8_t pdisc = gsm48_hdr_pdisc(gh);
115 uint8_t msg_type = gsm48_hdr_msg_type(gh);
Harald Welte51008772009-12-29 11:49:12 +0100116 int i;
117
118 /* if we're not part of a silent call, never reroute */
Holger Hans Peter Freyther758f4df2010-06-21 10:34:03 +0800119 if (!conn->silent_call)
Harald Welte51008772009-12-29 11:49:12 +0100120 return 0;
121
122 /* check if we are a special message that is handled in openbsc */
123 for (i = 0; i < ARRAY_SIZE(silent_call_accept); i++) {
124 if (silent_call_accept[i].pdisc == pdisc &&
Neels Hofmeyr531734a2016-03-14 16:13:24 +0100125 silent_call_accept[i].msg_type == msg_type)
Harald Welte51008772009-12-29 11:49:12 +0100126 return 0;
127 }
128
129 /* otherwise, reroute */
Jacob Erlbeck8e68b562014-01-30 21:01:12 +0100130 LOGP(DLSMS, LOGL_INFO, "Rerouting L3 message from a silent call.\n");
Harald Welte51008772009-12-29 11:49:12 +0100131 return 1;
132}
Philipp Maiere0d5caa2017-02-27 16:56:59 +0100133#endif
Harald Welte51008772009-12-29 11:49:12 +0100134
135
136/* initiate a silent call with a given subscriber */
Sylvain Munaut93558302019-02-14 20:13:08 +0100137int gsm_silent_call_start(struct vlr_subscr *vsub,
138 const struct gsm0808_channel_type *ct,
139 const char *traffic_dst_ip, uint16_t traffic_dst_port,
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100140 struct vty *vty)
Harald Weltea1482332009-11-14 10:08:40 +0100141{
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100142 struct gsm_network *net = vsub->vlr->user_ctx;
143 struct gsm_trans *trans = trans_alloc(net, vsub, TRANS_SILENT_CALL, 0, 0);
Harald Weltea1482332009-11-14 10:08:40 +0100144
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100145 trans->silent_call.ct = *ct;
Sylvain Munaut93558302019-02-14 20:13:08 +0100146 if (traffic_dst_ip) {
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100147 osmo_sockaddr_str_from_str(&trans->silent_call.rtp_cn, traffic_dst_ip, traffic_dst_port);
Sylvain Munaut93558302019-02-14 20:13:08 +0100148 }
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100149 trans->silent_call.from_vty = vty;
Sylvain Munaut93558302019-02-14 20:13:08 +0100150
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100151 if (!paging_request_start(vsub, PAGING_CAUSE_CALL_BACKGROUND, paging_cb_silent, trans,
152 "establish silent call")) {
153 trans_free(trans);
Neels Hofmeyrd656dff2018-03-09 14:59:44 +0100154 return -ENODEV;
Sylvain Munaut93558302019-02-14 20:13:08 +0100155 }
156
Neels Hofmeyrd656dff2018-03-09 14:59:44 +0100157 return 0;
Harald Weltea1482332009-11-14 10:08:40 +0100158}
159
Harald Welte51008772009-12-29 11:49:12 +0100160/* end a silent call with a given subscriber */
Harald Welte2483f1b2016-06-19 18:06:02 +0200161int gsm_silent_call_stop(struct vlr_subscr *vsub)
Harald Weltea1482332009-11-14 10:08:40 +0100162{
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100163 struct msc_a *msc_a = msc_a_for_vsub(vsub, true);
164 struct gsm_trans *trans;
165 if (!msc_a) {
Neels Hofmeyrd656dff2018-03-09 14:59:44 +0100166 LOGP(DMM, LOGL_ERROR, "%s: Cannot stop silent call, no connection for subscriber\n",
167 vlr_subscr_name(vsub));
168 return -ENODEV;
169 }
Harald Weltea1482332009-11-14 10:08:40 +0100170
Harald Welte83579ca2009-12-29 11:17:18 +0100171 /* did we actually establish a silent call for this guy? */
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100172 trans = trans_find_by_type(msc_a, TRANS_SILENT_CALL);
173 if (!trans) {
174 LOG_MSC_A(msc_a, LOGL_ERROR, "Cannot stop silent call, subscriber has no active silent call\n");
Neels Hofmeyrd656dff2018-03-09 14:59:44 +0100175 return -ENOENT;
176 }
Harald Welte83579ca2009-12-29 11:17:18 +0100177
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100178 trans_free(trans);
Harald Weltea1482332009-11-14 10:08:40 +0100179 return 0;
180}