Coverage Report

Created: 2025-04-04 10:29

/src/c-toxcore/toxcore/group.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 © 2014 Tox project.
4
 */
5
6
/**
7
 * Slightly better groupchats implementation.
8
 */
9
#include "group.h"
10
11
#include <assert.h>
12
#include <string.h>
13
14
#include "DHT.h"
15
#include "Messenger.h"
16
#include "attributes.h"
17
#include "ccompat.h"
18
#include "crypto_core.h"
19
#include "friend_connection.h"
20
#include "group_common.h"
21
#include "logger.h"
22
#include "mem.h"
23
#include "mono_time.h"
24
#include "net_crypto.h"
25
#include "network.h"
26
#include "sort.h"
27
#include "state.h"
28
#include "util.h"
29
30
enum {
31
    /** Connection is to one of the closest DESIRED_CLOSEST peers */
32
    GROUPCHAT_CONNECTION_REASON_CLOSEST     = 1 << 0,
33
34
    /** Connection is to a peer we are introducing to the conference */
35
    GROUPCHAT_CONNECTION_REASON_INTRODUCING = 1 << 1,
36
37
    /** Connection is to a peer who is introducing us to the conference */
38
    GROUPCHAT_CONNECTION_REASON_INTRODUCER  = 1 << 2,
39
};
40
41
typedef enum Groupchat_Connection_Type {
42
    GROUPCHAT_CONNECTION_NONE,
43
    GROUPCHAT_CONNECTION_CONNECTING,
44
    GROUPCHAT_CONNECTION_ONLINE,
45
} Groupchat_Connection_Type;
46
47
typedef enum Groupchat_Status {
48
    GROUPCHAT_STATUS_NONE,
49
    GROUPCHAT_STATUS_VALID,
50
    GROUPCHAT_STATUS_CONNECTED,
51
} Groupchat_Status;
52
53
14.5k
#define GROUP_ID_LENGTH CRYPTO_SYMMETRIC_KEY_SIZE
54
55
54
#define DESIRED_CLOSEST 4
56
40.9k
#define MAX_GROUP_CONNECTIONS 16
57
0
#define MAX_LAST_MESSAGE_INFOS 8
58
0
#define MAX_LOSSY_COUNT 256
59
60
/** Maximum number of frozen peers to store; `group_set_max_frozen()` overrides. */
61
5.87k
#define MAX_FROZEN_DEFAULT 128
62
63
typedef struct Message_Info {
64
    uint32_t message_number;
65
    uint8_t  message_id;
66
} Message_Info;
67
68
typedef struct Group_Peer {
69
    uint8_t     real_pk[CRYPTO_PUBLIC_KEY_SIZE];
70
    uint8_t     temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
71
    bool        temp_pk_updated;
72
    bool        is_friend;
73
74
    uint64_t    last_active;
75
76
    Message_Info
77
    last_message_infos[MAX_LAST_MESSAGE_INFOS]; /* received messages, strictly decreasing in message_number */
78
    uint8_t     num_last_message_infos;
79
80
    uint8_t     nick[MAX_NAME_LENGTH];
81
    uint8_t     nick_len;
82
    bool        nick_updated;
83
84
    uint16_t peer_number;
85
86
    uint8_t  recv_lossy[MAX_LOSSY_COUNT];
87
    uint16_t bottom_lossy_number;
88
    uint16_t top_lossy_number;
89
90
    void *object;
91
} Group_Peer;
92
93
typedef struct Groupchat_Connection {
94
    uint8_t type; /* `GROUPCHAT_CONNECTION_*` */
95
    uint8_t reasons; /* bit field with flags `GROUPCHAT_CONNECTION_REASON_*` */
96
    uint32_t number;
97
    uint16_t group_number;
98
} Groupchat_Connection;
99
100
typedef struct Groupchat_Closest {
101
    /**
102
     * Whether this peer is active in the closest_peers array.
103
     */
104
    bool active;
105
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
106
    uint8_t temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
107
} Groupchat_Closest;
108
109
typedef struct Group_c {
110
    uint8_t status;
111
112
    bool need_send_name;
113
    bool title_fresh;
114
115
    Group_Peer *group;
116
    uint32_t numpeers;
117
118
    Group_Peer *frozen;
119
    uint32_t numfrozen;
120
121
    uint32_t maxfrozen;
122
123
    Groupchat_Connection connections[MAX_GROUP_CONNECTIONS];
124
125
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
126
    Groupchat_Closest closest_peers[DESIRED_CLOSEST];
127
    uint8_t changed;
128
129
    uint8_t type;
130
    uint8_t id[GROUP_ID_LENGTH];
131
132
    uint8_t title[MAX_NAME_LENGTH];
133
    uint8_t title_len;
134
135
    uint32_t message_number;
136
    uint16_t lossy_message_number;
137
    uint16_t peer_number;
138
139
    uint64_t last_sent_ping;
140
141
    uint32_t num_introducer_connections;
142
143
    void *object;
144
145
    peer_on_join_cb *peer_on_join;
146
    peer_on_leave_cb *peer_on_leave;
147
    group_on_delete_cb *group_on_delete;
148
} Group_c;
149
150
struct Group_Chats {
151
    const Memory *mem;
152
    const Mono_Time *mono_time;
153
154
    Messenger *m;
155
    Friend_Connections *fr_c;
156
157
    Group_c *chats;
158
    uint16_t num_chats;
159
160
    g_conference_invite_cb *invite_callback;
161
    g_conference_connected_cb *connected_callback;
162
    g_conference_message_cb *message_callback;
163
    peer_name_cb *peer_name_callback;
164
    peer_list_changed_cb *peer_list_changed_callback;
165
    title_cb *title_callback;
166
167
    lossy_packet_cb *lossy_packethandlers[256];
168
};
169
170
static const Group_c empty_group_c = {0};
171
static const Group_Peer empty_group_peer = {{0}};
172
173
/**
174
 * Packet type IDs as per the protocol specification.
175
 */
176
typedef enum Group_Message_Id {
177
    GROUP_MESSAGE_PING_ID        = 0,
178
    GROUP_MESSAGE_NEW_PEER_ID    = 16,
179
    GROUP_MESSAGE_KILL_PEER_ID   = 17,
180
    GROUP_MESSAGE_FREEZE_PEER_ID = 18,
181
    GROUP_MESSAGE_NAME_ID        = 48,
182
    GROUP_MESSAGE_TITLE_ID       = 49,
183
} Group_Message_Id;
184
185
0
#define GROUP_MESSAGE_NEW_PEER_LENGTH (sizeof(uint16_t) + CRYPTO_PUBLIC_KEY_SIZE * 2)
186
0
#define GROUP_MESSAGE_KILL_PEER_LENGTH (sizeof(uint16_t))
187
188
1.20k
#define MAX_GROUP_MESSAGE_DATA_LEN (MAX_CRYPTO_DATA_SIZE - (1 + MIN_MESSAGE_PACKET_LEN))
189
190
typedef enum Invite_Id {
191
    INVITE_ID             = 0,
192
    INVITE_ACCEPT_ID      = 1,
193
    INVITE_MEMBER_ID      = 2,
194
} Invite_Id;
195
196
0
#define INVITE_PACKET_SIZE (1 + sizeof(uint16_t) + 1 + GROUP_ID_LENGTH)
197
0
#define INVITE_ACCEPT_PACKET_SIZE (1 + sizeof(uint16_t) * 2 + 1 + GROUP_ID_LENGTH)
198
0
#define INVITE_MEMBER_PACKET_SIZE (1 + sizeof(uint16_t) * 2 + 1 + GROUP_ID_LENGTH + sizeof(uint16_t))
199
200
0
#define ONLINE_PACKET_DATA_SIZE (sizeof(uint16_t) + 1 + GROUP_ID_LENGTH)
201
202
typedef enum Peer_Id {
203
    PEER_INTRODUCED_ID  = 1,
204
    PEER_QUERY_ID       = 8,
205
    PEER_RESPONSE_ID    = 9,
206
    PEER_TITLE_ID       = 10,
207
} Peer_Id;
208
209
1.20k
#define MIN_MESSAGE_PACKET_LEN (sizeof(uint16_t) * 2 + sizeof(uint32_t) + 1)
210
211
static_assert(GROUP_ID_LENGTH == CRYPTO_PUBLIC_KEY_SIZE,
212
              "GROUP_ID_LENGTH should be equal to CRYPTO_PUBLIC_KEY_SIZE");
213
214
const Mono_Time *g_mono_time(const Group_Chats *g_c)
215
0
{
216
0
    return g_c->mono_time;
217
0
}
218
219
non_null()
220
static bool group_id_eq(const uint8_t *a, const uint8_t *b)
221
0
{
222
0
    return pk_equal(a, b);
223
0
}
224
225
non_null()
226
static bool g_title_eq(const Group_c *g, const uint8_t *title, uint8_t title_len)
227
0
{
228
0
    return memeq(g->title, g->title_len, title, title_len);
229
0
}
230
231
non_null()
232
static bool g_peer_nick_eq(const Group_Peer *peer, const uint8_t *nick, uint8_t nick_len)
233
2.88k
{
234
2.88k
    return memeq(peer->nick, peer->nick_len, nick, nick_len);
235
2.88k
}
236
237
/**
238
 * @retval false if the groupnumber is not valid.
239
 * @retval true if the groupnumber is valid.
240
 */
241
non_null()
242
static bool is_groupnumber_valid(const Group_Chats *g_c, uint32_t groupnumber)
243
19.4k
{
244
19.4k
    return groupnumber < g_c->num_chats
245
19.4k
           && g_c->chats != nullptr
246
19.4k
           && g_c->chats[groupnumber].status != GROUPCHAT_STATUS_NONE;
247
19.4k
}
248
249
/** @brief Set the size of the groupchat list to num.
250
 *
251
 * @retval false if mem_vrealloc fails.
252
 * @retval true if it succeeds.
253
 */
254
non_null()
255
static bool realloc_conferences(Group_Chats *g_c, uint16_t num)
256
3.06k
{
257
3.06k
    if (num == 0) {
258
98
        mem_delete(g_c->mem, g_c->chats);
259
98
        g_c->chats = nullptr;
260
98
        return true;
261
98
    }
262
263
2.97k
    Group_c *newgroup_chats = (Group_c *)mem_vrealloc(g_c->mem, g_c->chats, num, sizeof(Group_c));
264
265
2.97k
    if (newgroup_chats == nullptr) {
266
0
        return false;
267
0
    }
268
269
2.97k
    g_c->chats = newgroup_chats;
270
2.97k
    return true;
271
2.97k
}
272
273
non_null()
274
static void setup_conference(Group_c *g)
275
5.87k
{
276
5.87k
    *g = empty_group_c;
277
5.87k
    g->maxfrozen = MAX_FROZEN_DEFAULT;
278
5.87k
}
279
280
/** @brief Create a new empty groupchat connection.
281
 *
282
 * @retval -1 on failure.
283
 * @return groupnumber on success.
284
 */
285
non_null()
286
static int32_t create_group_chat(Group_Chats *g_c)
287
2.93k
{
288
141k
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
289
138k
        if (g_c->chats[i].status == GROUPCHAT_STATUS_NONE) {
290
0
            return i;
291
0
        }
292
138k
    }
293
294
2.93k
    if (realloc_conferences(g_c, g_c->num_chats + 1)) {
295
2.93k
        const uint16_t id = g_c->num_chats;
296
2.93k
        ++g_c->num_chats;
297
2.93k
        setup_conference(&g_c->chats[id]);
298
2.93k
        return id;
299
2.93k
    }
300
301
0
    return -1;
302
2.93k
}
303
304
non_null()
305
static void wipe_group_c(const Memory *mem, Group_c *g)
306
2.93k
{
307
2.93k
    mem_delete(mem, g->frozen);
308
2.93k
    mem_delete(mem, g->group);
309
2.93k
    crypto_memzero(g, sizeof(Group_c));
310
2.93k
}
311
312
/** @brief Wipe a groupchat.
313
 *
314
 * @retval true on success.
315
 */
316
non_null()
317
static bool wipe_group_chat(Group_Chats *g_c, uint32_t groupnumber)
318
2.93k
{
319
2.93k
    if (groupnumber >= g_c->num_chats || g_c->chats == nullptr) {
320
0
        return false;
321
0
    }
322
323
2.93k
    wipe_group_c(g_c->mem, &g_c->chats[groupnumber]);
324
325
2.93k
    uint16_t i;
326
327
5.87k
    for (i = g_c->num_chats; i != 0; --i) {
328
5.77k
        if (g_c->chats[i - 1].status != GROUPCHAT_STATUS_NONE) {
329
2.83k
            break;
330
2.83k
        }
331
5.77k
    }
332
333
2.93k
    if (g_c->num_chats != i) {
334
131
        g_c->num_chats = i;
335
131
        realloc_conferences(g_c, g_c->num_chats);
336
131
    }
337
338
2.93k
    return true;
339
2.93k
}
340
341
non_null()
342
static Group_c *get_group_c(const Group_Chats *g_c, uint32_t groupnumber)
343
19.4k
{
344
19.4k
    if (!is_groupnumber_valid(g_c, groupnumber)) {
345
0
        return nullptr;
346
0
    }
347
348
19.4k
    return &g_c->chats[groupnumber];
349
19.4k
}
350
351
/**
352
 * check if peer with real_pk is in peer array.
353
 *
354
 * @return peer index if peer is in group.
355
 * @retval -1 if peer is not in group.
356
 *
357
 * TODO(irungentoo): make this more efficient.
358
 */
359
non_null()
360
static int peer_in_group(const Group_c *g, const uint8_t *real_pk)
361
2.85k
{
362
2.85k
    for (uint32_t i = 0; i < g->numpeers; ++i) {
363
0
        if (pk_equal(g->group[i].real_pk, real_pk)) {
364
0
            return i;
365
0
        }
366
0
    }
367
368
2.85k
    return -1;
369
2.85k
}
370
371
non_null()
372
static int frozen_in_group(const Group_c *g, const uint8_t *real_pk)
373
2.85k
{
374
2.99k
    for (uint32_t i = 0; i < g->numfrozen; ++i) {
375
171
        if (pk_equal(g->frozen[i].real_pk, real_pk)) {
376
31
            return i;
377
31
        }
378
171
    }
379
380
2.81k
    return -1;
381
2.85k
}
382
383
/**
384
 * check if group with the given type and id is in group array.
385
 *
386
 * @return group number if peer is in list.
387
 * @retval -1 if group is not in list.
388
 *
389
 * TODO(irungentoo): make this more efficient and maybe use constant time comparisons?
390
 */
391
non_null()
392
static int32_t get_group_num(const Group_Chats *g_c, const uint8_t type, const uint8_t *id)
393
0
{
394
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
395
0
        if (g_c->chats[i].type == type && group_id_eq(g_c->chats[i].id, id)) {
396
0
            return i;
397
0
        }
398
0
    }
399
400
0
    return -1;
401
0
}
402
403
int32_t conference_by_id(const Group_Chats *g_c, const uint8_t *id)
404
0
{
405
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
406
0
        if (group_id_eq(g_c->chats[i].id, id)) {
407
0
            return i;
408
0
        }
409
0
    }
410
411
0
    return -1;
412
0
}
413
414
/**
415
 * check if peer with peer_number is in peer array.
416
 *
417
 * @return peer index if peer is in chat.
418
 * @retval -1 if peer is not in chat.
419
 *
420
 * TODO(irungentoo): make this more efficient.
421
 */
422
non_null()
423
static int get_peer_index(const Group_c *g, uint16_t peer_number)
424
2.89k
{
425
2.89k
    for (uint32_t i = 0; i < g->numpeers; ++i) {
426
0
        if (g->group[i].peer_number == peer_number) {
427
0
            return i;
428
0
        }
429
0
    }
430
431
2.89k
    return -1;
432
2.89k
}
433
434
non_null()
435
static uint64_t calculate_comp_value(const uint8_t *pk1, const uint8_t *pk2)
436
0
{
437
0
    uint64_t cmp1 = 0;
438
0
    uint64_t cmp2 = 0;
439
440
0
    for (size_t i = 0; i < sizeof(uint64_t); ++i) {
441
0
        cmp1 = (cmp1 << 8) + (uint64_t)pk1[i];
442
0
        cmp2 = (cmp2 << 8) + (uint64_t)pk2[i];
443
0
    }
444
445
0
    return cmp1 - cmp2;
446
0
}
447
448
typedef enum Groupchat_Closest_Change {
449
    GROUPCHAT_CLOSEST_CHANGE_NONE,
450
    GROUPCHAT_CLOSEST_CHANGE_ADDED,
451
    GROUPCHAT_CLOSEST_CHANGE_REMOVED,
452
} Groupchat_Closest_Change;
453
454
non_null()
455
static bool add_to_closest(Group_c *g, const uint8_t *real_pk, const uint8_t *temp_pk)
456
2.89k
{
457
2.89k
    if (pk_equal(g->real_pk, real_pk)) {
458
2.88k
        return false;
459
2.88k
    }
460
461
6
    unsigned int index = DESIRED_CLOSEST;
462
463
30
    for (unsigned int i = 0; i < DESIRED_CLOSEST; ++i) {
464
24
        if (g->closest_peers[i].active && pk_equal(real_pk, g->closest_peers[i].real_pk)) {
465
0
            return true;
466
0
        }
467
24
    }
468
469
6
    for (unsigned int i = 0; i < DESIRED_CLOSEST; ++i) {
470
6
        if (!g->closest_peers[i].active) {
471
6
            index = i;
472
6
            break;
473
6
        }
474
6
    }
475
476
6
    if (index == DESIRED_CLOSEST) {
477
0
        uint64_t comp_val = calculate_comp_value(g->real_pk, real_pk);
478
0
        uint64_t comp_d = 0;
479
480
0
        for (unsigned int i = 0; i < (DESIRED_CLOSEST / 2); ++i) {
481
0
            const uint64_t comp = calculate_comp_value(g->real_pk, g->closest_peers[i].real_pk);
482
483
0
            if (comp > comp_val && comp > comp_d) {
484
0
                index = i;
485
0
                comp_d = comp;
486
0
            }
487
0
        }
488
489
0
        comp_val = calculate_comp_value(real_pk, g->real_pk);
490
491
0
        for (unsigned int i = DESIRED_CLOSEST / 2; i < DESIRED_CLOSEST; ++i) {
492
0
            const uint64_t comp = calculate_comp_value(g->closest_peers[i].real_pk, g->real_pk);
493
494
0
            if (comp > comp_val && comp > comp_d) {
495
0
                index = i;
496
0
                comp_d = comp;
497
0
            }
498
0
        }
499
0
    }
500
501
6
    if (index == DESIRED_CLOSEST) {
502
0
        return false;
503
0
    }
504
505
6
    uint8_t old_real_pk[CRYPTO_PUBLIC_KEY_SIZE];
506
6
    uint8_t old_temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
507
6
    bool old = false;
508
509
6
    if (g->closest_peers[index].active) {
510
0
        memcpy(old_real_pk, g->closest_peers[index].real_pk, CRYPTO_PUBLIC_KEY_SIZE);
511
0
        memcpy(old_temp_pk, g->closest_peers[index].temp_pk, CRYPTO_PUBLIC_KEY_SIZE);
512
0
        old = true;
513
0
    }
514
515
6
    g->closest_peers[index].active = true;
516
6
    memcpy(g->closest_peers[index].real_pk, real_pk, CRYPTO_PUBLIC_KEY_SIZE);
517
6
    memcpy(g->closest_peers[index].temp_pk, temp_pk, CRYPTO_PUBLIC_KEY_SIZE);
518
519
6
    if (old) {
520
0
        add_to_closest(g, old_real_pk, old_temp_pk);
521
0
    }
522
523
6
    if (g->changed == GROUPCHAT_CLOSEST_CHANGE_NONE) {
524
6
        g->changed = GROUPCHAT_CLOSEST_CHANGE_ADDED;
525
6
    }
526
527
6
    return true;
528
6
}
529
530
non_null()
531
static bool pk_in_closest_peers(const Group_c *g, const uint8_t *real_pk)
532
0
{
533
0
    for (unsigned int i = 0; i < DESIRED_CLOSEST; ++i) {
534
0
        if (!g->closest_peers[i].active) {
535
0
            continue;
536
0
        }
537
538
0
        if (pk_equal(g->closest_peers[i].real_pk, real_pk)) {
539
0
            return true;
540
0
        }
541
0
    }
542
543
0
    return false;
544
0
}
545
546
non_null()
547
static void remove_connection_reason(Group_Chats *g_c, Group_c *g, uint16_t i, uint8_t reason);
548
549
non_null()
550
static void purge_closest(Group_Chats *g_c, uint32_t groupnumber)
551
0
{
552
0
    Group_c *g = get_group_c(g_c, groupnumber);
553
554
0
    if (g == nullptr) {
555
0
        return;
556
0
    }
557
558
0
    for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
559
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
560
0
            continue;
561
0
        }
562
563
0
        if ((g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_CLOSEST) == 0) {
564
0
            continue;
565
0
        }
566
567
0
        uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
568
0
        get_friendcon_public_keys(real_pk, nullptr, g_c->fr_c, g->connections[i].number);
569
570
0
        if (!pk_in_closest_peers(g, real_pk)) {
571
0
            remove_connection_reason(g_c, g, i, GROUPCHAT_CONNECTION_REASON_CLOSEST);
572
0
        }
573
0
    }
574
0
}
575
576
non_null()
577
static bool send_packet_online(const Friend_Connections *fr_c, int friendcon_id, uint16_t group_num,
578
                               uint8_t type, const uint8_t *id);
579
580
non_null()
581
static int add_conn_to_groupchat(Group_Chats *g_c, int friendcon_id, Group_c *g, uint8_t reason,
582
                                 bool lock);
583
584
non_null(1) nullable(3)
585
static void add_closest_connections(Group_Chats *g_c, uint32_t groupnumber, void *userdata)
586
0
{
587
0
    Group_c *g = get_group_c(g_c, groupnumber);
588
589
0
    if (g == nullptr) {
590
0
        return;
591
0
    }
592
593
0
    for (uint32_t i = 0; i < DESIRED_CLOSEST; ++i) {
594
0
        if (!g->closest_peers[i].active) {
595
0
            continue;
596
0
        }
597
598
0
        int friendcon_id = getfriend_conn_id_pk(g_c->fr_c, g->closest_peers[i].real_pk);
599
600
0
        bool fresh = false;
601
602
0
        if (friendcon_id == -1) {
603
0
            friendcon_id = new_friend_connection(g_c->fr_c, g->closest_peers[i].real_pk);
604
0
            fresh = true;
605
606
0
            if (friendcon_id == -1) {
607
0
                continue;
608
0
            }
609
610
0
            set_dht_temp_pk(g_c->fr_c, friendcon_id, g->closest_peers[i].temp_pk, userdata);
611
0
        }
612
613
0
        const int connection_index = add_conn_to_groupchat(g_c, friendcon_id, g,
614
0
                                     GROUPCHAT_CONNECTION_REASON_CLOSEST, !fresh);
615
616
0
        if (connection_index == -1) {
617
0
            if (fresh) {
618
0
                kill_friend_connection(g_c->fr_c, friendcon_id);
619
0
            }
620
621
0
            continue;
622
0
        }
623
624
0
        if (friend_con_connected(g_c->fr_c, friendcon_id) == FRIENDCONN_STATUS_CONNECTED
625
0
                && g->connections[connection_index].type == GROUPCHAT_CONNECTION_CONNECTING) {
626
0
            send_packet_online(g_c->fr_c, friendcon_id, groupnumber, g->type, g->id);
627
0
        }
628
0
    }
629
0
}
630
631
non_null(1) nullable(3)
632
static bool connect_to_closest(Group_Chats *g_c, uint32_t groupnumber, void *userdata)
633
0
{
634
0
    Group_c *g = get_group_c(g_c, groupnumber);
635
636
0
    if (g == nullptr) {
637
0
        return false;
638
0
    }
639
640
0
    if (g->changed == GROUPCHAT_CLOSEST_CHANGE_NONE) {
641
0
        return true;
642
0
    }
643
644
0
    if (g->changed == GROUPCHAT_CLOSEST_CHANGE_REMOVED) {
645
0
        for (uint32_t i = 0; i < g->numpeers; ++i) {
646
0
            add_to_closest(g, g->group[i].real_pk, g->group[i].temp_pk);
647
0
        }
648
0
    }
649
650
0
    purge_closest(g_c, groupnumber);
651
652
0
    add_closest_connections(g_c, groupnumber, userdata);
653
654
0
    g->changed = GROUPCHAT_CLOSEST_CHANGE_NONE;
655
656
0
    return true;
657
0
}
658
659
non_null()
660
static int get_frozen_index(const Group_c *g, uint16_t peer_number)
661
2.89k
{
662
3.16k
    for (uint32_t i = 0; i < g->numfrozen; ++i) {
663
313
        if (g->frozen[i].peer_number == peer_number) {
664
40
            return i;
665
40
        }
666
313
    }
667
668
2.85k
    return -1;
669
2.89k
}
670
671
non_null()
672
static bool delete_frozen(const Memory *mem, Group_c *g, uint32_t frozen_index)
673
71
{
674
71
    if (frozen_index >= g->numfrozen) {
675
0
        return false;
676
0
    }
677
678
71
    --g->numfrozen;
679
680
71
    if (g->numfrozen == 0) {
681
56
        mem_delete(mem, g->frozen);
682
56
        g->frozen = nullptr;
683
56
    } else {
684
15
        if (g->numfrozen != frozen_index) {
685
12
            g->frozen[frozen_index] = g->frozen[g->numfrozen];
686
12
        }
687
688
15
        Group_Peer *const frozen_temp = (Group_Peer *)mem_vrealloc(mem, g->frozen, g->numfrozen, sizeof(Group_Peer));
689
690
15
        if (frozen_temp == nullptr) {
691
0
            return false;
692
0
        }
693
694
15
        g->frozen = frozen_temp;
695
15
    }
696
697
71
    return true;
698
71
}
699
700
/** @brief Update last_active timestamp on peer, and thaw the peer if it is frozen.
701
 *
702
 * @return peer index if peer is in the conference.
703
 * @retval -1 otherwise, and on error.
704
 */
705
non_null(1) nullable(4)
706
static int note_peer_active(Group_Chats *g_c, uint32_t groupnumber, uint16_t peer_number, void *userdata)
707
2.89k
{
708
2.89k
    Group_c *g = get_group_c(g_c, groupnumber);
709
710
2.89k
    if (g == nullptr) {
711
0
        return -1;
712
0
    }
713
714
2.89k
    const int peer_index = get_peer_index(g, peer_number);
715
716
2.89k
    if (peer_index != -1) {
717
0
        g->group[peer_index].last_active = mono_time_get(g_c->mono_time);
718
0
        return peer_index;
719
0
    }
720
721
2.89k
    const int frozen_index = get_frozen_index(g, peer_number);
722
723
2.89k
    if (frozen_index == -1) {
724
2.85k
        return -1;
725
2.85k
    }
726
727
    /* Now thaw the peer */
728
729
40
    Group_Peer *temp = (Group_Peer *)mem_vrealloc(g_c->mem, g->group, g->numpeers + 1, sizeof(Group_Peer));
730
731
40
    if (temp == nullptr) {
732
0
        return -1;
733
0
    }
734
735
40
    const uint32_t thawed_index = g->numpeers;
736
737
40
    g->group = temp;
738
40
    g->group[thawed_index] = g->frozen[frozen_index];
739
40
    g->group[thawed_index].temp_pk_updated = false;
740
40
    g->group[thawed_index].last_active = mono_time_get(g_c->mono_time);
741
742
40
    add_to_closest(g, g->group[thawed_index].real_pk, g->group[thawed_index].temp_pk);
743
744
40
    ++g->numpeers;
745
746
40
    delete_frozen(g_c->mem, g, frozen_index);
747
748
40
    if (g_c->peer_list_changed_callback != nullptr) {
749
0
        g_c->peer_list_changed_callback(g_c->m, groupnumber, userdata);
750
0
    }
751
752
40
    if (g->peer_on_join != nullptr) {
753
0
        g->peer_on_join(g->object, groupnumber, thawed_index);
754
0
    }
755
756
40
    g->need_send_name = true;
757
758
40
    return thawed_index;
759
40
}
760
761
non_null(1) nullable(4)
762
static bool delpeer(Group_Chats *g_c, uint32_t groupnumber, int peer_index, void *userdata);
763
764
non_null(1, 3) nullable(4)
765
static void delete_any_peer_with_pk(Group_Chats *g_c, uint32_t groupnumber, const uint8_t *real_pk, void *userdata)
766
2.85k
{
767
2.85k
    Group_c *g = get_group_c(g_c, groupnumber);
768
769
2.85k
    if (g == nullptr) {
770
0
        return;
771
0
    }
772
773
2.85k
    const int peer_index = peer_in_group(g, real_pk);
774
775
2.85k
    if (peer_index >= 0) {
776
0
        delpeer(g_c, groupnumber, peer_index, userdata);
777
0
    }
778
779
2.85k
    const int frozen_index = frozen_in_group(g, real_pk);
780
781
2.85k
    if (frozen_index >= 0) {
782
31
        delete_frozen(g_c->mem, g, frozen_index);
783
31
    }
784
2.85k
}
785
786
/** @brief Add a peer to the group chat, or update an existing peer.
787
 *
788
 * fresh indicates whether we should consider this information on the peer to
789
 * be current, and so should update temp_pk and consider the peer active.
790
 *
791
 * do_gc_callback indicates whether we want to trigger callbacks set by the client
792
 * via the public API. This should be set to false if this function is called
793
 * from outside of the `tox_iterate()` loop.
794
 *
795
 * @return peer_index if success or peer already in chat.
796
 * @retval -1 if error.
797
 */
798
non_null(1, 3, 4) nullable(6)
799
static int addpeer(Group_Chats *g_c, uint32_t groupnumber, const uint8_t *real_pk, const uint8_t *temp_pk,
800
                   uint16_t peer_number, void *userdata, bool fresh, bool do_gc_callback)
801
2.89k
{
802
2.89k
    Group_c *g = get_group_c(g_c, groupnumber);
803
804
2.89k
    if (g == nullptr) {
805
0
        return -1;
806
0
    }
807
808
2.89k
    const int peer_index = fresh ?
809
2.89k
                           note_peer_active(g_c, groupnumber, peer_number, userdata) :
810
2.89k
                           get_peer_index(g, peer_number);
811
812
2.89k
    if (peer_index != -1) {
813
40
        if (!pk_equal(g->group[peer_index].real_pk, real_pk)) {
814
6
            LOGGER_ERROR(g_c->m->log, "peer public key is incorrect for peer %d", peer_number);
815
6
            return -1;
816
6
        }
817
818
34
        if (fresh || !g->group[peer_index].temp_pk_updated) {
819
34
            pk_copy(g->group[peer_index].temp_pk, temp_pk);
820
34
            g->group[peer_index].temp_pk_updated = true;
821
34
        }
822
823
34
        return peer_index;
824
40
    }
825
826
2.85k
    if (!fresh) {
827
0
        const int frozen_index = get_frozen_index(g, peer_number);
828
829
0
        if (frozen_index != -1) {
830
0
            if (!pk_equal(g->frozen[frozen_index].real_pk, real_pk)) {
831
0
                return -1;
832
0
            }
833
834
0
            pk_copy(g->frozen[frozen_index].temp_pk, temp_pk);
835
836
0
            return -1;
837
0
        }
838
0
    }
839
840
2.85k
    delete_any_peer_with_pk(g_c, groupnumber, real_pk, userdata);
841
842
2.85k
    Group_Peer *temp = (Group_Peer *)mem_vrealloc(g_c->mem, g->group, g->numpeers + 1, sizeof(Group_Peer));
843
844
2.85k
    if (temp == nullptr) {
845
0
        return -1;
846
0
    }
847
848
2.85k
    temp[g->numpeers] = empty_group_peer;
849
2.85k
    g->group = temp;
850
851
2.85k
    const uint32_t new_index = g->numpeers;
852
853
2.85k
    pk_copy(g->group[new_index].real_pk, real_pk);
854
2.85k
    pk_copy(g->group[new_index].temp_pk, temp_pk);
855
2.85k
    g->group[new_index].temp_pk_updated = true;
856
2.85k
    g->group[new_index].peer_number = peer_number;
857
2.85k
    g->group[new_index].last_active = mono_time_get(g_c->mono_time);
858
2.85k
    g->group[new_index].is_friend = getfriend_id(g_c->m, real_pk) != -1;
859
2.85k
    ++g->numpeers;
860
861
2.85k
    add_to_closest(g, real_pk, temp_pk);
862
863
2.85k
    if (do_gc_callback && g_c->peer_list_changed_callback != nullptr) {
864
0
        g_c->peer_list_changed_callback(g_c->m, groupnumber, userdata);
865
0
    }
866
867
2.85k
    if (g->peer_on_join != nullptr) {
868
0
        g->peer_on_join(g->object, groupnumber, new_index);
869
0
    }
870
871
2.85k
    return new_index;
872
2.85k
}
873
874
non_null()
875
static void remove_connection(Group_Chats *g_c, Group_c *g, uint16_t i)
876
0
{
877
0
    if ((g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_INTRODUCER) != 0) {
878
0
        --g->num_introducer_connections;
879
0
    }
880
881
0
    kill_friend_connection(g_c->fr_c, g->connections[i].number);
882
0
    g->connections[i].type = GROUPCHAT_CONNECTION_NONE;
883
0
}
884
885
non_null()
886
static void remove_from_closest(Group_c *g, int peer_index)
887
0
{
888
0
    for (uint32_t i = 0; i < DESIRED_CLOSEST; ++i) {
889
0
        if (g->closest_peers[i].active
890
0
                && pk_equal(g->closest_peers[i].real_pk, g->group[peer_index].real_pk)) {
891
0
            g->closest_peers[i].active = false;
892
0
            g->changed = GROUPCHAT_CLOSEST_CHANGE_REMOVED;
893
0
            break;
894
0
        }
895
0
    }
896
0
}
897
898
/**
899
 * Delete a peer from the group chat.
900
 *
901
 * return true on success
902
 */
903
static bool delpeer(Group_Chats *g_c, uint32_t groupnumber, int peer_index, void *userdata)
904
0
{
905
0
    Group_c *g = get_group_c(g_c, groupnumber);
906
907
0
    if (g == nullptr) {
908
0
        return false;
909
0
    }
910
911
0
    remove_from_closest(g, peer_index);
912
913
0
    const int friendcon_id = getfriend_conn_id_pk(g_c->fr_c, g->group[peer_index].real_pk);
914
915
0
    if (friendcon_id != -1) {
916
0
        for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
917
0
            if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
918
0
                continue;
919
0
            }
920
921
0
            if (g->connections[i].number == (unsigned int)friendcon_id) {
922
0
                remove_connection(g_c, g, i);
923
0
                break;
924
0
            }
925
0
        }
926
0
    }
927
928
0
    --g->numpeers;
929
930
0
    void *peer_object = g->group[peer_index].object;
931
932
0
    if (g->numpeers == 0) {
933
0
        mem_delete(g_c->mem, g->group);
934
0
        g->group = nullptr;
935
0
    } else {
936
0
        if (g->numpeers != (uint32_t)peer_index) {
937
0
            g->group[peer_index] = g->group[g->numpeers];
938
0
        }
939
940
0
        Group_Peer *temp = (Group_Peer *)mem_vrealloc(g_c->mem, g->group, g->numpeers, sizeof(Group_Peer));
941
942
0
        if (temp == nullptr) {
943
0
            return false;
944
0
        }
945
946
0
        g->group = temp;
947
0
    }
948
949
0
    if (g_c->peer_list_changed_callback != nullptr) {
950
0
        g_c->peer_list_changed_callback(g_c->m, groupnumber, userdata);
951
0
    }
952
953
0
    if (g->peer_on_leave != nullptr) {
954
0
        g->peer_on_leave(g->object, groupnumber, peer_object);
955
0
    }
956
957
0
    return true;
958
0
}
959
960
/** Order peers with friends first and with more recently active earlier */
961
non_null()
962
static bool group_peer_less_handler(const void *object, const void *a, const void *b)
963
0
{
964
0
    const Group_Peer *pa = (const Group_Peer *)a;
965
0
    const Group_Peer *pb = (const Group_Peer *)b;
966
967
0
    if (((pa->is_friend ? 1 : 0) ^ (pb->is_friend ? 1 : 0)) != 0) {
968
0
        return pa->is_friend;
969
0
    }
970
971
0
    return cmp_uint(pb->last_active, pa->last_active) < 0;
972
0
}
973
974
non_null()
975
static const void *group_peer_get_handler(const void *arr, uint32_t index)
976
0
{
977
0
    const Group_Peer *entries = (const Group_Peer *)arr;
978
0
    return &entries[index];
979
0
}
980
981
non_null()
982
static void group_peer_set_handler(void *arr, uint32_t index, const void *val)
983
0
{
984
0
    Group_Peer *entries = (Group_Peer *)arr;
985
0
    const Group_Peer *entry = (const Group_Peer *)val;
986
0
    entries[index] = *entry;
987
0
}
988
989
non_null()
990
static void *group_peer_subarr_handler(void *arr, uint32_t index, uint32_t size)
991
0
{
992
0
    Group_Peer *entries = (Group_Peer *)arr;
993
0
    return &entries[index];
994
0
}
995
996
non_null()
997
static void *group_peer_alloc_handler(const void *object, uint32_t size)
998
0
{
999
0
    const Memory *mem = (const Memory *)object;
1000
0
    Group_Peer *tmp = (Group_Peer *)mem_valloc(mem, size, sizeof(Group_Peer));
1001
1002
0
    if (tmp == nullptr) {
1003
0
        return nullptr;
1004
0
    }
1005
1006
0
    return tmp;
1007
0
}
1008
1009
non_null()
1010
static void group_peer_delete_handler(const void *object, void *arr, uint32_t size)
1011
0
{
1012
0
    const Memory *mem = (const Memory *)object;
1013
0
    mem_delete(mem, arr);
1014
0
}
1015
1016
static const Sort_Funcs group_peer_cmp_funcs = {
1017
    group_peer_less_handler,
1018
    group_peer_get_handler,
1019
    group_peer_set_handler,
1020
    group_peer_subarr_handler,
1021
    group_peer_alloc_handler,
1022
    group_peer_delete_handler,
1023
};
1024
1025
/** @brief Delete frozen peers as necessary to ensure at most `g->maxfrozen` remain.
1026
 *
1027
 * @retval true if any frozen peers are removed.
1028
 */
1029
non_null()
1030
static bool delete_old_frozen(Group_c *g, const Memory *mem)
1031
0
{
1032
0
    if (g->numfrozen <= g->maxfrozen) {
1033
0
        return false;
1034
0
    }
1035
1036
0
    if (g->maxfrozen == 0) {
1037
0
        mem_delete(mem, g->frozen);
1038
0
        g->frozen = nullptr;
1039
0
        g->numfrozen = 0;
1040
0
        return true;
1041
0
    }
1042
1043
0
    merge_sort(g->frozen, g->numfrozen, mem, &group_peer_cmp_funcs);
1044
1045
0
    Group_Peer *temp = (Group_Peer *)mem_vrealloc(mem, g->frozen, g->maxfrozen, sizeof(Group_Peer));
1046
1047
0
    if (temp == nullptr) {
1048
0
        return false;
1049
0
    }
1050
1051
0
    g->frozen = temp;
1052
1053
0
    g->numfrozen = g->maxfrozen;
1054
1055
0
    return true;
1056
0
}
1057
1058
non_null()
1059
static bool try_send_rejoin(Group_Chats *g_c, Group_c *g, const uint8_t *real_pk);
1060
1061
non_null(1) nullable(4)
1062
static bool freeze_peer(Group_Chats *g_c, uint32_t groupnumber, int peer_index, void *userdata)
1063
0
{
1064
0
    Group_c *g = get_group_c(g_c, groupnumber);
1065
1066
0
    if (g == nullptr) {
1067
0
        return false;
1068
0
    }
1069
1070
0
    Group_Peer *temp = (Group_Peer *)mem_vrealloc(g_c->mem, g->frozen, g->numfrozen + 1, sizeof(Group_Peer));
1071
1072
0
    if (temp == nullptr) {
1073
0
        return false;
1074
0
    }
1075
1076
0
    g->frozen = temp;
1077
0
    g->frozen[g->numfrozen] = g->group[peer_index];
1078
0
    g->frozen[g->numfrozen].object = nullptr;
1079
1080
0
    if (!delpeer(g_c, groupnumber, peer_index, userdata)) {
1081
0
        return false;
1082
0
    }
1083
1084
0
    try_send_rejoin(g_c, g, g->frozen[g->numfrozen].real_pk);
1085
1086
0
    ++g->numfrozen;
1087
1088
0
    delete_old_frozen(g, g_c->mem);
1089
1090
0
    return true;
1091
0
}
1092
1093
/** @brief Set the nick for a peer.
1094
 *
1095
 * do_gc_callback indicates whether we want to trigger callbacks set by the client
1096
 * via the public API. This should be set to false if this function is called
1097
 * from outside of the `tox_iterate()` loop.
1098
 *
1099
 * @retval true on success.
1100
 */
1101
non_null(1, 4) nullable(6)
1102
static bool setnick(Group_Chats *g_c, uint32_t groupnumber, int peer_index, const uint8_t *nick, uint16_t nick_len,
1103
                    void *userdata, bool do_gc_callback)
1104
2.88k
{
1105
2.88k
    if (nick_len > MAX_NAME_LENGTH) {
1106
0
        return false;
1107
0
    }
1108
1109
2.88k
    Group_c *g = get_group_c(g_c, groupnumber);
1110
1111
2.88k
    if (g == nullptr) {
1112
0
        return false;
1113
0
    }
1114
1115
2.88k
    g->group[peer_index].nick_updated = true;
1116
1117
2.88k
    if (g_peer_nick_eq(&g->group[peer_index], nick, nick_len)) {
1118
        /* same name as already stored */
1119
2.40k
        return true;
1120
2.40k
    }
1121
1122
482
    if (nick_len > 0) {
1123
448
        memcpy(g->group[peer_index].nick, nick, nick_len);
1124
448
    }
1125
1126
482
    g->group[peer_index].nick_len = nick_len;
1127
1128
482
    if (do_gc_callback && g_c->peer_name_callback != nullptr) {
1129
0
        g_c->peer_name_callback(g_c->m, groupnumber, peer_index, nick, nick_len, userdata);
1130
0
    }
1131
1132
482
    return true;
1133
2.88k
}
1134
1135
/** @brief Set the title for a group.
1136
 *
1137
 * @retval true on success.
1138
 */
1139
non_null(1, 4) nullable(6)
1140
static bool settitle(Group_Chats *g_c, uint32_t groupnumber, int peer_index, const uint8_t *title, uint8_t title_len,
1141
                     void *userdata)
1142
0
{
1143
0
    if (title_len > MAX_NAME_LENGTH || title_len == 0) {
1144
0
        return false;
1145
0
    }
1146
1147
0
    Group_c *g = get_group_c(g_c, groupnumber);
1148
1149
0
    if (g == nullptr) {
1150
0
        return false;
1151
0
    }
1152
1153
0
    if (g_title_eq(g, title, title_len)) {
1154
        /* same title as already set */
1155
0
        return true;
1156
0
    }
1157
1158
0
    memcpy(g->title, title, title_len);
1159
0
    g->title_len = title_len;
1160
1161
0
    g->title_fresh = true;
1162
1163
0
    if (g_c->title_callback != nullptr) {
1164
0
        g_c->title_callback(g_c->m, groupnumber, peer_index, title, title_len, userdata);
1165
0
    }
1166
1167
0
    return true;
1168
0
}
1169
1170
/** Check if the group has no online connection, and freeze all peers if so */
1171
non_null(1) nullable(3)
1172
static void check_disconnected(Group_Chats *g_c, uint32_t groupnumber, void *userdata)
1173
0
{
1174
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1175
1176
0
    if (g == nullptr) {
1177
0
        return;
1178
0
    }
1179
1180
0
    for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
1181
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_ONLINE) {
1182
0
            return;
1183
0
        }
1184
0
    }
1185
1186
0
    for (uint32_t i = 0; i < g->numpeers; ++i) {
1187
0
        while (i < g->numpeers && !pk_equal(g->group[i].real_pk, g->real_pk)) {
1188
0
            freeze_peer(g_c, groupnumber, i, userdata);
1189
0
        }
1190
0
    }
1191
0
}
1192
1193
non_null(1) nullable(5)
1194
static void set_conns_type_connections(Group_Chats *g_c, uint32_t groupnumber, int friendcon_id, uint8_t type,
1195
                                       void *userdata)
1196
0
{
1197
0
    Group_c *g = get_group_c(g_c, groupnumber);
1198
1199
0
    if (g == nullptr) {
1200
0
        return;
1201
0
    }
1202
1203
0
    for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
1204
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
1205
0
            continue;
1206
0
        }
1207
1208
0
        if (g->connections[i].number != (unsigned int)friendcon_id) {
1209
0
            continue;
1210
0
        }
1211
1212
0
        if (type == GROUPCHAT_CONNECTION_ONLINE) {
1213
0
            send_packet_online(g_c->fr_c, friendcon_id, groupnumber, g->type, g->id);
1214
0
        } else {
1215
0
            g->connections[i].type = type;
1216
0
            check_disconnected(g_c, groupnumber, userdata);
1217
0
        }
1218
0
    }
1219
0
}
1220
1221
/** Set the type for all connections with friendcon_id */
1222
non_null(1) nullable(4)
1223
static void set_conns_status_groups(Group_Chats *g_c, int friendcon_id, uint8_t type, void *userdata)
1224
0
{
1225
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
1226
0
        set_conns_type_connections(g_c, i, friendcon_id, type, userdata);
1227
0
    }
1228
0
}
1229
1230
non_null()
1231
static void rejoin_frozen_friend(Group_Chats *g_c, int friendcon_id)
1232
0
{
1233
0
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
1234
0
    get_friendcon_public_keys(real_pk, nullptr, g_c->fr_c, friendcon_id);
1235
1236
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
1237
0
        Group_c *g = get_group_c(g_c, i);
1238
1239
0
        if (g == nullptr) {
1240
0
            continue;
1241
0
        }
1242
1243
0
        for (uint32_t j = 0; j < g->numfrozen; ++j) {
1244
0
            if (pk_equal(g->frozen[j].real_pk, real_pk)) {
1245
0
                try_send_rejoin(g_c, g, real_pk);
1246
0
                break;
1247
0
            }
1248
0
        }
1249
0
    }
1250
0
}
1251
1252
non_null(1) nullable(4)
1253
static int g_handle_any_status(void *object, int friendcon_id, bool status, void *userdata)
1254
0
{
1255
0
    Group_Chats *g_c = (Group_Chats *)object;
1256
1257
0
    if (status) {
1258
0
        rejoin_frozen_friend(g_c, friendcon_id);
1259
0
    }
1260
1261
0
    return 0;
1262
0
}
1263
1264
non_null(1) nullable(4)
1265
static int g_handle_status(void *object, int friendcon_id, bool status, void *userdata)
1266
0
{
1267
0
    Group_Chats *g_c = (Group_Chats *)object;
1268
1269
0
    if (status) { /* Went online */
1270
0
        set_conns_status_groups(g_c, friendcon_id, GROUPCHAT_CONNECTION_ONLINE, userdata);
1271
0
    } else { /* Went offline */
1272
0
        set_conns_status_groups(g_c, friendcon_id, GROUPCHAT_CONNECTION_CONNECTING, userdata);
1273
        // TODO(irungentoo): remove timedout connections?
1274
0
    }
1275
1276
0
    return 0;
1277
0
}
1278
1279
non_null(1, 3) nullable(5)
1280
static int g_handle_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata);
1281
non_null(1, 3) nullable(5)
1282
static int handle_lossy(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata);
1283
1284
/** @brief Add friend to group chat.
1285
 *
1286
 * @return connections index on success
1287
 * @retval -1 on failure.
1288
 */
1289
static int add_conn_to_groupchat(Group_Chats *g_c, int friendcon_id, Group_c *g, uint8_t reason,
1290
                                 bool lock)
1291
0
{
1292
0
    uint16_t empty = MAX_GROUP_CONNECTIONS;
1293
0
    uint16_t ind = MAX_GROUP_CONNECTIONS;
1294
1295
0
    for (uint16_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
1296
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
1297
0
            empty = i;
1298
0
            continue;
1299
0
        }
1300
1301
0
        if (g->connections[i].number == (uint32_t)friendcon_id) {
1302
0
            ind = i; /* Already in list. */
1303
0
            break;
1304
0
        }
1305
0
    }
1306
1307
0
    if (ind == MAX_GROUP_CONNECTIONS) {
1308
0
        if (empty == MAX_GROUP_CONNECTIONS) {
1309
0
            return -1;
1310
0
        }
1311
1312
0
        if (lock) {
1313
0
            friend_connection_lock(g_c->fr_c, friendcon_id);
1314
0
        }
1315
1316
0
        g->connections[empty].type = GROUPCHAT_CONNECTION_CONNECTING;
1317
0
        g->connections[empty].number = friendcon_id;
1318
0
        g->connections[empty].reasons = 0;
1319
        // TODO(irungentoo):
1320
0
        friend_connection_callbacks(g_c->m->fr_c, friendcon_id, GROUPCHAT_CALLBACK_INDEX, &g_handle_status, &g_handle_packet,
1321
0
                                    &handle_lossy, g_c, friendcon_id);
1322
0
        ind = empty;
1323
0
    }
1324
1325
0
    if ((g->connections[ind].reasons & reason) == 0) {
1326
0
        g->connections[ind].reasons |= reason;
1327
1328
0
        if (reason == GROUPCHAT_CONNECTION_REASON_INTRODUCER) {
1329
0
            ++g->num_introducer_connections;
1330
0
        }
1331
0
    }
1332
1333
0
    return ind;
1334
0
}
1335
1336
non_null()
1337
static bool send_peer_introduced(const Group_Chats *g_c, int friendcon_id, uint16_t group_num);
1338
1339
/** @brief Removes reason for keeping connection.
1340
 *
1341
 * Kills connection if this was the last reason.
1342
 */
1343
static void remove_connection_reason(Group_Chats *g_c, Group_c *g, uint16_t i, uint8_t reason)
1344
0
{
1345
0
    if ((g->connections[i].reasons & reason) == 0) {
1346
0
        return;
1347
0
    }
1348
1349
0
    g->connections[i].reasons &= ~reason;
1350
1351
0
    if (reason == GROUPCHAT_CONNECTION_REASON_INTRODUCER) {
1352
0
        --g->num_introducer_connections;
1353
1354
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_ONLINE) {
1355
0
            send_peer_introduced(g_c, g->connections[i].number, g->connections[i].group_number);
1356
0
        }
1357
0
    }
1358
1359
0
    if (g->connections[i].reasons == 0) {
1360
0
        kill_friend_connection(g_c->fr_c, g->connections[i].number);
1361
0
        g->connections[i].type = GROUPCHAT_CONNECTION_NONE;
1362
0
    }
1363
0
}
1364
1365
/** @brief Creates a new groupchat and puts it in the chats array.
1366
 *
1367
 * @param rng Random number generator used for generating the group ID.
1368
 * @param type is one of `GROUPCHAT_TYPE_*`
1369
 *
1370
 * @return group number on success.
1371
 * @retval -1 on failure.
1372
 */
1373
int add_groupchat(Group_Chats *g_c, const Random *rng, uint8_t type)
1374
0
{
1375
0
    const int32_t groupnumber = create_group_chat(g_c);
1376
1377
0
    if (groupnumber == -1) {
1378
0
        return -1;
1379
0
    }
1380
1381
0
    Group_c *g = &g_c->chats[groupnumber];
1382
1383
0
    g->status = GROUPCHAT_STATUS_CONNECTED;
1384
0
    g->type = type;
1385
0
    new_symmetric_key(rng, g->id);
1386
0
    g->peer_number = 0; /* Founder is peer 0. */
1387
0
    memcpy(g->real_pk, nc_get_self_public_key(g_c->m->net_crypto), CRYPTO_PUBLIC_KEY_SIZE);
1388
0
    const int peer_index = addpeer(g_c, groupnumber, g->real_pk, dht_get_self_public_key(g_c->m->dht), 0, nullptr, true,
1389
0
                                   false);
1390
1391
0
    if (peer_index == -1) {
1392
0
        return -1;
1393
0
    }
1394
1395
0
    setnick(g_c, groupnumber, peer_index, g_c->m->name, g_c->m->name_length, nullptr, false);
1396
1397
0
    return groupnumber;
1398
0
}
1399
1400
non_null()
1401
static bool group_leave(const Group_Chats *g_c, uint32_t groupnumber, bool permanent);
1402
1403
/** @brief Delete a groupchat from the chats array, informing the group first as
1404
 * appropriate.
1405
 *
1406
 * @retval true on success.
1407
 * @retval false if groupnumber is invalid.
1408
 */
1409
bool del_groupchat(Group_Chats *g_c, uint32_t groupnumber, bool leave_permanently)
1410
1.20k
{
1411
1.20k
    Group_c *g = get_group_c(g_c, groupnumber);
1412
1413
1.20k
    if (g == nullptr) {
1414
0
        return false;
1415
0
    }
1416
1417
1.20k
    group_leave(g_c, groupnumber, leave_permanently);
1418
1419
20.4k
    for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
1420
19.2k
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
1421
19.2k
            continue;
1422
19.2k
        }
1423
1424
0
        g->connections[i].type = GROUPCHAT_CONNECTION_NONE;
1425
0
        kill_friend_connection(g_c->fr_c, g->connections[i].number);
1426
0
    }
1427
1428
2.40k
    for (uint32_t i = 0; i < g->numpeers; ++i) {
1429
1.20k
        if (g->peer_on_leave != nullptr) {
1430
0
            g->peer_on_leave(g->object, groupnumber, g->group[i].object);
1431
0
        }
1432
1.20k
    }
1433
1434
1.20k
    if (g->group_on_delete != nullptr) {
1435
0
        g->group_on_delete(g->object, groupnumber);
1436
0
    }
1437
1438
1.20k
    return wipe_group_chat(g_c, groupnumber);
1439
1.20k
}
1440
1441
non_null()
1442
static const Group_Peer *peer_in_list(const Group_c *g, uint32_t peernumber, bool frozen)
1443
0
{
1444
0
    const Group_Peer *list = frozen ? g->frozen : g->group;
1445
0
    const uint32_t num = frozen ? g->numfrozen : g->numpeers;
1446
1447
0
    if (peernumber >= num) {
1448
0
        return nullptr;
1449
0
    }
1450
1451
0
    return &list[peernumber];
1452
0
}
1453
1454
/**
1455
 * @brief Copy the public key of (frozen, if frozen is true) peernumber who is in
1456
 *   groupnumber to pk.
1457
 *
1458
 * @param pk must be CRYPTO_PUBLIC_KEY_SIZE long.
1459
 *
1460
 * @retval 0 on success
1461
 * @retval -1 if groupnumber is invalid.
1462
 * @retval -2 if peernumber is invalid.
1463
 */
1464
int group_peer_pubkey(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber, uint8_t *pk, bool frozen)
1465
0
{
1466
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1467
1468
0
    if (g == nullptr) {
1469
0
        return -1;
1470
0
    }
1471
1472
0
    const Group_Peer *peer = peer_in_list(g, peernumber, frozen);
1473
1474
0
    if (peer == nullptr) {
1475
0
        return -2;
1476
0
    }
1477
1478
0
    memcpy(pk, peer->real_pk, CRYPTO_PUBLIC_KEY_SIZE);
1479
0
    return 0;
1480
0
}
1481
1482
/**
1483
 * @brief Return the size of (frozen, if frozen is true) peernumber's name.
1484
 *
1485
 * @retval -1 if groupnumber is invalid.
1486
 * @retval -2 if peernumber is invalid.
1487
 */
1488
int group_peername_size(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber, bool frozen)
1489
0
{
1490
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1491
1492
0
    if (g == nullptr) {
1493
0
        return -1;
1494
0
    }
1495
1496
0
    const Group_Peer *peer = peer_in_list(g, peernumber, frozen);
1497
1498
0
    if (peer == nullptr) {
1499
0
        return -2;
1500
0
    }
1501
1502
0
    return peer->nick_len;
1503
0
}
1504
1505
/**
1506
 * @brief Copy the name of (frozen, if frozen is true) peernumber who is in
1507
 *   groupnumber to name.
1508
 *
1509
 * @param  name must be at least MAX_NAME_LENGTH long.
1510
 *
1511
 * @return length of name if success
1512
 * @retval -1 if groupnumber is invalid.
1513
 * @retval -2 if peernumber is invalid.
1514
 */
1515
int group_peername(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber, uint8_t *name, bool frozen)
1516
0
{
1517
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1518
1519
0
    if (g == nullptr) {
1520
0
        return -1;
1521
0
    }
1522
1523
0
    const Group_Peer *peer = peer_in_list(g, peernumber, frozen);
1524
1525
0
    if (peer == nullptr) {
1526
0
        return -2;
1527
0
    }
1528
1529
0
    if (peer->nick_len > 0) {
1530
0
        memcpy(name, peer->nick, peer->nick_len);
1531
0
    }
1532
1533
0
    return peer->nick_len;
1534
0
}
1535
1536
/**
1537
 * @brief Copy last active timestamp of frozen peernumber who is in groupnumber to
1538
 *   last_active.
1539
 *
1540
 * @retval 0 on success.
1541
 * @retval -1 if groupnumber is invalid.
1542
 * @retval -2 if peernumber is invalid.
1543
 */
1544
int group_frozen_last_active(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber,
1545
                             uint64_t *last_active)
1546
0
{
1547
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1548
1549
0
    if (g == nullptr) {
1550
0
        return -1;
1551
0
    }
1552
1553
0
    if (peernumber >= g->numfrozen) {
1554
0
        return -2;
1555
0
    }
1556
1557
0
    *last_active = g->frozen[peernumber].last_active;
1558
0
    return 0;
1559
0
}
1560
1561
/** @brief Set maximum number of frozen peers.
1562
 *
1563
 * @retval 0 on success.
1564
 * @retval -1 if groupnumber is invalid.
1565
 */
1566
int group_set_max_frozen(const Group_Chats *g_c, uint32_t groupnumber, uint32_t maxfrozen)
1567
0
{
1568
0
    Group_c *g = get_group_c(g_c, groupnumber);
1569
1570
0
    if (g == nullptr) {
1571
0
        return -1;
1572
0
    }
1573
1574
0
    g->maxfrozen = maxfrozen;
1575
0
    delete_old_frozen(g, g_c->mem);
1576
0
    return 0;
1577
0
}
1578
1579
/**
1580
 * @return the number of (frozen, if frozen is true) peers in the group chat on success.
1581
 * @retval -1 if groupnumber is invalid.
1582
 */
1583
int group_number_peers(const Group_Chats *g_c, uint32_t groupnumber, bool frozen)
1584
0
{
1585
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1586
1587
0
    if (g == nullptr) {
1588
0
        return -1;
1589
0
    }
1590
1591
0
    return frozen ? g->numfrozen : g->numpeers;
1592
0
}
1593
1594
/**
1595
 * @retval 1 if the peernumber corresponds to ours.
1596
 * @retval 0 if the peernumber is not ours.
1597
 * @retval -1 if groupnumber is invalid.
1598
 * @retval -2 if peernumber is invalid.
1599
 * @retval -3 if we are not connected to the group chat.
1600
 */
1601
int group_peernumber_is_ours(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber)
1602
0
{
1603
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1604
1605
0
    if (g == nullptr) {
1606
0
        return -1;
1607
0
    }
1608
1609
0
    if (peernumber >= g->numpeers) {
1610
0
        return -2;
1611
0
    }
1612
1613
0
    if (g->status != GROUPCHAT_STATUS_CONNECTED) {
1614
0
        return -3;
1615
0
    }
1616
1617
0
    return (g->peer_number == g->group[peernumber].peer_number) ? 1 : 0;
1618
0
}
1619
1620
/** @brief return the type of groupchat (GROUPCHAT_TYPE_) that groupnumber is.
1621
 *
1622
 * @retval -1 on failure.
1623
 * @return type on success.
1624
 */
1625
int group_get_type(const Group_Chats *g_c, uint32_t groupnumber)
1626
0
{
1627
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1628
1629
0
    if (g == nullptr) {
1630
0
        return -1;
1631
0
    }
1632
1633
0
    return g->type;
1634
0
}
1635
1636
/** @brief Copies the unique id of `group_chat[groupnumber]` into `id`.
1637
 *
1638
 * @retval false on failure.
1639
 * @retval true on success.
1640
 */
1641
bool conference_get_id(const Group_Chats *g_c, uint32_t groupnumber, uint8_t *id)
1642
0
{
1643
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1644
1645
0
    if (g == nullptr) {
1646
0
        return false;
1647
0
    }
1648
1649
0
    if (id != nullptr) {
1650
0
        memcpy(id, g->id, sizeof(g->id));
1651
0
    }
1652
1653
0
    return true;
1654
0
}
1655
1656
/** @brief Send a group packet to friendcon_id.
1657
 *
1658
 * @retval true on success
1659
 * @retval false on failure
1660
 */
1661
non_null()
1662
static bool send_packet_group_peer(const Friend_Connections *fr_c, int friendcon_id, uint8_t packet_id,
1663
                                   uint16_t group_num, const uint8_t *data, uint16_t length)
1664
0
{
1665
0
    if (1 + sizeof(uint16_t) + length > MAX_CRYPTO_DATA_SIZE) {
1666
0
        return false;
1667
0
    }
1668
1669
0
    group_num = net_htons(group_num);
1670
0
    const uint32_t packet_size = 1 + sizeof(uint16_t) + length;
1671
0
    VLA(uint8_t, packet, packet_size);
1672
0
    packet[0] = packet_id;
1673
0
    memcpy(packet + 1, &group_num, sizeof(uint16_t));
1674
0
    memcpy(packet + 1 + sizeof(uint16_t), data, length);
1675
0
    return write_cryptpacket(friendconn_net_crypto(fr_c), friend_connection_crypt_connection_id(fr_c, friendcon_id),
1676
0
                             packet, packet_size, false) != -1;
1677
0
}
1678
1679
/** @brief Send a group lossy packet to friendcon_id.
1680
 *
1681
 * @retval true on success
1682
 * @retval false on failure
1683
 */
1684
non_null()
1685
static bool send_lossy_group_peer(const Friend_Connections *fr_c, int friendcon_id, uint8_t packet_id,
1686
                                  uint16_t group_num, const uint8_t *data, uint16_t length)
1687
0
{
1688
0
    if (1 + sizeof(uint16_t) + length > MAX_CRYPTO_DATA_SIZE) {
1689
0
        return false;
1690
0
    }
1691
1692
0
    group_num = net_htons(group_num);
1693
0
    const uint32_t packet_size = 1 + sizeof(uint16_t) + length;
1694
0
    VLA(uint8_t, packet, packet_size);
1695
0
    packet[0] = packet_id;
1696
0
    memcpy(packet + 1, &group_num, sizeof(uint16_t));
1697
0
    memcpy(packet + 1 + sizeof(uint16_t), data, length);
1698
0
    return send_lossy_cryptpacket(friendconn_net_crypto(fr_c), friend_connection_crypt_connection_id(fr_c, friendcon_id),
1699
0
                                  packet, packet_size) != -1;
1700
0
}
1701
1702
/** @brief invite friendnumber to groupnumber.
1703
 *
1704
 * @retval 0 on success.
1705
 * @retval -1 if groupnumber is invalid.
1706
 * @retval -2 if invite packet failed to send.
1707
 * @retval -3 if we are not connected to the group chat.
1708
 */
1709
int invite_friend(const Group_Chats *g_c, uint32_t friendnumber, uint32_t groupnumber)
1710
0
{
1711
0
    const Group_c *g = get_group_c(g_c, groupnumber);
1712
1713
0
    if (g == nullptr) {
1714
0
        return -1;
1715
0
    }
1716
1717
0
    if (g->status != GROUPCHAT_STATUS_CONNECTED) {
1718
0
        return -3;
1719
0
    }
1720
1721
0
    uint8_t invite[INVITE_PACKET_SIZE];
1722
0
    invite[0] = INVITE_ID;
1723
0
    const uint16_t groupchat_num = net_htons((uint16_t)groupnumber);
1724
0
    memcpy(invite + 1, &groupchat_num, sizeof(groupchat_num));
1725
0
    invite[1 + sizeof(groupchat_num)] = g->type;
1726
0
    memcpy(invite + 1 + sizeof(groupchat_num) + 1, g->id, GROUP_ID_LENGTH);
1727
1728
0
    if (send_conference_invite_packet(g_c->m, friendnumber, invite, sizeof(invite))) {
1729
0
        return 0;
1730
0
    }
1731
1732
0
    return -2;
1733
0
}
1734
1735
/** @brief Send a rejoin packet to a peer if we have a friend connection to the peer.
1736
 * @retval true if a packet was sent.
1737
 * @retval false otherwise.
1738
 */
1739
static bool try_send_rejoin(Group_Chats *g_c, Group_c *g, const uint8_t *real_pk)
1740
0
{
1741
0
    const int friendcon_id = getfriend_conn_id_pk(g_c->fr_c, real_pk);
1742
1743
0
    if (friendcon_id == -1) {
1744
0
        return false;
1745
0
    }
1746
1747
0
    uint8_t packet[1 + 1 + GROUP_ID_LENGTH];
1748
0
    packet[0] = PACKET_ID_REJOIN_CONFERENCE;
1749
0
    packet[1] = g->type;
1750
0
    memcpy(packet + 2, g->id, GROUP_ID_LENGTH);
1751
1752
0
    if (write_cryptpacket(friendconn_net_crypto(g_c->fr_c), friend_connection_crypt_connection_id(g_c->fr_c, friendcon_id),
1753
0
                          packet, sizeof(packet), false) == -1) {
1754
0
        return false;
1755
0
    }
1756
1757
0
    add_conn_to_groupchat(g_c, friendcon_id, g, GROUPCHAT_CONNECTION_REASON_INTRODUCER, true);
1758
1759
0
    return true;
1760
0
}
1761
1762
non_null()
1763
static bool send_peer_query(const Group_Chats *g_c, int friendcon_id, uint16_t group_num);
1764
1765
non_null()
1766
static bool send_invite_response(Group_Chats *g_c, int groupnumber, uint32_t friendnumber, const uint8_t *data,
1767
                                 uint16_t length);
1768
1769
/** @brief Join a group (we need to have been invited first).
1770
 *
1771
 * @param expected_type is the groupchat type we expect the chat we are joining
1772
 *   to have.
1773
 *
1774
 * @return group number on success.
1775
 * @retval -1 if data length is invalid.
1776
 * @retval -2 if group is not the expected type.
1777
 * @retval -3 if friendnumber is invalid.
1778
 * @retval -4 if client is already in this group.
1779
 * @retval -5 if group instance failed to initialize.
1780
 * @retval -6 if join packet fails to send.
1781
 */
1782
int join_groupchat(Group_Chats *g_c, uint32_t friendnumber, uint8_t expected_type, const uint8_t *data, uint16_t length)
1783
0
{
1784
0
    if (length != sizeof(uint16_t) + 1 + GROUP_ID_LENGTH) {
1785
0
        return -1;
1786
0
    }
1787
1788
0
    if (data[sizeof(uint16_t)] != expected_type) {
1789
0
        return -2;
1790
0
    }
1791
1792
0
    const int friendcon_id = getfriendcon_id(g_c->m, friendnumber);
1793
1794
0
    if (friendcon_id == -1) {
1795
0
        return -3;
1796
0
    }
1797
1798
0
    if (get_group_num(g_c, data[sizeof(uint16_t)], data + sizeof(uint16_t) + 1) != -1) {
1799
0
        return -4;
1800
0
    }
1801
1802
0
    const int groupnumber = create_group_chat(g_c);
1803
1804
0
    if (groupnumber == -1) {
1805
0
        return -5;
1806
0
    }
1807
1808
0
    Group_c *g = &g_c->chats[groupnumber];
1809
1810
0
    g->status = GROUPCHAT_STATUS_VALID;
1811
0
    memcpy(g->real_pk, nc_get_self_public_key(g_c->m->net_crypto), CRYPTO_PUBLIC_KEY_SIZE);
1812
1813
0
    if (!send_invite_response(g_c, groupnumber, friendnumber, data, length)) {
1814
0
        g->status = GROUPCHAT_STATUS_NONE;
1815
0
        return -6;
1816
0
    }
1817
1818
0
    return groupnumber;
1819
0
}
1820
1821
static bool send_invite_response(Group_Chats *g_c, int groupnumber, uint32_t friendnumber, const uint8_t *data,
1822
                                 uint16_t length)
1823
0
{
1824
0
    Group_c *g = get_group_c(g_c, groupnumber);
1825
1826
0
    if (g == nullptr) {
1827
0
        return false;
1828
0
    }
1829
1830
0
    const bool member = g->status == GROUPCHAT_STATUS_CONNECTED;
1831
1832
0
    const uint32_t response_size = member ? INVITE_MEMBER_PACKET_SIZE : INVITE_ACCEPT_PACKET_SIZE;
1833
0
    VLA(uint8_t, response, response_size);
1834
0
    response[0] = member ? INVITE_MEMBER_ID : INVITE_ACCEPT_ID;
1835
0
    net_pack_u16(response + 1, groupnumber);
1836
0
    memcpy(response + 1 + sizeof(uint16_t), data, length);
1837
1838
0
    if (member) {
1839
0
        net_pack_u16(response + 1 + sizeof(uint16_t) + length, g->peer_number);
1840
0
    }
1841
1842
0
    if (!send_conference_invite_packet(g_c->m, friendnumber, response, response_size)) {
1843
0
        return false;
1844
0
    }
1845
1846
0
    if (!member) {
1847
0
        g->type = data[sizeof(uint16_t)];
1848
0
        memcpy(g->id, data + sizeof(uint16_t) + 1, GROUP_ID_LENGTH);
1849
0
    }
1850
1851
0
    uint16_t other_groupnum;
1852
0
    net_unpack_u16(data, &other_groupnum);
1853
1854
0
    const int friendcon_id = getfriendcon_id(g_c->m, friendnumber);
1855
1856
0
    if (friendcon_id == -1) {
1857
0
        return false;
1858
0
    }
1859
1860
0
    const int connection_index = add_conn_to_groupchat(g_c, friendcon_id, g, GROUPCHAT_CONNECTION_REASON_INTRODUCER, true);
1861
1862
0
    if (member) {
1863
0
        add_conn_to_groupchat(g_c, friendcon_id, g, GROUPCHAT_CONNECTION_REASON_INTRODUCING, false);
1864
0
    }
1865
1866
0
    if (connection_index != -1) {
1867
0
        g->connections[connection_index].group_number = other_groupnum;
1868
0
        g->connections[connection_index].type = GROUPCHAT_CONNECTION_ONLINE;
1869
0
    }
1870
1871
0
    send_peer_query(g_c, friendcon_id, other_groupnum);
1872
1873
0
    return true;
1874
0
}
1875
1876
/** Set handlers for custom lossy packets. */
1877
void group_lossy_packet_registerhandler(Group_Chats *g_c, uint8_t byte, lossy_packet_cb *function)
1878
0
{
1879
0
    g_c->lossy_packethandlers[byte] = function;
1880
0
}
1881
1882
/** Set the callback for group invites. */
1883
void g_callback_group_invite(Group_Chats *g_c, g_conference_invite_cb *function)
1884
1.27k
{
1885
1.27k
    g_c->invite_callback = function;
1886
1.27k
}
1887
1888
/** Set the callback for group connection. */
1889
void g_callback_group_connected(Group_Chats *g_c, g_conference_connected_cb *function)
1890
1.27k
{
1891
1.27k
    g_c->connected_callback = function;
1892
1.27k
}
1893
1894
/** Set the callback for group messages. */
1895
void g_callback_group_message(Group_Chats *g_c, g_conference_message_cb *function)
1896
1.27k
{
1897
1.27k
    g_c->message_callback = function;
1898
1.27k
}
1899
1900
/** @brief Set callback function for peer nickname changes.
1901
 *
1902
 * It gets called every time a peer changes their nickname.
1903
 */
1904
void g_callback_peer_name(Group_Chats *g_c, peer_name_cb *function)
1905
1.27k
{
1906
1.27k
    g_c->peer_name_callback = function;
1907
1.27k
}
1908
1909
/** @brief Set callback function for peer list changes.
1910
 *
1911
 * It gets called every time the name list changes(new peer, deleted peer)
1912
 */
1913
void g_callback_peer_list_changed(Group_Chats *g_c, peer_list_changed_cb *function)
1914
1.27k
{
1915
1.27k
    g_c->peer_list_changed_callback = function;
1916
1.27k
}
1917
1918
/** Set callback function for title changes. */
1919
void g_callback_group_title(Group_Chats *g_c, title_cb *function)
1920
1.27k
{
1921
1.27k
    g_c->title_callback = function;
1922
1.27k
}
1923
1924
/** @brief Set a function to be called when a new peer joins a group chat.
1925
 *
1926
 * @retval 0 on success.
1927
 * @retval -1 on failure.
1928
 */
1929
int callback_groupchat_peer_new(const Group_Chats *g_c, uint32_t groupnumber, peer_on_join_cb *function)
1930
0
{
1931
0
    Group_c *g = get_group_c(g_c, groupnumber);
1932
1933
0
    if (g == nullptr) {
1934
0
        return -1;
1935
0
    }
1936
1937
0
    g->peer_on_join = function;
1938
0
    return 0;
1939
0
}
1940
1941
/** @brief Set a function to be called when a peer leaves a group chat.
1942
 *
1943
 * @retval 0 on success.
1944
 * @retval -1 on failure.
1945
 */
1946
int callback_groupchat_peer_delete(const Group_Chats *g_c, uint32_t groupnumber, peer_on_leave_cb *function)
1947
0
{
1948
0
    Group_c *g = get_group_c(g_c, groupnumber);
1949
1950
0
    if (g == nullptr) {
1951
0
        return -1;
1952
0
    }
1953
1954
0
    g->peer_on_leave = function;
1955
0
    return 0;
1956
0
}
1957
1958
/** @brief Set a function to be called when the group chat is deleted.
1959
 *
1960
 * @retval 0 on success.
1961
 * @retval -1 on failure.
1962
 */
1963
int callback_groupchat_delete(const Group_Chats *g_c, uint32_t groupnumber, group_on_delete_cb *function)
1964
0
{
1965
0
    Group_c *g = get_group_c(g_c, groupnumber);
1966
1967
0
    if (g == nullptr) {
1968
0
        return -1;
1969
0
    }
1970
1971
0
    g->group_on_delete = function;
1972
0
    return 0;
1973
0
}
1974
1975
non_null(1) nullable(4)
1976
static int send_message_group(const Group_Chats *g_c, uint32_t groupnumber, uint8_t message_id, const uint8_t *data,
1977
                              uint16_t len);
1978
1979
/** @brief send a ping message
1980
 * return true on success
1981
 */
1982
non_null()
1983
static bool group_ping_send(const Group_Chats *g_c, uint32_t groupnumber)
1984
0
{
1985
0
    return send_message_group(g_c, groupnumber, GROUP_MESSAGE_PING_ID, nullptr, 0) > 0;
1986
0
}
1987
1988
/** @brief send a new_peer message
1989
 * return true on success
1990
 */
1991
non_null()
1992
static bool group_new_peer_send(const Group_Chats *g_c, uint32_t groupnumber, uint16_t peer_num, const uint8_t *real_pk,
1993
                                const uint8_t *temp_pk)
1994
0
{
1995
0
    uint8_t packet[GROUP_MESSAGE_NEW_PEER_LENGTH];
1996
1997
0
    peer_num = net_htons(peer_num);
1998
0
    memcpy(packet, &peer_num, sizeof(uint16_t));
1999
0
    memcpy(packet + sizeof(uint16_t), real_pk, CRYPTO_PUBLIC_KEY_SIZE);
2000
0
    memcpy(packet + sizeof(uint16_t) + CRYPTO_PUBLIC_KEY_SIZE, temp_pk, CRYPTO_PUBLIC_KEY_SIZE);
2001
2002
0
    return send_message_group(g_c, groupnumber, GROUP_MESSAGE_NEW_PEER_ID, packet, sizeof(packet)) > 0;
2003
0
}
2004
2005
/** @brief send a kill_peer message
2006
 * return true on success
2007
 */
2008
non_null()
2009
static bool group_kill_peer_send(const Group_Chats *g_c, uint32_t groupnumber, uint16_t peer_num)
2010
0
{
2011
0
    uint8_t packet[GROUP_MESSAGE_KILL_PEER_LENGTH];
2012
2013
0
    peer_num = net_htons(peer_num);
2014
0
    memcpy(packet, &peer_num, sizeof(uint16_t));
2015
2016
0
    return send_message_group(g_c, groupnumber, GROUP_MESSAGE_KILL_PEER_ID, packet, sizeof(packet)) > 0;
2017
0
}
2018
2019
/** @brief send a freeze_peer message
2020
 * return true on success
2021
 */
2022
non_null()
2023
static bool group_freeze_peer_send(const Group_Chats *g_c, uint32_t groupnumber, uint16_t peer_num)
2024
1.20k
{
2025
1.20k
    uint8_t packet[GROUP_MESSAGE_KILL_PEER_LENGTH];
2026
2027
1.20k
    peer_num = net_htons(peer_num);
2028
1.20k
    memcpy(packet, &peer_num, sizeof(uint16_t));
2029
2030
1.20k
    return send_message_group(g_c, groupnumber, GROUP_MESSAGE_FREEZE_PEER_ID, packet, sizeof(packet)) > 0;
2031
1.20k
}
2032
2033
/** @brief send a name message
2034
 * return true on success
2035
 */
2036
non_null()
2037
static bool group_name_send(const Group_Chats *g_c, uint32_t groupnumber, const uint8_t *nick, uint16_t nick_len)
2038
0
{
2039
0
    if (nick_len > MAX_NAME_LENGTH) {
2040
0
        return false;
2041
0
    }
2042
2043
0
    return send_message_group(g_c, groupnumber, GROUP_MESSAGE_NAME_ID, nick, nick_len) > 0;
2044
0
}
2045
2046
/** @brief send message to announce leaving group
2047
 * return true on success
2048
 */
2049
static bool group_leave(const Group_Chats *g_c, uint32_t groupnumber, bool permanent)
2050
1.20k
{
2051
1.20k
    const Group_c *g = get_group_c(g_c, groupnumber);
2052
2053
1.20k
    if (g == nullptr) {
2054
0
        return false;
2055
0
    }
2056
2057
1.20k
    if (permanent) {
2058
0
        return group_kill_peer_send(g_c, groupnumber, g->peer_number);
2059
1.20k
    } else {
2060
1.20k
        return group_freeze_peer_send(g_c, groupnumber, g->peer_number);
2061
1.20k
    }
2062
1.20k
}
2063
2064
/** @brief set the group's title, limited to MAX_NAME_LENGTH.
2065
 * @retval 0 on success
2066
 * @retval -1 if groupnumber is invalid.
2067
 * @retval -2 if title is too long or empty.
2068
 * @retval -3 if packet fails to send.
2069
 */
2070
int group_title_send(const Group_Chats *g_c, uint32_t groupnumber, const uint8_t *title, uint8_t title_len)
2071
0
{
2072
0
    Group_c *g = get_group_c(g_c, groupnumber);
2073
2074
0
    if (g == nullptr) {
2075
0
        return -1;
2076
0
    }
2077
2078
0
    if (title_len > MAX_NAME_LENGTH || title_len == 0) {
2079
0
        return -2;
2080
0
    }
2081
2082
    /* same as already set? */
2083
0
    if (g_title_eq(g, title, title_len)) {
2084
0
        return 0;
2085
0
    }
2086
2087
0
    memcpy(g->title, title, title_len);
2088
0
    g->title_len = title_len;
2089
2090
0
    if (g->numpeers == 1) {
2091
0
        return 0;
2092
0
    }
2093
2094
0
    if (send_message_group(g_c, groupnumber, GROUP_MESSAGE_TITLE_ID, title, title_len) > 0) {
2095
0
        return 0;
2096
0
    }
2097
2098
0
    return -3;
2099
0
}
2100
2101
/** @brief return the group's title size.
2102
 * @retval -1 of groupnumber is invalid.
2103
 * @retval -2 if title is too long or empty.
2104
 */
2105
int group_title_get_size(const Group_Chats *g_c, uint32_t groupnumber)
2106
0
{
2107
0
    const Group_c *g = get_group_c(g_c, groupnumber);
2108
2109
0
    if (g == nullptr) {
2110
0
        return -1;
2111
0
    }
2112
2113
0
    if (g->title_len > MAX_NAME_LENGTH || g->title_len == 0) {
2114
0
        return -2;
2115
0
    }
2116
2117
0
    return g->title_len;
2118
0
}
2119
2120
/** @brief Get group title from groupnumber and put it in title.
2121
 *
2122
 * Title needs to be a valid memory location with a size of at least MAX_NAME_LENGTH (128) bytes.
2123
 *
2124
 * @return length of copied title if success.
2125
 * @retval -1 if groupnumber is invalid.
2126
 * @retval -2 if title is too long or empty.
2127
 */
2128
int group_title_get(const Group_Chats *g_c, uint32_t groupnumber, uint8_t *title)
2129
0
{
2130
0
    const Group_c *g = get_group_c(g_c, groupnumber);
2131
2132
0
    if (g == nullptr) {
2133
0
        return -1;
2134
0
    }
2135
2136
0
    if (g->title_len > MAX_NAME_LENGTH || g->title_len == 0) {
2137
0
        return -2;
2138
0
    }
2139
2140
0
    memcpy(title, g->title, g->title_len);
2141
0
    return g->title_len;
2142
0
}
2143
2144
non_null()
2145
static bool get_peer_number(const Group_c *g, const uint8_t *real_pk, uint16_t *peer_number)
2146
0
{
2147
0
    const int peer_index = peer_in_group(g, real_pk);
2148
2149
0
    if (peer_index >= 0) {
2150
0
        *peer_number = g->group[peer_index].peer_number;
2151
0
        return true;
2152
0
    }
2153
2154
0
    const int frozen_index = frozen_in_group(g, real_pk);
2155
2156
0
    if (frozen_index >= 0) {
2157
0
        *peer_number = g->frozen[frozen_index].peer_number;
2158
0
        return true;
2159
0
    }
2160
2161
0
    return false;
2162
0
}
2163
2164
non_null(1, 3) nullable(5)
2165
static void handle_friend_invite_packet(Messenger *m, uint32_t friend_number, const uint8_t *cookie, uint16_t length,
2166
                                        void *user_data)
2167
0
{
2168
0
    Group_Chats *g_c = m->conferences_object;
2169
2170
0
    if (length <= 1) {
2171
0
        return;
2172
0
    }
2173
2174
0
    switch (cookie[0]) {
2175
0
        case INVITE_ID: {
2176
0
            if (length != INVITE_PACKET_SIZE) {
2177
0
                return;
2178
0
            }
2179
2180
0
            const int groupnumber = get_group_num(g_c, cookie[1 + sizeof(uint16_t)], cookie + 1 + sizeof(uint16_t) + 1);
2181
2182
0
            const uint8_t *invite_data = cookie + 1;
2183
0
            const uint16_t invite_length = length - 1;
2184
2185
0
            if (groupnumber == -1) {
2186
0
                if (g_c->invite_callback != nullptr) {
2187
0
                    g_c->invite_callback(m, friend_number, invite_data[sizeof(uint16_t)], invite_data, invite_length, user_data);
2188
0
                }
2189
2190
0
                return;
2191
0
            } else {
2192
0
                const Group_c *g = get_group_c(g_c, groupnumber);
2193
2194
0
                if (g != nullptr && g->status == GROUPCHAT_STATUS_CONNECTED) {
2195
0
                    send_invite_response(g_c, groupnumber, friend_number, invite_data, invite_length);
2196
0
                }
2197
0
            }
2198
2199
0
            break;
2200
0
        }
2201
2202
0
        case INVITE_ACCEPT_ID:
2203
0
        case INVITE_MEMBER_ID: {
2204
0
            const bool member = cookie[0] == INVITE_MEMBER_ID;
2205
2206
0
            if (length != (member ? INVITE_MEMBER_PACKET_SIZE : INVITE_ACCEPT_PACKET_SIZE)) {
2207
0
                return;
2208
0
            }
2209
2210
0
            uint16_t other_groupnum;
2211
0
            uint16_t groupnum;
2212
0
            net_unpack_u16(cookie + 1, &other_groupnum);
2213
0
            net_unpack_u16(cookie + 1 + sizeof(uint16_t), &groupnum);
2214
2215
0
            Group_c *g = get_group_c(g_c, groupnum);
2216
2217
0
            if (g == nullptr) {
2218
0
                return;
2219
0
            }
2220
2221
0
            if (cookie[1 + sizeof(uint16_t) * 2] != g->type) {
2222
0
                return;
2223
0
            }
2224
2225
0
            if (!group_id_eq(cookie + 1 + sizeof(uint16_t) * 2 + 1, g->id)) {
2226
0
                return;
2227
0
            }
2228
2229
0
            uint16_t peer_number;
2230
2231
0
            if (member) {
2232
0
                net_unpack_u16(cookie + 1 + sizeof(uint16_t) * 2 + 1 + GROUP_ID_LENGTH, &peer_number);
2233
0
            } else {
2234
                /* TODO(irungentoo): what if two people enter the group at the
2235
                 * same time and are given the same peer_number by different
2236
                 * nodes? */
2237
0
                peer_number = random_u16(m->rng);
2238
2239
0
                unsigned int tries = 0;
2240
2241
0
                while (get_peer_index(g, peer_number) != -1 || get_frozen_index(g, peer_number) != -1) {
2242
0
                    peer_number = random_u16(m->rng);
2243
0
                    ++tries;
2244
2245
0
                    if (tries > 32) {
2246
0
                        return;
2247
0
                    }
2248
0
                }
2249
0
            }
2250
2251
0
            const int friendcon_id = getfriendcon_id(m, friend_number);
2252
2253
0
            if (friendcon_id == -1) {
2254
                // TODO(iphydf): Log something?
2255
0
                return;
2256
0
            }
2257
2258
0
            uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2259
0
            uint8_t temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
2260
0
            get_friendcon_public_keys(real_pk, temp_pk, g_c->fr_c, friendcon_id);
2261
2262
0
            addpeer(g_c, groupnum, real_pk, temp_pk, peer_number, user_data, true, true);
2263
0
            const int connection_index = add_conn_to_groupchat(g_c, friendcon_id, g,
2264
0
                                         GROUPCHAT_CONNECTION_REASON_INTRODUCING, true);
2265
2266
0
            if (member) {
2267
0
                add_conn_to_groupchat(g_c, friendcon_id, g, GROUPCHAT_CONNECTION_REASON_INTRODUCER, false);
2268
0
                send_peer_query(g_c, friendcon_id, other_groupnum);
2269
0
            }
2270
2271
0
            if (connection_index != -1) {
2272
0
                g->connections[connection_index].group_number = other_groupnum;
2273
0
                g->connections[connection_index].type = GROUPCHAT_CONNECTION_ONLINE;
2274
0
            }
2275
2276
0
            group_new_peer_send(g_c, groupnum, peer_number, real_pk, temp_pk);
2277
2278
0
            break;
2279
0
        }
2280
2281
0
        default: {
2282
0
            return;
2283
0
        }
2284
0
    }
2285
0
}
2286
2287
/** @brief Find index of friend in the connections list.
2288
 *
2289
 * return index on success
2290
 * return -1 on failure.
2291
 */
2292
non_null()
2293
static int friend_in_connections(const Group_c *g, int friendcon_id)
2294
0
{
2295
0
    for (unsigned int i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
2296
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
2297
0
            continue;
2298
0
        }
2299
2300
0
        if (g->connections[i].number == (uint32_t)friendcon_id) {
2301
0
            return i;
2302
0
        }
2303
0
    }
2304
2305
0
    return -1;
2306
0
}
2307
2308
/** return number of connections. */
2309
non_null()
2310
static unsigned int count_connected(const Group_c *g)
2311
1.20k
{
2312
1.20k
    unsigned int count = 0;
2313
2314
20.4k
    for (unsigned int i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
2315
19.2k
        if (g->connections[i].type == GROUPCHAT_CONNECTION_ONLINE) {
2316
0
            ++count;
2317
0
        }
2318
19.2k
    }
2319
2320
1.20k
    return count;
2321
1.20k
}
2322
2323
static bool send_packet_online(const Friend_Connections *fr_c, int friendcon_id, uint16_t group_num,
2324
                               uint8_t type, const uint8_t *id)
2325
0
{
2326
0
    uint8_t packet[1 + ONLINE_PACKET_DATA_SIZE];
2327
0
    group_num = net_htons(group_num);
2328
0
    packet[0] = PACKET_ID_ONLINE_PACKET;
2329
0
    memcpy(packet + 1, &group_num, sizeof(uint16_t));
2330
0
    packet[1 + sizeof(uint16_t)] = type;
2331
0
    memcpy(packet + 1 + sizeof(uint16_t) + 1, id, GROUP_ID_LENGTH);
2332
0
    return write_cryptpacket(friendconn_net_crypto(fr_c), friend_connection_crypt_connection_id(fr_c, friendcon_id), packet,
2333
0
                             sizeof(packet), false) != -1;
2334
0
}
2335
2336
non_null()
2337
static bool ping_groupchat(const Group_Chats *g_c, uint32_t groupnumber);
2338
2339
non_null()
2340
static int handle_packet_online(const Group_Chats *g_c, int friendcon_id, const uint8_t *data, uint16_t length)
2341
0
{
2342
0
    if (length != ONLINE_PACKET_DATA_SIZE) {
2343
0
        return -1;
2344
0
    }
2345
2346
0
    const int groupnumber = get_group_num(g_c, data[sizeof(uint16_t)], data + sizeof(uint16_t) + 1);
2347
2348
0
    if (groupnumber == -1) {
2349
0
        return -1;
2350
0
    }
2351
2352
0
    uint16_t other_groupnum;
2353
0
    memcpy(&other_groupnum, data, sizeof(uint16_t));
2354
0
    other_groupnum = net_ntohs(other_groupnum);
2355
2356
0
    Group_c *g = get_group_c(g_c, groupnumber);
2357
2358
0
    if (g == nullptr) {
2359
0
        return -1;
2360
0
    }
2361
2362
0
    const int index = friend_in_connections(g, friendcon_id);
2363
2364
0
    if (index == -1) {
2365
0
        return -1;
2366
0
    }
2367
2368
0
    if (g->connections[index].type == GROUPCHAT_CONNECTION_ONLINE) {
2369
0
        return -1;
2370
0
    }
2371
2372
0
    if (count_connected(g) == 0 || (g->connections[index].reasons & GROUPCHAT_CONNECTION_REASON_INTRODUCER) != 0) {
2373
0
        send_peer_query(g_c, friendcon_id, other_groupnum);
2374
0
    }
2375
2376
0
    g->connections[index].group_number = other_groupnum;
2377
0
    g->connections[index].type = GROUPCHAT_CONNECTION_ONLINE;
2378
0
    send_packet_online(g_c->fr_c, friendcon_id, groupnumber, g->type, g->id);
2379
2380
0
    if ((g->connections[index].reasons & GROUPCHAT_CONNECTION_REASON_INTRODUCING) != 0) {
2381
0
        uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2382
0
        uint8_t temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
2383
0
        get_friendcon_public_keys(real_pk, temp_pk, g_c->fr_c, friendcon_id);
2384
2385
0
        const int peer_index = peer_in_group(g, real_pk);
2386
2387
0
        if (peer_index != -1) {
2388
0
            group_new_peer_send(g_c, groupnumber, g->group[peer_index].peer_number, real_pk, temp_pk);
2389
0
        }
2390
2391
0
        g->need_send_name = true;
2392
0
    }
2393
2394
0
    ping_groupchat(g_c, groupnumber);
2395
2396
0
    return 0;
2397
0
}
2398
2399
non_null(1, 3) nullable(5)
2400
static int handle_packet_rejoin(Group_Chats *g_c, int friendcon_id, const uint8_t *data, uint16_t length,
2401
                                void *userdata)
2402
0
{
2403
0
    if (length < 1 + GROUP_ID_LENGTH) {
2404
0
        return -1;
2405
0
    }
2406
2407
0
    const int32_t groupnum = get_group_num(g_c, *data, data + 1);
2408
2409
0
    Group_c *g = get_group_c(g_c, groupnum);
2410
2411
0
    if (g == nullptr) {
2412
0
        return -1;
2413
0
    }
2414
2415
0
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2416
0
    uint8_t temp_pk[CRYPTO_PUBLIC_KEY_SIZE];
2417
0
    get_friendcon_public_keys(real_pk, temp_pk, g_c->fr_c, friendcon_id);
2418
2419
0
    uint16_t peer_number;
2420
2421
0
    if (!get_peer_number(g, real_pk, &peer_number)) {
2422
0
        return -1;
2423
0
    }
2424
2425
0
    addpeer(g_c, groupnum, real_pk, temp_pk, peer_number, userdata, true, true);
2426
0
    const int connection_index = add_conn_to_groupchat(g_c, friendcon_id, g,
2427
0
                                 GROUPCHAT_CONNECTION_REASON_INTRODUCING, true);
2428
2429
0
    if (connection_index != -1) {
2430
0
        send_packet_online(g_c->fr_c, friendcon_id, groupnum, g->type, g->id);
2431
0
    }
2432
2433
0
    return 0;
2434
0
}
2435
2436
// we could send title with invite, but then if it changes between sending and accepting inv, joinee won't see it
2437
2438
/**
2439
 * @retval true on success.
2440
 * @retval false on failure
2441
 */
2442
static bool send_peer_introduced(const Group_Chats *g_c, int friendcon_id, uint16_t group_num)
2443
0
{
2444
0
    uint8_t packet[1];
2445
0
    packet[0] = PEER_INTRODUCED_ID;
2446
0
    return send_packet_group_peer(g_c->fr_c, friendcon_id, PACKET_ID_DIRECT_CONFERENCE, group_num, packet, sizeof(packet));
2447
0
}
2448
2449
/**
2450
 * @retval true on success.
2451
 * @retval false on failure
2452
 */
2453
static bool send_peer_query(const Group_Chats *g_c, int friendcon_id, uint16_t group_num)
2454
0
{
2455
0
    uint8_t packet[1];
2456
0
    packet[0] = PEER_QUERY_ID;
2457
0
    return send_packet_group_peer(g_c->fr_c, friendcon_id, PACKET_ID_DIRECT_CONFERENCE, group_num, packet, sizeof(packet));
2458
0
}
2459
2460
/**
2461
 * @return number of peers sent on success.
2462
 * @retval 0 on failure.
2463
 */
2464
non_null()
2465
static unsigned int send_peers(const Group_Chats *g_c, const Group_c *g, int friendcon_id, uint16_t group_num)
2466
0
{
2467
0
    uint8_t response_packet[MAX_CRYPTO_DATA_SIZE - (1 + sizeof(uint16_t))];
2468
0
    response_packet[0] = PEER_RESPONSE_ID;
2469
0
    uint8_t *p = response_packet + 1;
2470
2471
0
    uint16_t sent = 0;
2472
2473
0
    for (uint32_t i = 0; i <= g->numpeers; ++i) {
2474
0
        if (i == g->numpeers
2475
0
                || (p - response_packet) + sizeof(uint16_t) + CRYPTO_PUBLIC_KEY_SIZE * 2 + 1 + g->group[i].nick_len >
2476
0
                sizeof(response_packet)) {
2477
0
            if (send_packet_group_peer(g_c->fr_c, friendcon_id, PACKET_ID_DIRECT_CONFERENCE, group_num, response_packet,
2478
0
                                       p - response_packet)) {
2479
0
                sent = i;
2480
0
            } else {
2481
0
                return sent;
2482
0
            }
2483
2484
0
            if (i == g->numpeers) {
2485
0
                break;
2486
0
            }
2487
2488
0
            p = response_packet + 1;
2489
0
        }
2490
2491
0
        const uint16_t peer_num = net_htons(g->group[i].peer_number);
2492
0
        memcpy(p, &peer_num, sizeof(peer_num));
2493
0
        p += sizeof(peer_num);
2494
0
        memcpy(p, g->group[i].real_pk, CRYPTO_PUBLIC_KEY_SIZE);
2495
0
        p += CRYPTO_PUBLIC_KEY_SIZE;
2496
0
        memcpy(p, g->group[i].temp_pk, CRYPTO_PUBLIC_KEY_SIZE);
2497
0
        p += CRYPTO_PUBLIC_KEY_SIZE;
2498
0
        *p = g->group[i].nick_len;
2499
0
        p += 1;
2500
0
        memcpy(p, g->group[i].nick, g->group[i].nick_len);
2501
0
        p += g->group[i].nick_len;
2502
0
    }
2503
2504
0
    if (g->title_len > 0) {
2505
0
        const uint32_t title_packet_size = 1 + g->title_len;
2506
0
        VLA(uint8_t, title_packet, title_packet_size);
2507
0
        title_packet[0] = PEER_TITLE_ID;
2508
0
        memcpy(title_packet + 1, g->title, g->title_len);
2509
0
        send_packet_group_peer(g_c->fr_c, friendcon_id, PACKET_ID_DIRECT_CONFERENCE, group_num,
2510
0
                               title_packet, title_packet_size);
2511
0
    }
2512
2513
0
    return sent;
2514
0
}
2515
2516
non_null(1, 3) nullable(5)
2517
static int handle_send_peers(Group_Chats *g_c, uint32_t groupnumber, const uint8_t *data, uint16_t length,
2518
                             void *userdata)
2519
0
{
2520
0
    if (length == 0) {
2521
0
        return -1;
2522
0
    }
2523
2524
0
    Group_c *g = get_group_c(g_c, groupnumber);
2525
2526
0
    if (g == nullptr) {
2527
0
        return -1;
2528
0
    }
2529
2530
0
    const uint8_t *d = data;
2531
2532
0
    while ((unsigned int)(length - (d - data)) >= sizeof(uint16_t) + CRYPTO_PUBLIC_KEY_SIZE * 2 + 1) {
2533
0
        uint16_t peer_num;
2534
0
        memcpy(&peer_num, d, sizeof(peer_num));
2535
0
        peer_num = net_ntohs(peer_num);
2536
0
        d += sizeof(uint16_t);
2537
2538
0
        if (g->status == GROUPCHAT_STATUS_VALID
2539
0
                && pk_equal(d, nc_get_self_public_key(g_c->m->net_crypto))) {
2540
0
            g->peer_number = peer_num;
2541
0
            g->status = GROUPCHAT_STATUS_CONNECTED;
2542
2543
0
            if (g_c->connected_callback != nullptr) {
2544
0
                g_c->connected_callback(g_c->m, groupnumber, userdata);
2545
0
            }
2546
2547
0
            g->need_send_name = true;
2548
0
        }
2549
2550
0
        const int peer_index = addpeer(g_c, groupnumber, d, d + CRYPTO_PUBLIC_KEY_SIZE, peer_num, userdata, false, true);
2551
2552
0
        if (peer_index == -1) {
2553
0
            return -1;
2554
0
        }
2555
2556
0
        d += CRYPTO_PUBLIC_KEY_SIZE * 2;
2557
0
        const uint8_t name_length = *d;
2558
0
        d += 1;
2559
2560
0
        if (name_length > (length - (d - data)) || name_length > MAX_NAME_LENGTH) {
2561
0
            return -1;
2562
0
        }
2563
2564
0
        if (!g->group[peer_index].nick_updated) {
2565
0
            setnick(g_c, groupnumber, peer_index, d, name_length, userdata, true);
2566
0
        }
2567
2568
0
        d += name_length;
2569
0
    }
2570
2571
0
    return 0;
2572
0
}
2573
2574
non_null(1, 3) nullable(6)
2575
static void handle_direct_packet(Group_Chats *g_c, uint32_t groupnumber, const uint8_t *data, uint16_t length,
2576
                                 uint32_t connection_index, void *userdata)
2577
0
{
2578
0
    if (length == 0) {
2579
0
        return;
2580
0
    }
2581
2582
0
    Group_c *g = get_group_c(g_c, groupnumber);
2583
2584
0
    if (g == nullptr) {
2585
0
        return;
2586
0
    }
2587
2588
0
    switch (data[0]) {
2589
0
        case PEER_INTRODUCED_ID: {
2590
0
            remove_connection_reason(g_c, g, connection_index, GROUPCHAT_CONNECTION_REASON_INTRODUCING);
2591
0
            break;
2592
0
        }
2593
2594
0
        case PEER_QUERY_ID: {
2595
0
            if (g->connections[connection_index].type != GROUPCHAT_CONNECTION_ONLINE) {
2596
0
                return;
2597
0
            }
2598
2599
0
            send_peers(g_c, g, g->connections[connection_index].number, g->connections[connection_index].group_number);
2600
0
            break;
2601
0
        }
2602
2603
0
        case PEER_RESPONSE_ID: {
2604
0
            handle_send_peers(g_c, groupnumber, data + 1, length - 1, userdata);
2605
0
            break;
2606
0
        }
2607
2608
0
        case PEER_TITLE_ID: {
2609
0
            if (!g->title_fresh) {
2610
0
                settitle(g_c, groupnumber, -1, data + 1, length - 1, userdata);
2611
0
            }
2612
2613
0
            break;
2614
0
        }
2615
0
    }
2616
0
}
2617
2618
/** @brief Send message to all connections except receiver (if receiver isn't -1)
2619
 *
2620
 * NOTE: this function appends the group chat number to the data passed to it.
2621
 *
2622
 * @return number of messages sent.
2623
 */
2624
non_null()
2625
static unsigned int send_message_all_connections(const Group_Chats *g_c, const Group_c *g, const uint8_t *data,
2626
        uint16_t length, int receiver)
2627
0
{
2628
0
    uint16_t sent = 0;
2629
2630
0
    for (uint16_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
2631
0
        if (g->connections[i].type != GROUPCHAT_CONNECTION_ONLINE) {
2632
0
            continue;
2633
0
        }
2634
2635
0
        if ((int)i == receiver) {
2636
0
            continue;
2637
0
        }
2638
2639
0
        if (send_packet_group_peer(g_c->fr_c, g->connections[i].number, PACKET_ID_MESSAGE_CONFERENCE,
2640
0
                                   g->connections[i].group_number, data, length)) {
2641
0
            ++sent;
2642
0
        }
2643
0
    }
2644
2645
0
    return sent;
2646
0
}
2647
2648
/** @brief Send lossy message to all connections except receiver (if receiver isn't -1)
2649
 *
2650
 * NOTE: this function appends the group chat number to the data passed to it.
2651
 *
2652
 * @return number of messages sent.
2653
 */
2654
non_null()
2655
static unsigned int send_lossy_all_connections(const Group_Chats *g_c, const Group_c *g, const uint8_t *data,
2656
        uint16_t length, int receiver)
2657
0
{
2658
0
    unsigned int sent = 0;
2659
0
    unsigned int num_connected_closest = 0;
2660
0
    unsigned int connected_closest[DESIRED_CLOSEST] = {0};
2661
2662
0
    for (unsigned int i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
2663
0
        if (g->connections[i].type != GROUPCHAT_CONNECTION_ONLINE) {
2664
0
            continue;
2665
0
        }
2666
2667
0
        if ((int)i == receiver) {
2668
0
            continue;
2669
0
        }
2670
2671
0
        if ((g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_CLOSEST) != 0) {
2672
0
            connected_closest[num_connected_closest] = i;
2673
0
            ++num_connected_closest;
2674
0
            continue;
2675
0
        }
2676
2677
0
        if (send_lossy_group_peer(g_c->fr_c, g->connections[i].number, PACKET_ID_LOSSY_CONFERENCE,
2678
0
                                  g->connections[i].group_number, data, length)) {
2679
0
            ++sent;
2680
0
        }
2681
0
    }
2682
2683
0
    if (num_connected_closest == 0) {
2684
0
        return sent;
2685
0
    }
2686
2687
0
    unsigned int to_send[2] = {0, 0};
2688
0
    uint64_t comp_val_old[2] = {(uint64_t) -1, (uint64_t) -1};
2689
2690
0
    for (unsigned int i = 0; i < num_connected_closest; ++i) {
2691
0
        uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE] = {0};
2692
0
        get_friendcon_public_keys(real_pk, nullptr, g_c->fr_c, g->connections[connected_closest[i]].number);
2693
0
        const uint64_t comp_val = calculate_comp_value(g->real_pk, real_pk);
2694
2695
0
        for (uint8_t j = 0; j < 2; ++j) {
2696
0
            if (j > 0 ? (comp_val > comp_val_old[j]) : (comp_val < comp_val_old[j])) {
2697
0
                to_send[j] = connected_closest[i];
2698
0
                comp_val_old[j] = comp_val;
2699
0
            }
2700
0
        }
2701
0
    }
2702
2703
0
    for (uint8_t j = 0; j < 2; ++j) {
2704
0
        if (j > 0 && to_send[1] == to_send[0]) {
2705
0
            break;
2706
0
        }
2707
2708
0
        if (send_lossy_group_peer(g_c->fr_c, g->connections[to_send[j]].number, PACKET_ID_LOSSY_CONFERENCE,
2709
0
                                  g->connections[to_send[j]].group_number, data, length)) {
2710
0
            ++sent;
2711
0
        }
2712
0
    }
2713
2714
0
    return sent;
2715
0
}
2716
2717
/** @brief Send data of len with message_id to groupnumber.
2718
 *
2719
 * @return number of peers it was sent to on success.
2720
 * @retval -1 if groupnumber is invalid.
2721
 * @retval -2 if message is too long.
2722
 * @retval -3 if we are not connected to the group.
2723
 * @retval -4 if message failed to send.
2724
 */
2725
static int send_message_group(const Group_Chats *g_c, uint32_t groupnumber, uint8_t message_id, const uint8_t *data,
2726
                              uint16_t len)
2727
1.20k
{
2728
1.20k
    assert(len == 0 || data != nullptr);
2729
1.20k
    Group_c *g = get_group_c(g_c, groupnumber);
2730
2731
1.20k
    if (g == nullptr) {
2732
0
        return -1;
2733
0
    }
2734
2735
1.20k
    if (len > MAX_GROUP_MESSAGE_DATA_LEN) {
2736
0
        return -2;
2737
0
    }
2738
2739
1.20k
    if (g->status != GROUPCHAT_STATUS_CONNECTED || count_connected(g) == 0) {
2740
1.20k
        return -3;
2741
1.20k
    }
2742
2743
0
    const uint16_t packet_size = sizeof(uint16_t) + sizeof(uint32_t) + 1 + len;
2744
0
    VLA(uint8_t, packet, packet_size);
2745
0
    const uint16_t peer_num = net_htons(g->peer_number);
2746
0
    memcpy(packet, &peer_num, sizeof(peer_num));
2747
2748
0
    ++g->message_number;
2749
2750
0
    if (g->message_number == 0) {
2751
0
        ++g->message_number;
2752
0
    }
2753
2754
0
    const uint32_t message_num = net_htonl(g->message_number);
2755
0
    memcpy(packet + sizeof(uint16_t), &message_num, sizeof(message_num));
2756
2757
0
    packet[sizeof(uint16_t) + sizeof(uint32_t)] = message_id;
2758
2759
0
    if (len != 0) {
2760
0
        memcpy(packet + sizeof(uint16_t) + sizeof(uint32_t) + 1, data, len);
2761
0
    }
2762
2763
0
    const unsigned int ret = send_message_all_connections(g_c, g, packet, packet_size, -1);
2764
2765
0
    if (ret == 0) {
2766
0
        return -4;
2767
0
    }
2768
2769
0
    return ret;
2770
0
}
2771
2772
/** @brief send a group message
2773
 * @retval 0 on success
2774
 * @see send_message_group for error codes.
2775
 */
2776
int group_message_send(const Group_Chats *g_c, uint32_t groupnumber, const uint8_t *message, uint16_t length)
2777
0
{
2778
0
    const int ret = send_message_group(g_c, groupnumber, PACKET_ID_MESSAGE, message, length);
2779
2780
0
    if (ret > 0) {
2781
0
        return 0;
2782
0
    }
2783
2784
0
    return ret;
2785
0
}
2786
2787
/** @brief send a group action
2788
 * @retval 0 on success
2789
 * @see send_message_group for error codes.
2790
 */
2791
int group_action_send(const Group_Chats *g_c, uint32_t groupnumber, const uint8_t *action, uint16_t length)
2792
0
{
2793
0
    const int ret = send_message_group(g_c, groupnumber, PACKET_ID_ACTION, action, length);
2794
2795
0
    if (ret > 0) {
2796
0
        return 0;
2797
0
    }
2798
2799
0
    return ret;
2800
0
}
2801
2802
/** @brief High level function to send custom lossy packets.
2803
 *
2804
 * @retval -1 on failure.
2805
 * @retval 0 on success.
2806
 */
2807
int send_group_lossy_packet(const Group_Chats *g_c, uint32_t groupnumber, const uint8_t *data, uint16_t length)
2808
0
{
2809
    // TODO(irungentoo): length check here?
2810
0
    Group_c *g = get_group_c(g_c, groupnumber);
2811
2812
0
    if (g == nullptr) {
2813
0
        return -1;
2814
0
    }
2815
2816
0
    const uint16_t packet_size = sizeof(uint16_t) * 2 + length;
2817
0
    VLA(uint8_t, packet, packet_size);
2818
0
    const uint16_t peer_number = net_htons(g->peer_number);
2819
0
    memcpy(packet, &peer_number, sizeof(uint16_t));
2820
0
    const uint16_t message_num = net_htons(g->lossy_message_number);
2821
0
    memcpy(packet + sizeof(uint16_t), &message_num, sizeof(uint16_t));
2822
0
    memcpy(packet + sizeof(uint16_t) * 2, data, length);
2823
2824
0
    if (send_lossy_all_connections(g_c, g, packet, packet_size, -1) == 0) {
2825
0
        return -1;
2826
0
    }
2827
2828
0
    ++g->lossy_message_number;
2829
0
    return 0;
2830
0
}
2831
2832
non_null()
2833
static Message_Info *find_message_slot_or_reject(uint32_t message_number, uint8_t message_id, Group_Peer *peer)
2834
0
{
2835
0
    const bool ignore_older = message_id == GROUP_MESSAGE_NAME_ID || message_id == GROUP_MESSAGE_TITLE_ID;
2836
2837
0
    Message_Info *i;
2838
2839
0
    for (i = peer->last_message_infos; i < peer->last_message_infos + peer->num_last_message_infos; ++i) {
2840
0
        if (message_number - (i->message_number + 1) <= ((uint32_t)1 << 31)) {
2841
0
            break;
2842
0
        }
2843
2844
0
        if (message_number == i->message_number) {
2845
0
            return nullptr;
2846
0
        }
2847
2848
0
        if (ignore_older && message_id == i->message_id) {
2849
0
            return nullptr;
2850
0
        }
2851
0
    }
2852
2853
0
    return i;
2854
0
}
2855
2856
/** @brief Stores message info in `peer->last_message_infos`.
2857
 *
2858
 * @retval true if message should be processed.
2859
 * @retval false otherwise.
2860
 */
2861
non_null()
2862
static bool check_message_info(uint32_t message_number, uint8_t message_id, Group_Peer *peer)
2863
0
{
2864
0
    Message_Info *const i = find_message_slot_or_reject(message_number, message_id, peer);
2865
2866
0
    if (i == nullptr) {
2867
0
        return false;
2868
0
    }
2869
2870
0
    if (i == peer->last_message_infos + MAX_LAST_MESSAGE_INFOS) {
2871
0
        return false;
2872
0
    }
2873
2874
0
    if (peer->num_last_message_infos < MAX_LAST_MESSAGE_INFOS) {
2875
0
        ++peer->num_last_message_infos;
2876
0
    }
2877
2878
0
    memmove(i + 1, i, (&peer->last_message_infos[peer->num_last_message_infos - 1] - i) * sizeof(Message_Info));
2879
2880
0
    i->message_number = message_number;
2881
0
    i->message_id = message_id;
2882
2883
0
    return true;
2884
0
}
2885
2886
non_null(1, 3) nullable(6)
2887
static void handle_message_packet_group(Group_Chats *g_c, uint32_t groupnumber, const uint8_t *data, uint16_t length,
2888
                                        uint32_t connection_index, void *userdata)
2889
0
{
2890
0
    if (length < sizeof(uint16_t) + sizeof(uint32_t) + 1) {
2891
0
        return;
2892
0
    }
2893
2894
0
    Group_c *g = get_group_c(g_c, groupnumber);
2895
2896
0
    if (g == nullptr) {
2897
0
        return;
2898
0
    }
2899
2900
0
    uint16_t peer_number;
2901
0
    memcpy(&peer_number, data, sizeof(uint16_t));
2902
0
    peer_number = net_ntohs(peer_number);
2903
2904
0
    uint32_t message_number;
2905
0
    memcpy(&message_number, data + sizeof(uint16_t), sizeof(message_number));
2906
0
    message_number = net_ntohl(message_number);
2907
2908
0
    const uint8_t message_id = data[sizeof(uint16_t) + sizeof(message_number)];
2909
0
    const uint8_t *msg_data = data + sizeof(uint16_t) + sizeof(message_number) + 1;
2910
0
    const uint16_t msg_data_len = length - (sizeof(uint16_t) + sizeof(message_number) + 1);
2911
2912
0
    const bool ignore_frozen = message_id == GROUP_MESSAGE_FREEZE_PEER_ID;
2913
2914
0
    const int index = ignore_frozen ? get_peer_index(g, peer_number)
2915
0
                      : note_peer_active(g_c, groupnumber, peer_number, userdata);
2916
2917
0
    if (index == -1) {
2918
0
        if (ignore_frozen) {
2919
0
            return;
2920
0
        }
2921
2922
0
        if (g->connections[connection_index].type != GROUPCHAT_CONNECTION_ONLINE) {
2923
0
            return;
2924
0
        }
2925
2926
        /* If we don't know the peer this packet came from, then we query the
2927
         * list of peers from the relaying peer.
2928
         * (They wouldn't have relayed it if they didn't know the peer.) */
2929
0
        send_peer_query(g_c, g->connections[connection_index].number, g->connections[connection_index].group_number);
2930
0
        return;
2931
0
    }
2932
2933
0
    if (g->num_introducer_connections > 0 && count_connected(g) > DESIRED_CLOSEST) {
2934
0
        for (uint32_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
2935
0
            if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE
2936
0
                    || (g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_INTRODUCER) == 0
2937
0
                    || i == connection_index) {
2938
0
                continue;
2939
0
            }
2940
2941
0
            uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2942
0
            get_friendcon_public_keys(real_pk, nullptr, g_c->fr_c, g->connections[i].number);
2943
2944
0
            if (pk_equal(g->group[index].real_pk, real_pk)) {
2945
                /* Received message from peer relayed via another peer, so
2946
                 * the introduction was successful */
2947
0
                remove_connection_reason(g_c, g, i, GROUPCHAT_CONNECTION_REASON_INTRODUCER);
2948
0
            }
2949
0
        }
2950
0
    }
2951
2952
0
    if (!check_message_info(message_number, message_id, &g->group[index])) {
2953
0
        return;
2954
0
    }
2955
2956
0
    uint8_t real_pk[CRYPTO_PUBLIC_KEY_SIZE];
2957
0
    get_friendcon_public_keys(real_pk, nullptr, g_c->fr_c, g->connections[connection_index].number);
2958
0
    const bool direct_from_sender = pk_equal(g->group[index].real_pk, real_pk);
2959
2960
0
    switch (message_id) {
2961
0
        case GROUP_MESSAGE_PING_ID: {
2962
0
            break;
2963
0
        }
2964
2965
0
        case GROUP_MESSAGE_NEW_PEER_ID: {
2966
0
            if (msg_data_len != GROUP_MESSAGE_NEW_PEER_LENGTH) {
2967
0
                return;
2968
0
            }
2969
2970
0
            uint16_t new_peer_number;
2971
0
            memcpy(&new_peer_number, msg_data, sizeof(uint16_t));
2972
0
            new_peer_number = net_ntohs(new_peer_number);
2973
0
            addpeer(g_c, groupnumber, msg_data + sizeof(uint16_t), msg_data + sizeof(uint16_t) + CRYPTO_PUBLIC_KEY_SIZE,
2974
0
                    new_peer_number, userdata, true, true);
2975
0
            break;
2976
0
        }
2977
2978
0
        case GROUP_MESSAGE_KILL_PEER_ID:
2979
0
        case GROUP_MESSAGE_FREEZE_PEER_ID: {
2980
0
            if (msg_data_len != GROUP_MESSAGE_KILL_PEER_LENGTH) {
2981
0
                return;
2982
0
            }
2983
2984
0
            uint16_t kill_peer_number;
2985
0
            memcpy(&kill_peer_number, msg_data, sizeof(uint16_t));
2986
0
            kill_peer_number = net_ntohs(kill_peer_number);
2987
2988
0
            if (peer_number == kill_peer_number) {
2989
0
                if (message_id == GROUP_MESSAGE_KILL_PEER_ID) {
2990
0
                    delpeer(g_c, groupnumber, index, userdata);
2991
0
                } else {
2992
0
                    freeze_peer(g_c, groupnumber, index, userdata);
2993
0
                }
2994
0
            } else {
2995
0
                return;
2996
                // TODO(irungentoo):
2997
0
            }
2998
2999
0
            break;
3000
0
        }
3001
3002
0
        case GROUP_MESSAGE_NAME_ID: {
3003
0
            if (!setnick(g_c, groupnumber, index, msg_data, msg_data_len, userdata, true)) {
3004
0
                return;
3005
0
            }
3006
3007
0
            break;
3008
0
        }
3009
3010
0
        case GROUP_MESSAGE_TITLE_ID: {
3011
0
            if (!settitle(g_c, groupnumber, index, msg_data, msg_data_len, userdata)) {
3012
0
                return;
3013
0
            }
3014
3015
0
            break;
3016
0
        }
3017
3018
0
        case PACKET_ID_MESSAGE: {
3019
0
            if (msg_data_len == 0) {
3020
0
                return;
3021
0
            }
3022
3023
0
            VLA(uint8_t, newmsg, msg_data_len + 1);
3024
0
            memcpy(newmsg, msg_data, msg_data_len);
3025
0
            newmsg[msg_data_len] = 0;
3026
3027
            // TODO(irungentoo):
3028
0
            if (g_c->message_callback != nullptr) {
3029
0
                g_c->message_callback(g_c->m, groupnumber, index, 0, newmsg, msg_data_len, userdata);
3030
0
            }
3031
3032
0
            break;
3033
0
        }
3034
3035
0
        case PACKET_ID_ACTION: {
3036
0
            if (msg_data_len == 0) {
3037
0
                return;
3038
0
            }
3039
3040
0
            VLA(uint8_t, newmsg, msg_data_len + 1);
3041
0
            memcpy(newmsg, msg_data, msg_data_len);
3042
0
            newmsg[msg_data_len] = 0;
3043
3044
            // TODO(irungentoo):
3045
0
            if (g_c->message_callback != nullptr) {
3046
0
                g_c->message_callback(g_c->m, groupnumber, index, 1, newmsg, msg_data_len, userdata);
3047
0
            }
3048
3049
0
            break;
3050
0
        }
3051
3052
0
        default: {
3053
0
            return;
3054
0
        }
3055
0
    }
3056
3057
    /* If the packet was received from the peer who sent the message, relay it
3058
     * back. When the sender only has one group connection (e.g. because there
3059
     * are only two peers in the group), this is the only way for them to
3060
     * receive their own message. */
3061
0
    send_message_all_connections(g_c, g, data, length, direct_from_sender ? -1 : connection_index);
3062
0
}
3063
3064
static int g_handle_packet(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata)
3065
0
{
3066
0
    Group_Chats *g_c = (Group_Chats *)object;
3067
3068
0
    if (length < 1 + sizeof(uint16_t) + 1) {
3069
0
        return -1;
3070
0
    }
3071
3072
0
    if (data[0] == PACKET_ID_ONLINE_PACKET) {
3073
0
        return handle_packet_online(g_c, friendcon_id, data + 1, length - 1);
3074
0
    }
3075
3076
0
    if (data[0] == PACKET_ID_REJOIN_CONFERENCE) {
3077
0
        return handle_packet_rejoin(g_c, friendcon_id, data + 1, length - 1, userdata);
3078
0
    }
3079
3080
0
    uint16_t groupnumber;
3081
0
    memcpy(&groupnumber, data + 1, sizeof(uint16_t));
3082
0
    groupnumber = net_ntohs(groupnumber);
3083
0
    const Group_c *g = get_group_c(g_c, groupnumber);
3084
3085
0
    if (g == nullptr) {
3086
0
        return -1;
3087
0
    }
3088
3089
0
    const int index = friend_in_connections(g, friendcon_id);
3090
3091
0
    if (index == -1) {
3092
0
        return -1;
3093
0
    }
3094
3095
0
    if (data[0] == PACKET_ID_DIRECT_CONFERENCE) {
3096
0
        handle_direct_packet(g_c, groupnumber, data + 1 + sizeof(uint16_t),
3097
0
                             length - (1 + sizeof(uint16_t)), index, userdata);
3098
0
        return 0;
3099
0
    }
3100
3101
0
    if (data[0] == PACKET_ID_MESSAGE_CONFERENCE) {
3102
0
        handle_message_packet_group(g_c, groupnumber, data + 1 + sizeof(uint16_t),
3103
0
                                    length - (1 + sizeof(uint16_t)), index, userdata);
3104
0
        return 0;
3105
0
    }
3106
3107
0
    return -1;
3108
0
}
3109
3110
/** @brief Did we already receive the lossy packet or not.
3111
 *
3112
 * @retval -1 on failure.
3113
 * @retval 0 if packet was not received.
3114
 * @retval 1 if packet was received.
3115
 *
3116
 * TODO(irungentoo): test this
3117
 */
3118
non_null()
3119
static int lossy_packet_not_received(const Group_c *g, int peer_index, uint16_t message_number)
3120
0
{
3121
0
    if (peer_index == -1) {
3122
0
        return -1;
3123
0
    }
3124
3125
0
    if (g->group[peer_index].bottom_lossy_number == g->group[peer_index].top_lossy_number) {
3126
0
        g->group[peer_index].top_lossy_number = message_number;
3127
0
        g->group[peer_index].bottom_lossy_number = (message_number - MAX_LOSSY_COUNT) + 1;
3128
0
        g->group[peer_index].recv_lossy[message_number % MAX_LOSSY_COUNT] = 1;
3129
0
        return 0;
3130
0
    }
3131
3132
0
    if ((uint16_t)(message_number - g->group[peer_index].bottom_lossy_number) < MAX_LOSSY_COUNT) {
3133
0
        if (g->group[peer_index].recv_lossy[message_number % MAX_LOSSY_COUNT] != 0) {
3134
0
            return 1;
3135
0
        }
3136
3137
0
        g->group[peer_index].recv_lossy[message_number % MAX_LOSSY_COUNT] = 1;
3138
0
        return 0;
3139
0
    }
3140
3141
0
    if ((uint16_t)(message_number - g->group[peer_index].bottom_lossy_number) > (1 << 15)) {
3142
0
        return -1;
3143
0
    }
3144
3145
0
    const uint16_t top_distance = message_number - g->group[peer_index].top_lossy_number;
3146
3147
0
    if (top_distance >= MAX_LOSSY_COUNT) {
3148
0
        crypto_memzero(g->group[peer_index].recv_lossy, sizeof(g->group[peer_index].recv_lossy));
3149
0
    } else {  // top_distance < MAX_LOSSY_COUNT
3150
0
        for (unsigned int i = g->group[peer_index].bottom_lossy_number;
3151
0
                i != g->group[peer_index].bottom_lossy_number + top_distance;
3152
0
                ++i) {
3153
0
            g->group[peer_index].recv_lossy[i % MAX_LOSSY_COUNT] = 0;
3154
0
        }
3155
0
    }
3156
3157
0
    g->group[peer_index].top_lossy_number = message_number;
3158
0
    g->group[peer_index].bottom_lossy_number = (message_number - MAX_LOSSY_COUNT) + 1;
3159
0
    g->group[peer_index].recv_lossy[message_number % MAX_LOSSY_COUNT] = 1;
3160
3161
0
    return 0;
3162
3163
0
}
3164
3165
/** Does this group type make use of lossy packets? */
3166
static bool type_uses_lossy(uint8_t type)
3167
0
{
3168
0
    return type == GROUPCHAT_TYPE_AV;
3169
0
}
3170
3171
static int handle_lossy(void *object, int friendcon_id, const uint8_t *data, uint16_t length, void *userdata)
3172
0
{
3173
0
    const Group_Chats *g_c = (const Group_Chats *)object;
3174
3175
0
    if (data[0] != PACKET_ID_LOSSY_CONFERENCE) {
3176
0
        return -1;
3177
0
    }
3178
3179
0
    if (length < 1 + sizeof(uint16_t) * 3 + 1) {
3180
0
        return -1;
3181
0
    }
3182
3183
0
    uint16_t groupnumber;
3184
0
    uint16_t peer_number;
3185
0
    uint16_t message_number;
3186
0
    memcpy(&groupnumber, data + 1, sizeof(uint16_t));
3187
0
    memcpy(&peer_number, data + 1 + sizeof(uint16_t), sizeof(uint16_t));
3188
0
    memcpy(&message_number, data + 1 + sizeof(uint16_t) * 2, sizeof(uint16_t));
3189
0
    groupnumber = net_ntohs(groupnumber);
3190
0
    peer_number = net_ntohs(peer_number);
3191
0
    message_number = net_ntohs(message_number);
3192
3193
0
    const Group_c *g = get_group_c(g_c, groupnumber);
3194
3195
0
    if (g == nullptr) {
3196
0
        return -1;
3197
0
    }
3198
3199
0
    if (!type_uses_lossy(g->type)) {
3200
0
        return -1;
3201
0
    }
3202
3203
0
    const int index = friend_in_connections(g, friendcon_id);
3204
3205
0
    if (index == -1) {
3206
0
        return -1;
3207
0
    }
3208
3209
0
    if (peer_number == g->peer_number) {
3210
0
        return -1;
3211
0
    }
3212
3213
0
    const int peer_index = get_peer_index(g, peer_number);
3214
3215
0
    if (peer_index == -1) {
3216
0
        return -1;
3217
0
    }
3218
3219
0
    if (lossy_packet_not_received(g, peer_index, message_number) != 0) {
3220
0
        return -1;
3221
0
    }
3222
3223
0
    const uint8_t *lossy_data = data + 1 + sizeof(uint16_t) * 3;
3224
0
    uint16_t lossy_length = length - (1 + sizeof(uint16_t) * 3);
3225
0
    const uint8_t message_id = lossy_data[0];
3226
0
    ++lossy_data;
3227
0
    --lossy_length;
3228
3229
0
    send_lossy_all_connections(g_c, g, data + 1 + sizeof(uint16_t), length - (1 + sizeof(uint16_t)), index);
3230
3231
0
    if (g_c->lossy_packethandlers[message_id] == nullptr) {
3232
0
        return -1;
3233
0
    }
3234
3235
0
    if (g_c->lossy_packethandlers[message_id](g->object, groupnumber, peer_index, g->group[peer_index].object,
3236
0
            lossy_data, lossy_length) == -1) {
3237
0
        return -1;
3238
0
    }
3239
3240
0
    return 0;
3241
0
}
3242
3243
/** @brief Set the object that is tied to the group chat.
3244
 *
3245
 * @retval 0 on success.
3246
 * @retval -1 on failure
3247
 */
3248
int group_set_object(const Group_Chats *g_c, uint32_t groupnumber, void *object)
3249
0
{
3250
0
    Group_c *g = get_group_c(g_c, groupnumber);
3251
3252
0
    if (g == nullptr) {
3253
0
        return -1;
3254
0
    }
3255
3256
0
    g->object = object;
3257
0
    return 0;
3258
0
}
3259
3260
/** @brief Set the object that is tied to the group peer.
3261
 *
3262
 * @retval 0 on success.
3263
 * @retval -1 on failure
3264
 */
3265
int group_peer_set_object(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber, void *object)
3266
0
{
3267
0
    const Group_c *g = get_group_c(g_c, groupnumber);
3268
3269
0
    if (g == nullptr) {
3270
0
        return -1;
3271
0
    }
3272
3273
0
    if (peernumber >= g->numpeers) {
3274
0
        return -1;
3275
0
    }
3276
3277
0
    g->group[peernumber].object = object;
3278
0
    return 0;
3279
0
}
3280
3281
/** @brief Return the object tied to the group chat previously set by group_set_object.
3282
 *
3283
 * @retval NULL on failure.
3284
 * @return object on success.
3285
 */
3286
void *group_get_object(const Group_Chats *g_c, uint32_t groupnumber)
3287
0
{
3288
0
    const Group_c *g = get_group_c(g_c, groupnumber);
3289
3290
0
    if (g == nullptr) {
3291
0
        return nullptr;
3292
0
    }
3293
3294
0
    return g->object;
3295
0
}
3296
3297
/** @brief Return the object tied to the group chat peer previously set by group_peer_set_object.
3298
 *
3299
 * @retval NULL on failure.
3300
 * @return object on success.
3301
 */
3302
void *group_peer_get_object(const Group_Chats *g_c, uint32_t groupnumber, uint32_t peernumber)
3303
0
{
3304
0
    const Group_c *g = get_group_c(g_c, groupnumber);
3305
3306
0
    if (g == nullptr) {
3307
0
        return nullptr;
3308
0
    }
3309
3310
0
    if (peernumber >= g->numpeers) {
3311
0
        return nullptr;
3312
0
    }
3313
3314
0
    return g->group[peernumber].object;
3315
0
}
3316
3317
/** Interval in seconds to send ping messages */
3318
0
#define GROUP_PING_INTERVAL 20
3319
3320
static bool ping_groupchat(const Group_Chats *g_c, uint32_t groupnumber)
3321
0
{
3322
0
    Group_c *g = get_group_c(g_c, groupnumber);
3323
3324
0
    if (g == nullptr) {
3325
0
        return false;
3326
0
    }
3327
3328
0
    if (mono_time_is_timeout(g_c->mono_time, g->last_sent_ping, GROUP_PING_INTERVAL)) {
3329
0
        if (group_ping_send(g_c, groupnumber)) {
3330
0
            g->last_sent_ping = mono_time_get(g_c->mono_time);
3331
0
        }
3332
0
    }
3333
3334
0
    return true;
3335
0
}
3336
3337
/** Seconds of inactivity after which to freeze a peer */
3338
0
#define FREEZE_TIMEOUT (GROUP_PING_INTERVAL * 3)
3339
3340
non_null(1) nullable(3)
3341
static bool groupchat_freeze_timedout(Group_Chats *g_c, uint32_t groupnumber, void *userdata)
3342
0
{
3343
0
    Group_c *g = get_group_c(g_c, groupnumber);
3344
3345
0
    if (g == nullptr) {
3346
0
        return false;
3347
0
    }
3348
3349
0
    for (uint32_t i = 0; i < g->numpeers; ++i) {
3350
0
        if (g->group[i].peer_number == g->peer_number) {
3351
0
            continue;
3352
0
        }
3353
3354
0
        if (mono_time_is_timeout(g_c->mono_time, g->group[i].last_active, FREEZE_TIMEOUT)) {
3355
0
            freeze_peer(g_c, groupnumber, i, userdata);
3356
0
        }
3357
0
    }
3358
3359
0
    if (g->numpeers <= 1) {
3360
0
        g->title_fresh = false;
3361
0
    }
3362
3363
0
    return true;
3364
0
}
3365
3366
/** Push non-empty slots to start. */
3367
non_null()
3368
static void squash_connections(Group_c *g)
3369
0
{
3370
0
    uint16_t num_connected = 0;
3371
3372
0
    for (uint16_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
3373
0
        if (g->connections[i].type != GROUPCHAT_CONNECTION_NONE) {
3374
0
            g->connections[num_connected] = g->connections[i];
3375
0
            ++num_connected;
3376
0
        }
3377
0
    }
3378
3379
0
    for (uint16_t i = num_connected; i < MAX_GROUP_CONNECTIONS; ++i) {
3380
0
        g->connections[i].type = GROUPCHAT_CONNECTION_NONE;
3381
0
    }
3382
0
}
3383
3384
0
#define MIN_EMPTY_CONNECTIONS (1 + MAX_GROUP_CONNECTIONS / 10)
3385
3386
non_null()
3387
static uint16_t empty_connection_count(const Group_c *g)
3388
0
{
3389
0
    uint16_t to_clear = MIN_EMPTY_CONNECTIONS;
3390
3391
0
    for (uint16_t i = 0; i < MAX_GROUP_CONNECTIONS; ++i) {
3392
0
        if (g->connections[i].type == GROUPCHAT_CONNECTION_NONE) {
3393
0
            --to_clear;
3394
3395
0
            if (to_clear == 0) {
3396
0
                break;
3397
0
            }
3398
0
        }
3399
0
    }
3400
3401
0
    return to_clear;
3402
0
}
3403
3404
/**
3405
 * @brief Remove old connections as necessary to ensure we have space for new
3406
 *   connections.
3407
 *
3408
 * This invalidates connections array indices (which is
3409
 * why we do this periodically rather than on adding a connection).
3410
 */
3411
non_null()
3412
static void clean_connections(Group_Chats *g_c, Group_c *g)
3413
0
{
3414
0
    for (uint16_t to_clear = empty_connection_count(g); to_clear > 0; --to_clear) {
3415
        // Remove a connection. Prefer non-closest connections, and given
3416
        // that prefer non-online connections, and given that prefer earlier
3417
        // slots.
3418
0
        uint16_t i = 0;
3419
3420
0
        while (i < MAX_GROUP_CONNECTIONS
3421
0
                && (g->connections[i].type != GROUPCHAT_CONNECTION_CONNECTING
3422
0
                    || (g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_CLOSEST) != 0)) {
3423
0
            ++i;
3424
0
        }
3425
3426
0
        if (i == MAX_GROUP_CONNECTIONS) {
3427
0
            i = 0;
3428
3429
0
            while (i < MAX_GROUP_CONNECTIONS - to_clear
3430
0
                    && (g->connections[i].type != GROUPCHAT_CONNECTION_ONLINE
3431
0
                        || (g->connections[i].reasons & GROUPCHAT_CONNECTION_REASON_CLOSEST) != 0)) {
3432
0
                ++i;
3433
0
            }
3434
0
        }
3435
3436
0
        if (g->connections[i].type != GROUPCHAT_CONNECTION_NONE) {
3437
0
            remove_connection(g_c, g, i);
3438
0
        }
3439
0
    }
3440
3441
0
    squash_connections(g);
3442
0
}
3443
3444
/** Send current name (set in messenger) to all online groups. */
3445
void send_name_all_groups(const Group_Chats *g_c)
3446
0
{
3447
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3448
0
        Group_c *g = get_group_c(g_c, i);
3449
3450
0
        if (g == nullptr) {
3451
0
            continue;
3452
0
        }
3453
3454
0
        if (g->status == GROUPCHAT_STATUS_CONNECTED) {
3455
0
            group_name_send(g_c, i, g_c->m->name, g_c->m->name_length);
3456
0
            g->need_send_name = false;
3457
0
        }
3458
0
    }
3459
0
}
3460
3461
1.07k
#define SAVED_PEER_SIZE_CONSTANT (2 * CRYPTO_PUBLIC_KEY_SIZE + sizeof(uint16_t) + sizeof(uint64_t) + 1)
3462
3463
non_null()
3464
static uint32_t saved_peer_size(const Group_Peer *peer)
3465
648
{
3466
648
    return SAVED_PEER_SIZE_CONSTANT + peer->nick_len;
3467
648
}
3468
3469
non_null()
3470
static uint8_t *save_peer(const Group_Peer *peer, uint8_t *data)
3471
216
{
3472
216
    memcpy(data, peer->real_pk, CRYPTO_PUBLIC_KEY_SIZE);
3473
216
    data += CRYPTO_PUBLIC_KEY_SIZE;
3474
3475
216
    memcpy(data, peer->temp_pk, CRYPTO_PUBLIC_KEY_SIZE);
3476
216
    data += CRYPTO_PUBLIC_KEY_SIZE;
3477
3478
216
    host_to_lendian_bytes16(data, peer->peer_number);
3479
216
    data += sizeof(uint16_t);
3480
3481
216
    host_to_lendian_bytes64(data, peer->last_active);
3482
216
    data += sizeof(uint64_t);
3483
3484
    // TODO(iphydf): This looks broken: nick_len can be > 255.
3485
216
    *data = peer->nick_len;
3486
216
    ++data;
3487
3488
216
    memcpy(data, peer->nick, peer->nick_len);
3489
216
    data += peer->nick_len;
3490
3491
216
    return data;
3492
216
}
3493
3494
6.45k
#define SAVED_CONF_SIZE_CONSTANT (1 + GROUP_ID_LENGTH + sizeof(uint32_t) \
3495
6.45k
      + sizeof(uint16_t) + sizeof(uint16_t) + sizeof(uint32_t) + 1)
3496
3497
non_null()
3498
static uint32_t saved_conf_size(const Group_c *g)
3499
3.25k
{
3500
3.25k
    uint32_t len = SAVED_CONF_SIZE_CONSTANT + g->title_len;
3501
3502
7.16k
    for (uint32_t j = 0; j < g->numpeers + g->numfrozen; ++j) {
3503
3.90k
        const Group_Peer *peer = (j < g->numpeers) ? &g->group[j] : &g->frozen[j - g->numpeers];
3504
3505
3.90k
        if (pk_equal(peer->real_pk, g->real_pk)) {
3506
3.25k
            continue;
3507
3.25k
        }
3508
3509
648
        len += saved_peer_size(peer);
3510
648
    }
3511
3512
3.25k
    return len;
3513
3.25k
}
3514
3515
/**
3516
 * Save a future message number. The save will remain valid until we have sent
3517
 * this many more messages.
3518
 */
3519
1.08k
#define SAVE_OFFSET_MESSAGE_NUMBER (1 << 16)
3520
1.08k
#define SAVE_OFFSET_LOSSY_MESSAGE_NUMBER (1 << 13)
3521
3522
non_null()
3523
static uint8_t *save_conf(const Group_c *g, uint8_t *data)
3524
1.08k
{
3525
1.08k
    *data = g->type;
3526
1.08k
    ++data;
3527
3528
1.08k
    memcpy(data, g->id, GROUP_ID_LENGTH);
3529
1.08k
    data += GROUP_ID_LENGTH;
3530
3531
1.08k
    host_to_lendian_bytes32(data, g->message_number + SAVE_OFFSET_MESSAGE_NUMBER);
3532
1.08k
    data += sizeof(uint32_t);
3533
3534
1.08k
    host_to_lendian_bytes16(data, g->lossy_message_number + SAVE_OFFSET_LOSSY_MESSAGE_NUMBER);
3535
1.08k
    data += sizeof(uint16_t);
3536
3537
1.08k
    host_to_lendian_bytes16(data, g->peer_number);
3538
1.08k
    data += sizeof(uint16_t);
3539
3540
1.08k
    uint8_t *const numsaved_location = data;
3541
1.08k
    data += sizeof(uint32_t);
3542
3543
1.08k
    *data = g->title_len;
3544
1.08k
    ++data;
3545
3546
1.08k
    memcpy(data, g->title, g->title_len);
3547
1.08k
    data += g->title_len;
3548
3549
1.08k
    uint32_t numsaved = 0;
3550
3551
2.38k
    for (uint32_t j = 0; j < g->numpeers + g->numfrozen; ++j) {
3552
1.30k
        const Group_Peer *peer = (j < g->numpeers) ? &g->group[j] : &g->frozen[j - g->numpeers];
3553
3554
1.30k
        if (pk_equal(peer->real_pk, g->real_pk)) {
3555
1.08k
            continue;
3556
1.08k
        }
3557
3558
216
        data = save_peer(peer, data);
3559
216
        ++numsaved;
3560
216
    }
3561
3562
1.08k
    host_to_lendian_bytes32(numsaved_location, numsaved);
3563
3564
1.08k
    return data;
3565
1.08k
}
3566
3567
non_null()
3568
static uint32_t conferences_section_size(const Group_Chats *g_c)
3569
3.82k
{
3570
3.82k
    uint32_t len = 0;
3571
3572
7.08k
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3573
3.25k
        const Group_c *g = get_group_c(g_c, i);
3574
3575
3.25k
        if (g == nullptr || g->status != GROUPCHAT_STATUS_CONNECTED) {
3576
0
            continue;
3577
0
        }
3578
3579
3.25k
        len += saved_conf_size(g);
3580
3.25k
    }
3581
3582
3.82k
    return len;
3583
3.82k
}
3584
3585
uint32_t conferences_size(const Group_Chats *g_c)
3586
2.55k
{
3587
2.55k
    return 2 * sizeof(uint32_t) + conferences_section_size(g_c);
3588
2.55k
}
3589
3590
uint8_t *conferences_save(const Group_Chats *g_c, uint8_t *data)
3591
1.27k
{
3592
1.27k
    const uint32_t len = conferences_section_size(g_c);
3593
1.27k
    data = state_write_section_header(data, STATE_COOKIE_TYPE, len, STATE_TYPE_CONFERENCES);
3594
3595
2.36k
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3596
1.08k
        const Group_c *g = get_group_c(g_c, i);
3597
3598
1.08k
        if (g == nullptr || g->status != GROUPCHAT_STATUS_CONNECTED) {
3599
0
            continue;
3600
0
        }
3601
3602
1.08k
        data = save_conf(g, data);
3603
1.08k
    }
3604
3605
1.27k
    return data;
3606
1.27k
}
3607
3608
/**
3609
 * @brief load_group Load a Group section from a save file
3610
 * @param g Group to load
3611
 * @param g_c Reference to all groupchats, need for utility functions
3612
 * @param data Start of the data to deserialze
3613
 * @param length Length of data
3614
 * @return 0 on error, number of consumed bytes otherwise
3615
 */
3616
non_null()
3617
static uint32_t load_group(Group_c *g, const Group_Chats *g_c, const uint8_t *data, uint32_t length)
3618
2.93k
{
3619
2.93k
    const uint8_t *init_data = data;
3620
3621
    // Initialize to default values so we can unconditionally free in case of an error
3622
2.93k
    setup_conference(g);
3623
3624
2.93k
    g->type = *data;
3625
2.93k
    ++data;
3626
3627
2.93k
    memcpy(g->id, data, GROUP_ID_LENGTH);
3628
2.93k
    data += GROUP_ID_LENGTH;
3629
3630
2.93k
    lendian_bytes_to_host32(&g->message_number, data);
3631
2.93k
    data += sizeof(uint32_t);
3632
3633
2.93k
    lendian_bytes_to_host16(&g->lossy_message_number, data);
3634
2.93k
    data += sizeof(uint16_t);
3635
3636
2.93k
    lendian_bytes_to_host16(&g->peer_number, data);
3637
2.93k
    data += sizeof(uint16_t);
3638
3639
2.93k
    lendian_bytes_to_host32(&g->numfrozen, data);
3640
2.93k
    data += sizeof(uint32_t);
3641
3642
2.93k
    g->title_len = *data;
3643
3644
2.93k
    if (g->title_len > MAX_NAME_LENGTH) {
3645
9
        return 0;
3646
9
    }
3647
3648
2.92k
    ++data;
3649
3650
2.92k
    assert((data - init_data) < UINT32_MAX);
3651
3652
2.92k
    if (length < (uint32_t)(data - init_data) + g->title_len) {
3653
2
        return 0;
3654
2
    }
3655
3656
2.92k
    memcpy(g->title, data, g->title_len);
3657
2.92k
    data += g->title_len;
3658
3659
3.31k
    for (uint32_t j = 0; j < g->numfrozen; ++j) {
3660
3661
429
        assert((data - init_data) < UINT32_MAX);
3662
3663
429
        if (length < (uint32_t)(data - init_data) + SAVED_PEER_SIZE_CONSTANT) {
3664
28
            return 0;
3665
28
        }
3666
3667
        // This is inefficient, but allows us to check data consistency before allocating memory
3668
401
        Group_Peer *tmp_frozen = (Group_Peer *)mem_vrealloc(g_c->mem, g->frozen, j + 1, sizeof(Group_Peer));
3669
3670
401
        if (tmp_frozen == nullptr) {
3671
            // Memory allocation failure
3672
0
            return 0;
3673
0
        }
3674
3675
401
        g->frozen = tmp_frozen;
3676
3677
401
        Group_Peer *peer = &g->frozen[j];
3678
401
        *peer = empty_group_peer;
3679
3680
401
        pk_copy(peer->real_pk, data);
3681
401
        data += CRYPTO_PUBLIC_KEY_SIZE;
3682
401
        pk_copy(peer->temp_pk, data);
3683
401
        data += CRYPTO_PUBLIC_KEY_SIZE;
3684
3685
401
        lendian_bytes_to_host16(&peer->peer_number, data);
3686
401
        data += sizeof(uint16_t);
3687
3688
401
        lendian_bytes_to_host64(&peer->last_active, data);
3689
401
        data += sizeof(uint64_t);
3690
3691
401
        peer->nick_len = *data;
3692
3693
401
        if (peer->nick_len > MAX_NAME_LENGTH) {
3694
5
            return 0;
3695
5
        }
3696
3697
396
        ++data;
3698
396
        assert((data - init_data) < UINT32_MAX);
3699
3700
396
        if (length < (uint32_t)(data - init_data) + peer->nick_len) {
3701
3
            return 0;
3702
3
        }
3703
3704
393
        memcpy(peer->nick, data, peer->nick_len);
3705
393
        data += peer->nick_len;
3706
3707
        // NOTE: this relies on friends being loaded before conferences.
3708
393
        peer->is_friend = getfriend_id(g_c->m, peer->real_pk) != -1;
3709
393
    }
3710
3711
2.89k
    if (g->numfrozen > g->maxfrozen) {
3712
0
        g->maxfrozen = g->numfrozen;
3713
0
    }
3714
3715
2.89k
    g->status = GROUPCHAT_STATUS_CONNECTED;
3716
3717
2.89k
    pk_copy(g->real_pk, nc_get_self_public_key(g_c->m->net_crypto));
3718
3719
2.89k
    assert((data - init_data) < UINT32_MAX);
3720
3721
2.89k
    return (uint32_t)(data - init_data);
3722
2.89k
}
3723
3724
non_null()
3725
static State_Load_Status load_conferences_helper(Group_Chats *g_c, const uint8_t *data, uint32_t length)
3726
313
{
3727
313
    const uint8_t *init_data = data;
3728
3729
3.19k
    while (length >= (uint32_t)(data - init_data) + SAVED_CONF_SIZE_CONSTANT) {
3730
2.93k
        const int groupnumber = create_group_chat(g_c);
3731
3732
        // Helpful for testing
3733
2.93k
        assert(groupnumber != -1);
3734
3735
2.93k
        if (groupnumber == -1) {
3736
            // If this fails there's a serious problem, don't bother with cleanup
3737
0
            LOGGER_ERROR(g_c->m->log, "conference creation failed");
3738
0
            return STATE_LOAD_STATUS_ERROR;
3739
0
        }
3740
3741
2.93k
        Group_c *g = &g_c->chats[groupnumber];
3742
3743
2.93k
        const uint32_t consumed = load_group(g, g_c, data, length - (uint32_t)(data - init_data));
3744
3745
2.93k
        if (consumed == 0) {
3746
            // remove partially loaded stuff, wipe_group_chat must be able to wipe a partially loaded group
3747
47
            const bool ret = wipe_group_chat(g_c, groupnumber);
3748
3749
            // HACK: suppress unused variable warning
3750
47
            if (!ret) {
3751
                // wipe_group_chat(...) must be able to wipe partially allocated groups
3752
0
                assert(ret);
3753
0
            }
3754
3755
47
            LOGGER_ERROR(g_c->m->log, "conference loading failed");
3756
47
            return STATE_LOAD_STATUS_ERROR;
3757
47
        }
3758
3759
2.89k
        data += consumed;
3760
3761
2.89k
        const int peer_index = addpeer(g_c, groupnumber, g->real_pk, dht_get_self_public_key(g_c->m->dht), g->peer_number,
3762
2.89k
                                       nullptr, true, false);
3763
3764
2.89k
        if (peer_index == -1) {
3765
6
            LOGGER_ERROR(g_c->m->log, "adding peer %d failed", g->peer_number);
3766
6
            return STATE_LOAD_STATUS_ERROR;
3767
6
        }
3768
3769
2.88k
        setnick(g_c, groupnumber, peer_index, g_c->m->name, g_c->m->name_length, nullptr, false);
3770
2.88k
    }
3771
3772
260
    return STATE_LOAD_STATUS_CONTINUE;
3773
313
}
3774
3775
non_null()
3776
static State_Load_Status load_conferences(Group_Chats *g_c, const uint8_t *data, uint32_t length)
3777
313
{
3778
313
    const State_Load_Status res = load_conferences_helper(g_c, data, length);
3779
3780
313
    if (res == STATE_LOAD_STATUS_CONTINUE) {
3781
260
        return res;
3782
260
    }
3783
3784
    // Loading failed, cleanup all Group_c
3785
3786
    // save locally, because wipe_group_chat(...) modifies it
3787
53
    const uint16_t num_groups = g_c->num_chats;
3788
3789
1.74k
    for (uint16_t i = 0; i < num_groups; ++i) {
3790
1.68k
        wipe_group_chat(g_c, i);
3791
1.68k
    }
3792
3793
53
    return res;
3794
313
}
3795
3796
bool conferences_load_state_section(Group_Chats *g_c, const uint8_t *data, uint32_t length, uint16_t type,
3797
                                    State_Load_Status *status)
3798
687
{
3799
687
    if (type != STATE_TYPE_CONFERENCES) {
3800
374
        return false;
3801
374
    }
3802
3803
313
    *status = load_conferences(g_c, data, length);
3804
313
    return true;
3805
687
}
3806
3807
/** Create new groupchat instance. */
3808
Group_Chats *new_groupchats(const Mono_Time *mono_time, const Memory *mem, Messenger *m)
3809
1.78k
{
3810
1.78k
    if (m == nullptr) {
3811
0
        return nullptr;
3812
0
    }
3813
3814
1.78k
    Group_Chats *temp = (Group_Chats *)mem_alloc(mem, sizeof(Group_Chats));
3815
3816
1.78k
    if (temp == nullptr) {
3817
0
        return nullptr;
3818
0
    }
3819
3820
1.78k
    temp->mem = mem;
3821
1.78k
    temp->mono_time = mono_time;
3822
1.78k
    temp->m = m;
3823
1.78k
    temp->fr_c = m->fr_c;
3824
1.78k
    m_callback_conference_invite(m, &handle_friend_invite_packet);
3825
3826
1.78k
    set_global_status_callback(m->fr_c, &g_handle_any_status, temp);
3827
3828
1.78k
    return temp;
3829
1.78k
}
3830
3831
/** main groupchats loop. */
3832
void do_groupchats(Group_Chats *g_c, void *userdata)
3833
0
{
3834
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3835
0
        Group_c *g = get_group_c(g_c, i);
3836
3837
0
        if (g == nullptr) {
3838
0
            continue;
3839
0
        }
3840
3841
0
        if (g->status == GROUPCHAT_STATUS_CONNECTED) {
3842
0
            connect_to_closest(g_c, i, userdata);
3843
0
            ping_groupchat(g_c, i);
3844
0
            groupchat_freeze_timedout(g_c, i, userdata);
3845
0
            clean_connections(g_c, g);
3846
3847
0
            if (g->need_send_name) {
3848
0
                group_name_send(g_c, i, g_c->m->name, g_c->m->name_length);
3849
0
                g->need_send_name = false;
3850
0
            }
3851
0
        }
3852
0
    }
3853
3854
    // TODO(irungentoo):
3855
0
}
3856
3857
/** Free everything related with group chats. */
3858
void kill_groupchats(Group_Chats *g_c)
3859
1.78k
{
3860
1.78k
    if (g_c == nullptr) {
3861
0
        return;
3862
0
    }
3863
3864
2.98k
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3865
1.20k
        del_groupchat(g_c, i, false);
3866
1.20k
    }
3867
3868
1.78k
    m_callback_conference_invite(g_c->m, nullptr);
3869
1.78k
    set_global_status_callback(g_c->m->fr_c, nullptr, nullptr);
3870
1.78k
    g_c->m->conferences_object = nullptr;
3871
1.78k
    mem_delete(g_c->mem, g_c);
3872
1.78k
}
3873
3874
/**
3875
 * @brief Return the number of chats in the instance m.
3876
 *
3877
 * You should use this to determine how much memory to allocate
3878
 * for copy_chatlist.
3879
 */
3880
uint32_t count_chatlist(const Group_Chats *g_c)
3881
0
{
3882
0
    uint32_t ret = 0;
3883
3884
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3885
0
        if (g_c->chats[i].status != GROUPCHAT_STATUS_NONE) {
3886
0
            ++ret;
3887
0
        }
3888
0
    }
3889
3890
0
    return ret;
3891
0
}
3892
3893
/** @brief Copy a list of valid chat IDs into the array out_list.
3894
 *
3895
 * If out_list is NULL, returns 0.
3896
 * Otherwise, returns the number of elements copied.
3897
 * If the array was too small, the contents
3898
 * of out_list will be truncated to list_size.
3899
 */
3900
uint32_t copy_chatlist(const Group_Chats *g_c, uint32_t *out_list, uint32_t list_size)
3901
0
{
3902
0
    if (out_list == nullptr) {
3903
0
        return 0;
3904
0
    }
3905
3906
0
    if (g_c->num_chats == 0) {
3907
0
        return 0;
3908
0
    }
3909
3910
0
    uint32_t ret = 0;
3911
3912
0
    for (uint16_t i = 0; i < g_c->num_chats; ++i) {
3913
0
        if (ret >= list_size) {
3914
0
            break;  /* Abandon ship */
3915
0
        }
3916
3917
0
        if (g_c->chats[i].status > GROUPCHAT_STATUS_NONE) {
3918
0
            out_list[ret] = i;
3919
0
            ++ret;
3920
0
        }
3921
0
    }
3922
3923
0
    return ret;
3924
0
}