blob: 7d59787f7ff0430e5281efc9bd1fcda38b8526b2 [file] [log] [blame]
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +01001/* SMS queue to continously attempt to deliver SMS */
2/*
3 * (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org>
4 * All Rights Reserved
5 *
6 * This program is free software; you can redistribute it and/or modify
Harald Welte9af6ddf2011-01-01 15:25:50 +01007 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation; either version 3 of the License, or
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +01009 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Harald Welte9af6ddf2011-01-01 15:25:50 +010014 * GNU Affero General Public License for more details.
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010015 *
Harald Welte9af6ddf2011-01-01 15:25:50 +010016 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010018 *
19 */
20
21/**
22 * The difficulty of such a queue is to send a lot of SMS without
23 * overloading the paging subsystem and the database and other users
24 * of the MSC. To make the best use we would need to know the number
25 * of pending paging requests, then throttle the number of SMS we
26 * want to send and such.
27 * We will start with a very simple SMS Queue and then try to speed
28 * things up by collecting data from other parts of the system.
29 */
30
Harald Welte2483f1b2016-06-19 18:06:02 +020031#include <limits.h>
32
Neels Hofmeyr90843962017-09-04 15:04:35 +020033#include <osmocom/msc/sms_queue.h>
Neels Hofmeyr90843962017-09-04 15:04:35 +020034#include <osmocom/msc/db.h>
35#include <osmocom/msc/debug.h>
36#include <osmocom/msc/gsm_data.h>
37#include <osmocom/msc/gsm_04_11.h>
38#include <osmocom/msc/gsm_subscriber.h>
39#include <osmocom/msc/signal.h>
40#include <osmocom/msc/vlr.h>
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010041
Pablo Neira Ayuso136f4532011-03-22 16:47:59 +010042#include <osmocom/core/talloc.h>
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010043
Holger Hans Peter Freyther81c0e252010-12-25 14:08:00 +010044#include <osmocom/vty/vty.h>
45
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010046/*
47 * One pending SMS that we wait for.
48 */
49struct gsm_sms_pending {
50 struct llist_head entry;
51
Harald Welte2483f1b2016-06-19 18:06:02 +020052 struct vlr_subscr *vsub;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010053 unsigned long long sms_id;
54 int failed_attempts;
55 int resend;
56};
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010057
58struct gsm_sms_queue {
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +020059 struct osmo_timer_list resend_pending;
60 struct osmo_timer_list push_queue;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010061 struct gsm_network *network;
Holger Hans Peter Freyther7e59c832010-12-25 14:46:54 +010062 int max_fail;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010063 int max_pending;
64 int pending;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010065
66 struct llist_head pending_sms;
Harald Welte2483f1b2016-06-19 18:06:02 +020067
68 char last_msisdn[GSM_EXTENSION_LENGTH+1];
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010069};
70
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010071static int sms_subscr_cb(unsigned int, unsigned int, void *, void *);
72static int sms_sms_cb(unsigned int, unsigned int, void *, void *);
73
74static struct gsm_sms_pending *sms_find_pending(struct gsm_sms_queue *smsq,
Stefan Sperling87cba1f2018-01-22 17:05:37 +010075 unsigned long long sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010076{
77 struct gsm_sms_pending *pending;
78
79 llist_for_each_entry(pending, &smsq->pending_sms, entry) {
Stefan Sperling87cba1f2018-01-22 17:05:37 +010080 if (pending->sms_id == sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010081 return pending;
82 }
83
84 return NULL;
85}
86
Stefan Sperling87cba1f2018-01-22 17:05:37 +010087int sms_queue_sms_is_pending(struct gsm_sms_queue *smsq, unsigned long long sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010088{
Stefan Sperling87cba1f2018-01-22 17:05:37 +010089 return sms_find_pending(smsq, sms_id) != NULL;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010090}
91
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +020092static struct gsm_sms_pending *sms_subscriber_find_pending(
93 struct gsm_sms_queue *smsq,
Harald Welte2483f1b2016-06-19 18:06:02 +020094 struct vlr_subscr *vsub)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010095{
96 struct gsm_sms_pending *pending;
97
98 llist_for_each_entry(pending, &smsq->pending_sms, entry) {
Harald Welte2483f1b2016-06-19 18:06:02 +020099 if (pending->vsub == vsub)
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200100 return pending;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100101 }
102
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200103 return NULL;
104}
105
106static int sms_subscriber_is_pending(struct gsm_sms_queue *smsq,
Harald Welte2483f1b2016-06-19 18:06:02 +0200107 struct vlr_subscr *vsub)
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200108{
Harald Welte2483f1b2016-06-19 18:06:02 +0200109 return sms_subscriber_find_pending(smsq, vsub) != NULL;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100110}
111
112static struct gsm_sms_pending *sms_pending_from(struct gsm_sms_queue *smsq,
113 struct gsm_sms *sms)
114{
115 struct gsm_sms_pending *pending;
116
117 pending = talloc_zero(smsq, struct gsm_sms_pending);
118 if (!pending)
119 return NULL;
120
Harald Welte2483f1b2016-06-19 18:06:02 +0200121 pending->vsub = vlr_subscr_get(sms->receiver);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100122 pending->sms_id = sms->id;
123 return pending;
124}
125
126static void sms_pending_free(struct gsm_sms_pending *pending)
127{
Harald Welte2483f1b2016-06-19 18:06:02 +0200128 vlr_subscr_put(pending->vsub);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100129 llist_del(&pending->entry);
130 talloc_free(pending);
131}
132
133static void sms_pending_resend(struct gsm_sms_pending *pending)
134{
Harald Welte2483f1b2016-06-19 18:06:02 +0200135 struct gsm_network *net = pending->vsub->vlr->user_ctx;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100136 struct gsm_sms_queue *smsq;
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100137 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100138 "Scheduling resend of SMS %llu.\n", pending->sms_id);
139
140 pending->resend = 1;
141
Harald Welte2483f1b2016-06-19 18:06:02 +0200142 smsq = net->sms_queue;
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200143 if (osmo_timer_pending(&smsq->resend_pending))
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100144 return;
145
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200146 osmo_timer_schedule(&smsq->resend_pending, 1, 0);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100147}
148
149static void sms_pending_failed(struct gsm_sms_pending *pending, int paging_error)
150{
Harald Welte2483f1b2016-06-19 18:06:02 +0200151 struct gsm_network *net = pending->vsub->vlr->user_ctx;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100152 struct gsm_sms_queue *smsq;
153
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100154 LOGP(DLSMS, LOGL_NOTICE, "Sending SMS %llu failed %d times.\n",
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100155 pending->sms_id, pending->failed_attempts);
156
Harald Welte2483f1b2016-06-19 18:06:02 +0200157 smsq = net->sms_queue;
Holger Hans Peter Freyther7e59c832010-12-25 14:46:54 +0100158 if (++pending->failed_attempts < smsq->max_fail)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100159 return sms_pending_resend(pending);
160
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100161 sms_pending_free(pending);
162 smsq->pending -= 1;
163 sms_queue_trigger(smsq);
164}
165
166/*
167 * Resend all SMS that are scheduled for a resend. This is done to
168 * avoid an immediate failure.
169 */
170static void sms_resend_pending(void *_data)
171{
172 struct gsm_sms_pending *pending, *tmp;
173 struct gsm_sms_queue *smsq = _data;
174
175 llist_for_each_entry_safe(pending, tmp, &smsq->pending_sms, entry) {
176 struct gsm_sms *sms;
177 if (!pending->resend)
178 continue;
179
180 sms = db_sms_get(smsq->network, pending->sms_id);
181
182 /* the sms is gone? Move to the next */
183 if (!sms) {
184 sms_pending_free(pending);
185 smsq->pending -= 1;
186 sms_queue_trigger(smsq);
187 } else {
188 pending->resend = 0;
189 gsm411_send_sms_subscr(sms->receiver, sms);
190 }
191 }
192}
193
Harald Welte2483f1b2016-06-19 18:06:02 +0200194/* Find the next pending SMS by cycling through the recipients. We could also
195 * cycle through the pending SMS, but that might cause us to keep trying to
196 * send SMS to the same few subscribers repeatedly while not servicing other
197 * subscribers for a long time. By walking the list of recipient MSISDNs, we
198 * ensure that all subscribers get their fair time to receive SMS. */
199struct gsm_sms *smsq_take_next_sms(struct gsm_network *net,
200 char *last_msisdn,
201 size_t last_msisdn_buflen)
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100202{
203 struct gsm_sms *sms;
Harald Welte2483f1b2016-06-19 18:06:02 +0200204 int wrapped = 0;
205 int sanity = 100;
206 char started_with_msisdn[last_msisdn_buflen];
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100207
Max98f74672018-02-05 12:57:06 +0100208 OSMO_STRLCPY_ARRAY(started_with_msisdn, last_msisdn);
Harald Welte2483f1b2016-06-19 18:06:02 +0200209
210 while (wrapped < 2 && (--sanity)) {
211 /* If we wrapped around and passed the first msisdn, we're
212 * through the entire SMS DB; end it. */
213 if (wrapped && strcmp(last_msisdn, started_with_msisdn) >= 0)
214 break;
215
216 sms = db_sms_get_next_unsent_rr_msisdn(net, last_msisdn, 9);
217 if (!sms) {
218 last_msisdn[0] = '\0';
Max5e2e9bd2018-02-06 19:31:08 +0100219 wrapped++;
Harald Welte2483f1b2016-06-19 18:06:02 +0200220 continue;
221 }
222
223 /* Whatever happens, next time around service another recipient
224 */
225 osmo_strlcpy(last_msisdn, sms->dst.addr, last_msisdn_buflen);
226
227 /* Is the subscriber attached? If not, go to next SMS */
228 if (!sms->receiver || !sms->receiver->lu_complete)
229 continue;
230
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100231 return sms;
232 }
233
Harald Welte2483f1b2016-06-19 18:06:02 +0200234 DEBUGP(DLSMS, "SMS queue: no SMS to be sent\n");
235 return NULL;
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100236}
237
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100238/**
239 * I will submit up to max_pending - pending SMS to the
240 * subsystem.
241 */
242static void sms_submit_pending(void *_data)
243{
244 struct gsm_sms_queue *smsq = _data;
245 int attempts = smsq->max_pending - smsq->pending;
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100246 int initialized = 0;
247 unsigned long long first_sub = 0;
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100248 int attempted = 0, rounds = 0;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100249
Harald Weltee5d2c602012-11-13 06:52:42 +0100250 LOGP(DLSMS, LOGL_DEBUG, "Attempting to send %d SMS\n", attempts);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100251
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100252 do {
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100253 struct gsm_sms_pending *pending;
254 struct gsm_sms *sms;
255
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100256
Harald Welte2483f1b2016-06-19 18:06:02 +0200257 sms = smsq_take_next_sms(smsq->network, smsq->last_msisdn,
258 sizeof(smsq->last_msisdn));
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200259 if (!sms) {
260 LOGP(DLSMS, LOGL_DEBUG, "Sending SMS done (%d attempted)\n",
261 attempted);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100262 break;
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200263 }
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100264
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100265 rounds += 1;
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200266 LOGP(DLSMS, LOGL_DEBUG, "Sending SMS round %d\n", rounds);
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100267
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100268 /*
269 * This code needs to detect a loop. It assumes that no SMS
270 * will vanish during the time this is executed. We will remember
271 * the id of the first GSM subscriber we see and then will
272 * compare this. The Database code should make sure that we will
273 * see all other subscribers first before seeing this one again.
274 *
275 * It is always scary to have an infinite loop like this.
276 */
277 if (!initialized) {
278 first_sub = sms->receiver->id;
279 initialized = 1;
280 } else if (first_sub == sms->receiver->id) {
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200281 LOGP(DLSMS, LOGL_DEBUG, "Sending SMS done (loop) (%d attempted)\n",
282 attempted);
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100283 sms_free(sms);
284 break;
285 }
286
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100287 /* no need to send a pending sms */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100288 if (sms_queue_sms_is_pending(smsq, sms->id)) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100289 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freytherdc53af62010-12-27 20:12:25 +0100290 "SMSqueue with pending sms: %llu. Skipping\n", sms->id);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100291 sms_free(sms);
292 continue;
293 }
294
295 /* no need to send a SMS with the same receiver */
296 if (sms_subscriber_is_pending(smsq, sms->receiver)) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100297 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100298 "SMSqueue with pending sub: %llu. Skipping\n", sms->receiver->id);
299 sms_free(sms);
300 continue;
301 }
302
303 pending = sms_pending_from(smsq, sms);
304 if (!pending) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100305 LOGP(DLSMS, LOGL_ERROR,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100306 "Failed to create pending SMS entry.\n");
307 sms_free(sms);
308 continue;
309 }
310
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100311 attempted += 1;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100312 smsq->pending += 1;
Holger Hans Peter Freyther701076e2010-12-28 14:25:25 +0100313 llist_add_tail(&pending->entry, &smsq->pending_sms);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100314 gsm411_send_sms_subscr(sms->receiver, sms);
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100315 } while (attempted < attempts && rounds < 1000);
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100316
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100317 LOGP(DLSMS, LOGL_DEBUG, "SMSqueue added %d messages in %d rounds\n", attempted, rounds);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100318}
319
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100320/**
321 * Send the next SMS or trigger the queue
322 */
Harald Welte2483f1b2016-06-19 18:06:02 +0200323static void sms_send_next(struct vlr_subscr *vsub)
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100324{
Harald Welte2483f1b2016-06-19 18:06:02 +0200325 struct gsm_network *net = vsub->vlr->user_ctx;
326 struct gsm_sms_queue *smsq = net->sms_queue;
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100327 struct gsm_sms_pending *pending;
328 struct gsm_sms *sms;
329
330 /* the subscriber should not be in the queue */
Harald Welte2483f1b2016-06-19 18:06:02 +0200331 OSMO_ASSERT(!sms_subscriber_is_pending(smsq, vsub));
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100332
333 /* check for more messages for this subscriber */
Harald Welte2483f1b2016-06-19 18:06:02 +0200334 sms = db_sms_get_unsent_for_subscr(vsub, UINT_MAX);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100335 if (!sms)
336 goto no_pending_sms;
337
Harald Welte2483f1b2016-06-19 18:06:02 +0200338 /* The sms should not be scheduled right now */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100339 OSMO_ASSERT(!sms_queue_sms_is_pending(smsq, sms->id));
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100340
341 /* Remember that we deliver this SMS and send it */
342 pending = sms_pending_from(smsq, sms);
343 if (!pending) {
344 LOGP(DLSMS, LOGL_ERROR,
345 "Failed to create pending SMS entry.\n");
346 sms_free(sms);
347 goto no_pending_sms;
348 }
349
350 smsq->pending += 1;
351 llist_add_tail(&pending->entry, &smsq->pending_sms);
352 gsm411_send_sms_subscr(sms->receiver, sms);
353 return;
354
355no_pending_sms:
356 /* Try to send the SMS to avoid the queue being stuck */
Harald Welte2483f1b2016-06-19 18:06:02 +0200357 sms_submit_pending(net->sms_queue);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100358}
359
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100360/*
361 * Kick off the queue again.
362 */
363int sms_queue_trigger(struct gsm_sms_queue *smsq)
364{
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200365 LOGP(DLSMS, LOGL_DEBUG, "Triggering SMS queue\n");
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200366 if (osmo_timer_pending(&smsq->push_queue))
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100367 return 0;
368
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200369 osmo_timer_schedule(&smsq->push_queue, 1, 0);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100370 return 0;
371}
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100372
373int sms_queue_start(struct gsm_network *network, int max_pending)
374{
375 struct gsm_sms_queue *sms = talloc_zero(network, struct gsm_sms_queue);
376 if (!sms) {
377 LOGP(DMSC, LOGL_ERROR, "Failed to create the SMS queue.\n");
378 return -1;
379 }
380
Pablo Neira Ayusobbc5b992011-05-06 12:12:31 +0200381 osmo_signal_register_handler(SS_SUBSCR, sms_subscr_cb, network);
382 osmo_signal_register_handler(SS_SMS, sms_sms_cb, network);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100383
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100384 network->sms_queue = sms;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100385 INIT_LLIST_HEAD(&sms->pending_sms);
Holger Hans Peter Freythera37e3bc2010-12-25 17:43:03 +0100386 sms->max_fail = 1;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100387 sms->network = network;
388 sms->max_pending = max_pending;
Pablo Neira Ayuso51215762017-05-08 20:57:52 +0200389 osmo_timer_setup(&sms->push_queue, sms_submit_pending, sms);
390 osmo_timer_setup(&sms->resend_pending, sms_resend_pending, sms);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100391
392 sms_submit_pending(sms);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100393
394 return 0;
395}
396
Harald Welte2483f1b2016-06-19 18:06:02 +0200397static int sub_ready_for_sm(struct gsm_network *net, struct vlr_subscr *vsub)
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100398{
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100399 struct gsm_sms *sms;
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200400 struct gsm_sms_pending *pending;
401 struct gsm_subscriber_connection *conn;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100402
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200403 /*
404 * The code used to be very clever and tried to submit
405 * a SMS during the Location Updating Request. This has
406 * two issues:
407 * 1.) The Phone might not be ready yet, e.g. the C155
408 * will not respond to the Submit when it is booting.
409 * 2.) The queue is already trying to submit SMS to the
410 * user and by not responding to the paging request
411 * we will set the LAC back to 0. We would have to
412 * stop the paging and move things over.
413 *
414 * We need to be careful in what we try here.
415 */
416
417 /* check if we have pending requests */
Harald Welte2483f1b2016-06-19 18:06:02 +0200418 pending = sms_subscriber_find_pending(net->sms_queue, vsub);
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200419 if (pending) {
420 LOGP(DMSC, LOGL_NOTICE,
421 "Pending paging while subscriber %llu attached.\n",
Harald Welte2483f1b2016-06-19 18:06:02 +0200422 vsub->id);
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200423 return 0;
424 }
425
Harald Welte2483f1b2016-06-19 18:06:02 +0200426 conn = connection_for_subscr(vsub);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100427 if (!conn)
428 return -1;
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200429
430 /* Now try to deliver any pending SMS to this sub */
Harald Welte2483f1b2016-06-19 18:06:02 +0200431 sms = db_sms_get_unsent_for_subscr(vsub, UINT_MAX);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100432 if (!sms)
433 return -1;
434 gsm411_send_sms(conn, sms);
435 return 0;
436}
437
438static int sms_subscr_cb(unsigned int subsys, unsigned int signal,
439 void *handler_data, void *signal_data)
440{
Harald Welte2483f1b2016-06-19 18:06:02 +0200441 struct vlr_subscr *vsub = signal_data;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100442
443 if (signal != S_SUBSCR_ATTACHED)
444 return 0;
445
446 /* this is readyForSM */
Harald Welte2483f1b2016-06-19 18:06:02 +0200447 return sub_ready_for_sm(handler_data, vsub);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100448}
449
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100450static int sms_sms_cb(unsigned int subsys, unsigned int signal,
451 void *handler_data, void *signal_data)
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100452{
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100453 struct gsm_network *network = handler_data;
454 struct sms_signal_data *sig_sms = signal_data;
455 struct gsm_sms_pending *pending;
Harald Welte2483f1b2016-06-19 18:06:02 +0200456 struct vlr_subscr *vsub;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100457
458 /* We got a new SMS and maybe should launch the queue again. */
459 if (signal == S_SMS_SUBMITTED || signal == S_SMS_SMMA) {
Holger Hans Peter Freyther024dc772014-02-24 14:29:27 +0100460 /* TODO: For SMMA we might want to re-use the radio connection. */
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100461 sms_queue_trigger(network->sms_queue);
462 return 0;
463 }
464
465 if (!sig_sms->sms)
466 return -1;
467
468
469 /*
470 * Find the entry of our queue. The SMS subsystem will submit
471 * sms that are not in our control as we just have a channel
472 * open anyway.
473 */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100474 pending = sms_find_pending(network->sms_queue, sig_sms->sms->id);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100475 if (!pending)
476 return 0;
477
478 switch (signal) {
479 case S_SMS_DELIVERED:
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100480 /* Remember the subscriber and clear the pending entry */
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100481 network->sms_queue->pending -= 1;
Harald Welte2483f1b2016-06-19 18:06:02 +0200482 vsub = vlr_subscr_get(pending->vsub);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100483 sms_pending_free(pending);
Stefan Sperling6ba2d5a2018-01-18 18:55:26 +0100484 db_sms_delete_sent_message_by_id(pending->sms_id);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100485 /* Attempt to send another SMS to this subscriber */
Harald Welte2483f1b2016-06-19 18:06:02 +0200486 sms_send_next(vsub);
487 vlr_subscr_put(vsub);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100488 break;
489 case S_SMS_MEM_EXCEEDED:
490 network->sms_queue->pending -= 1;
491 sms_pending_free(pending);
492 sms_queue_trigger(network->sms_queue);
493 break;
494 case S_SMS_UNKNOWN_ERROR:
495 /*
496 * There can be many reasons for this failure. E.g. the paging
497 * timed out, the subscriber was not paged at all, or there was
498 * a protocol error. The current strategy is to try sending the
499 * next SMS for busy/oom and to retransmit when we have paged.
500 *
501 * When the paging expires three times we will disable the
502 * subscriber. If we have some kind of other transmit error we
503 * should flag the SMS as bad.
504 */
505 switch (sig_sms->paging_result) {
506 case 0:
507 /* BAD SMS? */
508 db_sms_inc_deliver_attempts(sig_sms->sms);
509 sms_pending_failed(pending, 0);
510 break;
511 case GSM_PAGING_EXPIRED:
512 sms_pending_failed(pending, 1);
513 break;
514
515 case GSM_PAGING_OOM:
516 case GSM_PAGING_BUSY:
517 network->sms_queue->pending -= 1;
518 sms_pending_free(pending);
519 sms_queue_trigger(network->sms_queue);
520 break;
521 default:
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100522 LOGP(DLSMS, LOGL_ERROR, "Unhandled result: %d\n",
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100523 sig_sms->paging_result);
524 }
525 break;
526 default:
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100527 LOGP(DLSMS, LOGL_ERROR, "Unhandled result: %d\n",
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100528 sig_sms->paging_result);
529 }
530
Stefan Sperling14e05172018-01-22 17:31:20 +0100531 /* While here, attempt to remove an expired SMS from the DB. */
532 db_sms_delete_oldest_expired_message();
533
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100534 return 0;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100535}
Holger Hans Peter Freyther81c0e252010-12-25 14:08:00 +0100536
537/* VTY helper functions */
538int sms_queue_stats(struct gsm_sms_queue *smsq, struct vty *vty)
539{
540 struct gsm_sms_pending *pending;
541
542 vty_out(vty, "SMSqueue with max_pending: %d pending: %d%s",
543 smsq->max_pending, smsq->pending, VTY_NEWLINE);
544
545 llist_for_each_entry(pending, &smsq->pending_sms, entry)
Holger Hans Peter Freyther583e9ae2010-12-27 20:19:48 +0100546 vty_out(vty, " SMS Pending for Subscriber: %llu SMS: %llu Failed: %d.%s",
Harald Welte2483f1b2016-06-19 18:06:02 +0200547 pending->vsub->id, pending->sms_id,
Holger Hans Peter Freyther583e9ae2010-12-27 20:19:48 +0100548 pending->failed_attempts, VTY_NEWLINE);
Holger Hans Peter Freyther81c0e252010-12-25 14:08:00 +0100549 return 0;
550}
Holger Hans Peter Freyther3c6f6c22010-12-25 14:25:12 +0100551
552int sms_queue_set_max_pending(struct gsm_sms_queue *smsq, int max_pending)
553{
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100554 LOGP(DLSMS, LOGL_NOTICE, "SMSqueue old max: %d new: %d\n",
Holger Hans Peter Freyther3c6f6c22010-12-25 14:25:12 +0100555 smsq->max_pending, max_pending);
556 smsq->max_pending = max_pending;
557 return 0;
558}
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100559
Holger Hans Peter Freyther994dcbb2010-12-25 14:50:50 +0100560int sms_queue_set_max_failure(struct gsm_sms_queue *smsq, int max_fail)
561{
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100562 LOGP(DLSMS, LOGL_NOTICE, "SMSqueue max failure old: %d new: %d\n",
Holger Hans Peter Freyther994dcbb2010-12-25 14:50:50 +0100563 smsq->max_fail, max_fail);
564 smsq->max_fail = max_fail;
565 return 0;
566}
567
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100568int sms_queue_clear(struct gsm_sms_queue *smsq)
569{
570 struct gsm_sms_pending *pending, *tmp;
571
572 llist_for_each_entry_safe(pending, tmp, &smsq->pending_sms, entry) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100573 LOGP(DLSMS, LOGL_NOTICE,
Harald Welte2483f1b2016-06-19 18:06:02 +0200574 "SMSqueue clearing for sub %llu\n", pending->vsub->id);
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100575 sms_pending_free(pending);
576 }
577
Holger Hans Peter Freyther96e9f082010-12-28 14:09:07 +0100578 smsq->pending = 0;
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100579 return 0;
580}