blob: 6098a532483534f57a9a735d8d264f1339303cdb [file] [log] [blame]
Martin Hauke3f07dac2019-11-14 17:49:08 +01001/* SMS queue to continuously attempt to deliver SMS */
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +01002/*
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;
Neels Hofmeyrc4628a32018-12-07 14:47:34 +010053 struct msc_a *msc_a;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010054 unsigned long long sms_id;
55 int failed_attempts;
56 int resend;
57};
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010058
59struct gsm_sms_queue {
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +020060 struct osmo_timer_list resend_pending;
61 struct osmo_timer_list push_queue;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010062 struct gsm_network *network;
Holger Hans Peter Freyther7e59c832010-12-25 14:46:54 +010063 int max_fail;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010064 int max_pending;
65 int pending;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010066
67 struct llist_head pending_sms;
Harald Welte2483f1b2016-06-19 18:06:02 +020068
Vadim Yanitskiy8b0737f2019-05-25 19:27:17 +070069 char last_msisdn[GSM23003_MSISDN_MAX_DIGITS+1];
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +010070};
71
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010072static int sms_subscr_cb(unsigned int, unsigned int, void *, void *);
73static int sms_sms_cb(unsigned int, unsigned int, void *, void *);
74
75static struct gsm_sms_pending *sms_find_pending(struct gsm_sms_queue *smsq,
Stefan Sperling87cba1f2018-01-22 17:05:37 +010076 unsigned long long sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010077{
78 struct gsm_sms_pending *pending;
79
80 llist_for_each_entry(pending, &smsq->pending_sms, entry) {
Stefan Sperling87cba1f2018-01-22 17:05:37 +010081 if (pending->sms_id == sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010082 return pending;
83 }
84
85 return NULL;
86}
87
Stefan Sperling87cba1f2018-01-22 17:05:37 +010088int sms_queue_sms_is_pending(struct gsm_sms_queue *smsq, unsigned long long sms_id)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010089{
Stefan Sperling87cba1f2018-01-22 17:05:37 +010090 return sms_find_pending(smsq, sms_id) != NULL;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010091}
92
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +020093static struct gsm_sms_pending *sms_subscriber_find_pending(
94 struct gsm_sms_queue *smsq,
Harald Welte2483f1b2016-06-19 18:06:02 +020095 struct vlr_subscr *vsub)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +010096{
97 struct gsm_sms_pending *pending;
98
99 llist_for_each_entry(pending, &smsq->pending_sms, entry) {
Harald Welte2483f1b2016-06-19 18:06:02 +0200100 if (pending->vsub == vsub)
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200101 return pending;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100102 }
103
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200104 return NULL;
105}
106
107static int sms_subscriber_is_pending(struct gsm_sms_queue *smsq,
Harald Welte2483f1b2016-06-19 18:06:02 +0200108 struct vlr_subscr *vsub)
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200109{
Harald Welte2483f1b2016-06-19 18:06:02 +0200110 return sms_subscriber_find_pending(smsq, vsub) != NULL;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100111}
112
113static struct gsm_sms_pending *sms_pending_from(struct gsm_sms_queue *smsq,
114 struct gsm_sms *sms)
115{
116 struct gsm_sms_pending *pending;
117
118 pending = talloc_zero(smsq, struct gsm_sms_pending);
119 if (!pending)
120 return NULL;
121
Neels Hofmeyr7c5346c2019-02-19 02:36:35 +0100122 vlr_subscr_get(sms->receiver, VSUB_USE_SMS_PENDING);
123 pending->vsub = sms->receiver;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100124 pending->sms_id = sms->id;
125 return pending;
126}
127
128static void sms_pending_free(struct gsm_sms_pending *pending)
129{
Neels Hofmeyr7c5346c2019-02-19 02:36:35 +0100130 vlr_subscr_put(pending->vsub, VSUB_USE_SMS_PENDING);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100131 llist_del(&pending->entry);
132 talloc_free(pending);
133}
134
135static void sms_pending_resend(struct gsm_sms_pending *pending)
136{
Harald Welte2483f1b2016-06-19 18:06:02 +0200137 struct gsm_network *net = pending->vsub->vlr->user_ctx;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100138 struct gsm_sms_queue *smsq;
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100139 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100140 "Scheduling resend of SMS %llu.\n", pending->sms_id);
141
142 pending->resend = 1;
143
Harald Welte2483f1b2016-06-19 18:06:02 +0200144 smsq = net->sms_queue;
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200145 if (osmo_timer_pending(&smsq->resend_pending))
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100146 return;
147
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200148 osmo_timer_schedule(&smsq->resend_pending, 1, 0);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100149}
150
151static void sms_pending_failed(struct gsm_sms_pending *pending, int paging_error)
152{
Harald Welte2483f1b2016-06-19 18:06:02 +0200153 struct gsm_network *net = pending->vsub->vlr->user_ctx;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100154 struct gsm_sms_queue *smsq;
155
Neels Hofmeyr34a36da2019-03-06 16:00:41 +0100156 pending->failed_attempts++;
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100157 LOGP(DLSMS, LOGL_NOTICE, "Sending SMS %llu failed %d times.\n",
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100158 pending->sms_id, pending->failed_attempts);
159
Harald Welte2483f1b2016-06-19 18:06:02 +0200160 smsq = net->sms_queue;
Neels Hofmeyr34a36da2019-03-06 16:00:41 +0100161 if (pending->failed_attempts < smsq->max_fail)
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100162 return sms_pending_resend(pending);
163
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100164 sms_pending_free(pending);
165 smsq->pending -= 1;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100166}
167
168/*
169 * Resend all SMS that are scheduled for a resend. This is done to
170 * avoid an immediate failure.
171 */
172static void sms_resend_pending(void *_data)
173{
174 struct gsm_sms_pending *pending, *tmp;
175 struct gsm_sms_queue *smsq = _data;
176
177 llist_for_each_entry_safe(pending, tmp, &smsq->pending_sms, entry) {
178 struct gsm_sms *sms;
179 if (!pending->resend)
180 continue;
181
182 sms = db_sms_get(smsq->network, pending->sms_id);
183
184 /* the sms is gone? Move to the next */
185 if (!sms) {
186 sms_pending_free(pending);
187 smsq->pending -= 1;
188 sms_queue_trigger(smsq);
189 } else {
190 pending->resend = 0;
Vadim Yanitskiy24e025e2018-11-22 15:42:39 +0700191 gsm411_send_sms(smsq->network, sms->receiver, sms);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100192 }
193 }
194}
195
Harald Welte2483f1b2016-06-19 18:06:02 +0200196/* Find the next pending SMS by cycling through the recipients. We could also
197 * cycle through the pending SMS, but that might cause us to keep trying to
198 * send SMS to the same few subscribers repeatedly while not servicing other
199 * subscribers for a long time. By walking the list of recipient MSISDNs, we
200 * ensure that all subscribers get their fair time to receive SMS. */
201struct gsm_sms *smsq_take_next_sms(struct gsm_network *net,
202 char *last_msisdn,
203 size_t last_msisdn_buflen)
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100204{
205 struct gsm_sms *sms;
Harald Welte2483f1b2016-06-19 18:06:02 +0200206 int wrapped = 0;
207 int sanity = 100;
208 char started_with_msisdn[last_msisdn_buflen];
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100209
Max98f74672018-02-05 12:57:06 +0100210 OSMO_STRLCPY_ARRAY(started_with_msisdn, last_msisdn);
Harald Welte2483f1b2016-06-19 18:06:02 +0200211
212 while (wrapped < 2 && (--sanity)) {
213 /* If we wrapped around and passed the first msisdn, we're
214 * through the entire SMS DB; end it. */
215 if (wrapped && strcmp(last_msisdn, started_with_msisdn) >= 0)
216 break;
217
218 sms = db_sms_get_next_unsent_rr_msisdn(net, last_msisdn, 9);
219 if (!sms) {
220 last_msisdn[0] = '\0';
Max5e2e9bd2018-02-06 19:31:08 +0100221 wrapped++;
Harald Welte2483f1b2016-06-19 18:06:02 +0200222 continue;
223 }
224
225 /* Whatever happens, next time around service another recipient
226 */
227 osmo_strlcpy(last_msisdn, sms->dst.addr, last_msisdn_buflen);
228
229 /* Is the subscriber attached? If not, go to next SMS */
Vadim Yanitskiy96262a72019-03-28 21:25:14 +0700230 if (!sms->receiver || !sms->receiver->lu_complete) {
231 LOGP(DLSMS, LOGL_DEBUG,
Pau Espin Pedrolf8af7762019-04-09 12:44:35 +0200232 "Subscriber %s%s is not attached, skipping SMS %llu\n",
233 sms->receiver ? "" : "MSISDN-",
234 sms->receiver ? vlr_subscr_msisdn_or_name(sms->receiver)
235 : sms->dst.addr, sms->id);
Vadim Yanitskiy96262a72019-03-28 21:25:14 +0700236 sms_free(sms);
Harald Welte2483f1b2016-06-19 18:06:02 +0200237 continue;
Vadim Yanitskiy96262a72019-03-28 21:25:14 +0700238 }
Harald Welte2483f1b2016-06-19 18:06:02 +0200239
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100240 return sms;
241 }
242
Harald Welte2483f1b2016-06-19 18:06:02 +0200243 DEBUGP(DLSMS, "SMS queue: no SMS to be sent\n");
244 return NULL;
Holger Hans Peter Freytherf7e23892010-12-25 17:45:23 +0100245}
246
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100247/**
248 * I will submit up to max_pending - pending SMS to the
249 * subsystem.
250 */
251static void sms_submit_pending(void *_data)
252{
253 struct gsm_sms_queue *smsq = _data;
254 int attempts = smsq->max_pending - smsq->pending;
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100255 int initialized = 0;
256 unsigned long long first_sub = 0;
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100257 int attempted = 0, rounds = 0;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100258
Pau Espin Pedrolfe5b7042019-06-20 10:45:35 +0200259 LOGP(DLSMS, LOGL_DEBUG, "Attempting to send up to %d SMS\n", attempts);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100260
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100261 do {
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100262 struct gsm_sms_pending *pending;
263 struct gsm_sms *sms;
264
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100265
Harald Welte2483f1b2016-06-19 18:06:02 +0200266 sms = smsq_take_next_sms(smsq->network, smsq->last_msisdn,
267 sizeof(smsq->last_msisdn));
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200268 if (!sms) {
269 LOGP(DLSMS, LOGL_DEBUG, "Sending SMS done (%d attempted)\n",
270 attempted);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100271 break;
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200272 }
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100273
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100274 rounds += 1;
Neels Hofmeyr3d6a8be2019-11-18 07:15:56 +0100275 LOGP(DLSMS, LOGL_DEBUG, "Checking whether to send SMS %llu\n", sms->id);
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100276
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100277 /*
278 * This code needs to detect a loop. It assumes that no SMS
279 * will vanish during the time this is executed. We will remember
280 * the id of the first GSM subscriber we see and then will
281 * compare this. The Database code should make sure that we will
282 * see all other subscribers first before seeing this one again.
283 *
284 * It is always scary to have an infinite loop like this.
285 */
286 if (!initialized) {
287 first_sub = sms->receiver->id;
288 initialized = 1;
289 } else if (first_sub == sms->receiver->id) {
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200290 LOGP(DLSMS, LOGL_DEBUG, "Sending SMS done (loop) (%d attempted)\n",
291 attempted);
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100292 sms_free(sms);
293 break;
294 }
295
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100296 /* no need to send a pending sms */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100297 if (sms_queue_sms_is_pending(smsq, sms->id)) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100298 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freytherdc53af62010-12-27 20:12:25 +0100299 "SMSqueue with pending sms: %llu. Skipping\n", sms->id);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100300 sms_free(sms);
301 continue;
302 }
303
304 /* no need to send a SMS with the same receiver */
305 if (sms_subscriber_is_pending(smsq, sms->receiver)) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100306 LOGP(DLSMS, LOGL_DEBUG,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100307 "SMSqueue with pending sub: %llu. Skipping\n", sms->receiver->id);
308 sms_free(sms);
309 continue;
310 }
311
312 pending = sms_pending_from(smsq, sms);
313 if (!pending) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100314 LOGP(DLSMS, LOGL_ERROR,
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100315 "Failed to create pending SMS entry.\n");
316 sms_free(sms);
317 continue;
318 }
319
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100320 attempted += 1;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100321 smsq->pending += 1;
Holger Hans Peter Freyther701076e2010-12-28 14:25:25 +0100322 llist_add_tail(&pending->entry, &smsq->pending_sms);
Vadim Yanitskiy24e025e2018-11-22 15:42:39 +0700323 gsm411_send_sms(smsq->network, sms->receiver, sms);
Holger Hans Peter Freyther5479fc82010-12-25 19:32:12 +0100324 } while (attempted < attempts && rounds < 1000);
Holger Hans Peter Freyther20384572010-12-25 19:28:44 +0100325
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100326 LOGP(DLSMS, LOGL_DEBUG, "SMSqueue added %d messages in %d rounds\n", attempted, rounds);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100327}
328
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100329/**
330 * Send the next SMS or trigger the queue
331 */
Harald Welte2483f1b2016-06-19 18:06:02 +0200332static void sms_send_next(struct vlr_subscr *vsub)
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100333{
Harald Welte2483f1b2016-06-19 18:06:02 +0200334 struct gsm_network *net = vsub->vlr->user_ctx;
335 struct gsm_sms_queue *smsq = net->sms_queue;
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100336 struct gsm_sms_pending *pending;
337 struct gsm_sms *sms;
338
339 /* the subscriber should not be in the queue */
Harald Welte2483f1b2016-06-19 18:06:02 +0200340 OSMO_ASSERT(!sms_subscriber_is_pending(smsq, vsub));
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100341
342 /* check for more messages for this subscriber */
Harald Welte2483f1b2016-06-19 18:06:02 +0200343 sms = db_sms_get_unsent_for_subscr(vsub, UINT_MAX);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100344 if (!sms)
345 goto no_pending_sms;
346
Harald Welte2483f1b2016-06-19 18:06:02 +0200347 /* The sms should not be scheduled right now */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100348 OSMO_ASSERT(!sms_queue_sms_is_pending(smsq, sms->id));
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100349
350 /* Remember that we deliver this SMS and send it */
351 pending = sms_pending_from(smsq, sms);
352 if (!pending) {
353 LOGP(DLSMS, LOGL_ERROR,
354 "Failed to create pending SMS entry.\n");
355 sms_free(sms);
356 goto no_pending_sms;
357 }
358
359 smsq->pending += 1;
360 llist_add_tail(&pending->entry, &smsq->pending_sms);
Vadim Yanitskiy24e025e2018-11-22 15:42:39 +0700361 gsm411_send_sms(smsq->network, sms->receiver, sms);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100362 return;
363
364no_pending_sms:
365 /* Try to send the SMS to avoid the queue being stuck */
Harald Welte2483f1b2016-06-19 18:06:02 +0200366 sms_submit_pending(net->sms_queue);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100367}
368
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100369/*
370 * Kick off the queue again.
371 */
372int sms_queue_trigger(struct gsm_sms_queue *smsq)
373{
Neels Hofmeyr1e918c32016-05-09 21:48:53 +0200374 LOGP(DLSMS, LOGL_DEBUG, "Triggering SMS queue\n");
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200375 if (osmo_timer_pending(&smsq->push_queue))
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100376 return 0;
377
Pablo Neira Ayusobf540cb2011-05-06 12:11:06 +0200378 osmo_timer_schedule(&smsq->push_queue, 1, 0);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100379 return 0;
380}
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100381
382int sms_queue_start(struct gsm_network *network, int max_pending)
383{
384 struct gsm_sms_queue *sms = talloc_zero(network, struct gsm_sms_queue);
385 if (!sms) {
386 LOGP(DMSC, LOGL_ERROR, "Failed to create the SMS queue.\n");
387 return -1;
388 }
389
Pablo Neira Ayusobbc5b992011-05-06 12:12:31 +0200390 osmo_signal_register_handler(SS_SUBSCR, sms_subscr_cb, network);
391 osmo_signal_register_handler(SS_SMS, sms_sms_cb, network);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100392
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100393 network->sms_queue = sms;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100394 INIT_LLIST_HEAD(&sms->pending_sms);
Holger Hans Peter Freythera37e3bc2010-12-25 17:43:03 +0100395 sms->max_fail = 1;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100396 sms->network = network;
397 sms->max_pending = max_pending;
Pablo Neira Ayuso51215762017-05-08 20:57:52 +0200398 osmo_timer_setup(&sms->push_queue, sms_submit_pending, sms);
399 osmo_timer_setup(&sms->resend_pending, sms_resend_pending, sms);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100400
401 sms_submit_pending(sms);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100402
403 return 0;
404}
405
Harald Welte2483f1b2016-06-19 18:06:02 +0200406static int sub_ready_for_sm(struct gsm_network *net, struct vlr_subscr *vsub)
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100407{
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100408 struct gsm_sms *sms;
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200409 struct gsm_sms_pending *pending;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100410
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200411 /*
412 * The code used to be very clever and tried to submit
413 * a SMS during the Location Updating Request. This has
414 * two issues:
415 * 1.) The Phone might not be ready yet, e.g. the C155
416 * will not respond to the Submit when it is booting.
417 * 2.) The queue is already trying to submit SMS to the
418 * user and by not responding to the paging request
419 * we will set the LAC back to 0. We would have to
420 * stop the paging and move things over.
421 *
422 * We need to be careful in what we try here.
423 */
424
425 /* check if we have pending requests */
Harald Welte2483f1b2016-06-19 18:06:02 +0200426 pending = sms_subscriber_find_pending(net->sms_queue, vsub);
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200427 if (pending) {
428 LOGP(DMSC, LOGL_NOTICE,
429 "Pending paging while subscriber %llu attached.\n",
Harald Welte2483f1b2016-06-19 18:06:02 +0200430 vsub->id);
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200431 return 0;
432 }
433
Holger Hans Peter Freyther074b2b22011-07-25 00:13:06 +0200434 /* Now try to deliver any pending SMS to this sub */
Harald Welte2483f1b2016-06-19 18:06:02 +0200435 sms = db_sms_get_unsent_for_subscr(vsub, UINT_MAX);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100436 if (!sms)
437 return -1;
Vadim Yanitskiy24e025e2018-11-22 15:42:39 +0700438
439 gsm411_send_sms(net, vsub, sms);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100440 return 0;
441}
442
443static int sms_subscr_cb(unsigned int subsys, unsigned int signal,
444 void *handler_data, void *signal_data)
445{
Harald Welte2483f1b2016-06-19 18:06:02 +0200446 struct vlr_subscr *vsub = signal_data;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100447
448 if (signal != S_SUBSCR_ATTACHED)
449 return 0;
450
451 /* this is readyForSM */
Harald Welte2483f1b2016-06-19 18:06:02 +0200452 return sub_ready_for_sm(handler_data, vsub);
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100453}
454
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100455static int sms_sms_cb(unsigned int subsys, unsigned int signal,
456 void *handler_data, void *signal_data)
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100457{
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100458 struct gsm_network *network = handler_data;
459 struct sms_signal_data *sig_sms = signal_data;
460 struct gsm_sms_pending *pending;
Harald Welte2483f1b2016-06-19 18:06:02 +0200461 struct vlr_subscr *vsub;
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100462
463 /* We got a new SMS and maybe should launch the queue again. */
464 if (signal == S_SMS_SUBMITTED || signal == S_SMS_SMMA) {
Holger Hans Peter Freyther024dc772014-02-24 14:29:27 +0100465 /* TODO: For SMMA we might want to re-use the radio connection. */
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100466 sms_queue_trigger(network->sms_queue);
467 return 0;
468 }
469
470 if (!sig_sms->sms)
471 return -1;
472
473
474 /*
475 * Find the entry of our queue. The SMS subsystem will submit
476 * sms that are not in our control as we just have a channel
477 * open anyway.
478 */
Stefan Sperling87cba1f2018-01-22 17:05:37 +0100479 pending = sms_find_pending(network->sms_queue, sig_sms->sms->id);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100480 if (!pending)
481 return 0;
482
483 switch (signal) {
484 case S_SMS_DELIVERED:
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100485 /* Remember the subscriber and clear the pending entry */
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100486 network->sms_queue->pending -= 1;
Neels Hofmeyr7c5346c2019-02-19 02:36:35 +0100487 vsub = pending->vsub;
488 vlr_subscr_get(vsub, __func__);
Stefan Sperling6ba2d5a2018-01-18 18:55:26 +0100489 db_sms_delete_sent_message_by_id(pending->sms_id);
Harald Welte37a3ae82018-04-14 20:53:36 +0200490 sms_pending_free(pending);
Holger Hans Peter Freyther93de8b22014-02-24 16:13:04 +0100491 /* Attempt to send another SMS to this subscriber */
Harald Welte2483f1b2016-06-19 18:06:02 +0200492 sms_send_next(vsub);
Neels Hofmeyr7c5346c2019-02-19 02:36:35 +0100493 vlr_subscr_put(vsub, __func__);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100494 break;
495 case S_SMS_MEM_EXCEEDED:
496 network->sms_queue->pending -= 1;
497 sms_pending_free(pending);
498 sms_queue_trigger(network->sms_queue);
499 break;
500 case S_SMS_UNKNOWN_ERROR:
501 /*
502 * There can be many reasons for this failure. E.g. the paging
503 * timed out, the subscriber was not paged at all, or there was
504 * a protocol error. The current strategy is to try sending the
505 * next SMS for busy/oom and to retransmit when we have paged.
506 *
507 * When the paging expires three times we will disable the
508 * subscriber. If we have some kind of other transmit error we
509 * should flag the SMS as bad.
510 */
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100511 if (sig_sms->paging_result) {
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100512 /* BAD SMS? */
513 db_sms_inc_deliver_attempts(sig_sms->sms);
514 sms_pending_failed(pending, 0);
Neels Hofmeyrc4628a32018-12-07 14:47:34 +0100515 } else {
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100516 sms_pending_failed(pending, 1);
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100517 }
518 break;
519 default:
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100520 LOGP(DLSMS, LOGL_ERROR, "Unhandled result: %d\n",
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100521 sig_sms->paging_result);
522 }
523
Stefan Sperling14e05172018-01-22 17:31:20 +0100524 /* While here, attempt to remove an expired SMS from the DB. */
525 db_sms_delete_oldest_expired_message();
526
Holger Hans Peter Freyther17164062010-12-24 21:39:55 +0100527 return 0;
Holger Hans Peter Freyther11b28f92010-12-24 13:48:27 +0100528}
Holger Hans Peter Freyther81c0e252010-12-25 14:08:00 +0100529
530/* VTY helper functions */
531int sms_queue_stats(struct gsm_sms_queue *smsq, struct vty *vty)
532{
533 struct gsm_sms_pending *pending;
534
535 vty_out(vty, "SMSqueue with max_pending: %d pending: %d%s",
536 smsq->max_pending, smsq->pending, VTY_NEWLINE);
537
538 llist_for_each_entry(pending, &smsq->pending_sms, entry)
Holger Hans Peter Freyther583e9ae2010-12-27 20:19:48 +0100539 vty_out(vty, " SMS Pending for Subscriber: %llu SMS: %llu Failed: %d.%s",
Harald Welte2483f1b2016-06-19 18:06:02 +0200540 pending->vsub->id, pending->sms_id,
Holger Hans Peter Freyther583e9ae2010-12-27 20:19:48 +0100541 pending->failed_attempts, VTY_NEWLINE);
Holger Hans Peter Freyther81c0e252010-12-25 14:08:00 +0100542 return 0;
543}
Holger Hans Peter Freyther3c6f6c22010-12-25 14:25:12 +0100544
545int sms_queue_set_max_pending(struct gsm_sms_queue *smsq, int max_pending)
546{
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100547 LOGP(DLSMS, LOGL_NOTICE, "SMSqueue old max: %d new: %d\n",
Holger Hans Peter Freyther3c6f6c22010-12-25 14:25:12 +0100548 smsq->max_pending, max_pending);
549 smsq->max_pending = max_pending;
550 return 0;
551}
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100552
Holger Hans Peter Freyther994dcbb2010-12-25 14:50:50 +0100553int sms_queue_set_max_failure(struct gsm_sms_queue *smsq, int max_fail)
554{
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100555 LOGP(DLSMS, LOGL_NOTICE, "SMSqueue max failure old: %d new: %d\n",
Holger Hans Peter Freyther994dcbb2010-12-25 14:50:50 +0100556 smsq->max_fail, max_fail);
557 smsq->max_fail = max_fail;
558 return 0;
559}
560
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100561int sms_queue_clear(struct gsm_sms_queue *smsq)
562{
563 struct gsm_sms_pending *pending, *tmp;
564
565 llist_for_each_entry_safe(pending, tmp, &smsq->pending_sms, entry) {
Holger Hans Peter Freythereff409492012-11-10 19:46:58 +0100566 LOGP(DLSMS, LOGL_NOTICE,
Harald Welte2483f1b2016-06-19 18:06:02 +0200567 "SMSqueue clearing for sub %llu\n", pending->vsub->id);
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100568 sms_pending_free(pending);
569 }
570
Holger Hans Peter Freyther96e9f082010-12-28 14:09:07 +0100571 smsq->pending = 0;
Holger Hans Peter Freyther4dcc5e52010-12-25 14:38:30 +0100572 return 0;
573}