Coverage Report

Created: 2025-04-02 10:29

/src/c-toxcore/toxcore/Messenger.c
Line
Count
Source (jump to first uncovered line)
1
/* SPDX-License-Identifier: GPL-3.0-or-later
2
 * Copyright © 2016-2025 The TokTok team.
3
 * Copyright © 2013 Tox project.
4
 */
5
6
/**
7
 * An implementation of a simple text chat only messenger on the tox network core.
8
 */
9
#include "Messenger.h"
10
11
#include <assert.h>
12
#include <stdio.h>
13
#include <string.h>
14
#include <time.h>
15
16
#include "DHT.h"
17
#include "TCP_client.h"
18
#include "TCP_connection.h"
19
#include "TCP_server.h"
20
#include "announce.h"
21
#include "attributes.h"
22
#include "bin_pack.h"
23
#include "bin_unpack.h"
24
#include "ccompat.h"
25
#include "crypto_core.h"
26
#include "forwarding.h"
27
#include "friend_connection.h"
28
#include "friend_requests.h"
29
#include "group_announce.h"
30
#include "group_chats.h"
31
#include "group_common.h"
32
#include "group_onion_announce.h"
33
#include "logger.h"
34
#include "mem.h"
35
#include "mono_time.h"
36
#include "net_crypto.h"
37
#include "net_profile.h"
38
#include "network.h"
39
#include "onion.h"
40
#include "onion_announce.h"
41
#include "onion_client.h"
42
#include "state.h"
43
#include "util.h"
44
45
static_assert(MAX_CONCURRENT_FILE_PIPES <= UINT8_MAX + 1,
46
              "uint8_t cannot represent all file transfer numbers");
47
48
static const Friend empty_friend = {{0}};
49
50
/**
51
 * Determines if the friendnumber passed is valid in the Messenger object.
52
 *
53
 * @param friendnumber The index in the friend list.
54
 */
55
bool friend_is_valid(const Messenger *m, int32_t friendnumber)
56
0
{
57
0
    return (uint32_t)friendnumber < m->numfriends && m->friendlist[friendnumber].status != 0;
58
0
}
59
60
/** @brief Set the size of the friend list to numfriends.
61
 *
62
 * @retval -1 if mem_vrealloc fails.
63
 */
64
non_null()
65
static int realloc_friendlist(Messenger *m, uint32_t num)
66
247
{
67
247
    if (num == 0) {
68
0
        mem_delete(m->mem, m->friendlist);
69
0
        m->friendlist = nullptr;
70
0
        return 0;
71
0
    }
72
73
247
    Friend *newfriendlist = (Friend *)mem_vrealloc(m->mem, m->friendlist, num, sizeof(Friend));
74
75
247
    if (newfriendlist == nullptr) {
76
0
        return -1;
77
0
    }
78
79
247
    m->friendlist = newfriendlist;
80
247
    return 0;
81
247
}
82
83
/** @return the friend number associated to that public key.
84
 * @retval -1 if no such friend.
85
 */
86
int32_t getfriend_id(const Messenger *m, const uint8_t *real_pk)
87
3.49k
{
88
4.09k
    for (uint32_t i = 0; i < m->numfriends; ++i) {
89
730
        if (m->friendlist[i].status > 0 && pk_equal(real_pk, m->friendlist[i].real_pk)) {
90
126
            return i;
91
126
        }
92
730
    }
93
94
3.36k
    return -1;
95
3.49k
}
96
97
/** @brief Copies the public key associated to that friend id into real_pk buffer.
98
 *
99
 * Make sure that real_pk is of size CRYPTO_PUBLIC_KEY_SIZE.
100
 *
101
 * @retval 0 if success.
102
 * @retval -1 if failure.
103
 */
104
int get_real_pk(const Messenger *m, int32_t friendnumber, uint8_t *real_pk)
105
0
{
106
0
    if (!m_friend_exists(m, friendnumber)) {
107
0
        return -1;
108
0
    }
109
110
0
    memcpy(real_pk, m->friendlist[friendnumber].real_pk, CRYPTO_PUBLIC_KEY_SIZE);
111
0
    return 0;
112
0
}
113
114
/** @return friend connection id on success.
115
 * @retval -1 if failure.
116
 */
117
int getfriendcon_id(const Messenger *m, int32_t friendnumber)
118
0
{
119
0
    if (!m_friend_exists(m, friendnumber)) {
120
0
        return -1;
121
0
    }
122
123
0
    return m->friendlist[friendnumber].friendcon_id;
124
0
}
125
126
/**
127
 * Format: `[real_pk (32 bytes)][nospam number (4 bytes)][checksum (2 bytes)]`
128
 *
129
 * @param[out] address FRIEND_ADDRESS_SIZE byte address to give to others.
130
 */
131
void getaddress(const Messenger *m, uint8_t *address)
132
0
{
133
0
    pk_copy(address, nc_get_self_public_key(m->net_crypto));
134
0
    uint32_t nospam = get_nospam(m->fr);
135
0
    memcpy(address + CRYPTO_PUBLIC_KEY_SIZE, &nospam, sizeof(nospam));
136
0
    uint16_t checksum = data_checksum(address, FRIEND_ADDRESS_SIZE - sizeof(checksum));
137
0
    memcpy(address + CRYPTO_PUBLIC_KEY_SIZE + sizeof(nospam), &checksum, sizeof(checksum));
138
0
}
139
140
non_null()
141
static bool send_online_packet(const Messenger *m, int friendcon_id)
142
0
{
143
0
    const uint8_t packet[1] = {PACKET_ID_ONLINE};
144
0
    return write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c, friendcon_id), packet,
145
0
                             sizeof(packet), false) != -1;
146
0
}
147
148
non_null()
149
static bool send_offline_packet(const Messenger *m, int friendcon_id)
150
0
{
151
0
    const uint8_t packet[1] = {PACKET_ID_OFFLINE};
152
0
    return write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c, friendcon_id), packet,
153
0
                             sizeof(packet), false) != -1;
154
0
}
155
156
non_null(1) nullable(4)
157
static int m_handle_status(void *object, int friendcon_id, bool status, void *userdata);
158
non_null(1, 3) nullable(5)
159
static int m_handle_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata);
160
non_null(1, 3) nullable(5)
161
static int m_handle_lossy_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length,
162
                                 void *userdata);
163
164
non_null()
165
static int32_t init_new_friend(Messenger *m, const uint8_t *real_pk, uint8_t status)
166
247
{
167
247
    if (m->numfriends == UINT32_MAX) {
168
0
        LOGGER_ERROR(m->log, "Friend list full: we have more than 4 billion friends");
169
        /* This is technically incorrect, but close enough. */
170
0
        return FAERR_NOMEM;
171
0
    }
172
173
    /* Resize the friend list if necessary. */
174
247
    if (realloc_friendlist(m, m->numfriends + 1) != 0) {
175
0
        return FAERR_NOMEM;
176
0
    }
177
178
247
    m->friendlist[m->numfriends] = empty_friend;
179
180
247
    const int friendcon_id = new_friend_connection(m->fr_c, real_pk);
181
182
247
    if (friendcon_id == -1) {
183
0
        return FAERR_NOMEM;
184
0
    }
185
186
561
    for (uint32_t i = 0; i <= m->numfriends; ++i) {
187
561
        if (m->friendlist[i].status == NOFRIEND) {
188
247
            m->friendlist[i].status = status;
189
247
            m->friendlist[i].friendcon_id = friendcon_id;
190
247
            m->friendlist[i].friendrequest_lastsent = 0;
191
247
            pk_copy(m->friendlist[i].real_pk, real_pk);
192
247
            m->friendlist[i].statusmessage_length = 0;
193
247
            m->friendlist[i].userstatus = USERSTATUS_NONE;
194
247
            m->friendlist[i].is_typing = false;
195
247
            m->friendlist[i].message_id = 0;
196
247
            friend_connection_callbacks(m->fr_c, friendcon_id, MESSENGER_CALLBACK_INDEX, &m_handle_status, &m_handle_packet,
197
247
                                        &m_handle_lossy_packet, m, i);
198
199
247
            if (m->numfriends == i) {
200
247
                ++m->numfriends;
201
247
            }
202
203
247
            if (friend_con_connected(m->fr_c, friendcon_id) == FRIENDCONN_STATUS_CONNECTED) {
204
0
                send_online_packet(m, friendcon_id);
205
0
            }
206
207
247
            return i;
208
247
        }
209
561
    }
210
211
0
    return FAERR_NOMEM;
212
247
}
213
214
non_null()
215
static int32_t m_add_friend_contact_norequest(Messenger *m, const uint8_t *real_pk)
216
222
{
217
222
    if (getfriend_id(m, real_pk) != -1) {
218
11
        return FAERR_ALREADYSENT;
219
11
    }
220
221
211
    if (pk_equal(real_pk, nc_get_self_public_key(m->net_crypto))) {
222
0
        return FAERR_OWNKEY;
223
0
    }
224
225
211
    return init_new_friend(m, real_pk, FRIEND_CONFIRMED);
226
211
}
227
228
/**
229
 * Add a friend.
230
 *
231
 * Set the data that will be sent along with friend request.
232
 *
233
 * @param address is the address of the friend (returned by getaddress of the friend
234
 *   you wish to add) it must be FRIEND_ADDRESS_SIZE bytes.
235
 *   TODO(irungentoo): add checksum.
236
 * @param data is the data.
237
 * @param length is the length.
238
 *
239
 * @return the friend number if success.
240
 * @retval FA_TOOLONG if message length is too long.
241
 * @retval FAERR_NOMESSAGE if no message (message length must be >= 1 byte).
242
 * @retval FAERR_OWNKEY if user's own key.
243
 * @retval FAERR_ALREADYSENT if friend request already sent or already a friend.
244
 * @retval FAERR_BADCHECKSUM if bad checksum in address.
245
 * @retval FAERR_SETNEWNOSPAM if the friend was already there but the nospam was different.
246
 *   (the nospam for that friend was set to the new one).
247
 * @retval FAERR_NOMEM if increasing the friend list size fails.
248
 */
249
int32_t m_addfriend(Messenger *m, const uint8_t *address, const uint8_t *data, uint16_t length)
250
122
{
251
122
    if (length > MAX_FRIEND_REQUEST_DATA_SIZE) {
252
11
        return FAERR_TOOLONG;
253
11
    }
254
255
111
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
256
111
    pk_copy(real_pk, address);
257
258
111
    if (!public_key_valid(real_pk)) {
259
10
        return FAERR_BADCHECKSUM;
260
10
    }
261
262
101
    uint16_t check;
263
101
    const uint16_t checksum = data_checksum(address, FRIEND_ADDRESS_SIZE - sizeof(checksum));
264
101
    memcpy(&check, address + CRYPTO_PUBLIC_KEY_SIZE + sizeof(uint32_t), sizeof(check));
265
266
101
    if (check != checksum) {
267
0
        return FAERR_BADCHECKSUM;
268
0
    }
269
270
101
    if (length < 1) {
271
10
        return FAERR_NOMESSAGE;
272
10
    }
273
274
91
    if (pk_equal(real_pk, nc_get_self_public_key(m->net_crypto))) {
275
10
        return FAERR_OWNKEY;
276
10
    }
277
278
81
    const int32_t friend_id = getfriend_id(m, real_pk);
279
280
81
    if (friend_id != -1) {
281
45
        if (m->friendlist[friend_id].status >= FRIEND_CONFIRMED) {
282
10
            return FAERR_ALREADYSENT;
283
10
        }
284
285
35
        uint32_t nospam;
286
35
        memcpy(&nospam, address + CRYPTO_PUBLIC_KEY_SIZE, sizeof(nospam));
287
288
35
        if (m->friendlist[friend_id].friendrequest_nospam == nospam) {
289
18
            return FAERR_ALREADYSENT;
290
18
        }
291
292
17
        m->friendlist[friend_id].friendrequest_nospam = nospam;
293
17
        return FAERR_SETNEWNOSPAM;
294
35
    }
295
296
36
    const int32_t ret = init_new_friend(m, real_pk, FRIEND_ADDED);
297
298
36
    if (ret < 0) {
299
0
        return ret;
300
0
    }
301
302
36
    m->friendlist[ret].friendrequest_timeout = FRIENDREQUEST_TIMEOUT;
303
36
    memcpy(m->friendlist[ret].info, data, length);
304
36
    m->friendlist[ret].info_size = length;
305
36
    memcpy(&m->friendlist[ret].friendrequest_nospam, address + CRYPTO_PUBLIC_KEY_SIZE, sizeof(uint32_t));
306
307
36
    return ret;
308
36
}
309
310
int32_t m_addfriend_norequest(Messenger *m, const uint8_t *real_pk)
311
262
{
312
262
    if (!public_key_valid(real_pk)) {
313
30
        return FAERR_BADCHECKSUM;
314
30
    }
315
316
232
    if (pk_equal(real_pk, nc_get_self_public_key(m->net_crypto))) {
317
10
        return FAERR_OWNKEY;
318
10
    }
319
320
222
    return m_add_friend_contact_norequest(m, real_pk);
321
232
}
322
323
non_null()
324
static int clear_receipts(Messenger *m, int32_t friendnumber)
325
247
{
326
247
    if (!m_friend_exists(m, friendnumber)) {
327
0
        return -1;
328
0
    }
329
330
247
    struct Receipts *receipts = m->friendlist[friendnumber].receipts_start;
331
332
247
    while (receipts != nullptr) {
333
0
        struct Receipts *temp_r = receipts->next;
334
0
        mem_delete(m->mem, receipts);
335
0
        receipts = temp_r;
336
0
    }
337
338
247
    m->friendlist[friendnumber].receipts_start = nullptr;
339
247
    m->friendlist[friendnumber].receipts_end = nullptr;
340
247
    return 0;
341
247
}
342
343
non_null()
344
static int add_receipt(Messenger *m, int32_t friendnumber, uint32_t packet_num, uint32_t msg_id)
345
0
{
346
0
    if (!m_friend_exists(m, friendnumber)) {
347
0
        return -1;
348
0
    }
349
350
0
    struct Receipts *new_receipts = (struct Receipts *)mem_alloc(m->mem, sizeof(struct Receipts));
351
352
0
    if (new_receipts == nullptr) {
353
0
        return -1;
354
0
    }
355
356
0
    new_receipts->packet_num = packet_num;
357
0
    new_receipts->msg_id = msg_id;
358
359
0
    if (m->friendlist[friendnumber].receipts_start == nullptr) {
360
0
        m->friendlist[friendnumber].receipts_start = new_receipts;
361
0
    } else {
362
0
        m->friendlist[friendnumber].receipts_end->next = new_receipts;
363
0
    }
364
365
0
    m->friendlist[friendnumber].receipts_end = new_receipts;
366
0
    new_receipts->next = nullptr;
367
0
    return 0;
368
0
}
369
/**
370
 * return -1 on failure.
371
 * return 0 if packet was received.
372
 */
373
non_null()
374
static int friend_received_packet(const Messenger *m, int32_t friendnumber, uint32_t number)
375
0
{
376
0
    if (!m_friend_exists(m, friendnumber)) {
377
0
        return -1;
378
0
    }
379
380
0
    return cryptpacket_received(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
381
0
                                m->friendlist[friendnumber].friendcon_id), number);
382
0
}
383
384
bool m_create_group_connection(Messenger *m, GC_Chat *chat)
385
68
{
386
68
    random_bytes(m->rng, chat->m_group_public_key, CRYPTO_PUBLIC_KEY_SIZE);
387
68
    const int friendcon_id = new_friend_connection(m->fr_c, chat->m_group_public_key);
388
389
68
    if (friendcon_id == -1) {
390
0
        return false;
391
0
    }
392
393
68
    const Friend_Conn *connection = get_conn(m->fr_c, friendcon_id);
394
395
68
    if (connection == nullptr) {
396
0
        return false;
397
0
    }
398
399
68
    chat->friend_connection_id = friendcon_id;
400
401
68
    if (friend_con_connected(m->fr_c, friendcon_id) == FRIENDCONN_STATUS_CONNECTED) {
402
0
        send_online_packet(m, friendcon_id);
403
0
    }
404
405
68
    const int onion_friend_number = friend_conn_get_onion_friendnum(connection);
406
68
    Onion_Friend *onion_friend = onion_get_friend(m->onion_c, (uint16_t)onion_friend_number);
407
408
68
    onion_friend_set_gc_public_key(onion_friend, get_chat_id(&chat->chat_public_key));
409
68
    onion_friend_set_gc_data(onion_friend, nullptr, 0);
410
411
68
    return true;
412
68
}
413
414
/**
415
 * Kills the friend connection for a groupchat.
416
 */
417
void m_kill_group_connection(Messenger *m, const GC_Chat *chat)
418
68
{
419
68
    remove_request_received(m->fr, chat->m_group_public_key);
420
421
68
    friend_connection_callbacks(m->fr_c, chat->friend_connection_id, MESSENGER_CALLBACK_INDEX, nullptr,
422
68
                                nullptr, nullptr, nullptr, 0);
423
424
68
    if (friend_con_connected(m->fr_c, chat->friend_connection_id) == FRIENDCONN_STATUS_CONNECTED) {
425
0
        send_offline_packet(m, chat->friend_connection_id);
426
0
    }
427
428
68
    kill_friend_connection(m->fr_c, chat->friend_connection_id);
429
68
}
430
431
non_null(1) nullable(3)
432
static int do_receipts(Messenger *m, int32_t friendnumber, void *userdata)
433
0
{
434
0
    if (!m_friend_exists(m, friendnumber)) {
435
0
        return -1;
436
0
    }
437
438
0
    struct Receipts *receipts = m->friendlist[friendnumber].receipts_start;
439
440
0
    while (receipts != nullptr) {
441
0
        if (friend_received_packet(m, friendnumber, receipts->packet_num) == -1) {
442
0
            break;
443
0
        }
444
445
0
        if (m->read_receipt != nullptr) {
446
0
            m->read_receipt(m, friendnumber, receipts->msg_id, userdata);
447
0
        }
448
449
0
        struct Receipts *r_next = receipts->next;
450
451
0
        mem_delete(m->mem, receipts);
452
453
0
        m->friendlist[friendnumber].receipts_start = r_next;
454
455
0
        receipts = r_next;
456
0
    }
457
458
0
    if (m->friendlist[friendnumber].receipts_start == nullptr) {
459
0
        m->friendlist[friendnumber].receipts_end = nullptr;
460
0
    }
461
462
0
    return 0;
463
0
}
464
465
/** @brief Remove a friend.
466
 *
467
 * @retval 0 if success.
468
 * @retval -1 if failure.
469
 */
470
int m_delfriend(Messenger *m, int32_t friendnumber)
471
0
{
472
0
    if (!m_friend_exists(m, friendnumber)) {
473
0
        return -1;
474
0
    }
475
476
0
    clear_receipts(m, friendnumber);
477
0
    remove_request_received(m->fr, m->friendlist[friendnumber].real_pk);
478
0
    friend_connection_callbacks(m->fr_c, m->friendlist[friendnumber].friendcon_id, MESSENGER_CALLBACK_INDEX, nullptr,
479
0
                                nullptr, nullptr, nullptr, 0);
480
481
0
    if (friend_con_connected(m->fr_c, m->friendlist[friendnumber].friendcon_id) == FRIENDCONN_STATUS_CONNECTED) {
482
0
        send_offline_packet(m, m->friendlist[friendnumber].friendcon_id);
483
0
    }
484
485
0
    kill_friend_connection(m->fr_c, m->friendlist[friendnumber].friendcon_id);
486
0
    m->friendlist[friendnumber] = empty_friend;
487
488
0
    uint32_t i;
489
490
0
    for (i = m->numfriends; i != 0; --i) {
491
0
        if (m->friendlist[i - 1].status != NOFRIEND) {
492
0
            break;
493
0
        }
494
0
    }
495
496
0
    m->numfriends = i;
497
498
0
    if (realloc_friendlist(m, m->numfriends) != 0) {
499
0
        return FAERR_NOMEM;
500
0
    }
501
502
0
    return 0;
503
0
}
504
505
int m_get_friend_connectionstatus(const Messenger *m, int32_t friendnumber)
506
0
{
507
0
    if (!m_friend_exists(m, friendnumber)) {
508
0
        return -1;
509
0
    }
510
511
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
512
0
        return CONNECTION_NONE;
513
0
    }
514
515
0
    bool direct_connected = false;
516
0
    uint32_t num_online_relays = 0;
517
0
    const int crypt_conn_id = friend_connection_crypt_connection_id(m->fr_c, m->friendlist[friendnumber].friendcon_id);
518
519
0
    if (!crypto_connection_status(m->net_crypto, crypt_conn_id, &direct_connected, &num_online_relays)) {
520
0
        return CONNECTION_NONE;
521
0
    }
522
523
0
    if (direct_connected) {
524
0
        return CONNECTION_UDP;
525
0
    }
526
527
0
    if (num_online_relays != 0) {
528
0
        return CONNECTION_TCP;
529
0
    }
530
531
    /* if we have a valid friend connection but do not have an established connection
532
     * we leave the connection status unchanged until the friend connection is either
533
     * established or dropped.
534
     */
535
0
    return m->friendlist[friendnumber].last_connection_udp_tcp;
536
0
}
537
538
/**
539
 * Checks if there exists a friend with given friendnumber.
540
 *
541
 * @param friendnumber The index in the friend list.
542
 *
543
 * @retval true if friend exists.
544
 * @retval false if friend doesn't exist.
545
 */
546
bool m_friend_exists(const Messenger *m, int32_t friendnumber)
547
669
{
548
669
    return (unsigned int)friendnumber < m->numfriends && m->friendlist[friendnumber].status != 0;
549
669
}
550
551
/** @brief Send a message of type to an online friend.
552
 *
553
 * @retval -1 if friend not valid.
554
 * @retval -2 if too large.
555
 * @retval -3 if friend not online.
556
 * @retval -4 if send failed (because queue is full).
557
 * @retval -5 if bad type.
558
 * @retval 0 if success.
559
 *
560
 * The value in message_id will be passed to your read_receipt callback when the other receives the message.
561
 */
562
int m_send_message_generic(Messenger *m, int32_t friendnumber, uint8_t type, const uint8_t *message, uint32_t length,
563
                           uint32_t *message_id)
564
0
{
565
0
    if (type > MESSAGE_ACTION) {
566
0
        LOGGER_WARNING(m->log, "message type %d is invalid", type);
567
0
        return -5;
568
0
    }
569
570
0
    if (!m_friend_exists(m, friendnumber)) {
571
0
        LOGGER_WARNING(m->log, "friend number %d is invalid", friendnumber);
572
0
        return -1;
573
0
    }
574
575
0
    if (length >= MAX_CRYPTO_DATA_SIZE) {
576
0
        LOGGER_WARNING(m->log, "message length %u is too large", length);
577
0
        return -2;
578
0
    }
579
580
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
581
0
        LOGGER_WARNING(m->log, "friend %d is not online", friendnumber);
582
0
        return -3;
583
0
    }
584
585
0
    VLA(uint8_t, packet, length + 1);
586
0
    packet[0] = PACKET_ID_MESSAGE + type;
587
588
0
    assert(message != nullptr);
589
0
    memcpy(packet + 1, message, length);
590
591
0
    const int64_t packet_num = write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
592
0
                               m->friendlist[friendnumber].friendcon_id), packet, length + 1, false);
593
594
0
    if (packet_num == -1) {
595
0
        return -4;
596
0
    }
597
598
0
    const uint32_t msg_id = ++m->friendlist[friendnumber].message_id;
599
600
0
    add_receipt(m, friendnumber, packet_num, msg_id);
601
602
0
    if (message_id != nullptr) {
603
0
        *message_id = msg_id;
604
0
    }
605
606
0
    return 0;
607
0
}
608
609
non_null()
610
static bool write_cryptpacket_id(const Messenger *m, int32_t friendnumber, uint8_t packet_id, const uint8_t *data,
611
                                 uint32_t length, bool congestion_control)
612
0
{
613
0
    if (!m_friend_exists(m, friendnumber)) {
614
0
        return false;
615
0
    }
616
617
0
    if (length >= MAX_CRYPTO_DATA_SIZE || m->friendlist[friendnumber].status != FRIEND_ONLINE) {
618
0
        return false;
619
0
    }
620
621
0
    VLA(uint8_t, packet, length + 1);
622
0
    packet[0] = packet_id;
623
624
0
    assert(data != nullptr);
625
0
    memcpy(packet + 1, data, length);
626
627
0
    return write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
628
0
                             m->friendlist[friendnumber].friendcon_id), packet, length + 1, congestion_control) != -1;
629
0
}
630
631
/** @brief Send a name packet to friendnumber.
632
 * length is the length with the NULL terminator.
633
 */
634
non_null()
635
static bool m_sendname(const Messenger *m, int32_t friendnumber, const uint8_t *name, uint16_t length)
636
0
{
637
0
    if (length > MAX_NAME_LENGTH) {
638
0
        return false;
639
0
    }
640
641
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_NICKNAME, name, length, false);
642
0
}
643
644
/** @brief Set the name and name_length of a friend.
645
 *
646
 * name must be a string of maximum MAX_NAME_LENGTH length.
647
 * length must be at least 1 byte.
648
 * length is the length of name with the NULL terminator.
649
 *
650
 * @retval 0 if success.
651
 * @retval -1 if failure.
652
 */
653
int setfriendname(Messenger *m, int32_t friendnumber, const uint8_t *name, uint16_t length)
654
211
{
655
211
    if (!m_friend_exists(m, friendnumber)) {
656
0
        return -1;
657
0
    }
658
659
211
    if (length > MAX_NAME_LENGTH || length == 0) {
660
160
        return -1;
661
160
    }
662
663
51
    m->friendlist[friendnumber].name_length = length;
664
51
    memcpy(m->friendlist[friendnumber].name, name, length);
665
51
    return 0;
666
211
}
667
668
/** @brief Set our nickname.
669
 *
670
 * name must be a string of maximum MAX_NAME_LENGTH length.
671
 * length must be at least 1 byte.
672
 * length is the length of name with the NULL terminator.
673
 *
674
 * @retval 0 if success.
675
 * @retval -1 if failure.
676
 */
677
int setname(Messenger *m, const uint8_t *name, uint16_t length)
678
314
{
679
314
    if (length > MAX_NAME_LENGTH) {
680
0
        return -1;
681
0
    }
682
683
314
    if (m->name_length == length && (length == 0 || memcmp(name, m->name, length) == 0)) {
684
74
        return 0;
685
74
    }
686
687
240
    if (length > 0) {
688
240
        memcpy(m->name, name, length);
689
240
    }
690
691
240
    m->name_length = length;
692
693
570
    for (uint32_t i = 0; i < m->numfriends; ++i) {
694
330
        m->friendlist[i].name_sent = false;
695
330
    }
696
697
240
    return 0;
698
314
}
699
700
/**
701
 * @brief Get your nickname.
702
 *
703
 * m - The messenger context to use.
704
 * name needs to be a valid memory location with a size of at least MAX_NAME_LENGTH bytes.
705
 *
706
 * @return length of the name.
707
 * @retval 0 on error.
708
 */
709
uint16_t getself_name(const Messenger *m, uint8_t *name)
710
0
{
711
0
    if (name == nullptr) {
712
0
        return 0;
713
0
    }
714
715
0
    memcpy(name, m->name, m->name_length);
716
717
0
    return m->name_length;
718
0
}
719
720
/** @brief Get name of friendnumber and put it in name.
721
 *
722
 * name needs to be a valid memory location with a size of at least MAX_NAME_LENGTH (128) bytes.
723
 *
724
 * @return length of name if success.
725
 * @retval -1 if failure.
726
 */
727
int getname(const Messenger *m, int32_t friendnumber, uint8_t *name)
728
0
{
729
0
    if (!m_friend_exists(m, friendnumber)) {
730
0
        return -1;
731
0
    }
732
733
0
    memcpy(name, m->friendlist[friendnumber].name, m->friendlist[friendnumber].name_length);
734
0
    return m->friendlist[friendnumber].name_length;
735
0
}
736
737
int m_get_name_size(const Messenger *m, int32_t friendnumber)
738
0
{
739
0
    if (!m_friend_exists(m, friendnumber)) {
740
0
        return -1;
741
0
    }
742
743
0
    return m->friendlist[friendnumber].name_length;
744
0
}
745
746
int m_get_self_name_size(const Messenger *m)
747
0
{
748
0
    return m->name_length;
749
0
}
750
751
int m_set_statusmessage(Messenger *m, const uint8_t *status, uint16_t length)
752
557
{
753
557
    if (length > MAX_STATUSMESSAGE_LENGTH) {
754
0
        return -1;
755
0
    }
756
757
557
    if (m->statusmessage_length == length && (length == 0 || memcmp(m->statusmessage, status, length) == 0)) {
758
118
        return 0;
759
118
    }
760
761
439
    if (length > 0) {
762
439
        memcpy(m->statusmessage, status, length);
763
439
    }
764
765
439
    m->statusmessage_length = length;
766
767
778
    for (uint32_t i = 0; i < m->numfriends; ++i) {
768
339
        m->friendlist[i].statusmessage_sent = false;
769
339
    }
770
771
439
    return 0;
772
557
}
773
774
non_null()
775
static bool userstatus_from_int(uint8_t status, Userstatus *out_enum)
776
359
{
777
359
    switch (status) {
778
169
        case USERSTATUS_NONE: {
779
169
            *out_enum = USERSTATUS_NONE;
780
169
            return true;
781
0
        }
782
783
7
        case USERSTATUS_AWAY: {
784
7
            *out_enum = USERSTATUS_AWAY;
785
7
            return true;
786
0
        }
787
788
75
        case USERSTATUS_BUSY: {
789
75
            *out_enum = USERSTATUS_BUSY;
790
75
            return true;
791
0
        }
792
793
0
        case USERSTATUS_INVALID: {
794
0
            *out_enum = USERSTATUS_INVALID;
795
0
            return true;
796
0
        }
797
798
108
        default: {
799
108
            *out_enum = USERSTATUS_INVALID;
800
108
            return false;
801
0
        }
802
359
    }
803
359
}
804
805
int m_set_userstatus(Messenger *m, uint8_t status)
806
561
{
807
561
    if (status >= USERSTATUS_INVALID) {
808
264
        return -1;
809
264
    }
810
811
297
    if (m->userstatus == status) {
812
149
        return 0;
813
149
    }
814
815
148
    userstatus_from_int(status, &m->userstatus);
816
817
266
    for (uint32_t i = 0; i < m->numfriends; ++i) {
818
118
        m->friendlist[i].userstatus_sent = false;
819
118
    }
820
821
148
    return 0;
822
297
}
823
824
/**
825
 * Guaranteed to be at most MAX_STATUSMESSAGE_LENGTH.
826
 *
827
 * @return the length of friendnumber's status message, including null on success.
828
 * @retval -1 on failure.
829
 */
830
int m_get_statusmessage_size(const Messenger *m, int32_t friendnumber)
831
0
{
832
0
    if (!m_friend_exists(m, friendnumber)) {
833
0
        return -1;
834
0
    }
835
836
0
    return m->friendlist[friendnumber].statusmessage_length;
837
0
}
838
839
/** @brief Copy friendnumber's status message into buf, truncating if size is over maxlen.
840
 *
841
 * Get the size you need to allocate from m_get_statusmessage_size.
842
 * The self variant will copy our own status message.
843
 *
844
 * @return the length of the copied data on success
845
 * @retval -1 on failure.
846
 */
847
int m_copy_statusmessage(const Messenger *m, int32_t friendnumber, uint8_t *buf, uint32_t maxlen)
848
0
{
849
0
    if (!m_friend_exists(m, friendnumber)) {
850
0
        return -1;
851
0
    }
852
853
    // TODO(iphydf): This should be uint16_t and min_u16. If maxlen exceeds
854
    // uint16_t's range, it won't affect the result.
855
0
    const uint32_t msglen = min_u32(maxlen, m->friendlist[friendnumber].statusmessage_length);
856
857
0
    memcpy(buf, m->friendlist[friendnumber].statusmessage, msglen);
858
0
    memzero(buf + msglen, maxlen - msglen);
859
0
    return msglen;
860
0
}
861
862
/** @return the size of friendnumber's user status.
863
 * Guaranteed to be at most MAX_STATUSMESSAGE_LENGTH.
864
 */
865
int m_get_self_statusmessage_size(const Messenger *m)
866
0
{
867
0
    return m->statusmessage_length;
868
0
}
869
870
int m_copy_self_statusmessage(const Messenger *m, uint8_t *buf)
871
0
{
872
0
    memcpy(buf, m->statusmessage, m->statusmessage_length);
873
0
    return m->statusmessage_length;
874
0
}
875
876
uint8_t m_get_userstatus(const Messenger *m, int32_t friendnumber)
877
0
{
878
0
    if (!m_friend_exists(m, friendnumber)) {
879
0
        return USERSTATUS_INVALID;
880
0
    }
881
882
0
    uint8_t status = m->friendlist[friendnumber].userstatus;
883
884
0
    if (status >= USERSTATUS_INVALID) {
885
0
        status = USERSTATUS_NONE;
886
0
    }
887
888
0
    return status;
889
0
}
890
891
uint8_t m_get_self_userstatus(const Messenger *m)
892
0
{
893
0
    return m->userstatus;
894
0
}
895
896
uint64_t m_get_last_online(const Messenger *m, int32_t friendnumber)
897
0
{
898
0
    if (!m_friend_exists(m, friendnumber)) {
899
0
        return UINT64_MAX;
900
0
    }
901
902
0
    return m->friendlist[friendnumber].last_seen_time;
903
0
}
904
905
int m_set_usertyping(Messenger *m, int32_t friendnumber, bool is_typing)
906
0
{
907
0
    if (!m_friend_exists(m, friendnumber)) {
908
0
        return -1;
909
0
    }
910
911
0
    if (m->friendlist[friendnumber].user_istyping == is_typing) {
912
0
        return 0;
913
0
    }
914
915
0
    m->friendlist[friendnumber].user_istyping = is_typing;
916
0
    m->friendlist[friendnumber].user_istyping_sent = false;
917
918
0
    return 0;
919
0
}
920
921
int m_get_istyping(const Messenger *m, int32_t friendnumber)
922
0
{
923
0
    if (!m_friend_exists(m, friendnumber)) {
924
0
        return -1;
925
0
    }
926
927
0
    return m->friendlist[friendnumber].is_typing ? 1 : 0;
928
0
}
929
930
non_null()
931
static bool send_statusmessage(const Messenger *m, int32_t friendnumber, const uint8_t *status, uint16_t length)
932
0
{
933
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_STATUSMESSAGE, status, length, false);
934
0
}
935
936
non_null()
937
static bool send_userstatus(const Messenger *m, int32_t friendnumber, uint8_t status)
938
0
{
939
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_USERSTATUS, &status, sizeof(status), false);
940
0
}
941
942
non_null()
943
static bool send_user_istyping(const Messenger *m, int32_t friendnumber, bool is_typing)
944
0
{
945
0
    const uint8_t typing = is_typing ? 1 : 0;
946
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_TYPING, &typing, sizeof(typing), false);
947
0
}
948
949
non_null()
950
static int set_friend_statusmessage(const Messenger *m, int32_t friendnumber, const uint8_t *status, uint16_t length)
951
211
{
952
211
    if (!m_friend_exists(m, friendnumber)) {
953
0
        return -1;
954
0
    }
955
956
211
    if (length > MAX_STATUSMESSAGE_LENGTH) {
957
113
        return -1;
958
113
    }
959
960
98
    if (length > 0) {
961
52
        memcpy(m->friendlist[friendnumber].statusmessage, status, length);
962
52
    }
963
964
98
    m->friendlist[friendnumber].statusmessage_length = length;
965
98
    return 0;
966
211
}
967
968
non_null()
969
static void set_friend_userstatus(const Messenger *m, int32_t friendnumber, uint8_t status)
970
211
{
971
211
    userstatus_from_int(status, &m->friendlist[friendnumber].userstatus);
972
211
}
973
974
non_null()
975
static void set_friend_typing(const Messenger *m, int32_t friendnumber, bool is_typing)
976
0
{
977
0
    m->friendlist[friendnumber].is_typing = is_typing;
978
0
}
979
980
/** Set the function that will be executed when a friend request is received. */
981
void m_callback_friendrequest(Messenger *m, m_friend_request_cb *function)
982
1.27k
{
983
1.27k
    m->friend_request = function;
984
1.27k
}
985
986
/** Set the function that will be executed when a message from a friend is received. */
987
void m_callback_friendmessage(Messenger *m, m_friend_message_cb *function)
988
1.27k
{
989
1.27k
    m->friend_message = function;
990
1.27k
}
991
992
void m_callback_namechange(Messenger *m, m_friend_name_cb *function)
993
1.27k
{
994
1.27k
    m->friend_namechange = function;
995
1.27k
}
996
997
void m_callback_statusmessage(Messenger *m, m_friend_status_message_cb *function)
998
1.27k
{
999
1.27k
    m->friend_statusmessagechange = function;
1000
1.27k
}
1001
1002
void m_callback_userstatus(Messenger *m, m_friend_status_cb *function)
1003
1.27k
{
1004
1.27k
    m->friend_userstatuschange = function;
1005
1.27k
}
1006
1007
void m_callback_typingchange(Messenger *m, m_friend_typing_cb *function)
1008
1.27k
{
1009
1.27k
    m->friend_typingchange = function;
1010
1.27k
}
1011
1012
void m_callback_read_receipt(Messenger *m, m_friend_read_receipt_cb *function)
1013
1.27k
{
1014
1.27k
    m->read_receipt = function;
1015
1.27k
}
1016
1017
void m_callback_connectionstatus(Messenger *m, m_friend_connection_status_cb *function)
1018
1.27k
{
1019
1.27k
    m->friend_connectionstatuschange = function;
1020
1.27k
}
1021
1022
void m_callback_core_connection(Messenger *m, m_self_connection_status_cb *function)
1023
1.27k
{
1024
1.27k
    m->core_connection_change = function;
1025
1.27k
}
1026
1027
non_null(1) nullable(3)
1028
static void check_friend_tcp_udp(Messenger *m, int32_t friendnumber, void *userdata)
1029
0
{
1030
0
    const int last_connection_udp_tcp = m->friendlist[friendnumber].last_connection_udp_tcp;
1031
1032
0
    const int ret = m_get_friend_connectionstatus(m, friendnumber);
1033
1034
0
    if (ret == -1) {
1035
0
        return;
1036
0
    }
1037
1038
0
    if (last_connection_udp_tcp != ret) {
1039
0
        if (m->friend_connectionstatuschange != nullptr) {
1040
0
            m->friend_connectionstatuschange(m, friendnumber, ret, userdata);
1041
0
        }
1042
0
    }
1043
1044
0
    m->friendlist[friendnumber].last_connection_udp_tcp = (Connection_Status)ret;
1045
0
}
1046
1047
non_null()
1048
static void break_files(const Messenger *m, int32_t friendnumber);
1049
1050
non_null(1) nullable(4)
1051
static void check_friend_connectionstatus(Messenger *m, int32_t friendnumber, uint8_t status, void *userdata)
1052
0
{
1053
0
    if (status == NOFRIEND) {
1054
0
        return;
1055
0
    }
1056
1057
0
    const bool was_online = m->friendlist[friendnumber].status == FRIEND_ONLINE;
1058
0
    const bool is_online = status == FRIEND_ONLINE;
1059
1060
0
    if (is_online != was_online) {
1061
0
        if (was_online) {
1062
0
            break_files(m, friendnumber);
1063
0
            clear_receipts(m, friendnumber);
1064
0
        } else {
1065
0
            m->friendlist[friendnumber].name_sent = false;
1066
0
            m->friendlist[friendnumber].userstatus_sent = false;
1067
0
            m->friendlist[friendnumber].statusmessage_sent = false;
1068
0
            m->friendlist[friendnumber].user_istyping_sent = false;
1069
0
        }
1070
1071
0
        m->friendlist[friendnumber].status = status;
1072
1073
0
        check_friend_tcp_udp(m, friendnumber, userdata);
1074
0
    }
1075
0
}
1076
1077
non_null(1) nullable(4)
1078
static void set_friend_status(Messenger *m, int32_t friendnumber, uint8_t status, void *userdata)
1079
0
{
1080
0
    check_friend_connectionstatus(m, friendnumber, status, userdata);
1081
0
    m->friendlist[friendnumber].status = status;
1082
0
}
1083
1084
/*** CONFERENCES */
1085
1086
/** @brief Set the callback for conference invites. */
1087
void m_callback_conference_invite(Messenger *m, m_conference_invite_cb *function)
1088
3.53k
{
1089
3.53k
    m->conference_invite = function;
1090
3.53k
}
1091
1092
/** @brief the callback for group invites. */
1093
void m_callback_group_invite(Messenger *m, m_group_invite_cb *function)
1094
1.27k
{
1095
1.27k
    m->group_invite = function;
1096
1.27k
}
1097
1098
/** @brief Send a conference invite packet.
1099
 *
1100
 * return true on success
1101
 * return false on failure
1102
 */
1103
bool send_conference_invite_packet(const Messenger *m, int32_t friendnumber, const uint8_t *data, uint16_t length)
1104
0
{
1105
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_INVITE_CONFERENCE, data, length, false);
1106
0
}
1107
1108
/** @brief Send a group invite packet.
1109
 *
1110
 * @retval true if success
1111
 */
1112
bool send_group_invite_packet(const Messenger *m, uint32_t friendnumber, const uint8_t *packet, uint16_t length)
1113
0
{
1114
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_INVITE_GROUPCHAT, packet, length, false);
1115
0
}
1116
1117
/*** FILE SENDING */
1118
1119
/** @brief Set the callback for file send requests. */
1120
void callback_file_sendrequest(Messenger *m, m_file_recv_cb *function)
1121
1.27k
{
1122
1.27k
    m->file_sendrequest = function;
1123
1.27k
}
1124
1125
/** @brief Set the callback for file control requests. */
1126
void callback_file_control(Messenger *m, m_file_recv_control_cb *function)
1127
1.27k
{
1128
1.27k
    m->file_filecontrol = function;
1129
1.27k
}
1130
1131
/** @brief Set the callback for file data. */
1132
void callback_file_data(Messenger *m, m_file_recv_chunk_cb *function)
1133
1.27k
{
1134
1.27k
    m->file_filedata = function;
1135
1.27k
}
1136
1137
/** @brief Set the callback for file request chunk. */
1138
void callback_file_reqchunk(Messenger *m, m_file_chunk_request_cb *function)
1139
1.27k
{
1140
1.27k
    m->file_reqchunk = function;
1141
1.27k
}
1142
1143
0
#define MAX_FILENAME_LENGTH 255
1144
1145
/** @brief Copy the file transfer file id to file_id
1146
 *
1147
 * @retval 0 on success.
1148
 * @retval -1 if friend not valid.
1149
 * @retval -2 if filenumber not valid
1150
 */
1151
int file_get_id(const Messenger *m, int32_t friendnumber, uint32_t filenumber, uint8_t *file_id)
1152
0
{
1153
0
    if (!m_friend_exists(m, friendnumber)) {
1154
0
        return -1;
1155
0
    }
1156
1157
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1158
0
        return -2;
1159
0
    }
1160
1161
0
    uint32_t temp_filenum;
1162
0
    bool inbound;
1163
0
    uint8_t file_number;
1164
1165
0
    if (filenumber >= (1 << 16)) {
1166
0
        inbound = true;
1167
0
        temp_filenum = (filenumber >> 16) - 1;
1168
0
    } else {
1169
0
        inbound = false;
1170
0
        temp_filenum = filenumber;
1171
0
    }
1172
1173
0
    if (temp_filenum >= MAX_CONCURRENT_FILE_PIPES) {
1174
0
        return -2;
1175
0
    }
1176
1177
0
    file_number = temp_filenum;
1178
1179
0
    const struct File_Transfers *const ft = inbound
1180
0
                                                ? &m->friendlist[friendnumber].file_receiving[file_number]
1181
0
                                                : &m->friendlist[friendnumber].file_sending[file_number];
1182
1183
0
    if (ft->status == FILESTATUS_NONE) {
1184
0
        return -2;
1185
0
    }
1186
1187
0
    memcpy(file_id, ft->id, FILE_ID_LENGTH);
1188
0
    return 0;
1189
0
}
1190
1191
/** @brief Send a file send request.
1192
 * Maximum filename length is 255 bytes.
1193
 * @retval 1 on success
1194
 * @retval 0 on failure
1195
 */
1196
non_null()
1197
static bool file_sendrequest(const Messenger *m, int32_t friendnumber, uint8_t filenumber, uint32_t file_type,
1198
                             uint64_t filesize, const uint8_t *file_id, const uint8_t *filename, uint16_t filename_length)
1199
0
{
1200
0
    if (!m_friend_exists(m, friendnumber)) {
1201
0
        return false;
1202
0
    }
1203
1204
0
    if (filename_length > MAX_FILENAME_LENGTH) {
1205
0
        return false;
1206
0
    }
1207
1208
0
    const uint16_t packet_size = 1 + sizeof(file_type) + sizeof(filesize) + FILE_ID_LENGTH + filename_length;
1209
0
    VLA(uint8_t, packet, packet_size);
1210
0
    packet[0] = filenumber;
1211
0
    file_type = net_htonl(file_type);
1212
0
    memcpy(packet + 1, &file_type, sizeof(file_type));
1213
0
    net_pack_u64(packet + 1 + sizeof(file_type), filesize);
1214
0
    memcpy(packet + 1 + sizeof(file_type) + sizeof(filesize), file_id, FILE_ID_LENGTH);
1215
1216
0
    if (filename_length > 0) {
1217
0
        memcpy(packet + 1 + sizeof(file_type) + sizeof(filesize) + FILE_ID_LENGTH, filename, filename_length);
1218
0
    }
1219
1220
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_FILE_SENDREQUEST, packet, packet_size, false);
1221
0
}
1222
1223
/** @brief Send a file send request.
1224
 *
1225
 * Maximum filename length is 255 bytes.
1226
 *
1227
 * @return file number on success
1228
 * @retval -1 if friend not found.
1229
 * @retval -2 if filename length invalid.
1230
 * @retval -3 if no more file sending slots left.
1231
 * @retval -4 if could not send packet (friend offline).
1232
 */
1233
long int new_filesender(const Messenger *m, int32_t friendnumber, uint32_t file_type, uint64_t filesize,
1234
                        const uint8_t *file_id, const uint8_t *filename, uint16_t filename_length)
1235
0
{
1236
0
    if (!m_friend_exists(m, friendnumber)) {
1237
0
        return -1;
1238
0
    }
1239
1240
0
    if (filename_length > MAX_FILENAME_LENGTH) {
1241
0
        return -2;
1242
0
    }
1243
1244
0
    uint32_t i;
1245
1246
0
    for (i = 0; i < MAX_CONCURRENT_FILE_PIPES; ++i) {
1247
0
        if (m->friendlist[friendnumber].file_sending[i].status == FILESTATUS_NONE) {
1248
0
            break;
1249
0
        }
1250
0
    }
1251
1252
0
    if (i == MAX_CONCURRENT_FILE_PIPES) {
1253
0
        return -3;
1254
0
    }
1255
1256
0
    if (!file_sendrequest(m, friendnumber, i, file_type, filesize, file_id, filename, filename_length)) {
1257
0
        return -4;
1258
0
    }
1259
1260
0
    struct File_Transfers *ft = &m->friendlist[friendnumber].file_sending[i];
1261
1262
0
    ft->status = FILESTATUS_NOT_ACCEPTED;
1263
1264
0
    ft->size = filesize;
1265
1266
0
    ft->transferred = 0;
1267
1268
0
    ft->requested = 0;
1269
1270
0
    ft->paused = FILE_PAUSE_NOT;
1271
1272
0
    memcpy(ft->id, file_id, FILE_ID_LENGTH);
1273
1274
0
    return i;
1275
0
}
1276
1277
non_null(1) nullable(6)
1278
static bool send_file_control_packet(const Messenger *m, int32_t friendnumber, bool inbound, uint8_t filenumber,
1279
                                     uint8_t control_type, const uint8_t *data, uint16_t data_length)
1280
0
{
1281
0
    assert(data_length == 0 || data != nullptr);
1282
1283
0
    if ((unsigned int)(1 + 3 + data_length) > MAX_CRYPTO_DATA_SIZE) {
1284
0
        return false;
1285
0
    }
1286
1287
0
    const uint16_t packet_size = 3 + data_length;
1288
0
    VLA(uint8_t, packet, packet_size);
1289
1290
0
    packet[0] = inbound ? 1 : 0;
1291
0
    packet[1] = filenumber;
1292
0
    packet[2] = control_type;
1293
1294
0
    if (data_length > 0) {
1295
0
        memcpy(packet + 3, data, data_length);
1296
0
    }
1297
1298
0
    return write_cryptpacket_id(m, friendnumber, PACKET_ID_FILE_CONTROL, packet, packet_size, false);
1299
0
}
1300
1301
/** @brief Send a file control request.
1302
 *
1303
 * @retval 0 on success
1304
 * @retval -1 if friend not valid.
1305
 * @retval -2 if friend not online.
1306
 * @retval -3 if file number invalid.
1307
 * @retval -4 if file control is bad.
1308
 * @retval -5 if file already paused.
1309
 * @retval -6 if resume file failed because it was only paused by the other.
1310
 * @retval -7 if resume file failed because it wasn't paused.
1311
 * @retval -8 if packet failed to send.
1312
 */
1313
int file_control(const Messenger *m, int32_t friendnumber, uint32_t filenumber, unsigned int control)
1314
0
{
1315
0
    if (!m_friend_exists(m, friendnumber)) {
1316
0
        return -1;
1317
0
    }
1318
1319
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1320
0
        return -2;
1321
0
    }
1322
1323
0
    uint32_t temp_filenum;
1324
0
    bool inbound;
1325
0
    uint8_t file_number;
1326
1327
0
    if (filenumber >= (1 << 16)) {
1328
0
        inbound = true;
1329
0
        temp_filenum = (filenumber >> 16) - 1;
1330
0
    } else {
1331
0
        inbound = false;
1332
0
        temp_filenum = filenumber;
1333
0
    }
1334
1335
0
    if (temp_filenum >= MAX_CONCURRENT_FILE_PIPES) {
1336
0
        return -3;
1337
0
    }
1338
1339
0
    file_number = temp_filenum;
1340
1341
0
    struct File_Transfers *ft;
1342
1343
0
    if (inbound) {
1344
0
        ft = &m->friendlist[friendnumber].file_receiving[file_number];
1345
0
    } else {
1346
0
        ft = &m->friendlist[friendnumber].file_sending[file_number];
1347
0
    }
1348
1349
0
    if (ft->status == FILESTATUS_NONE) {
1350
0
        return -3;
1351
0
    }
1352
1353
0
    if (control > FILECONTROL_KILL) {
1354
0
        return -4;
1355
0
    }
1356
1357
0
    if (control == FILECONTROL_PAUSE && ((ft->paused & FILE_PAUSE_US) != 0 || ft->status != FILESTATUS_TRANSFERRING)) {
1358
0
        return -5;
1359
0
    }
1360
1361
0
    if (control == FILECONTROL_ACCEPT) {
1362
0
        if (ft->status == FILESTATUS_TRANSFERRING) {
1363
0
            if ((ft->paused & FILE_PAUSE_US) == 0) {
1364
0
                if ((ft->paused & FILE_PAUSE_OTHER) != 0) {
1365
0
                    return -6;
1366
0
                }
1367
1368
0
                return -7;
1369
0
            }
1370
0
        } else {
1371
0
            if (ft->status != FILESTATUS_NOT_ACCEPTED) {
1372
0
                return -7;
1373
0
            }
1374
1375
0
            if (!inbound) {
1376
0
                return -6;
1377
0
            }
1378
0
        }
1379
0
    }
1380
1381
0
    if (send_file_control_packet(m, friendnumber, inbound, file_number, control, nullptr, 0)) {
1382
0
        switch (control) {
1383
0
            case FILECONTROL_KILL: {
1384
0
                if (!inbound && (ft->status == FILESTATUS_TRANSFERRING || ft->status == FILESTATUS_FINISHED)) {
1385
                    // We are actively sending that file, remove from list
1386
0
                    --m->friendlist[friendnumber].num_sending_files;
1387
0
                }
1388
1389
0
                ft->status = FILESTATUS_NONE;
1390
0
                break;
1391
0
            }
1392
0
            case FILECONTROL_PAUSE: {
1393
0
                ft->paused |= FILE_PAUSE_US;
1394
0
                break;
1395
0
            }
1396
0
            case FILECONTROL_ACCEPT: {
1397
0
                ft->status = FILESTATUS_TRANSFERRING;
1398
1399
0
                if ((ft->paused & FILE_PAUSE_US) != 0) {
1400
0
                    ft->paused ^= FILE_PAUSE_US;
1401
0
                }
1402
0
                break;
1403
0
            }
1404
0
        }
1405
0
    } else {
1406
0
        return -8;
1407
0
    }
1408
1409
0
    return 0;
1410
0
}
1411
1412
/** @brief Send a seek file control request.
1413
 *
1414
 * @retval 0 on success
1415
 * @retval -1 if friend not valid.
1416
 * @retval -2 if friend not online.
1417
 * @retval -3 if file number invalid.
1418
 * @retval -4 if not receiving file.
1419
 * @retval -5 if file status wrong.
1420
 * @retval -6 if position bad.
1421
 * @retval -8 if packet failed to send.
1422
 */
1423
int file_seek(const Messenger *m, int32_t friendnumber, uint32_t filenumber, uint64_t position)
1424
0
{
1425
0
    if (!m_friend_exists(m, friendnumber)) {
1426
0
        return -1;
1427
0
    }
1428
1429
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1430
0
        return -2;
1431
0
    }
1432
1433
0
    if (filenumber < (1 << 16)) {
1434
        // Not receiving.
1435
0
        return -4;
1436
0
    }
1437
1438
0
    const uint32_t temp_filenum = (filenumber >> 16) - 1;
1439
1440
0
    if (temp_filenum >= MAX_CONCURRENT_FILE_PIPES) {
1441
0
        return -3;
1442
0
    }
1443
1444
0
    assert(temp_filenum <= UINT8_MAX);
1445
0
    const uint8_t file_number = temp_filenum;
1446
1447
    // We're always receiving at this point.
1448
0
    struct File_Transfers *ft = &m->friendlist[friendnumber].file_receiving[file_number];
1449
1450
0
    if (ft->status == FILESTATUS_NONE) {
1451
0
        return -3;
1452
0
    }
1453
1454
0
    if (ft->status != FILESTATUS_NOT_ACCEPTED) {
1455
0
        return -5;
1456
0
    }
1457
1458
0
    if (position >= ft->size) {
1459
0
        return -6;
1460
0
    }
1461
1462
0
    uint8_t sending_pos[sizeof(uint64_t)];
1463
0
    net_pack_u64(sending_pos, position);
1464
1465
0
    if (send_file_control_packet(m, friendnumber, true, file_number, FILECONTROL_SEEK, sending_pos,
1466
0
                                 sizeof(sending_pos))) {
1467
0
        ft->transferred = position;
1468
0
    } else {
1469
0
        return -8;
1470
0
    }
1471
1472
0
    return 0;
1473
0
}
1474
1475
/** @return packet number on success.
1476
 * @retval -1 on failure.
1477
 */
1478
non_null(1) nullable(4)
1479
static int64_t send_file_data_packet(const Messenger *m, int32_t friendnumber, uint8_t filenumber, const uint8_t *data,
1480
                                     uint16_t length)
1481
0
{
1482
0
    assert(length == 0 || data != nullptr);
1483
1484
0
    if (!m_friend_exists(m, friendnumber)) {
1485
0
        return -1;
1486
0
    }
1487
1488
0
    const uint16_t packet_size = 2 + length;
1489
0
    VLA(uint8_t, packet, packet_size);
1490
0
    packet[0] = PACKET_ID_FILE_DATA;
1491
0
    packet[1] = filenumber;
1492
1493
0
    if (length > 0) {
1494
0
        memcpy(packet + 2, data, length);
1495
0
    }
1496
1497
0
    return write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
1498
0
                             m->friendlist[friendnumber].friendcon_id), packet, packet_size, true);
1499
0
}
1500
1501
0
#define MAX_FILE_DATA_SIZE (MAX_CRYPTO_DATA_SIZE - 2)
1502
0
#define MIN_SLOTS_FREE (CRYPTO_MIN_QUEUE_LENGTH / 4)
1503
/** @brief Send file data.
1504
 *
1505
 * @retval 0 on success
1506
 * @retval -1 if friend not valid.
1507
 * @retval -2 if friend not online.
1508
 * @retval -3 if filenumber invalid.
1509
 * @retval -4 if file transfer not transferring.
1510
 * @retval -5 if bad data size.
1511
 * @retval -6 if packet queue full.
1512
 * @retval -7 if wrong position.
1513
 */
1514
int send_file_data(const Messenger *m, int32_t friendnumber, uint32_t filenumber, uint64_t position,
1515
                   const uint8_t *data, uint16_t length)
1516
0
{
1517
0
    assert(length == 0 || data != nullptr);
1518
1519
0
    if (!m_friend_exists(m, friendnumber)) {
1520
0
        return -1;
1521
0
    }
1522
1523
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1524
0
        return -2;
1525
0
    }
1526
1527
0
    if (filenumber >= MAX_CONCURRENT_FILE_PIPES) {
1528
0
        return -3;
1529
0
    }
1530
1531
0
    struct File_Transfers *ft = &m->friendlist[friendnumber].file_sending[filenumber];
1532
1533
0
    if (ft->status != FILESTATUS_TRANSFERRING) {
1534
0
        return -4;
1535
0
    }
1536
1537
0
    if (length > MAX_FILE_DATA_SIZE) {
1538
0
        return -5;
1539
0
    }
1540
1541
0
    if (ft->size - ft->transferred < length) {
1542
0
        return -5;
1543
0
    }
1544
1545
0
    if (ft->size != UINT64_MAX && length != MAX_FILE_DATA_SIZE && (ft->transferred + length) != ft->size) {
1546
0
        return -5;
1547
0
    }
1548
1549
0
    if (position != ft->transferred || (ft->requested <= position && ft->size != 0)) {
1550
0
        return -7;
1551
0
    }
1552
1553
    /* Prevent file sending from filling up the entire buffer preventing messages from being sent.
1554
     * TODO(irungentoo): remove */
1555
0
    if (crypto_num_free_sendqueue_slots(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
1556
0
                                        m->friendlist[friendnumber].friendcon_id)) < MIN_SLOTS_FREE) {
1557
0
        return -6;
1558
0
    }
1559
1560
0
    const int64_t ret = send_file_data_packet(m, friendnumber, filenumber, data, length);
1561
1562
0
    if (ret != -1) {
1563
        // TODO(irungentoo): record packet ids to check if other received complete file.
1564
0
        ft->transferred += length;
1565
1566
0
        if (length != MAX_FILE_DATA_SIZE || ft->size == ft->transferred) {
1567
0
            ft->status = FILESTATUS_FINISHED;
1568
0
            ft->last_packet_number = ret;
1569
0
        }
1570
1571
0
        return 0;
1572
0
    }
1573
1574
0
    return -6;
1575
0
}
1576
1577
/**
1578
 * Iterate over all file transfers and request chunks (from the client) for each
1579
 * of them.
1580
 *
1581
 * The free_slots parameter is updated by this function.
1582
 *
1583
 * @param m Our messenger object.
1584
 * @param friendnumber The friend we're sending files to.
1585
 * @param userdata The client userdata to pass along to chunk request callbacks.
1586
 * @param free_slots A pointer to the number of free send queue slots in the
1587
 *   crypto connection.
1588
 * @return true if there's still work to do, false otherwise.
1589
 *
1590
 */
1591
non_null()
1592
static bool do_all_filetransfers(Messenger *m, int32_t friendnumber, void *userdata, uint32_t *free_slots)
1593
0
{
1594
0
    Friend *const friendcon = &m->friendlist[friendnumber];
1595
1596
    // Iterate over file transfers as long as we're sending files
1597
0
    for (uint32_t i = 0; i < MAX_CONCURRENT_FILE_PIPES; ++i) {
1598
0
        if (friendcon->num_sending_files == 0) {
1599
            // no active file transfers anymore
1600
0
            return false;
1601
0
        }
1602
1603
0
        if (*free_slots == 0) {
1604
            // send buffer full enough
1605
0
            return false;
1606
0
        }
1607
1608
0
        struct File_Transfers *const ft = &friendcon->file_sending[i];
1609
1610
0
        if (ft->status == FILESTATUS_NONE || ft->status == FILESTATUS_NOT_ACCEPTED) {
1611
            // Filetransfers not actively sending, nothing to do
1612
0
            continue;
1613
0
        }
1614
1615
0
        if (max_speed_reached(m->net_crypto, friend_connection_crypt_connection_id(
1616
0
                                  m->fr_c, friendcon->friendcon_id))) {
1617
0
            LOGGER_DEBUG(m->log, "maximum connection speed reached");
1618
            // connection doesn't support any more data
1619
0
            return false;
1620
0
        }
1621
1622
        // If the file transfer is complete, we request a chunk of size 0.
1623
0
        if (ft->status == FILESTATUS_FINISHED && friend_received_packet(m, friendnumber, ft->last_packet_number) == 0) {
1624
0
            if (m->file_reqchunk != nullptr) {
1625
0
                m->file_reqchunk(m, friendnumber, i, ft->transferred, 0, userdata);
1626
0
            }
1627
1628
            // Now it's inactive, we're no longer sending this.
1629
0
            ft->status = FILESTATUS_NONE;
1630
0
            --friendcon->num_sending_files;
1631
0
        } else if (ft->status == FILESTATUS_TRANSFERRING && ft->paused == FILE_PAUSE_NOT) {
1632
0
            if (ft->size == 0) {
1633
                /* Send 0 data to friend if file is 0 length. */
1634
0
                send_file_data(m, friendnumber, i, 0, nullptr, 0);
1635
0
                continue;
1636
0
            }
1637
1638
0
            if (ft->size == ft->requested) {
1639
                // This file transfer is done.
1640
0
                continue;
1641
0
            }
1642
1643
0
            const uint16_t length = min_u64(ft->size - ft->requested, MAX_FILE_DATA_SIZE);
1644
0
            const uint64_t position = ft->requested;
1645
0
            ft->requested += length;
1646
1647
0
            if (m->file_reqchunk != nullptr) {
1648
0
                m->file_reqchunk(m, friendnumber, i, position, length, userdata);
1649
0
            }
1650
1651
            // The allocated slot is no longer free.
1652
0
            --*free_slots;
1653
0
        }
1654
0
    }
1655
1656
0
    return true;
1657
0
}
1658
1659
non_null(1) nullable(3)
1660
static void do_reqchunk_filecb(Messenger *m, int32_t friendnumber, void *userdata)
1661
0
{
1662
    // We're not currently doing any file transfers.
1663
0
    if (m->friendlist[friendnumber].num_sending_files == 0) {
1664
0
        return;
1665
0
    }
1666
1667
    // The number of packet slots left in the sendbuffer.
1668
    // This is a per friend count (CRYPTO_PACKET_BUFFER_SIZE).
1669
0
    uint32_t free_slots = crypto_num_free_sendqueue_slots(
1670
0
                              m->net_crypto,
1671
0
                              friend_connection_crypt_connection_id(
1672
0
                                  m->fr_c,
1673
0
                                  m->friendlist[friendnumber].friendcon_id));
1674
1675
    // We keep MIN_SLOTS_FREE slots free for other packets, otherwise file
1676
    // transfers might block other traffic for a long time.
1677
0
    free_slots = max_s32(0, (int32_t)free_slots - MIN_SLOTS_FREE);
1678
1679
    // Maximum number of outer loops below. If the client doesn't send file
1680
    // chunks from within the chunk request callback handler, we never realise
1681
    // that the file transfer has finished and may end up in an infinite loop.
1682
    //
1683
    // Request up to that number of chunks per file from the client
1684
    //
1685
    // TODO(Jfreegman): set this cap dynamically
1686
0
    const uint32_t max_ft_loops = 128;
1687
1688
0
    for (uint32_t i = 0; i < max_ft_loops; ++i) {
1689
0
        if (!do_all_filetransfers(m, friendnumber, userdata, &free_slots)) {
1690
0
            break;
1691
0
        }
1692
1693
0
        if (free_slots == 0) {
1694
            // stop when the buffer is full enough
1695
0
            break;
1696
0
        }
1697
0
    }
1698
0
}
1699
1700
/** @brief Run this when the friend disconnects.
1701
 * Kill all current file transfers.
1702
 */
1703
static void break_files(const Messenger *m, int32_t friendnumber)
1704
0
{
1705
0
    Friend *const f = &m->friendlist[friendnumber];
1706
1707
    // TODO(irungentoo): Inform the client which file transfers get killed with a callback?
1708
0
    for (uint32_t i = 0; i < MAX_CONCURRENT_FILE_PIPES; ++i) {
1709
0
        f->file_sending[i].status = FILESTATUS_NONE;
1710
0
        f->file_receiving[i].status = FILESTATUS_NONE;
1711
0
    }
1712
0
}
1713
1714
non_null()
1715
static struct File_Transfers *get_file_transfer(bool outbound, uint8_t filenumber,
1716
        uint32_t *real_filenumber, Friend *sender)
1717
0
{
1718
0
    struct File_Transfers *ft;
1719
1720
0
    if (outbound) {
1721
0
        *real_filenumber = filenumber;
1722
0
        ft = &sender->file_sending[filenumber];
1723
0
    } else {
1724
0
        *real_filenumber = (filenumber + 1) << 16;
1725
0
        ft = &sender->file_receiving[filenumber];
1726
0
    }
1727
1728
0
    if (ft->status == FILESTATUS_NONE) {
1729
0
        return nullptr;
1730
0
    }
1731
1732
0
    return ft;
1733
0
}
1734
1735
/** @retval -1 on failure
1736
 * @retval 0 on success.
1737
 */
1738
non_null(1, 6) nullable(8)
1739
static int handle_filecontrol(Messenger *m, int32_t friendnumber, bool outbound, uint8_t filenumber,
1740
                              uint8_t control_type, const uint8_t *data, uint16_t length, void *userdata)
1741
0
{
1742
0
    uint32_t real_filenumber;
1743
0
    struct File_Transfers *ft = get_file_transfer(outbound, filenumber, &real_filenumber, &m->friendlist[friendnumber]);
1744
1745
0
    if (ft == nullptr) {
1746
0
        LOGGER_DEBUG(m->log, "file control (friend %d, file %d): file transfer does not exist; telling the other to kill it",
1747
0
                     friendnumber, filenumber);
1748
0
        send_file_control_packet(m, friendnumber, !outbound, filenumber, FILECONTROL_KILL, nullptr, 0);
1749
0
        return -1;
1750
0
    }
1751
1752
0
    switch (control_type) {
1753
0
        case FILECONTROL_ACCEPT: {
1754
0
            if (outbound && ft->status == FILESTATUS_NOT_ACCEPTED) {
1755
0
                ft->status = FILESTATUS_TRANSFERRING;
1756
0
                ++m->friendlist[friendnumber].num_sending_files;
1757
0
            } else {
1758
0
                if ((ft->paused & FILE_PAUSE_OTHER) != 0) {
1759
0
                    ft->paused ^= FILE_PAUSE_OTHER;
1760
0
                } else {
1761
0
                    LOGGER_DEBUG(m->log, "file control (friend %d, file %d): friend told us to resume file transfer that wasn't paused",
1762
0
                                 friendnumber, filenumber);
1763
0
                    return -1;
1764
0
                }
1765
0
            }
1766
1767
0
            if (m->file_filecontrol != nullptr) {
1768
0
                m->file_filecontrol(m, friendnumber, real_filenumber, control_type, userdata);
1769
0
            }
1770
1771
0
            return 0;
1772
0
        }
1773
1774
0
        case FILECONTROL_PAUSE: {
1775
0
            if ((ft->paused & FILE_PAUSE_OTHER) != 0 || ft->status != FILESTATUS_TRANSFERRING) {
1776
0
                LOGGER_DEBUG(m->log, "file control (friend %d, file %d): friend told us to pause file transfer that is already paused",
1777
0
                             friendnumber, filenumber);
1778
0
                return -1;
1779
0
            }
1780
1781
0
            ft->paused |= FILE_PAUSE_OTHER;
1782
1783
0
            if (m->file_filecontrol != nullptr) {
1784
0
                m->file_filecontrol(m, friendnumber, real_filenumber, control_type, userdata);
1785
0
            }
1786
1787
0
            return 0;
1788
0
        }
1789
1790
0
        case FILECONTROL_KILL: {
1791
0
            if (m->file_filecontrol != nullptr) {
1792
0
                m->file_filecontrol(m, friendnumber, real_filenumber, control_type, userdata);
1793
0
            }
1794
1795
0
            if (outbound && (ft->status == FILESTATUS_TRANSFERRING || ft->status == FILESTATUS_FINISHED)) {
1796
0
                --m->friendlist[friendnumber].num_sending_files;
1797
0
            }
1798
1799
0
            ft->status = FILESTATUS_NONE;
1800
1801
0
            return 0;
1802
0
        }
1803
1804
0
        case FILECONTROL_SEEK: {
1805
0
            uint64_t position;
1806
1807
0
            if (length != sizeof(position)) {
1808
0
                LOGGER_DEBUG(m->log, "file control (friend %d, file %d): expected payload of length %d, but got %d",
1809
0
                             friendnumber, filenumber, (uint32_t)sizeof(position), length);
1810
0
                return -1;
1811
0
            }
1812
1813
            /* seek can only be sent by the receiver to seek before resuming broken transfers. */
1814
0
            if (ft->status != FILESTATUS_NOT_ACCEPTED || !outbound) {
1815
0
                LOGGER_DEBUG(m->log,
1816
0
                             "file control (friend %d, file %d): seek was either sent by a sender or by the receiver after accepting",
1817
0
                             friendnumber, filenumber);
1818
0
                return -1;
1819
0
            }
1820
1821
0
            net_unpack_u64(data, &position);
1822
1823
0
            if (position >= ft->size) {
1824
0
                LOGGER_DEBUG(m->log,
1825
0
                             "file control (friend %d, file %d): seek position %ld exceeds file size %ld",
1826
0
                             friendnumber, filenumber, (unsigned long)position, (unsigned long)ft->size);
1827
0
                return -1;
1828
0
            }
1829
1830
0
            ft->requested = position;
1831
0
            ft->transferred = position;
1832
0
            return 0;
1833
0
        }
1834
1835
0
        default: {
1836
0
            LOGGER_DEBUG(m->log, "file control (friend %d, file %d): invalid file control: %d",
1837
0
                         friendnumber, filenumber, control_type);
1838
0
            return -1;
1839
0
        }
1840
0
    }
1841
0
}
1842
1843
static int m_handle_lossy_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length,
1844
                                 void *userdata)
1845
0
{
1846
0
    Messenger *m = (Messenger *)object;
1847
1848
0
    if (!m_friend_exists(m, friendcon_id)) {
1849
0
        return 1;
1850
0
    }
1851
1852
0
    if (m->lossy_packethandler != nullptr) {
1853
0
        m->lossy_packethandler(m, friendcon_id, data[0], data, length, userdata);
1854
0
    }
1855
1856
0
    return 1;
1857
0
}
1858
1859
void custom_lossy_packet_registerhandler(Messenger *m, m_friend_lossy_packet_cb *lossy_packethandler)
1860
1.27k
{
1861
1.27k
    m->lossy_packethandler = lossy_packethandler;
1862
1.27k
}
1863
1864
int m_send_custom_lossy_packet(const Messenger *m, int32_t friendnumber, const uint8_t *data, uint32_t length)
1865
0
{
1866
0
    if (!m_friend_exists(m, friendnumber)) {
1867
0
        return -1;
1868
0
    }
1869
1870
0
    if (length == 0 || length > MAX_CRYPTO_DATA_SIZE) {
1871
0
        return -2;
1872
0
    }
1873
1874
0
    if (data[0] < PACKET_ID_RANGE_LOSSY_START || data[0] > PACKET_ID_RANGE_LOSSY_END) {
1875
0
        return -3;
1876
0
    }
1877
1878
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1879
0
        return -4;
1880
0
    }
1881
1882
0
    if (send_lossy_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
1883
0
                               m->friendlist[friendnumber].friendcon_id), data, length) == -1) {
1884
0
        return -5;
1885
0
    }
1886
1887
0
    return 0;
1888
0
}
1889
1890
non_null(1, 3) nullable(5)
1891
static int handle_custom_lossless_packet(void *object, int friend_num, const uint8_t *packet, uint16_t length,
1892
        void *userdata)
1893
0
{
1894
0
    Messenger *m = (Messenger *)object;
1895
1896
0
    if (!m_friend_exists(m, friend_num)) {
1897
0
        return -1;
1898
0
    }
1899
1900
0
    if (packet[0] < PACKET_ID_RANGE_LOSSLESS_CUSTOM_START || packet[0] > PACKET_ID_RANGE_LOSSLESS_CUSTOM_END) {
1901
        // allow PACKET_ID_MSI packets to be handled by custom packet handler
1902
0
        if (packet[0] != PACKET_ID_MSI) {
1903
0
            return -1;
1904
0
        }
1905
0
    }
1906
1907
0
    if (m->lossless_packethandler != nullptr) {
1908
0
        m->lossless_packethandler(m, friend_num, packet[0], packet, length, userdata);
1909
0
    }
1910
1911
0
    return 1;
1912
0
}
1913
1914
void custom_lossless_packet_registerhandler(Messenger *m, m_friend_lossless_packet_cb *lossless_packethandler)
1915
1.27k
{
1916
1.27k
    m->lossless_packethandler = lossless_packethandler;
1917
1.27k
}
1918
1919
int send_custom_lossless_packet(const Messenger *m, int32_t friendnumber, const uint8_t *data, uint32_t length)
1920
0
{
1921
0
    if (!m_friend_exists(m, friendnumber)) {
1922
0
        return -1;
1923
0
    }
1924
1925
0
    if (length == 0 || length > MAX_CRYPTO_DATA_SIZE) {
1926
0
        return -2;
1927
0
    }
1928
1929
0
    if ((data[0] < PACKET_ID_RANGE_LOSSLESS_CUSTOM_START || data[0] > PACKET_ID_RANGE_LOSSLESS_CUSTOM_END)
1930
0
            && data[0] != PACKET_ID_MSI) {
1931
0
        return -3;
1932
0
    }
1933
1934
0
    if (m->friendlist[friendnumber].status != FRIEND_ONLINE) {
1935
0
        return -4;
1936
0
    }
1937
1938
0
    if (write_cryptpacket(m->net_crypto, friend_connection_crypt_connection_id(m->fr_c,
1939
0
                          m->friendlist[friendnumber].friendcon_id), data, length, true) == -1) {
1940
0
        return -5;
1941
0
    }
1942
1943
0
    return 0;
1944
0
}
1945
1946
/** Function to filter out some friend requests*/
1947
non_null()
1948
static int friend_already_added(void *object, const uint8_t *public_key)
1949
0
{
1950
0
    const Messenger *m = (const Messenger *)object;
1951
1952
0
    if (getfriend_id(m, public_key) == -1) {
1953
0
        return 0;
1954
0
    }
1955
1956
0
    return -1;
1957
0
}
1958
1959
/** @brief Check for and handle a timed-out friend request.
1960
 *
1961
 * If the request has timed-out then the friend status is set back to FRIEND_ADDED.
1962
 * @param friendcon_id friendlist index of the timed-out friend
1963
 * @param t time
1964
 */
1965
non_null(1) nullable(4)
1966
static void check_friend_request_timed_out(Messenger *m, uint32_t friendcon_id, uint64_t t, void *userdata)
1967
0
{
1968
0
    Friend *f = &m->friendlist[friendcon_id];
1969
1970
0
    if (f->friendrequest_lastsent + f->friendrequest_timeout < t) {
1971
0
        set_friend_status(m, friendcon_id, FRIEND_ADDED, userdata);
1972
        /* Double the default timeout every time if friendrequest is assumed
1973
         * to have been sent unsuccessfully.
1974
         */
1975
0
        f->friendrequest_timeout *= 2;
1976
0
    }
1977
0
}
1978
1979
non_null(1) nullable(4)
1980
static int m_handle_status(void *object, int friendcon_id, bool status, void *userdata)
1981
0
{
1982
0
    Messenger *m = (Messenger *)object;
1983
1984
0
    if (status) { /* Went online. */
1985
0
        send_online_packet(m, m->friendlist[friendcon_id].friendcon_id);
1986
0
    } else { /* Went offline. */
1987
0
        if (m->friendlist[friendcon_id].status == FRIEND_ONLINE) {
1988
0
            set_friend_status(m, friendcon_id, FRIEND_CONFIRMED, userdata);
1989
0
        }
1990
0
    }
1991
1992
0
    return 0;
1993
0
}
1994
1995
non_null(1, 3) nullable(5)
1996
static int m_handle_packet_offline(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
1997
0
{
1998
0
    if (data_length == 0) {
1999
0
        set_friend_status(m, friendcon_id, FRIEND_CONFIRMED, userdata);
2000
0
    }
2001
2002
0
    return 0;
2003
0
}
2004
2005
non_null(1, 3) nullable(5)
2006
static int m_handle_packet_nickname(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2007
0
{
2008
0
    if (data_length > MAX_NAME_LENGTH) {
2009
0
        return 0;
2010
0
    }
2011
2012
    /* Make sure the NULL terminator is present. */
2013
0
    VLA(uint8_t, data_terminated, data_length + 1);
2014
0
    memcpy(data_terminated, data, data_length);
2015
0
    data_terminated[data_length] = 0;
2016
2017
    /* inform of namechange before we overwrite the old name */
2018
0
    if (m->friend_namechange != nullptr) {
2019
0
        m->friend_namechange(m, friendcon_id, data_terminated, data_length, userdata);
2020
0
    }
2021
2022
0
    memcpy(m->friendlist[friendcon_id].name, data_terminated, data_length);
2023
0
    m->friendlist[friendcon_id].name_length = data_length;
2024
2025
0
    return 0;
2026
0
}
2027
2028
non_null(1, 3) nullable(5)
2029
static int m_handle_packet_statusmessage(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2030
0
{
2031
0
    if (data_length > MAX_STATUSMESSAGE_LENGTH) {
2032
0
        return 0;
2033
0
    }
2034
2035
    /* Make sure the NULL terminator is present. */
2036
0
    VLA(uint8_t, data_terminated, data_length + 1);
2037
0
    memcpy(data_terminated, data, data_length);
2038
0
    data_terminated[data_length] = 0;
2039
2040
0
    if (m->friend_statusmessagechange != nullptr) {
2041
0
        m->friend_statusmessagechange(m, friendcon_id, data_terminated, data_length, userdata);
2042
0
    }
2043
2044
0
    set_friend_statusmessage(m, friendcon_id, data_terminated, data_length);
2045
2046
0
    return 0;
2047
0
}
2048
2049
non_null(1, 3) nullable(5)
2050
static int m_handle_packet_userstatus(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2051
0
{
2052
0
    if (data_length != 1) {
2053
0
        return 0;
2054
0
    }
2055
2056
0
    Userstatus status;
2057
0
    if (!userstatus_from_int(data[0], &status)) {
2058
0
        return 0;
2059
0
    }
2060
2061
0
    if (m->friend_userstatuschange != nullptr) {
2062
0
        m->friend_userstatuschange(m, friendcon_id, status, userdata);
2063
0
    }
2064
2065
0
    set_friend_userstatus(m, friendcon_id, status);
2066
2067
0
    return 0;
2068
0
}
2069
2070
non_null(1, 3) nullable(5)
2071
static int m_handle_packet_typing(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2072
0
{
2073
0
    if (data_length != 1) {
2074
0
        return 0;
2075
0
    }
2076
2077
0
    const bool typing = data[0] != 0;
2078
2079
0
    set_friend_typing(m, friendcon_id, typing);
2080
2081
0
    if (m->friend_typingchange != nullptr) {
2082
0
        m->friend_typingchange(m, friendcon_id, typing, userdata);
2083
0
    }
2084
2085
0
    return 0;
2086
0
}
2087
2088
non_null(1, 3) nullable(6)
2089
static int m_handle_packet_message(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, const Message_Type message_type, void *userdata)
2090
0
{
2091
0
    if (data_length == 0) {
2092
0
        return 0;
2093
0
    }
2094
2095
0
    const uint8_t *message = data;
2096
0
    const uint16_t message_length = data_length;
2097
2098
    /* Make sure the NULL terminator is present. */
2099
0
    VLA(uint8_t, message_terminated, message_length + 1);
2100
0
    memcpy(message_terminated, message, message_length);
2101
0
    message_terminated[message_length] = 0;
2102
2103
0
    if (m->friend_message != nullptr) {
2104
0
        m->friend_message(m, friendcon_id, message_type, message_terminated, message_length, userdata);
2105
0
    }
2106
2107
0
    return 0;
2108
0
}
2109
2110
non_null(1, 3) nullable(5)
2111
static int m_handle_packet_invite_conference(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2112
0
{
2113
0
    if (data_length == 0) {
2114
0
        return 0;
2115
0
    }
2116
2117
0
    if (m->conference_invite != nullptr) {
2118
0
        m->conference_invite(m, friendcon_id, data, data_length, userdata);
2119
0
    }
2120
2121
0
    return 0;
2122
0
}
2123
2124
non_null(1, 3) nullable(5)
2125
static int m_handle_packet_file_sendrequest(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2126
0
{
2127
0
    const unsigned int head_length = 1 + sizeof(uint32_t) + sizeof(uint64_t) + FILE_ID_LENGTH;
2128
2129
0
    if (data_length < head_length) {
2130
0
        return 0;
2131
0
    }
2132
2133
0
    const uint8_t filenumber = data[0];
2134
2135
#if UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES
2136
2137
    if (filenumber >= MAX_CONCURRENT_FILE_PIPES) {
2138
        return 0;
2139
    }
2140
2141
#endif /* UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES */
2142
2143
0
    uint64_t filesize;
2144
0
    uint32_t file_type;
2145
0
    const uint16_t filename_length = data_length - head_length;
2146
2147
0
    if (filename_length > MAX_FILENAME_LENGTH) {
2148
0
        return 0;
2149
0
    }
2150
2151
0
    memcpy(&file_type, data + 1, sizeof(file_type));
2152
0
    file_type = net_ntohl(file_type);
2153
2154
0
    net_unpack_u64(data + 1 + sizeof(uint32_t), &filesize);
2155
0
    struct File_Transfers *ft = &m->friendlist[friendcon_id].file_receiving[filenumber];
2156
2157
0
    if (ft->status != FILESTATUS_NONE) {
2158
0
        return 0;
2159
0
    }
2160
2161
0
    ft->status = FILESTATUS_NOT_ACCEPTED;
2162
0
    ft->size = filesize;
2163
0
    ft->transferred = 0;
2164
0
    ft->paused = FILE_PAUSE_NOT;
2165
0
    memcpy(ft->id, data + 1 + sizeof(uint32_t) + sizeof(uint64_t), FILE_ID_LENGTH);
2166
2167
0
    VLA(uint8_t, filename_terminated, filename_length + 1);
2168
0
    const uint8_t *filename = nullptr;
2169
2170
0
    if (filename_length > 0) {
2171
        /* Force NULL terminate file name. */
2172
0
        memcpy(filename_terminated, data + head_length, filename_length);
2173
0
        filename_terminated[filename_length] = 0;
2174
0
        filename = filename_terminated;
2175
0
    }
2176
2177
0
    uint32_t real_filenumber = filenumber;
2178
0
    real_filenumber += 1;
2179
0
    real_filenumber <<= 16;
2180
2181
0
    if (m->file_sendrequest != nullptr) {
2182
0
        m->file_sendrequest(m, friendcon_id, real_filenumber, file_type, filesize, filename, filename_length,
2183
0
                            userdata);
2184
0
    }
2185
2186
0
    return 0;
2187
0
}
2188
2189
non_null(1, 3) nullable(5)
2190
static int m_handle_packet_file_control(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2191
0
{
2192
0
    if (data_length < 3) {
2193
0
        return 0;
2194
0
    }
2195
2196
    // On the other side, "outbound" is "inbound", i.e. if they send 1,
2197
    // that means "inbound" on their side, but we call it "outbound"
2198
    // here.
2199
0
    const bool outbound = data[0] == 1;
2200
0
    const uint8_t filenumber = data[1];
2201
0
    const uint8_t control_type = data[2];
2202
2203
#if UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES
2204
2205
    if (filenumber >= MAX_CONCURRENT_FILE_PIPES) {
2206
        return 0;
2207
    }
2208
2209
#endif /* UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES */
2210
2211
0
    if (handle_filecontrol(m, friendcon_id, outbound, filenumber, control_type, data + 3, data_length - 3, userdata) == -1) {
2212
        // TODO(iphydf): Do something different here? Right now, this
2213
        // check is pointless.
2214
0
        return 0;
2215
0
    }
2216
2217
0
    return 0;
2218
0
}
2219
2220
non_null(1, 3) nullable(5)
2221
static int m_handle_packet_file_data(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2222
0
{
2223
0
    if (data_length < 1) {
2224
0
        return 0;
2225
0
    }
2226
2227
0
    const uint8_t filenumber = data[0];
2228
2229
#if UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES
2230
2231
    if (filenumber >= MAX_CONCURRENT_FILE_PIPES) {
2232
        return 0;
2233
    }
2234
2235
#endif /* UINT8_MAX >= MAX_CONCURRENT_FILE_PIPES */
2236
2237
0
    struct File_Transfers *ft = &m->friendlist[friendcon_id].file_receiving[filenumber];
2238
2239
0
    if (ft->status != FILESTATUS_TRANSFERRING) {
2240
0
        return 0;
2241
0
    }
2242
2243
0
    uint64_t position = ft->transferred;
2244
0
    uint32_t real_filenumber = filenumber;
2245
0
    real_filenumber += 1;
2246
0
    real_filenumber <<= 16;
2247
0
    uint16_t file_data_length = data_length - 1;
2248
0
    const uint8_t *file_data;
2249
2250
0
    if (file_data_length == 0) {
2251
0
        file_data = nullptr;
2252
0
    } else {
2253
0
        file_data = data + 1;
2254
0
    }
2255
2256
    /* Prevent more data than the filesize from being passed to clients. */
2257
0
    if ((ft->transferred + file_data_length) > ft->size) {
2258
0
        file_data_length = ft->size - ft->transferred;
2259
0
    }
2260
2261
0
    if (m->file_filedata != nullptr) {
2262
0
        m->file_filedata(m, friendcon_id, real_filenumber, position, file_data, file_data_length, userdata);
2263
0
    }
2264
2265
0
    ft->transferred += file_data_length;
2266
2267
0
    if (file_data_length > 0 && (ft->transferred >= ft->size || file_data_length != MAX_FILE_DATA_SIZE)) {
2268
0
        file_data_length = 0;
2269
0
        file_data = nullptr;
2270
0
        position = ft->transferred;
2271
2272
        /* Full file received. */
2273
0
        if (m->file_filedata != nullptr) {
2274
0
            m->file_filedata(m, friendcon_id, real_filenumber, position, file_data, file_data_length, userdata);
2275
0
        }
2276
0
    }
2277
2278
    /* Data is zero, filetransfer is over. */
2279
0
    if (file_data_length == 0) {
2280
0
        ft->status = FILESTATUS_NONE;
2281
0
    }
2282
2283
0
    return 0;
2284
0
}
2285
2286
non_null(1, 3) nullable(5)
2287
static int m_handle_packet_invite_groupchat(Messenger *m, const int friendcon_id, const uint8_t *data, const uint16_t data_length, void *userdata)
2288
0
{
2289
    // first two bytes are messenger packet type and group invite type
2290
0
    if (data_length < 2 + GC_JOIN_DATA_LENGTH) {
2291
0
        return 0;
2292
0
    }
2293
2294
0
    const uint8_t invite_type = data[1];
2295
0
    const uint8_t *join_data = data + 2;
2296
0
    const uint32_t join_data_len = data_length - 2;
2297
2298
0
    if (m->group_invite != nullptr && data[1] == GROUP_INVITE && data_length != 2 + GC_JOIN_DATA_LENGTH) {
2299
0
        if (group_not_added(m->group_handler, join_data, join_data_len)) {
2300
0
            m->group_invite(m, friendcon_id, join_data, GC_JOIN_DATA_LENGTH,
2301
0
                            join_data + GC_JOIN_DATA_LENGTH, join_data_len - GC_JOIN_DATA_LENGTH, userdata);
2302
0
        }
2303
0
    } else if (invite_type == GROUP_INVITE_ACCEPTED) {
2304
0
        handle_gc_invite_accepted_packet(m->group_handler, friendcon_id, join_data, join_data_len);
2305
0
    } else if (invite_type == GROUP_INVITE_CONFIRMATION) {
2306
0
        handle_gc_invite_confirmed_packet(m->group_handler, friendcon_id, join_data, join_data_len);
2307
0
    }
2308
2309
0
    return 0;
2310
0
}
2311
2312
non_null(1, 3) nullable(5)
2313
static int m_handle_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata)
2314
0
{
2315
0
    Messenger *m = (Messenger *)object;
2316
2317
0
    if (length == 0) {
2318
0
        return -1;
2319
0
    }
2320
2321
0
    const uint8_t packet_id = data[0];
2322
0
    const uint8_t *payload = data + 1;
2323
0
    const uint16_t payload_length = length - 1;
2324
2325
0
    if (m->friendlist[friendcon_id].status != FRIEND_ONLINE) {
2326
0
        if (packet_id == PACKET_ID_ONLINE && length == 1) {
2327
0
            set_friend_status(m, friendcon_id, FRIEND_ONLINE, userdata);
2328
0
            send_online_packet(m, m->friendlist[friendcon_id].friendcon_id);
2329
0
        } else {
2330
0
            return -1;
2331
0
        }
2332
0
    }
2333
2334
0
    switch (packet_id) {
2335
        // TODO(Green-Sky): now all return 0 on error AND success, make errors errors?
2336
0
        case PACKET_ID_OFFLINE:
2337
0
            return m_handle_packet_offline(m, friendcon_id, payload, payload_length, userdata);
2338
0
        case PACKET_ID_NICKNAME:
2339
0
            return m_handle_packet_nickname(m, friendcon_id, payload, payload_length, userdata);
2340
0
        case PACKET_ID_STATUSMESSAGE:
2341
0
            return m_handle_packet_statusmessage(m, friendcon_id, payload, payload_length, userdata);
2342
0
        case PACKET_ID_USERSTATUS:
2343
0
            return m_handle_packet_userstatus(m, friendcon_id, payload, payload_length, userdata);
2344
0
        case PACKET_ID_TYPING:
2345
0
            return m_handle_packet_typing(m, friendcon_id, payload, payload_length, userdata);
2346
0
        case PACKET_ID_MESSAGE:
2347
0
            return m_handle_packet_message(m, friendcon_id, payload, payload_length, MESSAGE_NORMAL, userdata);
2348
0
        case PACKET_ID_ACTION:
2349
0
            return m_handle_packet_message(m, friendcon_id, payload, payload_length, MESSAGE_ACTION, userdata);
2350
0
        case PACKET_ID_INVITE_CONFERENCE:
2351
0
            return m_handle_packet_invite_conference(m, friendcon_id, payload, payload_length, userdata);
2352
0
        case PACKET_ID_FILE_SENDREQUEST:
2353
0
            return m_handle_packet_file_sendrequest(m, friendcon_id, payload, payload_length, userdata);
2354
0
        case PACKET_ID_FILE_CONTROL:
2355
0
            return m_handle_packet_file_control(m, friendcon_id, payload, payload_length, userdata);
2356
0
        case PACKET_ID_FILE_DATA:
2357
0
            return m_handle_packet_file_data(m, friendcon_id, payload, payload_length, userdata);
2358
0
        case PACKET_ID_MSI:
2359
0
            return handle_custom_lossless_packet(object, friendcon_id, data, length, userdata);
2360
0
        case PACKET_ID_INVITE_GROUPCHAT:
2361
0
            return m_handle_packet_invite_groupchat(m, friendcon_id, payload, payload_length, userdata);
2362
0
    }
2363
2364
0
    return handle_custom_lossless_packet(object, friendcon_id, data, length, userdata);
2365
0
}
2366
2367
non_null(1) nullable(2)
2368
static void do_friends(Messenger *m, void *userdata)
2369
0
{
2370
0
    const uint64_t temp_time = mono_time_get(m->mono_time);
2371
2372
0
    for (uint32_t i = 0; i < m->numfriends; ++i) {
2373
0
        if (m->friendlist[i].status == FRIEND_ADDED) {
2374
0
            const int fr = send_friend_request_packet(m->fr_c, m->friendlist[i].friendcon_id, m->friendlist[i].friendrequest_nospam,
2375
0
                           m->friendlist[i].info,
2376
0
                           m->friendlist[i].info_size);
2377
2378
0
            if (fr >= 0) {
2379
0
                set_friend_status(m, i, FRIEND_REQUESTED, userdata);
2380
0
                m->friendlist[i].friendrequest_lastsent = temp_time;
2381
0
            }
2382
0
        }
2383
2384
0
        if (m->friendlist[i].status == FRIEND_REQUESTED) {
2385
            /* If we didn't connect to friend after successfully sending him a friend
2386
             * request the request is deemed unsuccessful so we set the status back to
2387
             * FRIEND_ADDED and try again.
2388
             */
2389
0
            check_friend_request_timed_out(m, i, temp_time, userdata);
2390
0
        }
2391
2392
0
        if (m->friendlist[i].status == FRIEND_ONLINE) { /* friend is online. */
2393
0
            if (!m->friendlist[i].name_sent) {
2394
0
                if (m_sendname(m, i, m->name, m->name_length)) {
2395
0
                    m->friendlist[i].name_sent = true;
2396
0
                }
2397
0
            }
2398
2399
0
            if (!m->friendlist[i].statusmessage_sent) {
2400
0
                if (send_statusmessage(m, i, m->statusmessage, m->statusmessage_length)) {
2401
0
                    m->friendlist[i].statusmessage_sent = true;
2402
0
                }
2403
0
            }
2404
2405
0
            if (!m->friendlist[i].userstatus_sent) {
2406
0
                if (send_userstatus(m, i, m->userstatus)) {
2407
0
                    m->friendlist[i].userstatus_sent = true;
2408
0
                }
2409
0
            }
2410
2411
0
            if (!m->friendlist[i].user_istyping_sent) {
2412
0
                if (send_user_istyping(m, i, m->friendlist[i].user_istyping)) {
2413
0
                    m->friendlist[i].user_istyping_sent = true;
2414
0
                }
2415
0
            }
2416
2417
0
            check_friend_tcp_udp(m, i, userdata);
2418
0
            do_receipts(m, i, userdata);
2419
0
            do_reqchunk_filecb(m, i, userdata);
2420
2421
0
            m->friendlist[i].last_seen_time = (uint64_t) time(nullptr);
2422
0
        }
2423
0
    }
2424
0
}
2425
2426
non_null(1) nullable(2)
2427
static void m_connection_status_callback(Messenger *m, void *userdata)
2428
0
{
2429
0
    const Onion_Connection_Status conn_status = onion_connection_status(m->onion_c);
2430
2431
0
    if (conn_status != m->last_connection_status) {
2432
0
        if (m->core_connection_change != nullptr) {
2433
0
            m->core_connection_change(m, conn_status, userdata);
2434
0
        }
2435
2436
0
        m->last_connection_status = conn_status;
2437
0
    }
2438
0
}
2439
2440
0
#define DUMPING_CLIENTS_FRIENDS_EVERY_N_SECONDS 60UL
2441
2442
#define IDSTRING_LEN (CRYPTO_PUBLIC_KEY_SIZE * 2 + 1)
2443
/** id_str should be of length at least IDSTRING_LEN */
2444
non_null()
2445
static char *id_to_string(const uint8_t *pk, char *id_str, size_t length)
2446
0
{
2447
0
    if (length < IDSTRING_LEN) {
2448
0
        snprintf(id_str, length, "Bad buf length");
2449
0
        return id_str;
2450
0
    }
2451
0
2452
0
    for (uint32_t i = 0; i < CRYPTO_PUBLIC_KEY_SIZE; ++i) {
2453
0
        snprintf(&id_str[i * 2], length - i * 2, "%02X", pk[i]);
2454
0
    }
2455
0
2456
0
    id_str[CRYPTO_PUBLIC_KEY_SIZE * 2] = '\0';
2457
0
    return id_str;
2458
0
}
2459
2460
/** @brief Minimum messenger run interval in ms
2461
 * TODO(mannol): A/V
2462
 */
2463
0
#define MIN_RUN_INTERVAL 50
2464
2465
/**
2466
 * @brief Return the time in milliseconds before `do_messenger()` should be called again
2467
 *   for optimal performance.
2468
 *
2469
 * @return time (in ms) before the next `do_messenger()` needs to be run on success.
2470
 */
2471
uint32_t messenger_run_interval(const Messenger *m)
2472
0
{
2473
0
    const uint32_t crypto_interval = crypto_run_interval(m->net_crypto);
2474
2475
0
    if (crypto_interval > MIN_RUN_INTERVAL) {
2476
0
        return MIN_RUN_INTERVAL;
2477
0
    }
2478
2479
0
    return crypto_interval;
2480
0
}
2481
2482
/** @brief Attempts to create a DHT announcement for a group chat with our connection info. An
2483
 * announcement can only be created if we either have a UDP or TCP connection to the network.
2484
 *
2485
 * @retval true if success.
2486
 */
2487
non_null()
2488
static bool self_announce_group(const Messenger *m, GC_Chat *chat, Onion_Friend *onion_friend)
2489
0
{
2490
0
    GC_Public_Announce announce = {{{{{0}}}}};
2491
2492
0
    const bool ip_port_is_set = chat->self_udp_status != SELF_UDP_STATUS_NONE;
2493
0
    const int tcp_num = tcp_copy_connected_relays(chat->tcp_conn, announce.base_announce.tcp_relays,
2494
0
                        GCA_MAX_ANNOUNCED_TCP_RELAYS);
2495
2496
0
    if (tcp_num == 0 && !ip_port_is_set) {
2497
0
        onion_friend_set_gc_data(onion_friend, nullptr, 0);
2498
0
        return false;
2499
0
    }
2500
2501
0
    announce.base_announce.tcp_relays_count = (uint8_t)tcp_num;
2502
0
    announce.base_announce.ip_port_is_set = ip_port_is_set;
2503
2504
0
    if (ip_port_is_set) {
2505
0
        memcpy(&announce.base_announce.ip_port, &chat->self_ip_port, sizeof(IP_Port));
2506
0
    }
2507
2508
0
    memcpy(announce.base_announce.peer_public_key, chat->self_public_key.enc, ENC_PUBLIC_KEY_SIZE);
2509
0
    memcpy(announce.chat_public_key, get_chat_id(&chat->chat_public_key), ENC_PUBLIC_KEY_SIZE);
2510
2511
0
    uint8_t gc_data[GCA_MAX_DATA_LENGTH];
2512
0
    const int length = gca_pack_public_announce(m->log, gc_data, GCA_MAX_DATA_LENGTH, &announce);
2513
2514
0
    if (length <= 0) {
2515
0
        onion_friend_set_gc_data(onion_friend, nullptr, 0);
2516
0
        return false;
2517
0
    }
2518
2519
0
    if (gca_add_announce(m->mem, m->mono_time, m->group_announce, &announce) == nullptr) {
2520
0
        onion_friend_set_gc_data(onion_friend, nullptr, 0);
2521
0
        return false;
2522
0
    }
2523
2524
0
    onion_friend_set_gc_data(onion_friend, gc_data, (uint16_t)length);
2525
0
    chat->update_self_announces = false;
2526
0
    chat->last_time_self_announce = mono_time_get(chat->mono_time);
2527
2528
0
    if (tcp_num > 0) {
2529
0
        pk_copy(chat->announced_tcp_relay_pk, announce.base_announce.tcp_relays[0].public_key);
2530
0
    } else {
2531
0
        memzero(chat->announced_tcp_relay_pk, sizeof(chat->announced_tcp_relay_pk));
2532
0
    }
2533
2534
0
    LOGGER_DEBUG(chat->log, "Published group announce. TCP relays: %d, UDP status: %d", tcp_num,
2535
0
                 chat->self_udp_status);
2536
0
    return true;
2537
0
}
2538
2539
non_null()
2540
static void do_gc_onion_friends(const Messenger *m)
2541
0
{
2542
0
    const uint16_t num_friends = onion_get_friend_count(m->onion_c);
2543
2544
0
    for (uint16_t i = 0; i < num_friends; ++i) {
2545
0
        Onion_Friend *onion_friend = onion_get_friend(m->onion_c, i);
2546
2547
0
        if (!onion_friend_is_groupchat(onion_friend)) {
2548
0
            continue;
2549
0
        }
2550
2551
0
        GC_Chat *chat = gc_get_group_by_public_key(m->group_handler, onion_friend_get_gc_public_key(onion_friend));
2552
2553
0
        if (chat == nullptr) {
2554
0
            continue;
2555
0
        }
2556
2557
0
        if (chat->update_self_announces) {
2558
0
            self_announce_group(m, chat, onion_friend);
2559
0
        }
2560
0
    }
2561
0
}
2562
2563
/** @brief The main loop that needs to be run at least 20 times per second. */
2564
void do_messenger(Messenger *m, void *userdata)
2565
0
{
2566
    // Add the TCP relays, but only if this is the first time calling do_messenger
2567
0
    if (!m->has_added_relays) {
2568
0
        m->has_added_relays = true;
2569
2570
0
        for (uint16_t i = 0; i < m->num_loaded_relays; ++i) {
2571
0
            add_tcp_relay(m->net_crypto, &m->loaded_relays[i].ip_port, m->loaded_relays[i].public_key);
2572
0
        }
2573
2574
0
        m->num_loaded_relays = 0;
2575
2576
0
        if (m->tcp_server != nullptr) {
2577
            /* Add self tcp server. */
2578
0
            IP_Port local_ip_port;
2579
0
            local_ip_port.port = net_htons(m->options.tcp_server_port);
2580
0
            local_ip_port.ip.family = net_family_ipv4();
2581
0
            local_ip_port.ip.ip.v4 = get_ip4_loopback();
2582
0
            add_tcp_relay(m->net_crypto, &local_ip_port, tcp_server_public_key(m->tcp_server));
2583
0
        }
2584
0
    }
2585
2586
0
    if (!m->options.udp_disabled) {
2587
0
        networking_poll(m->net, userdata);
2588
0
        do_dht(m->dht);
2589
0
    }
2590
2591
0
    if (m->tcp_server != nullptr) {
2592
0
        do_tcp_server(m->tcp_server, m->mono_time);
2593
0
    }
2594
2595
0
    do_net_crypto(m->net_crypto, userdata);
2596
0
    do_onion_client(m->onion_c);
2597
0
    do_friend_connections(m->fr_c, userdata);
2598
0
    do_friends(m, userdata);
2599
0
    do_gc(m->group_handler, userdata);
2600
0
    do_gca(m->mono_time, m->group_announce);
2601
0
    do_gc_onion_friends(m);
2602
0
    m_connection_status_callback(m, userdata);
2603
2604
0
    if (mono_time_get(m->mono_time) > m->lastdump + DUMPING_CLIENTS_FRIENDS_EVERY_N_SECONDS) {
2605
0
        m->lastdump = mono_time_get(m->mono_time);
2606
0
        uint32_t last_pinged;
2607
2608
0
        for (uint32_t client = 0; client < LCLIENT_LIST; ++client) {
2609
0
            const Client_data *cptr = dht_get_close_client(m->dht, client);
2610
0
            const IPPTsPng *const assocs[] = { &cptr->assoc4, &cptr->assoc6, nullptr };
2611
2612
0
            for (const IPPTsPng * const *it = assocs; *it != nullptr; ++it) {
2613
0
                const IPPTsPng *const assoc = *it;
2614
2615
0
                if (ip_isset(&assoc->ip_port.ip)) {
2616
0
                    last_pinged = m->lastdump - assoc->last_pinged;
2617
2618
0
                    if (last_pinged > 999) {
2619
0
                        last_pinged = 999;
2620
0
                    }
2621
2622
0
                    Ip_Ntoa ip_str;
2623
0
                    char id_str[IDSTRING_LEN];
2624
0
                    LOGGER_TRACE(m->log, "C[%2u] %s:%u [%3u] %s",
2625
0
                                 client, net_ip_ntoa(&assoc->ip_port.ip, &ip_str),
2626
0
                                 net_ntohs(assoc->ip_port.port), last_pinged,
2627
0
                                 id_to_string(cptr->public_key, id_str, sizeof(id_str)));
2628
0
                }
2629
0
            }
2630
0
        }
2631
2632
        /* dht contains additional "friends" (requests) */
2633
0
        const uint32_t num_dhtfriends = dht_get_num_friends(m->dht);
2634
0
        VLA(int32_t, m2dht, num_dhtfriends);
2635
0
        VLA(int32_t, dht2m, num_dhtfriends);
2636
2637
0
        for (uint32_t friend_idx = 0; friend_idx < num_dhtfriends; ++friend_idx) {
2638
0
            m2dht[friend_idx] = -1;
2639
0
            dht2m[friend_idx] = -1;
2640
2641
0
            if (friend_idx >= m->numfriends) {
2642
0
                continue;
2643
0
            }
2644
2645
0
            for (uint32_t dhtfriend = 0; dhtfriend < dht_get_num_friends(m->dht); ++dhtfriend) {
2646
0
                if (pk_equal(m->friendlist[friend_idx].real_pk, dht_get_friend_public_key(m->dht, dhtfriend))) {
2647
0
                    assert(dhtfriend < INT32_MAX);
2648
0
                    m2dht[friend_idx] = (int32_t)dhtfriend;
2649
0
                    break;
2650
0
                }
2651
0
            }
2652
0
        }
2653
2654
0
        for (uint32_t friend_idx = 0; friend_idx < num_dhtfriends; ++friend_idx) {
2655
0
            if (m2dht[friend_idx] >= 0) {
2656
0
                assert(friend_idx < INT32_MAX);
2657
0
                dht2m[m2dht[friend_idx]] = (int32_t)friend_idx;
2658
0
            }
2659
0
        }
2660
2661
0
        if (m->numfriends != dht_get_num_friends(m->dht)) {
2662
0
            LOGGER_TRACE(m->log, "Friend num in DHT %u != friend num in msger %u", dht_get_num_friends(m->dht), m->numfriends);
2663
0
        }
2664
2665
0
        for (uint32_t friend_idx = 0; friend_idx < num_dhtfriends; ++friend_idx) {
2666
0
            const Friend *const msgfptr = dht2m[friend_idx] >= 0 ?  &m->friendlist[dht2m[friend_idx]] : nullptr;
2667
0
            const DHT_Friend *const dhtfptr = dht_get_friend(m->dht, friend_idx);
2668
2669
0
            if (msgfptr != nullptr) {
2670
0
                char id_str[IDSTRING_LEN];
2671
0
                LOGGER_TRACE(m->log, "F[%2u:%2u] <%s> %s",
2672
0
                             dht2m[friend_idx], friend_idx, msgfptr->name,
2673
0
                             id_to_string(msgfptr->real_pk, id_str, sizeof(id_str)));
2674
0
            } else {
2675
0
                char id_str[IDSTRING_LEN];
2676
0
                LOGGER_TRACE(m->log, "F[--:%2u] %s", friend_idx,
2677
0
                             id_to_string(dht_friend_public_key(dhtfptr), id_str, sizeof(id_str)));
2678
0
            }
2679
2680
0
            for (uint32_t client = 0; client < MAX_FRIEND_CLIENTS; ++client) {
2681
0
                const Client_data *cptr = dht_friend_client(dhtfptr, client);
2682
0
                const IPPTsPng *const assocs[] = {&cptr->assoc4, &cptr->assoc6};
2683
2684
0
                for (size_t a = 0; a < sizeof(assocs) / sizeof(assocs[0]); ++a) {
2685
0
                    const IPPTsPng *const assoc = assocs[a];
2686
2687
0
                    if (ip_isset(&assoc->ip_port.ip)) {
2688
0
                        last_pinged = m->lastdump - assoc->last_pinged;
2689
2690
0
                        if (last_pinged > 999) {
2691
0
                            last_pinged = 999;
2692
0
                        }
2693
2694
0
                        Ip_Ntoa ip_str;
2695
0
                        char id_str[IDSTRING_LEN];
2696
0
                        LOGGER_TRACE(m->log, "F[%2u] => C[%2u] %s:%u [%3u] %s",
2697
0
                                     friend_idx, client, net_ip_ntoa(&assoc->ip_port.ip, &ip_str),
2698
0
                                     net_ntohs(assoc->ip_port.port), last_pinged,
2699
0
                                     id_to_string(cptr->public_key, id_str, sizeof(id_str)));
2700
0
                    }
2701
0
                }
2702
0
            }
2703
0
        }
2704
0
    }
2705
0
}
2706
2707
/** new messenger format for load/save, more robust and forward compatible */
2708
2709
29
#define SAVED_FRIEND_REQUEST_SIZE 1024
2710
5.43k
#define NUM_SAVED_PATH_NODES 8
2711
2712
struct Saved_Friend {
2713
    uint8_t status;
2714
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2715
    uint8_t info[SAVED_FRIEND_REQUEST_SIZE]; // the data that is sent during the friend requests we do.
2716
    uint16_t info_size; // Length of the info.
2717
    uint8_t name[MAX_NAME_LENGTH];
2718
    uint16_t name_length;
2719
    uint8_t statusmessage[MAX_STATUSMESSAGE_LENGTH];
2720
    uint16_t statusmessage_length;
2721
    uint8_t userstatus;
2722
    uint32_t friendrequest_nospam;
2723
    uint8_t last_seen_time[sizeof(uint64_t)];
2724
};
2725
2726
static uint32_t friend_size(void)
2727
5.58k
{
2728
5.58k
    uint32_t data = 0;
2729
5.58k
    const struct Saved_Friend *const temp = nullptr;
2730
2731
5.58k
#define VALUE_MEMBER(data, name) \
2732
33.5k
    do {                         \
2733
33.5k
        data += sizeof(name);    \
2734
33.5k
    } while (0)
2735
5.58k
#define ARRAY_MEMBER(data, name) \
2736
27.9k
    do {                         \
2737
27.9k
        data += sizeof(name);    \
2738
27.9k
    } while (0)
2739
2740
    // Exactly the same in friend_load, friend_save, and friend_size
2741
5.58k
    VALUE_MEMBER(data, temp->status);
2742
5.58k
    ARRAY_MEMBER(data, temp->real_pk);
2743
5.58k
    ARRAY_MEMBER(data, temp->info);
2744
5.58k
    ++data; // padding
2745
5.58k
    VALUE_MEMBER(data, temp->info_size);
2746
5.58k
    ARRAY_MEMBER(data, temp->name);
2747
5.58k
    VALUE_MEMBER(data, temp->name_length);
2748
5.58k
    ARRAY_MEMBER(data, temp->statusmessage);
2749
5.58k
    ++data; // padding
2750
5.58k
    VALUE_MEMBER(data, temp->statusmessage_length);
2751
5.58k
    VALUE_MEMBER(data, temp->userstatus);
2752
5.58k
    data += 3; // padding
2753
5.58k
    VALUE_MEMBER(data, temp->friendrequest_nospam);
2754
5.58k
    ARRAY_MEMBER(data, temp->last_seen_time);
2755
2756
5.58k
#undef VALUE_MEMBER
2757
5.58k
#undef ARRAY_MEMBER
2758
2759
5.58k
    return data;
2760
5.58k
}
2761
2762
non_null()
2763
static uint8_t *friend_save(const struct Saved_Friend *temp, uint8_t *data)
2764
120
{
2765
120
#define VALUE_MEMBER(data, name)           \
2766
720
    do {                                   \
2767
720
        memcpy(data, &name, sizeof(name)); \
2768
720
        data += sizeof(name);              \
2769
720
    } while (0)
2770
2771
120
#define ARRAY_MEMBER(data, name)          \
2772
600
    do {                                  \
2773
600
        memcpy(data, name, sizeof(name)); \
2774
600
        data += sizeof(name);             \
2775
600
    } while (0)
2776
2777
    // Exactly the same in friend_load, friend_save, and friend_size
2778
120
    VALUE_MEMBER(data, temp->status);
2779
120
    ARRAY_MEMBER(data, temp->real_pk);
2780
120
    ARRAY_MEMBER(data, temp->info);
2781
120
    ++data; // padding
2782
120
    VALUE_MEMBER(data, temp->info_size);
2783
120
    ARRAY_MEMBER(data, temp->name);
2784
120
    VALUE_MEMBER(data, temp->name_length);
2785
120
    ARRAY_MEMBER(data, temp->statusmessage);
2786
120
    ++data; // padding
2787
120
    VALUE_MEMBER(data, temp->statusmessage_length);
2788
120
    VALUE_MEMBER(data, temp->userstatus);
2789
120
    data += 3; // padding
2790
120
    VALUE_MEMBER(data, temp->friendrequest_nospam);
2791
120
    ARRAY_MEMBER(data, temp->last_seen_time);
2792
2793
120
#undef VALUE_MEMBER
2794
120
#undef ARRAY_MEMBER
2795
2796
120
    return data;
2797
120
}
2798
2799
non_null()
2800
static const uint8_t *friend_load(struct Saved_Friend *temp, const uint8_t *data)
2801
457
{
2802
457
#define VALUE_MEMBER(data, name)           \
2803
2.74k
    do {                                   \
2804
2.74k
        memcpy(&name, data, sizeof(name)); \
2805
2.74k
        data += sizeof(name);              \
2806
2.74k
    } while (0)
2807
2808
457
#define ARRAY_MEMBER(data, name)          \
2809
2.28k
    do {                                  \
2810
2.28k
        memcpy(name, data, sizeof(name)); \
2811
2.28k
        data += sizeof(name);             \
2812
2.28k
    } while (0)
2813
2814
    // Exactly the same in friend_load, friend_save, and friend_size
2815
457
    VALUE_MEMBER(data, temp->status);
2816
457
    ARRAY_MEMBER(data, temp->real_pk);
2817
457
    ARRAY_MEMBER(data, temp->info);
2818
457
    ++data; // padding
2819
457
    VALUE_MEMBER(data, temp->info_size);
2820
457
    ARRAY_MEMBER(data, temp->name);
2821
457
    VALUE_MEMBER(data, temp->name_length);
2822
457
    ARRAY_MEMBER(data, temp->statusmessage);
2823
457
    ++data; // padding
2824
457
    VALUE_MEMBER(data, temp->statusmessage_length);
2825
457
    VALUE_MEMBER(data, temp->userstatus);
2826
457
    data += 3; // padding
2827
457
    VALUE_MEMBER(data, temp->friendrequest_nospam);
2828
457
    ARRAY_MEMBER(data, temp->last_seen_time);
2829
2830
457
#undef VALUE_MEMBER
2831
457
#undef ARRAY_MEMBER
2832
2833
457
    return data;
2834
457
}
2835
2836
non_null()
2837
static uint32_t m_state_plugins_size(const Messenger *m)
2838
2.54k
{
2839
2.54k
    const uint32_t size32 = sizeof(uint32_t);
2840
2.54k
    const uint32_t sizesubhead = size32 * 2;
2841
2842
2.54k
    uint32_t size = 0;
2843
2844
2.54k
    for (const Messenger_State_Plugin *plugin = m->options.state_plugins;
2845
25.4k
            plugin != m->options.state_plugins + m->options.state_plugins_length;
2846
22.8k
            ++plugin) {
2847
22.8k
        size += sizesubhead + plugin->size(m);
2848
22.8k
    }
2849
2850
2.54k
    return size;
2851
2.54k
}
2852
2853
/** @brief Registers a state plugin for saving, loading, and getting the size of a section of the save.
2854
 *
2855
 * @retval true on success
2856
 * @retval false on error
2857
 */
2858
bool m_register_state_plugin(Messenger *m, State_Type type, m_state_size_cb *size_callback,
2859
                             m_state_load_cb *load_callback,
2860
                             m_state_save_cb *save_callback)
2861
15.9k
{
2862
15.9k
    const uint32_t new_length = m->options.state_plugins_length + 1;
2863
15.9k
    Messenger_State_Plugin *temp = (Messenger_State_Plugin *)mem_vrealloc(
2864
15.9k
                                       m->mem, m->options.state_plugins, new_length, sizeof(Messenger_State_Plugin));
2865
2866
15.9k
    if (temp == nullptr) {
2867
0
        return false;
2868
0
    }
2869
2870
15.9k
    m->options.state_plugins = temp;
2871
15.9k
    m->options.state_plugins_length = new_length;
2872
2873
15.9k
    const uint8_t index = m->options.state_plugins_length - 1;
2874
15.9k
    m->options.state_plugins[index].type = type;
2875
15.9k
    m->options.state_plugins[index].size = size_callback;
2876
15.9k
    m->options.state_plugins[index].load = load_callback;
2877
15.9k
    m->options.state_plugins[index].save = save_callback;
2878
2879
15.9k
    return true;
2880
15.9k
}
2881
2882
non_null()
2883
static uint32_t m_plugin_size(const Messenger *m, State_Type type)
2884
8.27k
{
2885
28.0k
    for (uint8_t i = 0; i < m->options.state_plugins_length; ++i) {
2886
28.0k
        const Messenger_State_Plugin plugin = m->options.state_plugins[i];
2887
2888
28.0k
        if (plugin.type == type) {
2889
8.27k
            return plugin.size(m);
2890
8.27k
        }
2891
28.0k
    }
2892
2893
0
    LOGGER_ERROR(m->log, "Unknown type encountered: %u", type);
2894
2895
0
    return UINT32_MAX;
2896
8.27k
}
2897
2898
/** return size of the messenger data (for saving). */
2899
uint32_t messenger_size(const Messenger *m)
2900
2.54k
{
2901
2.54k
    return m_state_plugins_size(m);
2902
2.54k
}
2903
2904
/** Save the messenger in data (must be allocated memory of size at least `Messenger_size()`) */
2905
uint8_t *messenger_save(const Messenger *m, uint8_t *data)
2906
1.27k
{
2907
12.7k
    for (uint8_t i = 0; i < m->options.state_plugins_length; ++i) {
2908
11.4k
        const Messenger_State_Plugin plugin = m->options.state_plugins[i];
2909
11.4k
        data = plugin.save(m, data);
2910
11.4k
    }
2911
2912
1.27k
    return data;
2913
1.27k
}
2914
2915
// nospam state plugin
2916
non_null()
2917
static uint32_t nospam_keys_size(const Messenger *m)
2918
4.34k
{
2919
4.34k
    return sizeof(uint32_t) + CRYPTO_PUBLIC_KEY_SIZE + CRYPTO_SECRET_KEY_SIZE;
2920
4.34k
}
2921
2922
non_null()
2923
static State_Load_Status load_nospam_keys(Messenger *m, const uint8_t *data, uint32_t length)
2924
532
{
2925
532
    if (length != m_plugin_size(m, STATE_TYPE_NOSPAMKEYS)) {
2926
1
        return STATE_LOAD_STATUS_ERROR;
2927
1
    }
2928
2929
531
    uint32_t nospam;
2930
531
    lendian_bytes_to_host32(&nospam, data);
2931
531
    set_nospam(m->fr, nospam);
2932
531
    load_secret_key(m->net_crypto, data + sizeof(uint32_t) + CRYPTO_PUBLIC_KEY_SIZE);
2933
2934
531
    if (!pk_equal(data + sizeof(uint32_t), nc_get_self_public_key(m->net_crypto))) {
2935
7
        LOGGER_ERROR(m->log, "public key stored in savedata does not match its secret key");
2936
7
        return STATE_LOAD_STATUS_ERROR;
2937
7
    }
2938
2939
524
    return STATE_LOAD_STATUS_CONTINUE;
2940
531
}
2941
2942
non_null()
2943
static uint8_t *save_nospam_keys(const Messenger *m, uint8_t *data)
2944
1.27k
{
2945
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_NOSPAMKEYS);
2946
1.27k
    static_assert(sizeof(get_nospam(m->fr)) == sizeof(uint32_t), "nospam doesn't fit in a 32 bit int");
2947
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_NOSPAMKEYS);
2948
1.27k
    const uint32_t nospam = get_nospam(m->fr);
2949
1.27k
    host_to_lendian_bytes32(data, nospam);
2950
1.27k
    save_keys(m->net_crypto, data + sizeof(uint32_t));
2951
1.27k
    data += len;
2952
1.27k
    return data;
2953
1.27k
}
2954
2955
// DHT state plugin
2956
non_null()
2957
static uint32_t m_dht_size(const Messenger *m)
2958
3.81k
{
2959
3.81k
    return dht_size(m->dht);
2960
3.81k
}
2961
2962
non_null()
2963
static uint8_t *save_dht(const Messenger *m, uint8_t *data)
2964
1.27k
{
2965
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_DHT);
2966
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_DHT);
2967
1.27k
    dht_save(m->dht, data);
2968
1.27k
    data += len;
2969
1.27k
    return data;
2970
1.27k
}
2971
2972
non_null()
2973
static State_Load_Status m_dht_load(Messenger *m, const uint8_t *data, uint32_t length)
2974
704
{
2975
704
    dht_load(m->dht, data, length); // TODO(endoffile78): Should we throw an error if dht_load fails?
2976
704
    return STATE_LOAD_STATUS_CONTINUE;
2977
704
}
2978
2979
// friendlist state plugin
2980
non_null()
2981
static uint32_t saved_friendslist_size(const Messenger *m)
2982
3.81k
{
2983
3.81k
    return count_friendlist(m) * friend_size();
2984
3.81k
}
2985
2986
non_null()
2987
static uint8_t *friends_list_save(const Messenger *m, uint8_t *data)
2988
1.27k
{
2989
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_FRIENDS);
2990
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_FRIENDS);
2991
2992
1.27k
    uint32_t num = 0;
2993
1.27k
    uint8_t *cur_data = data;
2994
2995
1.39k
    for (uint32_t i = 0; i < m->numfriends; ++i) {
2996
120
        if (m->friendlist[i].status > 0) {
2997
120
            struct Saved_Friend temp = { 0 };
2998
120
            temp.status = m->friendlist[i].status;
2999
120
            memcpy(temp.real_pk, m->friendlist[i].real_pk, CRYPTO_PUBLIC_KEY_SIZE);
3000
3001
120
            if (temp.status < 3) {
3002
                // TODO(iphydf): Use uint16_t and min_u16 here.
3003
29
                const size_t friendrequest_length =
3004
29
                    min_u32(m->friendlist[i].info_size,
3005
29
                            min_u32(SAVED_FRIEND_REQUEST_SIZE, MAX_FRIEND_REQUEST_DATA_SIZE));
3006
29
                memcpy(temp.info, m->friendlist[i].info, friendrequest_length);
3007
3008
29
                temp.info_size = net_htons(m->friendlist[i].info_size);
3009
29
                temp.friendrequest_nospam = m->friendlist[i].friendrequest_nospam;
3010
91
            } else {
3011
91
                temp.status = 3;
3012
91
                memcpy(temp.name, m->friendlist[i].name, m->friendlist[i].name_length);
3013
91
                temp.name_length = net_htons(m->friendlist[i].name_length);
3014
91
                memcpy(temp.statusmessage, m->friendlist[i].statusmessage, m->friendlist[i].statusmessage_length);
3015
91
                temp.statusmessage_length = net_htons(m->friendlist[i].statusmessage_length);
3016
91
                temp.userstatus = m->friendlist[i].userstatus;
3017
3018
91
                net_pack_u64(temp.last_seen_time, m->friendlist[i].last_seen_time);
3019
91
            }
3020
3021
120
            uint8_t *next_data = friend_save(&temp, cur_data);
3022
120
            assert(next_data - cur_data == friend_size());
3023
120
#ifdef __LP64__
3024
120
            assert(memcmp(cur_data, &temp, friend_size()) == 0);
3025
120
#endif /* __LP64__ */
3026
120
            cur_data = next_data;
3027
120
            ++num;
3028
120
        }
3029
120
    }
3030
3031
1.27k
    assert(cur_data - data == num * friend_size());
3032
1.27k
    data += len;
3033
3034
1.27k
    return data;
3035
1.27k
}
3036
3037
non_null()
3038
static State_Load_Status friends_list_load(Messenger *m, const uint8_t *data, uint32_t length)
3039
267
{
3040
267
    const uint32_t l_friend_size = friend_size();
3041
3042
267
    if (length % l_friend_size != 0) {
3043
1
        return STATE_LOAD_STATUS_ERROR; // TODO(endoffile78): error or continue?
3044
1
    }
3045
3046
266
    const uint32_t num = length / l_friend_size;
3047
266
    const uint8_t *cur_data = data;
3048
3049
723
    for (uint32_t i = 0; i < num; ++i) {
3050
457
        struct Saved_Friend temp = { 0 };
3051
457
        const uint8_t *next_data = friend_load(&temp, cur_data);
3052
457
        assert(next_data - cur_data == l_friend_size);
3053
3054
457
        cur_data = next_data;
3055
3056
457
        if (temp.status >= 3) {
3057
262
            const int fnum = m_addfriend_norequest(m, temp.real_pk);
3058
3059
262
            if (fnum < 0) {
3060
51
                continue;
3061
51
            }
3062
3063
211
            setfriendname(m, fnum, temp.name, net_ntohs(temp.name_length));
3064
211
            set_friend_statusmessage(m, fnum, temp.statusmessage, net_ntohs(temp.statusmessage_length));
3065
211
            set_friend_userstatus(m, fnum, temp.userstatus);
3066
211
            net_unpack_u64(temp.last_seen_time, &m->friendlist[fnum].last_seen_time);
3067
211
        } else if (temp.status != 0) {
3068
            /* TODO(irungentoo): This is not a good way to do this. */
3069
122
            uint8_t address[FRIEND_ADDRESS_SIZE];
3070
122
            pk_copy(address, temp.real_pk);
3071
122
            memcpy(address + CRYPTO_PUBLIC_KEY_SIZE, &temp.friendrequest_nospam, sizeof(uint32_t));
3072
122
            uint16_t checksum = data_checksum(address, FRIEND_ADDRESS_SIZE - sizeof(checksum));
3073
122
            memcpy(address + CRYPTO_PUBLIC_KEY_SIZE + sizeof(uint32_t), &checksum, sizeof(checksum));
3074
122
            m_addfriend(m, address, temp.info, net_ntohs(temp.info_size));
3075
122
        }
3076
457
    }
3077
3078
266
    return STATE_LOAD_STATUS_CONTINUE;
3079
266
}
3080
3081
non_null()
3082
static void pack_groupchats(const GC_Session *c, Bin_Pack *bp)
3083
2.78k
{
3084
2.78k
    assert(bp != nullptr && c != nullptr);
3085
2.78k
    bin_pack_array(bp, gc_count_groups(c));
3086
3087
5.76k
    for (uint32_t i = 0; i < c->chats_index; ++i) { // this loop must match the one in gc_count_groups()
3088
2.98k
        const GC_Chat *chat = &c->chats[i];
3089
3090
2.98k
        if (!gc_group_is_valid(chat)) {
3091
1.97k
            continue;
3092
1.97k
        }
3093
3094
1.01k
        gc_group_save(chat, bp);
3095
1.01k
    }
3096
2.78k
}
3097
3098
non_null()
3099
static bool pack_groupchats_handler(const void *obj, const Logger *logger, Bin_Pack *bp)
3100
2.78k
{
3101
2.78k
    const GC_Session *session = (const GC_Session *)obj;
3102
2.78k
    pack_groupchats(session, bp);
3103
2.78k
    return true;  // TODO(iphydf): Return bool from pack functions.
3104
2.78k
}
3105
3106
non_null()
3107
static uint32_t saved_groups_size(const Messenger *m)
3108
2.66k
{
3109
2.66k
    const GC_Session *session = m->group_handler;
3110
2.66k
    return bin_pack_obj_size(pack_groupchats_handler, session, m->log);
3111
2.66k
}
3112
3113
non_null()
3114
static uint8_t *groups_save(const Messenger *m, uint8_t *data)
3115
1.27k
{
3116
1.27k
    const GC_Session *c = m->group_handler;
3117
3118
1.27k
    const uint32_t num_groups = gc_count_groups(c);
3119
3120
1.27k
    if (num_groups == 0) {
3121
1.15k
        return data;
3122
1.15k
    }
3123
3124
120
    const uint32_t len = m_plugin_size(m, STATE_TYPE_GROUPS);
3125
3126
120
    if (len == 0) {
3127
0
        return data;
3128
0
    }
3129
3130
120
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_GROUPS);
3131
3132
120
    if (!bin_pack_obj(pack_groupchats_handler, c, m->log, data, len)) {
3133
0
        LOGGER_FATAL(m->log, "failed to pack group chats into buffer of length %u", len);
3134
0
        return data;
3135
0
    }
3136
3137
120
    data += len;
3138
3139
120
    LOGGER_DEBUG(m->log, "Saved %u groups (length %u)", num_groups, len);
3140
3141
120
    return data;
3142
120
}
3143
3144
non_null()
3145
static bool handle_groups_load(void *obj, Bin_Unpack *bu)
3146
26.0k
{
3147
26.0k
    Messenger *m = (Messenger *)obj;
3148
3149
26.0k
    uint32_t num_groups;
3150
26.0k
    if (!bin_unpack_array(bu, &num_groups)) {
3151
33
        LOGGER_ERROR(m->log, "msgpack failed to unpack groupchats array: expected array");
3152
33
        return false;
3153
33
    }
3154
3155
26.0k
    LOGGER_DEBUG(m->log, "Loading %u groups", num_groups);
3156
3157
26.1k
    for (uint32_t i = 0; i < num_groups; ++i) {
3158
26.0k
        const int group_number = gc_group_load(m->group_handler, bu);
3159
3160
26.0k
        if (group_number < 0) {
3161
25.9k
            LOGGER_WARNING(m->log, "Failed to load group %u", i);
3162
            // Can't recover trivially. We may need to skip over some data here.
3163
25.9k
            break;
3164
25.9k
        }
3165
26.0k
    }
3166
3167
26.0k
    LOGGER_DEBUG(m->log, "Successfully loaded %u groups", gc_count_groups(m->group_handler));
3168
3169
26.0k
    return true;
3170
26.0k
}
3171
3172
non_null()
3173
static State_Load_Status groups_load(Messenger *m, const uint8_t *data, uint32_t length)
3174
26.0k
{
3175
26.0k
    if (!bin_unpack_obj(m->mem, handle_groups_load, m, data, length)) {
3176
33
        LOGGER_ERROR(m->log, "msgpack failed to unpack groupchats array");
3177
33
        return STATE_LOAD_STATUS_ERROR;
3178
33
    }
3179
3180
26.0k
    return STATE_LOAD_STATUS_CONTINUE;
3181
26.0k
}
3182
3183
// name state plugin
3184
non_null()
3185
static uint32_t name_size(const Messenger *m)
3186
3.81k
{
3187
3.81k
    return m->name_length;
3188
3.81k
}
3189
3190
non_null()
3191
static uint8_t *save_name(const Messenger *m, uint8_t *data)
3192
1.27k
{
3193
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_NAME);
3194
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_NAME);
3195
1.27k
    memcpy(data, m->name, len);
3196
1.27k
    data += len;
3197
1.27k
    return data;
3198
1.27k
}
3199
3200
non_null()
3201
static State_Load_Status load_name(Messenger *m, const uint8_t *data, uint32_t length)
3202
446
{
3203
446
    if (length > 0 && length <= MAX_NAME_LENGTH) {
3204
314
        setname(m, data, length);
3205
314
    }
3206
3207
446
    return STATE_LOAD_STATUS_CONTINUE;
3208
446
}
3209
3210
// status message state plugin
3211
non_null()
3212
static uint32_t status_message_size(const Messenger *m)
3213
3.81k
{
3214
3.81k
    return m->statusmessage_length;
3215
3.81k
}
3216
3217
non_null()
3218
static uint8_t *save_status_message(const Messenger *m, uint8_t *data)
3219
1.27k
{
3220
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_STATUSMESSAGE);
3221
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_STATUSMESSAGE);
3222
1.27k
    memcpy(data, m->statusmessage, len);
3223
1.27k
    data += len;
3224
1.27k
    return data;
3225
1.27k
}
3226
3227
non_null()
3228
static State_Load_Status load_status_message(Messenger *m, const uint8_t *data, uint32_t length)
3229
670
{
3230
670
    if (length > 0 && length <= MAX_STATUSMESSAGE_LENGTH) {
3231
557
        m_set_statusmessage(m, data, length);
3232
557
    }
3233
3234
670
    return STATE_LOAD_STATUS_CONTINUE;
3235
670
}
3236
3237
// status state plugin
3238
non_null()
3239
static uint32_t status_size(const Messenger *m)
3240
3.81k
{
3241
3.81k
    return 1;
3242
3.81k
}
3243
3244
non_null()
3245
static uint8_t *save_status(const Messenger *m, uint8_t *data)
3246
1.27k
{
3247
1.27k
    const uint32_t len = m_plugin_size(m, STATE_TYPE_STATUS);
3248
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_STATUS);
3249
1.27k
    *data = m->userstatus;
3250
1.27k
    data += len;
3251
1.27k
    return data;
3252
1.27k
}
3253
3254
non_null()
3255
static State_Load_Status load_status(Messenger *m, const uint8_t *data, uint32_t length)
3256
596
{
3257
596
    if (length == 1) {
3258
561
        m_set_userstatus(m, *data);
3259
561
    }
3260
3261
596
    return STATE_LOAD_STATUS_CONTINUE;
3262
596
}
3263
3264
// TCP Relay state plugin
3265
non_null()
3266
static uint32_t tcp_relay_size(const Messenger *m)
3267
2.54k
{
3268
2.54k
    return NUM_SAVED_TCP_RELAYS * packed_node_size(net_family_tcp_ipv6());
3269
2.54k
}
3270
3271
non_null()
3272
static uint8_t *save_tcp_relays(const Messenger *m, uint8_t *data)
3273
1.27k
{
3274
1.27k
    Node_format relays[NUM_SAVED_TCP_RELAYS] = {{{0}}};
3275
1.27k
    uint8_t *temp_data = data;
3276
1.27k
    data = state_write_section_header(temp_data, STATE_COOKIE_TYPE, 0, STATE_TYPE_TCP_RELAY);
3277
3278
1.27k
    if (m->num_loaded_relays > 0) {
3279
38
        memcpy(relays, m->loaded_relays, sizeof(Node_format) * m->num_loaded_relays);
3280
38
    }
3281
3282
1.27k
    uint32_t num = m->num_loaded_relays;
3283
1.27k
    num += copy_connected_tcp_relays(m->net_crypto, relays + num, NUM_SAVED_TCP_RELAYS - num);
3284
3285
1.27k
    const int l = pack_nodes(m->log, data, NUM_SAVED_TCP_RELAYS * packed_node_size(net_family_tcp_ipv6()), relays, num);
3286
3287
1.27k
    if (l > 0) {
3288
38
        const uint32_t len = l;
3289
38
        data = state_write_section_header(temp_data, STATE_COOKIE_TYPE, len, STATE_TYPE_TCP_RELAY);
3290
38
        data += len;
3291
38
    }
3292
3293
1.27k
    return data;
3294
1.27k
}
3295
3296
non_null()
3297
static State_Load_Status load_tcp_relays(Messenger *m, const uint8_t *data, uint32_t length)
3298
587
{
3299
587
    if (length > 0) {
3300
521
        const int num = unpack_nodes(m->loaded_relays, NUM_SAVED_TCP_RELAYS, nullptr, data, length, true);
3301
3302
521
        if (num == -1) {
3303
99
            m->num_loaded_relays = 0;
3304
99
            return STATE_LOAD_STATUS_CONTINUE;
3305
99
        }
3306
3307
422
        m->num_loaded_relays = num;
3308
422
        m->has_added_relays = false;
3309
422
    }
3310
3311
488
    return STATE_LOAD_STATUS_CONTINUE;
3312
587
}
3313
3314
// path node state plugin
3315
non_null()
3316
static uint32_t path_node_size(const Messenger *m)
3317
2.54k
{
3318
2.54k
    return NUM_SAVED_PATH_NODES * packed_node_size(net_family_tcp_ipv6());
3319
2.54k
}
3320
3321
non_null()
3322
static uint8_t *save_path_nodes(const Messenger *m, uint8_t *data)
3323
1.27k
{
3324
1.27k
    Node_format nodes[NUM_SAVED_PATH_NODES] = {{{0}}};
3325
1.27k
    uint8_t *temp_data = data;
3326
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, 0, STATE_TYPE_PATH_NODE);
3327
1.27k
    const unsigned int num = onion_backup_nodes(m->onion_c, nodes, NUM_SAVED_PATH_NODES);
3328
1.27k
    const int l = pack_nodes(m->log, data, NUM_SAVED_PATH_NODES * packed_node_size(net_family_tcp_ipv6()), nodes, num);
3329
3330
1.27k
    if (l > 0) {
3331
36
        const uint32_t len = l;
3332
36
        data = state_write_section_header(temp_data, STATE_COOKIE_TYPE, len, STATE_TYPE_PATH_NODE);
3333
36
        data += len;
3334
36
    }
3335
3336
1.27k
    return data;
3337
1.27k
}
3338
3339
non_null()
3340
static State_Load_Status load_path_nodes(Messenger *m, const uint8_t *data, uint32_t length)
3341
427
{
3342
427
    if (length > 0) {
3343
359
        Node_format nodes[NUM_SAVED_PATH_NODES];
3344
359
        const int num = unpack_nodes(nodes, NUM_SAVED_PATH_NODES, nullptr, data, length, false);
3345
3346
359
        if (num == -1) {
3347
53
            return STATE_LOAD_STATUS_CONTINUE;
3348
53
        }
3349
3350
2.23k
        for (int i = 0; i < num; ++i) {
3351
1.93k
            onion_add_bs_path_node(m->onion_c, &nodes[i].ip_port, nodes[i].public_key);
3352
1.93k
        }
3353
306
    }
3354
3355
374
    return STATE_LOAD_STATUS_CONTINUE;
3356
427
}
3357
3358
non_null()
3359
static void m_register_default_plugins(Messenger *m)
3360
1.76k
{
3361
1.76k
    m_register_state_plugin(m, STATE_TYPE_NOSPAMKEYS, nospam_keys_size, load_nospam_keys, save_nospam_keys);
3362
1.76k
    m_register_state_plugin(m, STATE_TYPE_DHT, m_dht_size, m_dht_load, save_dht);
3363
1.76k
    m_register_state_plugin(m, STATE_TYPE_FRIENDS, saved_friendslist_size, friends_list_load, friends_list_save);
3364
1.76k
    m_register_state_plugin(m, STATE_TYPE_NAME, name_size, load_name, save_name);
3365
1.76k
    m_register_state_plugin(m, STATE_TYPE_STATUSMESSAGE, status_message_size, load_status_message,
3366
1.76k
                            save_status_message);
3367
1.76k
    m_register_state_plugin(m, STATE_TYPE_STATUS, status_size, load_status, save_status);
3368
1.76k
    if (m->options.groups_persistence_enabled) {
3369
1.76k
        m_register_state_plugin(m, STATE_TYPE_GROUPS, saved_groups_size, groups_load, groups_save);
3370
1.76k
    }
3371
1.76k
    m_register_state_plugin(m, STATE_TYPE_TCP_RELAY, tcp_relay_size, load_tcp_relays, save_tcp_relays);
3372
1.76k
    m_register_state_plugin(m, STATE_TYPE_PATH_NODE, path_node_size, load_path_nodes, save_path_nodes);
3373
1.76k
}
3374
3375
bool messenger_load_state_section(Messenger *m, const uint8_t *data, uint32_t length, uint16_t type,
3376
                                  State_Load_Status *status)
3377
31.0k
{
3378
209k
    for (uint8_t i = 0; i < m->options.state_plugins_length; ++i) {
3379
208k
        const Messenger_State_Plugin *const plugin = &m->options.state_plugins[i];
3380
3381
208k
        if (plugin->type == type) {
3382
30.3k
            *status = plugin->load(m, data, length);
3383
30.3k
            return true;
3384
30.3k
        }
3385
208k
    }
3386
3387
685
    return false;
3388
31.0k
}
3389
3390
/** @brief Return the number of friends in the instance m.
3391
 *
3392
 * You should use this to determine how much memory to allocate
3393
 * for copy_friendlist.
3394
 */
3395
uint32_t count_friendlist(const Messenger *m)
3396
3.81k
{
3397
3.81k
    uint32_t ret = 0;
3398
3399
4.17k
    for (uint32_t i = 0; i < m->numfriends; ++i) {
3400
360
        if (m->friendlist[i].status > 0) {
3401
360
            ++ret;
3402
360
        }
3403
360
    }
3404
3405
3.81k
    return ret;
3406
3.81k
}
3407
3408
/** @brief Copy a list of valid friend IDs into the array out_list.
3409
 * If out_list is NULL, returns 0.
3410
 * Otherwise, returns the number of elements copied.
3411
 * If the array was too small, the contents
3412
 * of out_list will be truncated to list_size.
3413
 */
3414
uint32_t copy_friendlist(Messenger const *m, uint32_t *out_list, uint32_t list_size)
3415
0
{
3416
0
    if (out_list == nullptr) {
3417
0
        return 0;
3418
0
    }
3419
3420
0
    if (m->numfriends == 0) {
3421
0
        return 0;
3422
0
    }
3423
3424
0
    uint32_t ret = 0;
3425
3426
0
    for (uint32_t i = 0; i < m->numfriends; ++i) {
3427
0
        if (ret >= list_size) {
3428
0
            break; /* Abandon ship */
3429
0
        }
3430
3431
0
        if (m->friendlist[i].status > 0) {
3432
0
            out_list[ret] = i;
3433
0
            ++ret;
3434
0
        }
3435
0
    }
3436
3437
0
    return ret;
3438
0
}
3439
3440
static fr_friend_request_cb m_handle_friend_request;
3441
non_null(1, 2, 3) nullable(5)
3442
static void m_handle_friend_request(
3443
    void *object, const uint8_t *public_key, const uint8_t *message, size_t length, void *user_data)
3444
0
{
3445
0
    Messenger *m = (Messenger *)object;
3446
0
    assert(m != nullptr);
3447
0
    m->friend_request(m, public_key, message, length, user_data);
3448
0
}
3449
3450
/** @brief Run this at startup.
3451
 *
3452
 * @return allocated instance of Messenger on success.
3453
 * @retval 0 if there are problems.
3454
 *
3455
 * if error is not NULL it will be set to one of the values in the enum above.
3456
 */
3457
Messenger *new_messenger(Mono_Time *mono_time, const Memory *mem, const Random *rng, const Network *ns,
3458
                         Messenger_Options *options, Messenger_Error *error)
3459
1.78k
{
3460
1.78k
    if (options == nullptr) {
3461
0
        return nullptr;
3462
0
    }
3463
3464
1.78k
    if (error != nullptr) {
3465
1.78k
        *error = MESSENGER_ERROR_OTHER;
3466
1.78k
    }
3467
3468
1.78k
    Messenger *m = (Messenger *)mem_alloc(mem, sizeof(Messenger));
3469
3470
1.78k
    if (m == nullptr) {
3471
1
        return nullptr;
3472
1
    }
3473
3474
1.78k
    m->mono_time = mono_time;
3475
1.78k
    m->mem = mem;
3476
1.78k
    m->rng = rng;
3477
1.78k
    m->ns = ns;
3478
3479
1.78k
    m->fr = friendreq_new(mem);
3480
3481
1.78k
    if (m->fr == nullptr) {
3482
1
        mem_delete(mem, m);
3483
1
        return nullptr;
3484
1
    }
3485
3486
1.78k
    m->log = logger_new(mem);
3487
3488
1.78k
    if (m->log == nullptr) {
3489
1
        friendreq_kill(m->fr);
3490
1
        mem_delete(mem, m);
3491
1
        return nullptr;
3492
1
    }
3493
3494
1.78k
    logger_callback_log(m->log, options->log_callback, options->log_context, options->log_user_data);
3495
3496
1.78k
    unsigned int net_err = 0;
3497
3498
1.78k
    if (!options->udp_disabled && options->proxy_info.proxy_type != TCP_PROXY_NONE) {
3499
        // We don't currently support UDP over proxy.
3500
10
        LOGGER_INFO(m->log, "UDP enabled and proxy set: disabling UDP");
3501
10
        options->udp_disabled = true;
3502
10
    }
3503
3504
1.78k
    if (options->udp_disabled) {
3505
10
        m->net = new_networking_no_udp(m->log, m->mem, m->ns);
3506
1.77k
    } else {
3507
1.77k
        IP ip;
3508
1.77k
        ip_init(&ip, options->ipv6enabled);
3509
1.77k
        m->net = new_networking_ex(m->log, m->mem, m->ns, &ip, options->port_range[0], options->port_range[1], &net_err);
3510
1.77k
    }
3511
3512
1.78k
    if (m->net == nullptr) {
3513
3
        friendreq_kill(m->fr);
3514
3515
3
        if (error != nullptr && net_err == 1) {
3516
0
            LOGGER_WARNING(m->log, "network initialisation failed (no ports available)");
3517
0
            *error = MESSENGER_ERROR_PORT;
3518
0
        }
3519
3520
3
        logger_kill(m->log);
3521
3
        mem_delete(mem, m);
3522
3
        return nullptr;
3523
3
    }
3524
3525
1.78k
    m->dht = new_dht(m->log, m->mem, m->rng, m->ns, m->mono_time, m->net, options->hole_punching_enabled, options->local_discovery_enabled);
3526
3527
1.78k
    if (m->dht == nullptr) {
3528
14
        kill_networking(m->net);
3529
14
        friendreq_kill(m->fr);
3530
14
        logger_kill(m->log);
3531
14
        mem_delete(mem, m);
3532
14
        return nullptr;
3533
14
    }
3534
3535
1.76k
    m->tcp_np = netprof_new(m->log, mem);
3536
3537
1.76k
    if (m->tcp_np == nullptr) {
3538
0
        LOGGER_WARNING(m->log, "TCP netprof initialisation failed");
3539
0
        kill_dht(m->dht);
3540
0
        kill_networking(m->net);
3541
0
        friendreq_kill(m->fr);
3542
0
        logger_kill(m->log);
3543
0
        mem_delete(mem, m);
3544
0
        return nullptr;
3545
0
    }
3546
3547
1.76k
    m->net_crypto = new_net_crypto(m->log, m->mem, m->rng, m->ns, m->mono_time, m->dht, &options->proxy_info, m->tcp_np);
3548
3549
1.76k
    if (m->net_crypto == nullptr) {
3550
0
        LOGGER_WARNING(m->log, "net_crypto initialisation failed");
3551
3552
0
        netprof_kill(mem, m->tcp_np);
3553
0
        kill_dht(m->dht);
3554
0
        kill_networking(m->net);
3555
0
        friendreq_kill(m->fr);
3556
0
        logger_kill(m->log);
3557
0
        mem_delete(mem, m);
3558
0
        return nullptr;
3559
0
    }
3560
3561
1.76k
    m->group_announce = new_gca_list(m->mem);
3562
3563
1.76k
    if (m->group_announce == nullptr) {
3564
0
        LOGGER_WARNING(m->log, "DHT group chats initialisation failed");
3565
3566
0
        kill_net_crypto(m->net_crypto);
3567
0
        netprof_kill(mem, m->tcp_np);
3568
0
        kill_dht(m->dht);
3569
0
        kill_networking(m->net);
3570
0
        friendreq_kill(m->fr);
3571
0
        logger_kill(m->log);
3572
0
        mem_delete(mem, m);
3573
0
        return nullptr;
3574
0
    }
3575
3576
1.76k
    if (options->dht_announcements_enabled) {
3577
1.76k
        m->forwarding = new_forwarding(m->log, m->mem, m->rng, m->mono_time, m->dht);
3578
1.76k
        if (m->forwarding != nullptr) {
3579
1.76k
            m->announce = new_announcements(m->log, m->mem, m->rng, m->mono_time, m->forwarding);
3580
1.76k
        } else {
3581
0
            m->announce = nullptr;
3582
0
        }
3583
1.76k
    } else {
3584
0
        m->forwarding = nullptr;
3585
0
        m->announce = nullptr;
3586
0
    }
3587
3588
1.76k
    m->onion = new_onion(m->log, m->mem, m->mono_time, m->rng, m->dht);
3589
1.76k
    m->onion_a = new_onion_announce(m->log, m->mem, m->rng, m->mono_time, m->dht);
3590
1.76k
    m->onion_c = new_onion_client(m->log, m->mem, m->rng, m->mono_time, m->net_crypto);
3591
1.76k
    if (m->onion_c != nullptr) {
3592
1.76k
        m->fr_c = new_friend_connections(m->log, m->mem, m->mono_time, m->ns, m->onion_c, options->local_discovery_enabled);
3593
1.76k
    }
3594
3595
1.76k
    if ((options->dht_announcements_enabled && (m->forwarding == nullptr || m->announce == nullptr)) ||
3596
1.76k
            m->onion == nullptr || m->onion_a == nullptr || m->onion_c == nullptr || m->fr_c == nullptr) {
3597
0
        LOGGER_WARNING(m->log, "onion initialisation failed");
3598
3599
0
        kill_onion(m->onion);
3600
0
        kill_onion_announce(m->onion_a);
3601
0
        kill_onion_client(m->onion_c);
3602
0
        kill_gca(m->group_announce);
3603
0
        kill_friend_connections(m->fr_c);
3604
0
        kill_announcements(m->announce);
3605
0
        kill_forwarding(m->forwarding);
3606
0
        kill_net_crypto(m->net_crypto);
3607
0
        netprof_kill(mem, m->tcp_np);
3608
0
        kill_dht(m->dht);
3609
0
        kill_networking(m->net);
3610
0
        friendreq_kill(m->fr);
3611
0
        logger_kill(m->log);
3612
0
        mem_delete(mem, m);
3613
0
        return nullptr;
3614
0
    }
3615
3616
1.76k
    gca_onion_init(m->group_announce, m->onion_a);
3617
3618
1.76k
    m->group_handler = new_dht_groupchats(m);
3619
3620
1.76k
    if (m->group_handler == nullptr) {
3621
0
        LOGGER_WARNING(m->log, "conferences initialisation failed");
3622
3623
0
        kill_onion(m->onion);
3624
0
        kill_onion_announce(m->onion_a);
3625
0
        kill_onion_client(m->onion_c);
3626
0
        kill_gca(m->group_announce);
3627
0
        kill_friend_connections(m->fr_c);
3628
0
        kill_announcements(m->announce);
3629
0
        kill_forwarding(m->forwarding);
3630
0
        kill_net_crypto(m->net_crypto);
3631
0
        netprof_kill(mem, m->tcp_np);
3632
0
        kill_dht(m->dht);
3633
0
        kill_networking(m->net);
3634
0
        friendreq_kill(m->fr);
3635
0
        logger_kill(m->log);
3636
0
        mem_delete(mem, m);
3637
0
        return nullptr;
3638
0
    }
3639
3640
1.76k
    if (options->tcp_server_port != 0) {
3641
0
        m->tcp_server = new_tcp_server(m->log, m->mem, m->rng, m->ns, options->ipv6enabled, 1,
3642
0
                                       &options->tcp_server_port, dht_get_self_secret_key(m->dht),
3643
0
                                       m->onion, m->forwarding);
3644
3645
0
        if (m->tcp_server == nullptr) {
3646
0
            LOGGER_WARNING(m->log, "TCP server initialisation failed");
3647
3648
0
            kill_onion(m->onion);
3649
0
            kill_onion_announce(m->onion_a);
3650
0
            kill_dht_groupchats(m->group_handler);
3651
0
            kill_friend_connections(m->fr_c);
3652
0
            kill_onion_client(m->onion_c);
3653
0
            kill_gca(m->group_announce);
3654
0
            kill_announcements(m->announce);
3655
0
            kill_forwarding(m->forwarding);
3656
0
            kill_net_crypto(m->net_crypto);
3657
0
            netprof_kill(mem, m->tcp_np);
3658
0
            kill_dht(m->dht);
3659
0
            kill_networking(m->net);
3660
0
            friendreq_kill(m->fr);
3661
0
            logger_kill(m->log);
3662
0
            mem_delete(mem, m);
3663
3664
0
            if (error != nullptr) {
3665
0
                *error = MESSENGER_ERROR_TCP_SERVER;
3666
0
            }
3667
3668
0
            return nullptr;
3669
0
        }
3670
0
    }
3671
3672
1.76k
    m->options = *options;
3673
1.76k
    friendreq_init(m->fr, m->fr_c);
3674
1.76k
    set_nospam(m->fr, random_u32(m->rng));
3675
1.76k
    set_filter_function(m->fr, &friend_already_added, m);
3676
3677
1.76k
    m->lastdump = 0;
3678
1.76k
    m->is_receiving_file = 0;
3679
3680
1.76k
    m_register_default_plugins(m);
3681
1.76k
    callback_friendrequest(m->fr, m_handle_friend_request, m);
3682
3683
1.76k
    if (error != nullptr) {
3684
1.76k
        *error = MESSENGER_ERROR_NONE;
3685
1.76k
    }
3686
3687
1.76k
    return m;
3688
1.76k
}
3689
3690
/** @brief Run this before closing shop.
3691
 *
3692
 * Free all datastructures.
3693
 */
3694
void kill_messenger(Messenger *m)
3695
1.76k
{
3696
1.76k
    if (m == nullptr) {
3697
0
        return;
3698
0
    }
3699
3700
1.76k
    if (m->tcp_server != nullptr) {
3701
0
        kill_tcp_server(m->tcp_server);
3702
0
    }
3703
3704
1.76k
    kill_onion(m->onion);
3705
1.76k
    kill_onion_announce(m->onion_a);
3706
1.76k
    kill_dht_groupchats(m->group_handler);
3707
1.76k
    kill_friend_connections(m->fr_c);
3708
1.76k
    kill_onion_client(m->onion_c);
3709
1.76k
    kill_gca(m->group_announce);
3710
1.76k
    kill_announcements(m->announce);
3711
1.76k
    kill_forwarding(m->forwarding);
3712
1.76k
    kill_net_crypto(m->net_crypto);
3713
1.76k
    netprof_kill(m->mem, m->tcp_np);
3714
1.76k
    kill_dht(m->dht);
3715
1.76k
    kill_networking(m->net);
3716
3717
2.01k
    for (uint32_t i = 0; i < m->numfriends; ++i) {
3718
247
        clear_receipts(m, i);
3719
247
    }
3720
3721
1.76k
    mem_delete(m->mem, m->friendlist);
3722
1.76k
    friendreq_kill(m->fr);
3723
3724
1.76k
    mem_delete(m->mem, m->options.state_plugins);
3725
1.76k
    logger_kill(m->log);
3726
1.76k
    mem_delete(m->mem, m);
3727
1.76k
}
3728
3729
bool m_is_receiving_file(Messenger *m)
3730
0
{
3731
    // Only run the expensive loop below once every 64 tox_iterate calls.
3732
0
    const uint8_t skip_count = 64;
3733
3734
0
    if (m->is_receiving_file != 0) {
3735
0
        --m->is_receiving_file;
3736
0
        return true;
3737
0
    }
3738
3739
    // TODO(iphydf): This is a very expensive loop. Consider keeping track of
3740
    // the number of live file transfers.
3741
0
    for (size_t friend_number = 0; friend_number < m->numfriends; ++friend_number) {
3742
0
        for (size_t i = 0; i < MAX_CONCURRENT_FILE_PIPES; ++i) {
3743
0
            if (m->friendlist[friend_number].file_receiving[i].status == FILESTATUS_TRANSFERRING) {
3744
0
                m->is_receiving_file = skip_count;
3745
0
                return true;
3746
0
            }
3747
0
        }
3748
0
    }
3749
3750
0
    return false;
3751
0
}