Coverage Report

Created: 2025-04-02 10:29

/src/c-toxcore/toxcore/tox.c
Line
Count
Source (jump to first uncovered line)
1
/* SPDX-License-Identifier: GPL-3.0-or-later
2
 * Copyright © 2016-2025 The TokTok team.
3
 * Copyright © 2013 Tox project.
4
 */
5
6
/**
7
 * The Tox public API.
8
 */
9
#ifndef _XOPEN_SOURCE
10
#define _XOPEN_SOURCE 600
11
#endif /* _XOPEN_SOURCE */
12
13
#include "tox.h"
14
15
#include <assert.h>
16
#include <string.h>
17
18
#include "DHT.h"
19
#include "Messenger.h"
20
#include "TCP_client.h"
21
#include "attributes.h"
22
#include "ccompat.h"
23
#include "crypto_core.h"
24
#include "friend_requests.h"
25
#include "group.h"
26
#include "group_chats.h"
27
#include "group_common.h"
28
#include "logger.h"
29
#include "mem.h"
30
#include "mono_time.h"
31
#include "net_crypto.h"
32
#include "network.h"
33
#include "onion_client.h"
34
#include "state.h"
35
#include "tox_log_level.h"
36
#include "tox_options.h"
37
#include "tox_private.h"
38
#include "tox_struct.h" // IWYU pragma: keep
39
#include "util.h"
40
41
#include "../toxencryptsave/defines.h"
42
43
#define SET_ERROR_PARAMETER(param, x) \
44
3.58k
    do {                              \
45
3.58k
        if (param != nullptr) {       \
46
46
            *param = x;               \
47
46
        }                             \
48
3.58k
    } while (0)
49
50
static_assert(TOX_HASH_LENGTH == CRYPTO_SHA256_SIZE,
51
              "TOX_HASH_LENGTH is assumed to be equal to CRYPTO_SHA256_SIZE");
52
static_assert(FILE_ID_LENGTH == CRYPTO_SYMMETRIC_KEY_SIZE,
53
              "FILE_ID_LENGTH is assumed to be equal to CRYPTO_SYMMETRIC_KEY_SIZE");
54
static_assert(TOX_DHT_NODE_IP_STRING_SIZE == IP_NTOA_LEN,
55
              "TOX_DHT_NODE_IP_STRING_SIZE is assumed to be equal to IP_NTOA_LEN");
56
static_assert(TOX_GROUP_PEER_IP_STRING_MAX_LENGTH == IP_NTOA_LEN,
57
              "TOX_GROUP_PEER_IP_STRING_MAX_LENGTH is assumed to be equal to IP_NTOA_LEN");
58
static_assert(TOX_DHT_NODE_PUBLIC_KEY_SIZE == CRYPTO_PUBLIC_KEY_SIZE,
59
              "TOX_DHT_NODE_PUBLIC_KEY_SIZE is assumed to be equal to CRYPTO_PUBLIC_KEY_SIZE");
60
static_assert(TOX_FILE_ID_LENGTH == CRYPTO_SYMMETRIC_KEY_SIZE,
61
              "TOX_FILE_ID_LENGTH is assumed to be equal to CRYPTO_SYMMETRIC_KEY_SIZE");
62
static_assert(TOX_FILE_ID_LENGTH == TOX_HASH_LENGTH,
63
              "TOX_FILE_ID_LENGTH is assumed to be equal to TOX_HASH_LENGTH");
64
static_assert(TOX_PUBLIC_KEY_SIZE == CRYPTO_PUBLIC_KEY_SIZE,
65
              "TOX_PUBLIC_KEY_SIZE is assumed to be equal to CRYPTO_PUBLIC_KEY_SIZE");
66
static_assert(TOX_SECRET_KEY_SIZE == CRYPTO_SECRET_KEY_SIZE,
67
              "TOX_SECRET_KEY_SIZE is assumed to be equal to CRYPTO_SECRET_KEY_SIZE");
68
static_assert(TOX_MAX_NAME_LENGTH == MAX_NAME_LENGTH,
69
              "TOX_MAX_NAME_LENGTH is assumed to be equal to MAX_NAME_LENGTH");
70
static_assert(TOX_MAX_STATUS_MESSAGE_LENGTH == MAX_STATUSMESSAGE_LENGTH,
71
              "TOX_MAX_STATUS_MESSAGE_LENGTH is assumed to be equal to MAX_STATUSMESSAGE_LENGTH");
72
static_assert(TOX_GROUP_MAX_MESSAGE_LENGTH == GROUP_MAX_MESSAGE_LENGTH,
73
              "TOX_GROUP_MAX_MESSAGE_LENGTH is assumed to be equal to GROUP_MAX_MESSAGE_LENGTH");
74
static_assert(TOX_MAX_CUSTOM_PACKET_SIZE == MAX_GC_CUSTOM_LOSSLESS_PACKET_SIZE,
75
              "TOX_MAX_CUSTOM_PACKET_SIZE is assumed to be equal to MAX_GC_CUSTOM_LOSSLESS_PACKET_SIZE");
76
77
struct Tox_Userdata {
78
    Tox *tox;
79
    void *user_data;
80
};
81
82
static logger_cb tox_log_handler;
83
non_null(1, 3, 5, 6) nullable(7)
84
static void tox_log_handler(void *context, Logger_Level level, const char *file, uint32_t line, const char *func,
85
                            const char *message, void *userdata)
86
80.3k
{
87
80.3k
    Tox *tox = (Tox *)context;
88
80.3k
    assert(tox != nullptr);
89
90
80.3k
    if (tox->log_callback != nullptr) {
91
25
        tox->log_callback(tox, (Tox_Log_Level)level, file, line, func, message, userdata);
92
25
    }
93
80.3k
}
94
95
static m_self_connection_status_cb tox_self_connection_status_handler;
96
non_null(1) nullable(3)
97
static void tox_self_connection_status_handler(Messenger *m, Onion_Connection_Status connection_status, void *user_data)
98
0
{
99
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
100
101
0
    if (tox_data->tox->self_connection_status_callback != nullptr) {
102
0
        tox_unlock(tox_data->tox);
103
0
        tox_data->tox->self_connection_status_callback(tox_data->tox, (Tox_Connection)connection_status, tox_data->user_data);
104
0
        tox_lock(tox_data->tox);
105
0
    }
106
0
}
107
108
static m_friend_name_cb tox_friend_name_handler;
109
non_null(1, 3) nullable(5)
110
static void tox_friend_name_handler(Messenger *m, uint32_t friend_number, const uint8_t *name, size_t length,
111
                                    void *user_data)
112
0
{
113
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
114
115
0
    if (tox_data->tox->friend_name_callback != nullptr) {
116
0
        tox_unlock(tox_data->tox);
117
0
        tox_data->tox->friend_name_callback(tox_data->tox, friend_number, name, length, tox_data->user_data);
118
0
        tox_lock(tox_data->tox);
119
0
    }
120
0
}
121
122
static m_friend_status_message_cb tox_friend_status_message_handler;
123
non_null(1, 3) nullable(5)
124
static void tox_friend_status_message_handler(Messenger *m, uint32_t friend_number, const uint8_t *message,
125
        size_t length, void *user_data)
126
0
{
127
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
128
129
0
    if (tox_data->tox->friend_status_message_callback != nullptr) {
130
0
        tox_unlock(tox_data->tox);
131
0
        tox_data->tox->friend_status_message_callback(tox_data->tox, friend_number, message, length, tox_data->user_data);
132
0
        tox_lock(tox_data->tox);
133
0
    }
134
0
}
135
136
static m_friend_status_cb tox_friend_status_handler;
137
non_null(1) nullable(4)
138
static void tox_friend_status_handler(Messenger *m, uint32_t friend_number, unsigned int status, void *user_data)
139
0
{
140
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
141
142
0
    if (tox_data->tox->friend_status_callback != nullptr) {
143
0
        tox_unlock(tox_data->tox);
144
0
        tox_data->tox->friend_status_callback(tox_data->tox, friend_number, (Tox_User_Status)status, tox_data->user_data);
145
0
        tox_lock(tox_data->tox);
146
0
    }
147
0
}
148
149
static m_friend_connection_status_cb tox_friend_connection_status_handler;
150
non_null(1) nullable(4)
151
static void tox_friend_connection_status_handler(Messenger *m, uint32_t friend_number, unsigned int connection_status,
152
        void *user_data)
153
0
{
154
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
155
156
0
    if (tox_data->tox->friend_connection_status_callback != nullptr) {
157
0
        tox_unlock(tox_data->tox);
158
0
        tox_data->tox->friend_connection_status_callback(tox_data->tox, friend_number, (Tox_Connection)connection_status,
159
0
                tox_data->user_data);
160
0
        tox_lock(tox_data->tox);
161
0
    }
162
0
}
163
164
static m_friend_typing_cb tox_friend_typing_handler;
165
non_null(1) nullable(4)
166
static void tox_friend_typing_handler(Messenger *m, uint32_t friend_number, bool is_typing, void *user_data)
167
0
{
168
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
169
170
0
    if (tox_data->tox->friend_typing_callback != nullptr) {
171
0
        tox_unlock(tox_data->tox);
172
0
        tox_data->tox->friend_typing_callback(tox_data->tox, friend_number, is_typing, tox_data->user_data);
173
0
        tox_lock(tox_data->tox);
174
0
    }
175
0
}
176
177
static m_friend_read_receipt_cb tox_friend_read_receipt_handler;
178
non_null(1) nullable(4)
179
static void tox_friend_read_receipt_handler(Messenger *m, uint32_t friend_number, uint32_t message_id, void *user_data)
180
0
{
181
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
182
183
0
    if (tox_data->tox->friend_read_receipt_callback != nullptr) {
184
0
        tox_unlock(tox_data->tox);
185
0
        tox_data->tox->friend_read_receipt_callback(tox_data->tox, friend_number, message_id, tox_data->user_data);
186
0
        tox_lock(tox_data->tox);
187
0
    }
188
0
}
189
190
static m_friend_request_cb tox_friend_request_handler;
191
non_null(1, 2, 3) nullable(5)
192
static void tox_friend_request_handler(Messenger *m, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE], const uint8_t *message, size_t length,
193
                                       void *user_data)
194
0
{
195
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
196
197
0
    if (tox_data->tox->friend_request_callback != nullptr) {
198
0
        tox_unlock(tox_data->tox);
199
0
        tox_data->tox->friend_request_callback(tox_data->tox, public_key, message, length, tox_data->user_data);
200
0
        tox_lock(tox_data->tox);
201
0
    }
202
0
}
203
204
static m_friend_message_cb tox_friend_message_handler;
205
non_null(1, 4) nullable(6)
206
static void tox_friend_message_handler(Messenger *m, uint32_t friend_number, unsigned int message_type,
207
                                       const uint8_t *message, size_t length, void *user_data)
208
0
{
209
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
210
211
0
    if (tox_data->tox->friend_message_callback != nullptr) {
212
0
        tox_unlock(tox_data->tox);
213
0
        tox_data->tox->friend_message_callback(tox_data->tox, friend_number, (Tox_Message_Type)message_type, message, length,
214
0
                                               tox_data->user_data);
215
0
        tox_lock(tox_data->tox);
216
0
    }
217
0
}
218
219
static m_file_recv_control_cb tox_file_recv_control_handler;
220
non_null(1) nullable(5)
221
static void tox_file_recv_control_handler(Messenger *m, uint32_t friend_number, uint32_t file_number,
222
        unsigned int control, void *user_data)
223
0
{
224
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
225
226
0
    if (tox_data->tox->file_recv_control_callback != nullptr) {
227
0
        tox_unlock(tox_data->tox);
228
0
        tox_data->tox->file_recv_control_callback(tox_data->tox, friend_number, file_number, (Tox_File_Control)control,
229
0
                tox_data->user_data);
230
0
        tox_lock(tox_data->tox);
231
0
    }
232
0
}
233
234
static m_file_chunk_request_cb tox_file_chunk_request_handler;
235
non_null(1) nullable(6)
236
static void tox_file_chunk_request_handler(Messenger *m, uint32_t friend_number, uint32_t file_number,
237
        uint64_t position, size_t length, void *user_data)
238
0
{
239
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
240
241
0
    if (tox_data->tox->file_chunk_request_callback != nullptr) {
242
0
        tox_unlock(tox_data->tox);
243
0
        tox_data->tox->file_chunk_request_callback(tox_data->tox, friend_number, file_number, position, length,
244
0
                tox_data->user_data);
245
0
        tox_lock(tox_data->tox);
246
0
    }
247
0
}
248
249
static m_file_recv_cb tox_file_recv_handler;
250
non_null(1, 6) nullable(8)
251
static void tox_file_recv_handler(Messenger *m, uint32_t friend_number, uint32_t file_number, uint32_t kind,
252
                                  uint64_t file_size, const uint8_t *filename, size_t filename_length, void *user_data)
253
0
{
254
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
255
256
0
    if (tox_data->tox->file_recv_callback != nullptr) {
257
0
        tox_unlock(tox_data->tox);
258
0
        tox_data->tox->file_recv_callback(tox_data->tox, friend_number, file_number, kind, file_size, filename, filename_length,
259
0
                                          tox_data->user_data);
260
0
        tox_lock(tox_data->tox);
261
0
    }
262
0
}
263
264
static m_file_recv_chunk_cb tox_file_recv_chunk_handler;
265
non_null(1, 5) nullable(7)
266
static void tox_file_recv_chunk_handler(Messenger *m, uint32_t friend_number, uint32_t file_number, uint64_t position,
267
                                        const uint8_t *data, size_t length, void *user_data)
268
0
{
269
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
270
271
0
    if (tox_data->tox->file_recv_chunk_callback != nullptr) {
272
0
        tox_unlock(tox_data->tox);
273
0
        tox_data->tox->file_recv_chunk_callback(tox_data->tox, friend_number, file_number, position, data, length,
274
0
                                                tox_data->user_data);
275
0
        tox_lock(tox_data->tox);
276
0
    }
277
0
}
278
279
static g_conference_invite_cb tox_conference_invite_handler;
280
non_null(1, 4) nullable(6)
281
static void tox_conference_invite_handler(Messenger *m, uint32_t friend_number, int type, const uint8_t *cookie,
282
        size_t length, void *user_data)
283
0
{
284
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
285
286
0
    if (tox_data->tox->conference_invite_callback != nullptr) {
287
0
        tox_unlock(tox_data->tox);
288
0
        tox_data->tox->conference_invite_callback(tox_data->tox, friend_number, (Tox_Conference_Type)type, cookie, length,
289
0
                tox_data->user_data);
290
0
        tox_lock(tox_data->tox);
291
0
    }
292
0
}
293
294
static g_conference_connected_cb tox_conference_connected_handler;
295
non_null(1) nullable(3)
296
static void tox_conference_connected_handler(Messenger *m, uint32_t conference_number, void *user_data)
297
0
{
298
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
299
300
0
    if (tox_data->tox->conference_connected_callback != nullptr) {
301
0
        tox_unlock(tox_data->tox);
302
0
        tox_data->tox->conference_connected_callback(tox_data->tox, conference_number, tox_data->user_data);
303
0
        tox_lock(tox_data->tox);
304
0
    }
305
0
}
306
307
static g_conference_message_cb tox_conference_message_handler;
308
non_null(1, 5) nullable(7)
309
static void tox_conference_message_handler(Messenger *m, uint32_t conference_number, uint32_t peer_number, int type,
310
        const uint8_t *message, size_t length, void *user_data)
311
0
{
312
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
313
314
0
    if (tox_data->tox->conference_message_callback != nullptr) {
315
0
        tox_unlock(tox_data->tox);
316
0
        tox_data->tox->conference_message_callback(tox_data->tox, conference_number, peer_number, (Tox_Message_Type)type,
317
0
                message, length, tox_data->user_data);
318
0
        tox_lock(tox_data->tox);
319
0
    }
320
0
}
321
322
static title_cb tox_conference_title_handler;
323
non_null(1, 4) nullable(6)
324
static void tox_conference_title_handler(Messenger *m, uint32_t conference_number, uint32_t peer_number,
325
        const uint8_t *title, size_t length, void *user_data)
326
0
{
327
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
328
329
0
    if (tox_data->tox->conference_title_callback != nullptr) {
330
0
        tox_unlock(tox_data->tox);
331
0
        tox_data->tox->conference_title_callback(tox_data->tox, conference_number, peer_number, title, length,
332
0
                tox_data->user_data);
333
0
        tox_lock(tox_data->tox);
334
0
    }
335
0
}
336
337
static peer_name_cb tox_conference_peer_name_handler;
338
non_null(1, 4) nullable(6)
339
static void tox_conference_peer_name_handler(Messenger *m, uint32_t conference_number, uint32_t peer_number,
340
        const uint8_t *name, size_t length, void *user_data)
341
0
{
342
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
343
344
0
    if (tox_data->tox->conference_peer_name_callback != nullptr) {
345
0
        tox_unlock(tox_data->tox);
346
0
        tox_data->tox->conference_peer_name_callback(tox_data->tox, conference_number, peer_number, name, length,
347
0
                tox_data->user_data);
348
0
        tox_lock(tox_data->tox);
349
0
    }
350
0
}
351
352
static peer_list_changed_cb tox_conference_peer_list_changed_handler;
353
non_null(1) nullable(3)
354
static void tox_conference_peer_list_changed_handler(Messenger *m, uint32_t conference_number, void *user_data)
355
0
{
356
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
357
358
0
    if (tox_data->tox->conference_peer_list_changed_callback != nullptr) {
359
0
        tox_unlock(tox_data->tox);
360
0
        tox_data->tox->conference_peer_list_changed_callback(tox_data->tox, conference_number, tox_data->user_data);
361
0
        tox_lock(tox_data->tox);
362
0
    }
363
0
}
364
365
static dht_nodes_response_cb tox_dht_nodes_response_handler;
366
non_null(1, 2) nullable(3)
367
static void tox_dht_nodes_response_handler(const DHT *dht, const Node_format *node, void *user_data)
368
0
{
369
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
370
371
0
    if (tox_data->tox->dht_nodes_response_callback == nullptr) {
372
0
        return;
373
0
    }
374
375
0
    Ip_Ntoa ip_str;
376
0
    net_ip_ntoa(&node->ip_port.ip, &ip_str);
377
378
0
    tox_unlock(tox_data->tox);
379
0
    tox_data->tox->dht_nodes_response_callback(
380
0
        tox_data->tox, node->public_key, ip_str.buf, ip_str.length, net_ntohs(node->ip_port.port),
381
0
        tox_data->user_data);
382
0
    tox_lock(tox_data->tox);
383
0
}
384
385
static m_friend_lossy_packet_cb tox_friend_lossy_packet_handler;
386
non_null(1, 4) nullable(6)
387
static void tox_friend_lossy_packet_handler(Messenger *m, uint32_t friend_number, uint8_t packet_id,
388
        const uint8_t *data, size_t length, void *user_data)
389
0
{
390
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
391
392
0
    assert(data != nullptr);
393
0
    assert(length > 0);
394
395
0
    if (tox_data->tox->friend_lossy_packet_callback_per_pktid[packet_id] != nullptr) {
396
0
        tox_unlock(tox_data->tox);
397
0
        tox_data->tox->friend_lossy_packet_callback_per_pktid[packet_id](tox_data->tox, friend_number, data, length,
398
0
                tox_data->user_data);
399
0
        tox_lock(tox_data->tox);
400
0
    }
401
0
}
402
403
static m_friend_lossless_packet_cb tox_friend_lossless_packet_handler;
404
non_null(1, 4) nullable(6)
405
static void tox_friend_lossless_packet_handler(Messenger *m, uint32_t friend_number, uint8_t packet_id,
406
        const uint8_t *data, size_t length, void *user_data)
407
0
{
408
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
409
410
0
    assert(data != nullptr);
411
0
    assert(length > 0);
412
413
0
    if (tox_data->tox->friend_lossless_packet_callback_per_pktid[packet_id] != nullptr) {
414
0
        tox_unlock(tox_data->tox);
415
0
        tox_data->tox->friend_lossless_packet_callback_per_pktid[packet_id](tox_data->tox, friend_number, data, length,
416
0
                tox_data->user_data);
417
0
        tox_lock(tox_data->tox);
418
0
    }
419
0
}
420
421
non_null(1, 4) nullable(6)
422
static void tox_group_peer_name_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
423
                                        const uint8_t *name, size_t length, void *user_data)
424
0
{
425
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
426
427
0
    if (tox_data->tox->group_peer_name_callback != nullptr) {
428
0
        tox_unlock(tox_data->tox);
429
0
        tox_data->tox->group_peer_name_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), name, length, tox_data->user_data);
430
0
        tox_lock(tox_data->tox);
431
0
    }
432
0
}
433
434
non_null(1) nullable(5)
435
static void tox_group_peer_status_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
436
        unsigned int status, void *user_data)
437
0
{
438
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
439
440
0
    if (tox_data->tox->group_peer_status_callback != nullptr) {
441
0
        tox_unlock(tox_data->tox);
442
0
        tox_data->tox->group_peer_status_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), (Tox_User_Status)status,
443
0
                tox_data->user_data);
444
0
        tox_lock(tox_data->tox);
445
0
    }
446
0
}
447
448
non_null(1, 4) nullable(6)
449
static void tox_group_topic_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id, const uint8_t *topic,
450
                                    size_t length, void *user_data)
451
0
{
452
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
453
454
0
    if (tox_data->tox->group_topic_callback != nullptr) {
455
0
        tox_unlock(tox_data->tox);
456
0
        tox_data->tox->group_topic_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), topic, length, tox_data->user_data);
457
0
        tox_lock(tox_data->tox);
458
0
    }
459
0
}
460
461
non_null(1) nullable(4)
462
static void tox_group_topic_lock_handler(const Messenger *m, uint32_t group_number, unsigned int topic_lock,
463
        void *user_data)
464
0
{
465
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
466
467
0
    if (tox_data->tox->group_topic_lock_callback != nullptr) {
468
0
        tox_unlock(tox_data->tox);
469
0
        tox_data->tox->group_topic_lock_callback(tox_data->tox, group_number, (Tox_Group_Topic_Lock)topic_lock,
470
0
                tox_data->user_data);
471
0
        tox_lock(tox_data->tox);
472
0
    }
473
0
}
474
475
non_null(1) nullable(4)
476
static void tox_group_voice_state_handler(const Messenger *m, uint32_t group_number, unsigned int voice_state,
477
        void *user_data)
478
0
{
479
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
480
481
0
    if (tox_data->tox->group_voice_state_callback != nullptr) {
482
0
        tox_unlock(tox_data->tox);
483
0
        tox_data->tox->group_voice_state_callback(tox_data->tox, group_number, (Tox_Group_Voice_State)voice_state,
484
0
                tox_data->user_data);
485
0
        tox_lock(tox_data->tox);
486
0
    }
487
0
}
488
489
non_null(1) nullable(4)
490
static void tox_group_peer_limit_handler(const Messenger *m, uint32_t group_number, uint32_t peer_limit,
491
        void *user_data)
492
0
{
493
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
494
495
0
    if (tox_data->tox->group_peer_limit_callback != nullptr) {
496
0
        tox_unlock(tox_data->tox);
497
0
        tox_data->tox->group_peer_limit_callback(tox_data->tox, group_number, peer_limit, tox_data->user_data);
498
0
        tox_lock(tox_data->tox);
499
0
    }
500
0
}
501
502
non_null(1) nullable(4)
503
static void tox_group_privacy_state_handler(const Messenger *m, uint32_t group_number, unsigned int privacy_state,
504
        void *user_data)
505
0
{
506
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
507
508
0
    if (tox_data->tox->group_privacy_state_callback != nullptr) {
509
0
        tox_unlock(tox_data->tox);
510
0
        tox_data->tox->group_privacy_state_callback(tox_data->tox, group_number, (Tox_Group_Privacy_State)privacy_state,
511
0
                tox_data->user_data);
512
0
        tox_lock(tox_data->tox);
513
0
    }
514
0
}
515
516
non_null(1) nullable(3, 5)
517
static void tox_group_password_handler(const Messenger *m, uint32_t group_number, const uint8_t *password,
518
                                       size_t length, void *user_data)
519
0
{
520
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
521
522
0
    if (tox_data->tox->group_password_callback != nullptr) {
523
0
        tox_unlock(tox_data->tox);
524
0
        tox_data->tox->group_password_callback(tox_data->tox, group_number, password, length, tox_data->user_data);
525
0
        tox_lock(tox_data->tox);
526
0
    }
527
0
}
528
529
non_null(1, 5) nullable(8)
530
static void tox_group_message_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id, unsigned int type,
531
                                      const uint8_t *message, size_t length, Tox_Group_Message_Id message_id, void *user_data)
532
0
{
533
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
534
535
0
    if (tox_data->tox->group_message_callback != nullptr) {
536
0
        tox_unlock(tox_data->tox);
537
0
        tox_data->tox->group_message_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), (Tox_Message_Type)type, message, length,
538
0
                                              message_id, tox_data->user_data);
539
0
        tox_lock(tox_data->tox);
540
0
    }
541
0
}
542
543
non_null(1, 5) nullable(8)
544
static void tox_group_private_message_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
545
        unsigned int type, const uint8_t *message, size_t length, Tox_Group_Message_Id message_id, void *user_data)
546
0
{
547
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
548
549
0
    if (tox_data->tox->group_private_message_callback != nullptr) {
550
0
        tox_unlock(tox_data->tox);
551
0
        tox_data->tox->group_private_message_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), (Tox_Message_Type)type, message,
552
0
                length, message_id, tox_data->user_data);
553
0
        tox_lock(tox_data->tox);
554
0
    }
555
0
}
556
557
non_null(1, 4) nullable(6)
558
static void tox_group_custom_packet_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
559
        const uint8_t *data, size_t length, void *user_data)
560
0
{
561
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
562
563
0
    if (tox_data->tox->group_custom_packet_callback != nullptr) {
564
0
        tox_unlock(tox_data->tox);
565
0
        tox_data->tox->group_custom_packet_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), data, length, tox_data->user_data);
566
0
        tox_lock(tox_data->tox);
567
0
    }
568
0
}
569
570
non_null(1, 4) nullable(6)
571
static void tox_group_custom_private_packet_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
572
        const uint8_t *data, size_t length, void *user_data)
573
0
{
574
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
575
576
0
    if (tox_data->tox->group_custom_private_packet_callback != nullptr) {
577
0
        tox_unlock(tox_data->tox);
578
0
        tox_data->tox->group_custom_private_packet_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), data, length,
579
0
                tox_data->user_data);
580
0
        tox_lock(tox_data->tox);
581
0
    }
582
0
}
583
584
non_null(1, 3, 5) nullable(7)
585
static void tox_group_invite_handler(const Messenger *m, uint32_t friend_number, const uint8_t *invite_data,
586
                                     size_t length, const uint8_t *group_name, size_t group_name_length, void *user_data)
587
0
{
588
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
589
590
0
    if (tox_data->tox->group_invite_callback != nullptr) {
591
0
        tox_unlock(tox_data->tox);
592
0
        tox_data->tox->group_invite_callback(tox_data->tox, friend_number, invite_data, length, group_name, group_name_length,
593
0
                                             tox_data->user_data);
594
0
        tox_lock(tox_data->tox);
595
0
    }
596
0
}
597
598
non_null(1) nullable(4)
599
static void tox_group_peer_join_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id, void *user_data)
600
0
{
601
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
602
603
0
    if (tox_data->tox->group_peer_join_callback != nullptr) {
604
0
        tox_unlock(tox_data->tox);
605
0
        tox_data->tox->group_peer_join_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id), tox_data->user_data);
606
0
        tox_lock(tox_data->tox);
607
0
    }
608
0
}
609
610
non_null(1, 5) nullable(7, 9)
611
static void tox_group_peer_exit_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id peer_id,
612
                                        unsigned int exit_type, const uint8_t *name, size_t name_length,
613
                                        const uint8_t *part_message, size_t length, void *user_data)
614
0
{
615
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
616
617
0
    if (tox_data->tox->group_peer_exit_callback != nullptr) {
618
0
        tox_unlock(tox_data->tox);
619
0
        tox_data->tox->group_peer_exit_callback(tox_data->tox, group_number, gc_peer_id_to_int(peer_id),
620
0
                                                (Tox_Group_Exit_Type)exit_type, name, name_length,
621
0
                                                part_message, length, tox_data->user_data);
622
0
        tox_lock(tox_data->tox);
623
0
    }
624
0
}
625
626
non_null(1) nullable(3)
627
static void tox_group_self_join_handler(const Messenger *m, uint32_t group_number, void *user_data)
628
0
{
629
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
630
631
0
    if (tox_data->tox->group_self_join_callback != nullptr) {
632
0
        tox_unlock(tox_data->tox);
633
0
        tox_data->tox->group_self_join_callback(tox_data->tox, group_number, tox_data->user_data);
634
0
        tox_lock(tox_data->tox);
635
0
    }
636
0
}
637
638
non_null(1) nullable(4)
639
static void tox_group_join_fail_handler(const Messenger *m, uint32_t group_number, unsigned int fail_type,
640
                                        void *user_data)
641
0
{
642
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
643
644
0
    if (tox_data->tox->group_join_fail_callback != nullptr) {
645
0
        tox_unlock(tox_data->tox);
646
0
        tox_data->tox->group_join_fail_callback(tox_data->tox, group_number, (Tox_Group_Join_Fail)fail_type,
647
0
                                                tox_data->user_data);
648
0
        tox_lock(tox_data->tox);
649
0
    }
650
0
}
651
652
non_null(1) nullable(6)
653
static void tox_group_moderation_handler(const Messenger *m, uint32_t group_number, GC_Peer_Id source_peer_number,
654
        GC_Peer_Id target_peer_number, unsigned int mod_type, void *user_data)
655
0
{
656
0
    struct Tox_Userdata *tox_data = (struct Tox_Userdata *)user_data;
657
658
0
    if (tox_data->tox->group_moderation_callback != nullptr) {
659
0
        tox_unlock(tox_data->tox);
660
0
        tox_data->tox->group_moderation_callback(tox_data->tox, group_number,
661
0
                gc_peer_id_to_int(source_peer_number), gc_peer_id_to_int(target_peer_number),
662
0
                (Tox_Group_Mod_Event)mod_type, tox_data->user_data);
663
0
        tox_lock(tox_data->tox);
664
0
    }
665
0
}
666
667
bool tox_version_is_compatible(uint32_t major, uint32_t minor, uint32_t patch)
668
0
{
669
0
    return TOX_VERSION_IS_API_COMPATIBLE(major, minor, patch);
670
0
}
671
672
non_null()
673
static State_Load_Status state_load_callback(void *outer, const uint8_t *data, uint32_t length, uint16_t type)
674
31.0k
{
675
31.0k
    const Tox *tox = (const Tox *)outer;
676
31.0k
    State_Load_Status status = STATE_LOAD_STATUS_CONTINUE;
677
678
31.0k
    if (messenger_load_state_section(tox->m, data, length, type, &status)
679
31.0k
            || conferences_load_state_section(tox->m->conferences_object, data, length, type, &status)) {
680
30.6k
        return status;
681
30.6k
    }
682
683
375
    if (type == STATE_TYPE_END) {
684
6
        if (length != 0) {
685
1
            return STATE_LOAD_STATUS_ERROR;
686
1
        }
687
688
5
        return STATE_LOAD_STATUS_END;
689
6
    }
690
691
369
    LOGGER_ERROR(tox->m->log, "Load state: contains unrecognized part (len %u, type %u)",
692
369
                 length, type);
693
694
369
    return STATE_LOAD_STATUS_CONTINUE;
695
375
}
696
697
/** Load tox from data of size length. */
698
non_null()
699
static int tox_load(Tox *tox, const uint8_t *data, uint32_t length)
700
1.76k
{
701
1.76k
    uint32_t data32[2];
702
1.76k
    const uint32_t cookie_len = sizeof(data32);
703
704
1.76k
    if (length < cookie_len) {
705
0
        return -1;
706
0
    }
707
708
1.76k
    memcpy(data32, data, sizeof(uint32_t));
709
1.76k
    lendian_bytes_to_host32(data32 + 1, data + sizeof(uint32_t));
710
711
1.76k
    if (data32[0] != 0 || data32[1] != STATE_COOKIE_GLOBAL) {
712
2
        return -1;
713
2
    }
714
715
1.76k
    return state_load(tox->m->log, state_load_callback, tox, data + cookie_len,
716
1.76k
                      length - cookie_len, STATE_COOKIE_TYPE);
717
1.76k
}
718
719
nullable(1, 2, 3)
720
static Tox *tox_new_system(const struct Tox_Options *options, Tox_Err_New *error, const Tox_System *sys)
721
1.79k
{
722
1.79k
    struct Tox_Options *default_options = nullptr;
723
724
1.79k
    if (options == nullptr) {
725
0
        Tox_Err_Options_New err;
726
0
        default_options = tox_options_new(&err);
727
728
0
        switch (err) {
729
0
            case TOX_ERR_OPTIONS_NEW_OK: {
730
0
                assert(default_options != nullptr);
731
0
                break;
732
0
            }
733
734
0
            case TOX_ERR_OPTIONS_NEW_MALLOC: {
735
0
                SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
736
0
                return nullptr;
737
0
            }
738
0
        }
739
0
    }
740
741
1.79k
    const struct Tox_Options *const opts = options != nullptr ? options : default_options;
742
1.79k
    assert(opts != nullptr);
743
744
1.79k
    const Tox_System default_system = tox_default_system();
745
746
1.79k
    if (sys == nullptr) {
747
0
        sys = &default_system;
748
0
    }
749
750
1.79k
    if (sys->rng == nullptr || sys->ns == nullptr || sys->mem == nullptr) {
751
        // TODO(iphydf): Not quite right, but similar.
752
0
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
753
0
        tox_options_free(default_options);
754
0
        return nullptr;
755
0
    }
756
757
1.79k
    Messenger_Options m_options = {false};
758
759
1.79k
    m_options.dns_enabled = !tox_options_get_experimental_disable_dns(opts);
760
761
1.79k
    bool load_savedata_sk = false;
762
1.79k
    bool load_savedata_tox = false;
763
764
1.79k
    if (tox_options_get_savedata_type(opts) != TOX_SAVEDATA_TYPE_NONE) {
765
1.77k
        if (tox_options_get_savedata_data(opts) == nullptr || tox_options_get_savedata_length(opts) == 0) {
766
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_LOAD_BAD_FORMAT);
767
0
            tox_options_free(default_options);
768
0
            return nullptr;
769
0
        }
770
1.77k
    }
771
772
1.79k
    if (tox_options_get_savedata_type(opts) == TOX_SAVEDATA_TYPE_SECRET_KEY) {
773
0
        if (tox_options_get_savedata_length(opts) != TOX_SECRET_KEY_SIZE) {
774
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_LOAD_BAD_FORMAT);
775
0
            tox_options_free(default_options);
776
0
            return nullptr;
777
0
        }
778
779
0
        load_savedata_sk = true;
780
1.79k
    } else if (tox_options_get_savedata_type(opts) == TOX_SAVEDATA_TYPE_TOX_SAVE) {
781
1.77k
        if (tox_options_get_savedata_length(opts) < TOX_ENC_SAVE_MAGIC_LENGTH) {
782
1
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_LOAD_BAD_FORMAT);
783
1
            tox_options_free(default_options);
784
1
            return nullptr;
785
1
        }
786
787
1.77k
        if (memcmp(tox_options_get_savedata_data(opts), TOX_ENC_SAVE_MAGIC_NUMBER, TOX_ENC_SAVE_MAGIC_LENGTH) == 0) {
788
1
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_LOAD_ENCRYPTED);
789
1
            tox_options_free(default_options);
790
1
            return nullptr;
791
1
        }
792
793
1.76k
        load_savedata_tox = true;
794
1.76k
    }
795
796
1.79k
    m_options.ipv6enabled = tox_options_get_ipv6_enabled(opts);
797
1.79k
    m_options.udp_disabled = !tox_options_get_udp_enabled(opts);
798
1.79k
    m_options.port_range[0] = tox_options_get_start_port(opts);
799
1.79k
    m_options.port_range[1] = tox_options_get_end_port(opts);
800
1.79k
    m_options.tcp_server_port = tox_options_get_tcp_port(opts);
801
1.79k
    m_options.hole_punching_enabled = tox_options_get_hole_punching_enabled(opts);
802
1.79k
    m_options.local_discovery_enabled = tox_options_get_local_discovery_enabled(opts);
803
1.79k
    m_options.dht_announcements_enabled = tox_options_get_dht_announcements_enabled(opts);
804
1.79k
    m_options.groups_persistence_enabled = tox_options_get_experimental_groups_persistence(opts);
805
806
1.79k
    if (m_options.udp_disabled) {
807
0
        m_options.local_discovery_enabled = false;
808
0
    }
809
810
1.79k
    Tox *tox = (Tox *)mem_alloc(sys->mem, sizeof(Tox));
811
812
1.79k
    if (tox == nullptr) {
813
1
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
814
1
        tox_options_free(default_options);
815
1
        return nullptr;
816
1
    }
817
818
1.79k
    tox->log_callback = tox_options_get_log_callback(opts);
819
1.79k
    m_options.log_callback = tox_log_handler;
820
1.79k
    m_options.log_context = tox;
821
1.79k
    m_options.log_user_data = tox_options_get_log_user_data(opts);
822
823
1.79k
    switch (tox_options_get_proxy_type(opts)) {
824
4
        case TOX_PROXY_TYPE_HTTP: {
825
4
            m_options.proxy_info.proxy_type = TCP_PROXY_HTTP;
826
4
            break;
827
0
        }
828
829
7
        case TOX_PROXY_TYPE_SOCKS5: {
830
7
            m_options.proxy_info.proxy_type = TCP_PROXY_SOCKS5;
831
7
            break;
832
0
        }
833
834
1.78k
        case TOX_PROXY_TYPE_NONE: {
835
1.78k
            m_options.proxy_info.proxy_type = TCP_PROXY_NONE;
836
1.78k
            break;
837
0
        }
838
839
0
        default: {
840
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_PROXY_BAD_TYPE);
841
0
            mem_delete(sys->mem, tox);
842
0
            tox_options_free(default_options);
843
0
            return nullptr;
844
0
        }
845
1.79k
    }
846
847
1.79k
    tox->sys = *sys;
848
849
1.79k
    if (m_options.proxy_info.proxy_type != TCP_PROXY_NONE) {
850
11
        if (tox_options_get_proxy_port(opts) == 0) {
851
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_PROXY_BAD_PORT);
852
0
            mem_delete(sys->mem, tox);
853
0
            tox_options_free(default_options);
854
0
            return nullptr;
855
0
        }
856
857
11
        ip_init(&m_options.proxy_info.ip_port.ip, m_options.ipv6enabled);
858
859
11
        if (m_options.ipv6enabled) {
860
11
            m_options.proxy_info.ip_port.ip.family = net_family_unspec();
861
11
        }
862
863
11
        const char *const proxy_host = tox_options_get_proxy_host(opts);
864
11
        const bool dns_enabled = !tox_options_get_experimental_disable_dns(opts);
865
866
11
        if (proxy_host == nullptr
867
11
                || !addr_resolve_or_parse_ip(tox->sys.ns, tox->sys.mem, proxy_host, &m_options.proxy_info.ip_port.ip, nullptr, dns_enabled)) {
868
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_PROXY_BAD_HOST);
869
            // TODO(irungentoo): TOX_ERR_NEW_PROXY_NOT_FOUND if domain.
870
0
            mem_delete(sys->mem, tox);
871
0
            tox_options_free(default_options);
872
0
            return nullptr;
873
0
        }
874
875
11
        m_options.proxy_info.ip_port.port = net_htons(tox_options_get_proxy_port(opts));
876
11
    }
877
878
1.79k
    tox->mono_time = mono_time_new(tox->sys.mem, sys->mono_time_callback, sys->mono_time_user_data);
879
880
1.79k
    if (tox->mono_time == nullptr) {
881
2
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
882
2
        mem_delete(sys->mem, tox);
883
2
        tox_options_free(default_options);
884
2
        return nullptr;
885
2
    }
886
887
1.78k
    if (tox_options_get_experimental_thread_safety(opts)) {
888
0
        pthread_mutex_t *mutex = (pthread_mutex_t *)mem_alloc(sys->mem, sizeof(pthread_mutex_t));
889
890
0
        if (mutex == nullptr) {
891
0
            SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
892
0
            mem_delete(sys->mem, tox);
893
0
            tox_options_free(default_options);
894
0
            return nullptr;
895
0
        }
896
897
0
        pthread_mutex_init(mutex, nullptr);
898
899
0
        tox->mutex = mutex;
900
1.78k
    } else {
901
1.78k
        tox->mutex = nullptr;
902
1.78k
    }
903
904
1.78k
    tox_lock(tox);
905
906
1.78k
    Messenger_Error m_error;
907
1.78k
    tox->m = new_messenger(tox->mono_time, tox->sys.mem, tox->sys.rng, tox->sys.ns, &m_options, &m_error);
908
909
1.78k
    if (tox->m == nullptr) {
910
20
        switch (m_error) {
911
0
            case MESSENGER_ERROR_PORT:
912
0
            case MESSENGER_ERROR_TCP_SERVER: {
913
0
                SET_ERROR_PARAMETER(error, TOX_ERR_NEW_PORT_ALLOC);
914
0
                break;
915
0
            }
916
20
            case MESSENGER_ERROR_OTHER:
917
20
            case MESSENGER_ERROR_NONE: {
918
20
                SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
919
20
                break;
920
20
            }
921
20
        }
922
923
20
        mono_time_free(tox->sys.mem, tox->mono_time);
924
20
        tox_unlock(tox);
925
926
20
        if (tox->mutex != nullptr) {
927
0
            pthread_mutex_destroy(tox->mutex);
928
0
        }
929
930
20
        mem_delete(sys->mem, tox->mutex);
931
20
        mem_delete(sys->mem, tox);
932
20
        tox_options_free(default_options);
933
20
        return nullptr;
934
20
    }
935
936
1.76k
    tox->m->conferences_object = new_groupchats(tox->mono_time, sys->mem, tox->m);
937
938
1.76k
    if (tox->m->conferences_object == nullptr) {
939
0
        kill_messenger(tox->m);
940
941
0
        mono_time_free(tox->sys.mem, tox->mono_time);
942
0
        tox_unlock(tox);
943
944
0
        if (tox->mutex != nullptr) {
945
0
            pthread_mutex_destroy(tox->mutex);
946
0
        }
947
948
0
        mem_delete(sys->mem, tox->mutex);
949
0
        mem_delete(sys->mem, tox);
950
951
0
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_MALLOC);
952
0
        tox_options_free(default_options);
953
0
        return nullptr;
954
0
    }
955
956
1.76k
    if (load_savedata_tox
957
1.76k
            && tox_load(tox, tox_options_get_savedata_data(opts), tox_options_get_savedata_length(opts)) == -1) {
958
499
        kill_groupchats(tox->m->conferences_object);
959
499
        kill_messenger(tox->m);
960
961
499
        mono_time_free(tox->sys.mem, tox->mono_time);
962
499
        tox_unlock(tox);
963
964
499
        if (tox->mutex != nullptr) {
965
0
            pthread_mutex_destroy(tox->mutex);
966
0
        }
967
968
499
        mem_delete(sys->mem, tox->mutex);
969
499
        mem_delete(sys->mem, tox);
970
971
499
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_LOAD_BAD_FORMAT);
972
499
        tox_options_free(default_options);
973
499
        return nullptr;
974
499
    }
975
976
1.27k
    if (load_savedata_sk) {
977
0
        load_secret_key(tox->m->net_crypto, tox_options_get_savedata_data(opts));
978
0
    }
979
980
1.27k
    m_callback_namechange(tox->m, tox_friend_name_handler);
981
1.27k
    m_callback_core_connection(tox->m, tox_self_connection_status_handler);
982
1.27k
    m_callback_statusmessage(tox->m, tox_friend_status_message_handler);
983
1.27k
    m_callback_userstatus(tox->m, tox_friend_status_handler);
984
1.27k
    m_callback_connectionstatus(tox->m, tox_friend_connection_status_handler);
985
1.27k
    m_callback_typingchange(tox->m, tox_friend_typing_handler);
986
1.27k
    m_callback_read_receipt(tox->m, tox_friend_read_receipt_handler);
987
1.27k
    m_callback_friendrequest(tox->m, tox_friend_request_handler);
988
1.27k
    m_callback_friendmessage(tox->m, tox_friend_message_handler);
989
1.27k
    callback_file_control(tox->m, tox_file_recv_control_handler);
990
1.27k
    callback_file_reqchunk(tox->m, tox_file_chunk_request_handler);
991
1.27k
    callback_file_sendrequest(tox->m, tox_file_recv_handler);
992
1.27k
    callback_file_data(tox->m, tox_file_recv_chunk_handler);
993
1.27k
    dht_callback_nodes_response(tox->m->dht, tox_dht_nodes_response_handler);
994
1.27k
    g_callback_group_invite(tox->m->conferences_object, tox_conference_invite_handler);
995
1.27k
    g_callback_group_connected(tox->m->conferences_object, tox_conference_connected_handler);
996
1.27k
    g_callback_group_message(tox->m->conferences_object, tox_conference_message_handler);
997
1.27k
    g_callback_group_title(tox->m->conferences_object, tox_conference_title_handler);
998
1.27k
    g_callback_peer_name(tox->m->conferences_object, tox_conference_peer_name_handler);
999
1.27k
    g_callback_peer_list_changed(tox->m->conferences_object, tox_conference_peer_list_changed_handler);
1000
1.27k
    custom_lossy_packet_registerhandler(tox->m, tox_friend_lossy_packet_handler);
1001
1.27k
    custom_lossless_packet_registerhandler(tox->m, tox_friend_lossless_packet_handler);
1002
1003
1.27k
    m_callback_group_invite(tox->m, tox_group_invite_handler);
1004
1.27k
    gc_callback_message(tox->m, tox_group_message_handler);
1005
1.27k
    gc_callback_private_message(tox->m, tox_group_private_message_handler);
1006
1.27k
    gc_callback_custom_packet(tox->m, tox_group_custom_packet_handler);
1007
1.27k
    gc_callback_custom_private_packet(tox->m, tox_group_custom_private_packet_handler);
1008
1.27k
    gc_callback_moderation(tox->m, tox_group_moderation_handler);
1009
1.27k
    gc_callback_nick_change(tox->m, tox_group_peer_name_handler);
1010
1.27k
    gc_callback_status_change(tox->m, tox_group_peer_status_handler);
1011
1.27k
    gc_callback_topic_change(tox->m, tox_group_topic_handler);
1012
1.27k
    gc_callback_peer_limit(tox->m, tox_group_peer_limit_handler);
1013
1.27k
    gc_callback_privacy_state(tox->m, tox_group_privacy_state_handler);
1014
1.27k
    gc_callback_topic_lock(tox->m, tox_group_topic_lock_handler);
1015
1.27k
    gc_callback_password(tox->m, tox_group_password_handler);
1016
1.27k
    gc_callback_peer_join(tox->m, tox_group_peer_join_handler);
1017
1.27k
    gc_callback_peer_exit(tox->m, tox_group_peer_exit_handler);
1018
1.27k
    gc_callback_self_join(tox->m, tox_group_self_join_handler);
1019
1.27k
    gc_callback_rejected(tox->m, tox_group_join_fail_handler);
1020
1.27k
    gc_callback_voice_state(tox->m, tox_group_voice_state_handler);
1021
1022
1.27k
    tox_unlock(tox);
1023
1024
1.27k
    SET_ERROR_PARAMETER(error, TOX_ERR_NEW_OK);
1025
1026
1.27k
    tox_options_free(default_options);
1027
1.27k
    return tox;
1028
1.76k
}
1029
1030
Tox *tox_new(const struct Tox_Options *options, Tox_Err_New *error)
1031
0
{
1032
0
    return tox_new_system(options, error, nullptr);
1033
0
}
1034
1035
Tox *tox_new_testing(const Tox_Options *options, Tox_Err_New *error, const Tox_Options_Testing *testing, Tox_Err_New_Testing *testing_error)
1036
1.79k
{
1037
1.79k
    if (testing == nullptr) {
1038
0
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_NULL);
1039
0
        SET_ERROR_PARAMETER(testing_error, TOX_ERR_NEW_TESTING_NULL);
1040
0
        return nullptr;
1041
0
    }
1042
1043
1.79k
    if (testing->operating_system == nullptr) {
1044
0
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_NULL);
1045
0
        SET_ERROR_PARAMETER(testing_error, TOX_ERR_NEW_TESTING_NULL);
1046
0
        return nullptr;
1047
0
    }
1048
1049
1.79k
    const Tox_System *sys = testing->operating_system;
1050
1051
1.79k
    if (sys->rng == nullptr || sys->ns == nullptr || sys->mem == nullptr) {
1052
0
        SET_ERROR_PARAMETER(error, TOX_ERR_NEW_NULL);
1053
0
        SET_ERROR_PARAMETER(testing_error, TOX_ERR_NEW_TESTING_NULL);
1054
0
        return nullptr;
1055
0
    }
1056
1057
1.79k
    SET_ERROR_PARAMETER(testing_error, TOX_ERR_NEW_TESTING_OK);
1058
1.79k
    return tox_new_system(options, error, sys);
1059
1.79k
}
1060
1061
void tox_kill(Tox *tox)
1062
1.27k
{
1063
1.27k
    if (tox == nullptr) {
1064
0
        return;
1065
0
    }
1066
1067
1.27k
    tox_lock(tox);
1068
1.27k
    LOGGER_ASSERT(tox->m->log, tox->toxav_object == nullptr, "Attempted to kill tox while toxav is still alive");
1069
1.27k
    kill_groupchats(tox->m->conferences_object);
1070
1.27k
    kill_messenger(tox->m);
1071
1.27k
    mono_time_free(tox->sys.mem, tox->mono_time);
1072
1.27k
    tox_unlock(tox);
1073
1074
1.27k
    if (tox->mutex != nullptr) {
1075
0
        pthread_mutex_destroy(tox->mutex);
1076
0
        mem_delete(tox->sys.mem, tox->mutex);
1077
0
    }
1078
1079
1.27k
    mem_delete(tox->sys.mem, tox);
1080
1.27k
}
1081
1082
static uint32_t end_size(void)
1083
2.54k
{
1084
2.54k
    return 2 * sizeof(uint32_t);
1085
2.54k
}
1086
1087
non_null()
1088
static void end_save(uint8_t *data)
1089
1.27k
{
1090
1.27k
    state_write_section_header(data, STATE_COOKIE_TYPE, 0, STATE_TYPE_END);
1091
1.27k
}
1092
1093
size_t tox_get_savedata_size(const Tox *tox)
1094
2.54k
{
1095
2.54k
    assert(tox != nullptr);
1096
2.54k
    tox_lock(tox);
1097
2.54k
    const size_t ret = 2 * sizeof(uint32_t)
1098
2.54k
                       + messenger_size(tox->m)
1099
2.54k
                       + conferences_size(tox->m->conferences_object)
1100
2.54k
                       + end_size();
1101
2.54k
    tox_unlock(tox);
1102
2.54k
    return ret;
1103
2.54k
}
1104
1105
void tox_get_savedata(const Tox *tox, uint8_t *savedata)
1106
1.27k
{
1107
1.27k
    assert(tox != nullptr);
1108
1109
1.27k
    if (savedata == nullptr) {
1110
0
        return;
1111
0
    }
1112
1113
1.27k
    memzero(savedata, tox_get_savedata_size(tox));
1114
1115
1.27k
    tox_lock(tox);
1116
1117
1.27k
    const uint32_t size32 = sizeof(uint32_t);
1118
1119
    // write cookie
1120
1.27k
    memzero(savedata, size32);
1121
1.27k
    savedata += size32;
1122
1.27k
    host_to_lendian_bytes32(savedata, STATE_COOKIE_GLOBAL);
1123
1.27k
    savedata += size32;
1124
1125
1.27k
    savedata = messenger_save(tox->m, savedata);
1126
1.27k
    savedata = conferences_save(tox->m->conferences_object, savedata);
1127
1.27k
    end_save(savedata);
1128
1129
1.27k
    tox_unlock(tox);
1130
1.27k
}
1131
1132
non_null(5) nullable(1, 2, 4, 6)
1133
static int32_t resolve_bootstrap_node(Tox *tox, const char *host, uint16_t port, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE],
1134
                                      IP_Port **root, Tox_Err_Bootstrap *error)
1135
0
{
1136
0
    assert(tox != nullptr);
1137
0
    assert(root != nullptr);
1138
1139
0
    if (host == nullptr || public_key == nullptr) {
1140
0
        SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_NULL);
1141
0
        return -1;
1142
0
    }
1143
1144
0
    if (port == 0) {
1145
0
        SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_BAD_PORT);
1146
0
        return -1;
1147
0
    }
1148
1149
0
    const int32_t count = net_getipport(tox->sys.ns, tox->sys.mem, host, root, TOX_SOCK_DGRAM, tox->m->options.dns_enabled);
1150
1151
0
    if (count < 1) {
1152
0
        LOGGER_DEBUG(tox->m->log, "could not resolve bootstrap node '%s'", host);
1153
0
        net_freeipport(tox->sys.mem, *root);
1154
0
        SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_BAD_HOST);
1155
0
        return -1;
1156
0
    }
1157
1158
0
    assert(*root != nullptr);
1159
0
    return count;
1160
0
}
1161
1162
bool tox_bootstrap(Tox *tox, const char *host, uint16_t port, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE], Tox_Err_Bootstrap *error)
1163
0
{
1164
0
    IP_Port *root;
1165
0
    const int32_t count = resolve_bootstrap_node(tox, host, port, public_key, &root, error);
1166
1167
0
    if (count == -1) {
1168
0
        return false;
1169
0
    }
1170
1171
0
    tox_lock(tox);
1172
0
    assert(count >= 0);
1173
0
    bool onion_success = false;
1174
    // UDP bootstrap is default success if it's disabled (because we don't even try).
1175
0
    bool udp_success = tox->m->options.udp_disabled;
1176
1177
0
    for (int32_t i = 0; i < count; ++i) {
1178
0
        if (!tox->m->options.ipv6enabled && net_family_is_ipv6(root[i].ip.family)) {
1179
            // We can't use ipv6 when it's disabled.
1180
0
            continue;
1181
0
        }
1182
1183
0
        root[i].port = net_htons(port);
1184
1185
0
        if (onion_add_bs_path_node(tox->m->onion_c, &root[i], public_key)) {
1186
            // If UDP is enabled, the caller cares about whether any of the
1187
            // bootstrap calls below will succeed. In TCP-only mode, adding
1188
            // onion path nodes successfully is sufficient.
1189
0
            onion_success = true;
1190
0
        }
1191
1192
0
        if (!tox->m->options.udp_disabled) {
1193
0
            if (dht_bootstrap(tox->m->dht, &root[i], public_key)) {
1194
                // If any of the bootstrap calls worked, we call it success.
1195
0
                udp_success = true;
1196
0
            }
1197
0
        }
1198
0
    }
1199
1200
0
    tox_unlock(tox);
1201
1202
0
    net_freeipport(tox->sys.mem, root);
1203
1204
0
    if (count == 0 || !onion_success || !udp_success) {
1205
0
        LOGGER_DEBUG(tox->m->log, "bootstrap node '%s' resolved to %d IP_Ports%s (onion: %s, UDP: %s)",
1206
0
                     host, count,
1207
0
                     count > 0 ? ", but failed to bootstrap with any of them" : "",
1208
0
                     onion_success ? "success" : "FAILURE",
1209
0
                     tox->m->options.udp_disabled ? "disabled" : (udp_success ? "success" : "FAILURE"));
1210
0
        SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_BAD_HOST);
1211
0
        return false;
1212
0
    }
1213
1214
0
    SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_OK);
1215
0
    return true;
1216
0
}
1217
1218
bool tox_add_tcp_relay(Tox *tox, const char *host, uint16_t port, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE],
1219
                       Tox_Err_Bootstrap *error)
1220
0
{
1221
0
    IP_Port *root;
1222
0
    const int32_t count = resolve_bootstrap_node(tox, host, port, public_key, &root, error);
1223
1224
0
    if (count == -1) {
1225
0
        return false;
1226
0
    }
1227
1228
0
    tox_lock(tox);
1229
0
    assert(count >= 0);
1230
1231
0
    for (int32_t i = 0; i < count; ++i) {
1232
0
        root[i].port = net_htons(port);
1233
1234
0
        add_tcp_relay(tox->m->net_crypto, &root[i], public_key);
1235
0
    }
1236
1237
0
    tox_unlock(tox);
1238
1239
0
    net_freeipport(tox->sys.mem, root);
1240
1241
0
    if (count == 0) {
1242
0
        SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_BAD_HOST);
1243
0
        return false;
1244
0
    }
1245
1246
0
    SET_ERROR_PARAMETER(error, TOX_ERR_BOOTSTRAP_OK);
1247
0
    return true;
1248
0
}
1249
1250
Tox_Connection tox_self_get_connection_status(const Tox *tox)
1251
0
{
1252
0
    assert(tox != nullptr);
1253
0
    tox_lock(tox);
1254
0
    const Onion_Connection_Status ret = onion_connection_status(tox->m->onion_c);
1255
0
    tox_unlock(tox);
1256
1257
0
    switch (ret) {
1258
0
        case ONION_CONNECTION_STATUS_NONE:
1259
0
            return TOX_CONNECTION_NONE;
1260
1261
0
        case ONION_CONNECTION_STATUS_TCP:
1262
0
            return TOX_CONNECTION_TCP;
1263
1264
0
        case ONION_CONNECTION_STATUS_UDP:
1265
0
            return TOX_CONNECTION_UDP;
1266
0
    }
1267
1268
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
1269
0
    return TOX_CONNECTION_NONE;
1270
0
}
1271
1272
void tox_callback_self_connection_status(Tox *tox, tox_self_connection_status_cb *callback)
1273
0
{
1274
0
    assert(tox != nullptr);
1275
0
    tox->self_connection_status_callback = callback;
1276
0
}
1277
1278
uint32_t tox_iteration_interval(const Tox *tox)
1279
0
{
1280
0
    assert(tox != nullptr);
1281
0
    tox_lock(tox);
1282
0
    uint32_t ret = messenger_run_interval(tox->m);
1283
1284
0
    if (m_is_receiving_file(tox->m)) {
1285
0
        ret = 1;
1286
0
    }
1287
1288
0
    tox_unlock(tox);
1289
0
    return ret;
1290
0
}
1291
1292
void tox_iterate(Tox *tox, void *user_data)
1293
0
{
1294
0
    assert(tox != nullptr);
1295
0
    tox_lock(tox);
1296
1297
0
    mono_time_update(tox->mono_time);
1298
1299
0
    struct Tox_Userdata tox_data = { tox, user_data };
1300
0
    do_messenger(tox->m, &tox_data);
1301
0
    do_groupchats(tox->m->conferences_object, &tox_data);
1302
1303
0
    tox_unlock(tox);
1304
0
}
1305
1306
void tox_self_get_address(const Tox *tox, uint8_t address[TOX_ADDRESS_SIZE])
1307
0
{
1308
0
    assert(tox != nullptr);
1309
1310
0
    if (address != nullptr) {
1311
0
        tox_lock(tox);
1312
0
        getaddress(tox->m, address);
1313
0
        tox_unlock(tox);
1314
0
    }
1315
0
}
1316
1317
void tox_self_set_nospam(Tox *tox, uint32_t nospam)
1318
0
{
1319
0
    assert(tox != nullptr);
1320
0
    tox_lock(tox);
1321
0
    set_nospam(tox->m->fr, net_htonl(nospam));
1322
0
    tox_unlock(tox);
1323
0
}
1324
1325
uint32_t tox_self_get_nospam(const Tox *tox)
1326
0
{
1327
0
    assert(tox != nullptr);
1328
0
    tox_lock(tox);
1329
0
    const uint32_t ret = net_ntohl(get_nospam(tox->m->fr));
1330
0
    tox_unlock(tox);
1331
0
    return ret;
1332
0
}
1333
1334
void tox_self_get_public_key(const Tox *tox, uint8_t public_key[TOX_PUBLIC_KEY_SIZE])
1335
0
{
1336
0
    assert(tox != nullptr);
1337
1338
0
    if (public_key != nullptr) {
1339
0
        tox_lock(tox);
1340
0
        memcpy(public_key, nc_get_self_public_key(tox->m->net_crypto), CRYPTO_PUBLIC_KEY_SIZE);
1341
0
        tox_unlock(tox);
1342
0
    }
1343
0
}
1344
1345
void tox_self_get_secret_key(const Tox *tox, uint8_t secret_key[TOX_SECRET_KEY_SIZE])
1346
0
{
1347
0
    assert(tox != nullptr);
1348
1349
0
    if (secret_key != nullptr) {
1350
0
        tox_lock(tox);
1351
0
        memcpy(secret_key, nc_get_self_secret_key(tox->m->net_crypto), CRYPTO_SECRET_KEY_SIZE);
1352
0
        tox_unlock(tox);
1353
0
    }
1354
0
}
1355
1356
bool tox_self_set_name(Tox *tox, const uint8_t *name, size_t length, Tox_Err_Set_Info *error)
1357
0
{
1358
0
    assert(tox != nullptr);
1359
1360
0
    if (name == nullptr && length != 0) {
1361
0
        SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_NULL);
1362
0
        return false;
1363
0
    }
1364
1365
0
    tox_lock(tox);
1366
1367
0
    if (setname(tox->m, name, length) == 0) {
1368
        // TODO(irungentoo): function to set different per group names?
1369
0
        send_name_all_groups(tox->m->conferences_object);
1370
0
        SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_OK);
1371
0
        tox_unlock(tox);
1372
0
        return true;
1373
0
    }
1374
1375
0
    SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_TOO_LONG);
1376
0
    tox_unlock(tox);
1377
0
    return false;
1378
0
}
1379
1380
size_t tox_self_get_name_size(const Tox *tox)
1381
0
{
1382
0
    assert(tox != nullptr);
1383
0
    tox_lock(tox);
1384
0
    const size_t ret = m_get_self_name_size(tox->m);
1385
0
    tox_unlock(tox);
1386
0
    return ret;
1387
0
}
1388
1389
void tox_self_get_name(const Tox *tox, uint8_t *name)
1390
0
{
1391
0
    assert(tox != nullptr);
1392
1393
0
    if (name != nullptr) {
1394
0
        tox_lock(tox);
1395
0
        getself_name(tox->m, name);
1396
0
        tox_unlock(tox);
1397
0
    }
1398
0
}
1399
1400
bool tox_self_set_status_message(Tox *tox, const uint8_t *status_message, size_t length, Tox_Err_Set_Info *error)
1401
0
{
1402
0
    assert(tox != nullptr);
1403
1404
0
    if (status_message == nullptr && length != 0) {
1405
0
        SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_NULL);
1406
0
        return false;
1407
0
    }
1408
1409
0
    tox_lock(tox);
1410
1411
0
    if (m_set_statusmessage(tox->m, status_message, length) == 0) {
1412
0
        SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_OK);
1413
0
        tox_unlock(tox);
1414
0
        return true;
1415
0
    }
1416
1417
0
    SET_ERROR_PARAMETER(error, TOX_ERR_SET_INFO_TOO_LONG);
1418
0
    tox_unlock(tox);
1419
0
    return false;
1420
0
}
1421
1422
size_t tox_self_get_status_message_size(const Tox *tox)
1423
0
{
1424
0
    assert(tox != nullptr);
1425
0
    tox_lock(tox);
1426
0
    const size_t ret = m_get_self_statusmessage_size(tox->m);
1427
0
    tox_unlock(tox);
1428
0
    return ret;
1429
0
}
1430
1431
void tox_self_get_status_message(const Tox *tox, uint8_t *status_message)
1432
0
{
1433
0
    assert(tox != nullptr);
1434
1435
0
    if (status_message != nullptr) {
1436
0
        tox_lock(tox);
1437
0
        m_copy_self_statusmessage(tox->m, status_message);
1438
0
        tox_unlock(tox);
1439
0
    }
1440
0
}
1441
1442
void tox_self_set_status(Tox *tox, Tox_User_Status status)
1443
0
{
1444
0
    assert(tox != nullptr);
1445
0
    tox_lock(tox);
1446
0
    m_set_userstatus(tox->m, status);
1447
0
    tox_unlock(tox);
1448
0
}
1449
1450
Tox_User_Status tox_self_get_status(const Tox *tox)
1451
0
{
1452
0
    assert(tox != nullptr);
1453
0
    tox_lock(tox);
1454
0
    const uint8_t status = m_get_self_userstatus(tox->m);
1455
0
    tox_unlock(tox);
1456
0
    return (Tox_User_Status)status;
1457
0
}
1458
1459
non_null(1) nullable(3)
1460
static void set_friend_error(const Logger *log, int32_t ret, Tox_Err_Friend_Add *error)
1461
0
{
1462
0
    switch (ret) {
1463
0
        case FAERR_TOOLONG: {
1464
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_TOO_LONG);
1465
0
            break;
1466
0
        }
1467
1468
0
        case FAERR_NOMESSAGE: {
1469
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_NO_MESSAGE);
1470
0
            break;
1471
0
        }
1472
1473
0
        case FAERR_OWNKEY: {
1474
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_OWN_KEY);
1475
0
            break;
1476
0
        }
1477
1478
0
        case FAERR_ALREADYSENT: {
1479
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_ALREADY_SENT);
1480
0
            break;
1481
0
        }
1482
1483
0
        case FAERR_BADCHECKSUM: {
1484
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_BAD_CHECKSUM);
1485
0
            break;
1486
0
        }
1487
1488
0
        case FAERR_SETNEWNOSPAM: {
1489
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_SET_NEW_NOSPAM);
1490
0
            break;
1491
0
        }
1492
1493
0
        case FAERR_NOMEM: {
1494
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_MALLOC);
1495
0
            break;
1496
0
        }
1497
1498
0
        default: {
1499
            /* can't happen */
1500
0
            LOGGER_FATAL(log, "impossible return value: %d", ret);
1501
0
            break;
1502
0
        }
1503
0
    }
1504
0
}
1505
1506
uint32_t tox_friend_add(Tox *tox, const uint8_t address[TOX_ADDRESS_SIZE], const uint8_t *message, size_t length,
1507
                        Tox_Err_Friend_Add *error)
1508
0
{
1509
0
    assert(tox != nullptr);
1510
1511
0
    if (address == nullptr || message == nullptr) {
1512
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_NULL);
1513
0
        return UINT32_MAX;
1514
0
    }
1515
1516
0
    tox_lock(tox);
1517
0
    const int32_t ret = m_addfriend(tox->m, address, message, length);
1518
1519
0
    if (ret >= 0) {
1520
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_OK);
1521
0
        tox_unlock(tox);
1522
0
        return (uint32_t)ret;
1523
0
    }
1524
1525
0
    set_friend_error(tox->m->log, ret, error);
1526
0
    tox_unlock(tox);
1527
0
    return UINT32_MAX;
1528
0
}
1529
1530
uint32_t tox_friend_add_norequest(Tox *tox, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE], Tox_Err_Friend_Add *error)
1531
0
{
1532
0
    assert(tox != nullptr);
1533
1534
0
    if (public_key == nullptr) {
1535
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_NULL);
1536
0
        return UINT32_MAX;
1537
0
    }
1538
1539
0
    tox_lock(tox);
1540
0
    const int32_t ret = m_addfriend_norequest(tox->m, public_key);
1541
1542
0
    if (ret >= 0) {
1543
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_ADD_OK);
1544
0
        tox_unlock(tox);
1545
0
        return (uint32_t)ret;
1546
0
    }
1547
1548
0
    set_friend_error(tox->m->log, ret, error);
1549
0
    tox_unlock(tox);
1550
0
    return UINT32_MAX;
1551
0
}
1552
1553
bool tox_friend_delete(Tox *tox, uint32_t friend_number, Tox_Err_Friend_Delete *error)
1554
0
{
1555
0
    assert(tox != nullptr);
1556
0
    tox_lock(tox);
1557
0
    const int ret = m_delfriend(tox->m, friend_number);
1558
0
    tox_unlock(tox);
1559
1560
    // TODO(irungentoo): handle if realloc fails?
1561
0
    if (ret == -1) {
1562
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_DELETE_FRIEND_NOT_FOUND);
1563
0
        return false;
1564
0
    }
1565
1566
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_DELETE_OK);
1567
0
    return true;
1568
0
}
1569
1570
uint32_t tox_friend_by_public_key(const Tox *tox, const uint8_t public_key[TOX_PUBLIC_KEY_SIZE], Tox_Err_Friend_By_Public_Key *error)
1571
0
{
1572
0
    assert(tox != nullptr);
1573
1574
0
    if (public_key == nullptr) {
1575
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_BY_PUBLIC_KEY_NULL);
1576
0
        return UINT32_MAX;
1577
0
    }
1578
1579
0
    tox_lock(tox);
1580
0
    const int32_t ret = getfriend_id(tox->m, public_key);
1581
0
    tox_unlock(tox);
1582
1583
0
    if (ret == -1) {
1584
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_BY_PUBLIC_KEY_NOT_FOUND);
1585
0
        return UINT32_MAX;
1586
0
    }
1587
1588
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_BY_PUBLIC_KEY_OK);
1589
0
    assert(ret >= 0);
1590
0
    return (uint32_t)ret;
1591
0
}
1592
1593
bool tox_friend_get_public_key(const Tox *tox, uint32_t friend_number, uint8_t public_key[TOX_PUBLIC_KEY_SIZE],
1594
                               Tox_Err_Friend_Get_Public_Key *error)
1595
0
{
1596
0
    assert(tox != nullptr);
1597
1598
0
    if (public_key == nullptr) {
1599
0
        return false;
1600
0
    }
1601
1602
0
    tox_lock(tox);
1603
1604
0
    if (get_real_pk(tox->m, friend_number, public_key) == -1) {
1605
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_GET_PUBLIC_KEY_FRIEND_NOT_FOUND);
1606
0
        tox_unlock(tox);
1607
0
        return false;
1608
0
    }
1609
1610
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_GET_PUBLIC_KEY_OK);
1611
0
    tox_unlock(tox);
1612
0
    return true;
1613
0
}
1614
1615
bool tox_friend_exists(const Tox *tox, uint32_t friend_number)
1616
0
{
1617
0
    assert(tox != nullptr);
1618
0
    tox_lock(tox);
1619
0
    const bool ret = m_friend_exists(tox->m, friend_number);
1620
0
    tox_unlock(tox);
1621
0
    return ret;
1622
0
}
1623
1624
uint64_t tox_friend_get_last_online(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Get_Last_Online *error)
1625
0
{
1626
0
    assert(tox != nullptr);
1627
0
    tox_lock(tox);
1628
0
    const uint64_t timestamp = m_get_last_online(tox->m, friend_number);
1629
0
    tox_unlock(tox);
1630
1631
0
    if (timestamp == UINT64_MAX) {
1632
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_GET_LAST_ONLINE_FRIEND_NOT_FOUND);
1633
0
        return UINT64_MAX;
1634
0
    }
1635
1636
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_GET_LAST_ONLINE_OK);
1637
0
    return timestamp;
1638
0
}
1639
1640
size_t tox_self_get_friend_list_size(const Tox *tox)
1641
0
{
1642
0
    assert(tox != nullptr);
1643
0
    tox_lock(tox);
1644
0
    const size_t ret = count_friendlist(tox->m);
1645
0
    tox_unlock(tox);
1646
0
    return ret;
1647
0
}
1648
1649
void tox_self_get_friend_list(const Tox *tox, uint32_t *friend_list)
1650
0
{
1651
0
    assert(tox != nullptr);
1652
1653
0
    if (friend_list != nullptr) {
1654
0
        tox_lock(tox);
1655
        // TODO(irungentoo): size parameter?
1656
0
        copy_friendlist(tox->m, friend_list, count_friendlist(tox->m));
1657
0
        tox_unlock(tox);
1658
0
    }
1659
0
}
1660
1661
size_t tox_friend_get_name_size(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Query *error)
1662
0
{
1663
0
    assert(tox != nullptr);
1664
0
    tox_lock(tox);
1665
0
    const int ret = m_get_name_size(tox->m, friend_number);
1666
0
    tox_unlock(tox);
1667
1668
0
    if (ret == -1) {
1669
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1670
0
        return SIZE_MAX;
1671
0
    }
1672
1673
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1674
0
    return ret;
1675
0
}
1676
1677
bool tox_friend_get_name(const Tox *tox, uint32_t friend_number, uint8_t *name, Tox_Err_Friend_Query *error)
1678
0
{
1679
0
    assert(tox != nullptr);
1680
1681
0
    if (name == nullptr) {
1682
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_NULL);
1683
0
        return false;
1684
0
    }
1685
1686
0
    tox_lock(tox);
1687
0
    const int ret = getname(tox->m, friend_number, name);
1688
0
    tox_unlock(tox);
1689
1690
0
    if (ret == -1) {
1691
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1692
0
        return false;
1693
0
    }
1694
1695
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1696
0
    return true;
1697
0
}
1698
1699
void tox_callback_friend_name(Tox *tox, tox_friend_name_cb *callback)
1700
0
{
1701
0
    assert(tox != nullptr);
1702
0
    tox->friend_name_callback = callback;
1703
0
}
1704
1705
size_t tox_friend_get_status_message_size(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Query *error)
1706
0
{
1707
0
    assert(tox != nullptr);
1708
0
    tox_lock(tox);
1709
0
    const int ret = m_get_statusmessage_size(tox->m, friend_number);
1710
0
    tox_unlock(tox);
1711
1712
0
    if (ret == -1) {
1713
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1714
0
        return SIZE_MAX;
1715
0
    }
1716
1717
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1718
0
    return ret;
1719
0
}
1720
1721
bool tox_friend_get_status_message(const Tox *tox, uint32_t friend_number, uint8_t *status_message,
1722
                                   Tox_Err_Friend_Query *error)
1723
0
{
1724
0
    assert(tox != nullptr);
1725
1726
0
    if (status_message == nullptr) {
1727
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_NULL);
1728
0
        return false;
1729
0
    }
1730
1731
0
    tox_lock(tox);
1732
0
    const int size = m_get_statusmessage_size(tox->m, friend_number);
1733
1734
0
    if (size == -1) {
1735
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1736
0
        tox_unlock(tox);
1737
0
        return false;
1738
0
    }
1739
1740
0
    const int ret = m_copy_statusmessage(tox->m, friend_number, status_message, size);
1741
0
    LOGGER_ASSERT(tox->m->log, ret == size, "concurrency problem: friend status message changed");
1742
1743
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1744
0
    tox_unlock(tox);
1745
0
    return ret == size;
1746
0
}
1747
1748
void tox_callback_friend_status_message(Tox *tox, tox_friend_status_message_cb *callback)
1749
0
{
1750
0
    assert(tox != nullptr);
1751
0
    tox->friend_status_message_callback = callback;
1752
0
}
1753
1754
Tox_User_Status tox_friend_get_status(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Query *error)
1755
0
{
1756
0
    assert(tox != nullptr);
1757
0
    tox_lock(tox);
1758
0
    const int ret = m_get_userstatus(tox->m, friend_number);
1759
0
    tox_unlock(tox);
1760
1761
0
    if (ret == USERSTATUS_INVALID) {
1762
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1763
0
        return TOX_USER_STATUS_NONE;
1764
0
    }
1765
1766
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1767
0
    return (Tox_User_Status)ret;
1768
0
}
1769
1770
void tox_callback_friend_status(Tox *tox, tox_friend_status_cb *callback)
1771
0
{
1772
0
    assert(tox != nullptr);
1773
0
    tox->friend_status_callback = callback;
1774
0
}
1775
1776
Tox_Connection tox_friend_get_connection_status(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Query *error)
1777
0
{
1778
0
    assert(tox != nullptr);
1779
0
    tox_lock(tox);
1780
0
    const int ret = m_get_friend_connectionstatus(tox->m, friend_number);
1781
0
    tox_unlock(tox);
1782
1783
0
    if (ret == -1) {
1784
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1785
0
        return TOX_CONNECTION_NONE;
1786
0
    }
1787
1788
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1789
0
    return (Tox_Connection)ret;
1790
0
}
1791
1792
void tox_callback_friend_connection_status(Tox *tox, tox_friend_connection_status_cb *callback)
1793
0
{
1794
0
    assert(tox != nullptr);
1795
0
    tox->friend_connection_status_callback = callback;
1796
0
}
1797
1798
bool tox_friend_get_typing(const Tox *tox, uint32_t friend_number, Tox_Err_Friend_Query *error)
1799
0
{
1800
0
    assert(tox != nullptr);
1801
0
    tox_lock(tox);
1802
0
    const int ret = m_get_istyping(tox->m, friend_number);
1803
0
    tox_unlock(tox);
1804
1805
0
    if (ret == -1) {
1806
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_FRIEND_NOT_FOUND);
1807
0
        return false;
1808
0
    }
1809
1810
0
    SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_QUERY_OK);
1811
0
    return ret != 0;
1812
0
}
1813
1814
void tox_callback_friend_typing(Tox *tox, tox_friend_typing_cb *callback)
1815
0
{
1816
0
    assert(tox != nullptr);
1817
0
    tox->friend_typing_callback = callback;
1818
0
}
1819
1820
bool tox_self_set_typing(Tox *tox, uint32_t friend_number, bool typing, Tox_Err_Set_Typing *error)
1821
0
{
1822
0
    assert(tox != nullptr);
1823
0
    tox_lock(tox);
1824
1825
0
    if (m_set_usertyping(tox->m, friend_number, typing) == -1) {
1826
0
        SET_ERROR_PARAMETER(error, TOX_ERR_SET_TYPING_FRIEND_NOT_FOUND);
1827
0
        tox_unlock(tox);
1828
0
        return false;
1829
0
    }
1830
1831
0
    SET_ERROR_PARAMETER(error, TOX_ERR_SET_TYPING_OK);
1832
0
    tox_unlock(tox);
1833
0
    return true;
1834
0
}
1835
1836
non_null(1) nullable(3)
1837
static void set_message_error(const Logger *log, int ret, Tox_Err_Friend_Send_Message *error)
1838
0
{
1839
0
    switch (ret) {
1840
0
        case 0: {
1841
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_OK);
1842
0
            break;
1843
0
        }
1844
1845
0
        case -1: {
1846
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_FRIEND_NOT_FOUND);
1847
0
            break;
1848
0
        }
1849
1850
0
        case -2: {
1851
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_TOO_LONG);
1852
0
            break;
1853
0
        }
1854
1855
0
        case -3: {
1856
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_FRIEND_NOT_CONNECTED);
1857
0
            break;
1858
0
        }
1859
1860
0
        case -4: {
1861
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_SENDQ);
1862
0
            break;
1863
0
        }
1864
1865
0
        case -5: {
1866
0
            LOGGER_FATAL(log, "impossible: Messenger and Tox disagree on message types");
1867
0
            break;
1868
0
        }
1869
1870
0
        default: {
1871
            /* can't happen */
1872
0
            LOGGER_FATAL(log, "impossible return value: %d", ret);
1873
0
            break;
1874
0
        }
1875
0
    }
1876
0
}
1877
1878
uint32_t tox_friend_send_message(Tox *tox, uint32_t friend_number, Tox_Message_Type type, const uint8_t *message,
1879
                                 size_t length, Tox_Err_Friend_Send_Message *error)
1880
0
{
1881
0
    assert(tox != nullptr);
1882
1883
0
    if (message == nullptr) {
1884
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_NULL);
1885
0
        return 0;
1886
0
    }
1887
1888
0
    if (length == 0) {
1889
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_SEND_MESSAGE_EMPTY);
1890
0
        return 0;
1891
0
    }
1892
1893
0
    uint32_t message_id = 0;
1894
0
    tox_lock(tox);
1895
0
    set_message_error(tox->m->log, m_send_message_generic(tox->m, friend_number, type, message, length, &message_id),
1896
0
                      error);
1897
0
    tox_unlock(tox);
1898
0
    return message_id;
1899
0
}
1900
1901
void tox_callback_friend_read_receipt(Tox *tox, tox_friend_read_receipt_cb *callback)
1902
0
{
1903
0
    assert(tox != nullptr);
1904
0
    tox->friend_read_receipt_callback = callback;
1905
0
}
1906
1907
void tox_callback_friend_request(Tox *tox, tox_friend_request_cb *callback)
1908
0
{
1909
0
    assert(tox != nullptr);
1910
0
    tox->friend_request_callback = callback;
1911
0
}
1912
1913
void tox_callback_friend_message(Tox *tox, tox_friend_message_cb *callback)
1914
0
{
1915
0
    assert(tox != nullptr);
1916
0
    tox->friend_message_callback = callback;
1917
0
}
1918
1919
bool tox_hash(uint8_t hash[TOX_HASH_LENGTH], const uint8_t *data, size_t length)
1920
0
{
1921
0
    if (hash == nullptr || (data == nullptr && length != 0)) {
1922
0
        return false;
1923
0
    }
1924
1925
0
    crypto_sha256(hash, data, length);
1926
0
    return true;
1927
0
}
1928
1929
bool tox_file_control(Tox *tox, uint32_t friend_number, uint32_t file_number, Tox_File_Control control,
1930
                      Tox_Err_File_Control *error)
1931
0
{
1932
0
    assert(tox != nullptr);
1933
0
    tox_lock(tox);
1934
0
    const int ret = file_control(tox->m, friend_number, file_number, control);
1935
0
    tox_unlock(tox);
1936
1937
0
    if (ret == 0) {
1938
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_OK);
1939
0
        return true;
1940
0
    }
1941
1942
0
    switch (ret) {
1943
0
        case -1: {
1944
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_FRIEND_NOT_FOUND);
1945
0
            return false;
1946
0
        }
1947
1948
0
        case -2: {
1949
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_FRIEND_NOT_CONNECTED);
1950
0
            return false;
1951
0
        }
1952
1953
0
        case -3: {
1954
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_NOT_FOUND);
1955
0
            return false;
1956
0
        }
1957
1958
0
        case -4: {
1959
            /* can't happen (this code is returned if `control` is invalid type) */
1960
0
            LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
1961
0
            return false;
1962
0
        }
1963
1964
0
        case -5: {
1965
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_ALREADY_PAUSED);
1966
0
            return false;
1967
0
        }
1968
1969
0
        case -6: {
1970
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_DENIED);
1971
0
            return false;
1972
0
        }
1973
1974
0
        case -7: {
1975
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_NOT_PAUSED);
1976
0
            return false;
1977
0
        }
1978
1979
0
        case -8: {
1980
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_CONTROL_SENDQ);
1981
0
            return false;
1982
0
        }
1983
0
    }
1984
1985
    /* can't happen */
1986
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
1987
1988
0
    return false;
1989
0
}
1990
1991
bool tox_file_seek(Tox *tox, uint32_t friend_number, uint32_t file_number, uint64_t position,
1992
                   Tox_Err_File_Seek *error)
1993
0
{
1994
0
    assert(tox != nullptr);
1995
0
    tox_lock(tox);
1996
0
    const int ret = file_seek(tox->m, friend_number, file_number, position);
1997
0
    tox_unlock(tox);
1998
1999
0
    if (ret == 0) {
2000
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_OK);
2001
0
        return true;
2002
0
    }
2003
2004
0
    switch (ret) {
2005
0
        case -1: {
2006
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_FRIEND_NOT_FOUND);
2007
0
            return false;
2008
0
        }
2009
2010
0
        case -2: {
2011
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_FRIEND_NOT_CONNECTED);
2012
0
            return false;
2013
0
        }
2014
2015
0
        case -3: {
2016
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_NOT_FOUND);
2017
0
            return false;
2018
0
        }
2019
2020
0
        case -4: // fall-through
2021
0
        case -5: {
2022
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_DENIED);
2023
0
            return false;
2024
0
        }
2025
2026
0
        case -6: {
2027
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_INVALID_POSITION);
2028
0
            return false;
2029
0
        }
2030
2031
0
        case -8: {
2032
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEEK_SENDQ);
2033
0
            return false;
2034
0
        }
2035
0
    }
2036
2037
    /* can't happen */
2038
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
2039
2040
0
    return false;
2041
0
}
2042
2043
void tox_callback_file_recv_control(Tox *tox, tox_file_recv_control_cb *callback)
2044
0
{
2045
0
    assert(tox != nullptr);
2046
0
    tox->file_recv_control_callback = callback;
2047
0
}
2048
2049
bool tox_file_get_file_id(const Tox *tox, uint32_t friend_number, uint32_t file_number, uint8_t file_id[TOX_FILE_ID_LENGTH],
2050
                          Tox_Err_File_Get *error)
2051
0
{
2052
0
    assert(tox != nullptr);
2053
2054
0
    if (file_id == nullptr) {
2055
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_GET_NULL);
2056
0
        return false;
2057
0
    }
2058
2059
0
    tox_lock(tox);
2060
0
    const int ret = file_get_id(tox->m, friend_number, file_number, file_id);
2061
0
    tox_unlock(tox);
2062
2063
0
    if (ret == 0) {
2064
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_GET_OK);
2065
0
        return true;
2066
0
    }
2067
2068
0
    if (ret == -1) {
2069
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_GET_FRIEND_NOT_FOUND);
2070
0
    } else {
2071
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_GET_NOT_FOUND);
2072
0
    }
2073
2074
0
    return false;
2075
0
}
2076
2077
uint32_t tox_file_send(Tox *tox, uint32_t friend_number, uint32_t kind, uint64_t file_size, const uint8_t file_id[TOX_FILE_ID_LENGTH],
2078
                       const uint8_t *filename, size_t filename_length, Tox_Err_File_Send *error)
2079
0
{
2080
0
    assert(tox != nullptr);
2081
2082
0
    if (filename == nullptr && filename_length != 0) {
2083
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_NULL);
2084
0
        return UINT32_MAX;
2085
0
    }
2086
2087
0
    uint8_t f_id[FILE_ID_LENGTH];
2088
2089
0
    if (file_id == nullptr) {
2090
        /* Tox keys are 32 bytes like FILE_ID_LENGTH. */
2091
0
        new_symmetric_key(tox->sys.rng, f_id);
2092
0
        file_id = f_id;
2093
0
    }
2094
2095
0
    tox_lock(tox);
2096
0
    const long int file_num = new_filesender(tox->m, friend_number, kind, file_size, file_id, filename, filename_length);
2097
0
    tox_unlock(tox);
2098
2099
0
    if (file_num >= 0) {
2100
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_OK);
2101
0
        return file_num;
2102
0
    }
2103
2104
0
    switch (file_num) {
2105
0
        case -1: {
2106
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_FRIEND_NOT_FOUND);
2107
0
            return UINT32_MAX;
2108
0
        }
2109
2110
0
        case -2: {
2111
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_NAME_TOO_LONG);
2112
0
            return UINT32_MAX;
2113
0
        }
2114
2115
0
        case -3: {
2116
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_TOO_MANY);
2117
0
            return UINT32_MAX;
2118
0
        }
2119
2120
0
        case -4: {
2121
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_FRIEND_NOT_CONNECTED);
2122
0
            return UINT32_MAX;
2123
0
        }
2124
0
    }
2125
2126
    /* can't happen */
2127
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %ld", file_num);
2128
2129
0
    return UINT32_MAX;
2130
0
}
2131
2132
bool tox_file_send_chunk(Tox *tox, uint32_t friend_number, uint32_t file_number, uint64_t position, const uint8_t *data,
2133
                         size_t length, Tox_Err_File_Send_Chunk *error)
2134
0
{
2135
0
    assert(tox != nullptr);
2136
0
    tox_lock(tox);
2137
0
    const int ret = send_file_data(tox->m, friend_number, file_number, position, data, length);
2138
0
    tox_unlock(tox);
2139
2140
0
    if (ret == 0) {
2141
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_OK);
2142
0
        return true;
2143
0
    }
2144
2145
0
    switch (ret) {
2146
0
        case -1: {
2147
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_FRIEND_NOT_FOUND);
2148
0
            return false;
2149
0
        }
2150
2151
0
        case -2: {
2152
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_FRIEND_NOT_CONNECTED);
2153
0
            return false;
2154
0
        }
2155
2156
0
        case -3: {
2157
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_NOT_FOUND);
2158
0
            return false;
2159
0
        }
2160
2161
0
        case -4: {
2162
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_NOT_TRANSFERRING);
2163
0
            return false;
2164
0
        }
2165
2166
0
        case -5: {
2167
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_INVALID_LENGTH);
2168
0
            return false;
2169
0
        }
2170
2171
0
        case -6: {
2172
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_SENDQ);
2173
0
            return false;
2174
0
        }
2175
2176
0
        case -7: {
2177
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FILE_SEND_CHUNK_WRONG_POSITION);
2178
0
            return false;
2179
0
        }
2180
0
    }
2181
2182
    /* can't happen */
2183
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
2184
2185
0
    return false;
2186
0
}
2187
2188
void tox_callback_file_chunk_request(Tox *tox, tox_file_chunk_request_cb *callback)
2189
0
{
2190
0
    assert(tox != nullptr);
2191
0
    tox->file_chunk_request_callback = callback;
2192
0
}
2193
2194
void tox_callback_file_recv(Tox *tox, tox_file_recv_cb *callback)
2195
0
{
2196
0
    assert(tox != nullptr);
2197
0
    tox->file_recv_callback = callback;
2198
0
}
2199
2200
void tox_callback_file_recv_chunk(Tox *tox, tox_file_recv_chunk_cb *callback)
2201
0
{
2202
0
    assert(tox != nullptr);
2203
0
    tox->file_recv_chunk_callback = callback;
2204
0
}
2205
2206
void tox_callback_conference_invite(Tox *tox, tox_conference_invite_cb *callback)
2207
0
{
2208
0
    assert(tox != nullptr);
2209
0
    tox->conference_invite_callback = callback;
2210
0
}
2211
2212
void tox_callback_conference_connected(Tox *tox, tox_conference_connected_cb *callback)
2213
0
{
2214
0
    assert(tox != nullptr);
2215
0
    tox->conference_connected_callback = callback;
2216
0
}
2217
2218
void tox_callback_conference_message(Tox *tox, tox_conference_message_cb *callback)
2219
0
{
2220
0
    assert(tox != nullptr);
2221
0
    tox->conference_message_callback = callback;
2222
0
}
2223
2224
void tox_callback_conference_title(Tox *tox, tox_conference_title_cb *callback)
2225
0
{
2226
0
    assert(tox != nullptr);
2227
0
    tox->conference_title_callback = callback;
2228
0
}
2229
2230
void tox_callback_conference_peer_name(Tox *tox, tox_conference_peer_name_cb *callback)
2231
0
{
2232
0
    assert(tox != nullptr);
2233
0
    tox->conference_peer_name_callback = callback;
2234
0
}
2235
2236
void tox_callback_conference_peer_list_changed(Tox *tox, tox_conference_peer_list_changed_cb *callback)
2237
0
{
2238
0
    assert(tox != nullptr);
2239
0
    tox->conference_peer_list_changed_callback = callback;
2240
0
}
2241
2242
uint32_t tox_conference_new(Tox *tox, Tox_Err_Conference_New *error)
2243
0
{
2244
0
    assert(tox != nullptr);
2245
0
    tox_lock(tox);
2246
0
    const int ret = add_groupchat(tox->m->conferences_object, tox->sys.rng, GROUPCHAT_TYPE_TEXT);
2247
0
    tox_unlock(tox);
2248
2249
0
    if (ret == -1) {
2250
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_NEW_INIT);
2251
0
        return UINT32_MAX;
2252
0
    }
2253
2254
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_NEW_OK);
2255
0
    return ret;
2256
0
}
2257
2258
bool tox_conference_delete(Tox *tox, uint32_t conference_number, Tox_Err_Conference_Delete *error)
2259
0
{
2260
0
    assert(tox != nullptr);
2261
0
    tox_lock(tox);
2262
0
    const bool ret = del_groupchat(tox->m->conferences_object, conference_number, true);
2263
0
    tox_unlock(tox);
2264
2265
0
    if (!ret) {
2266
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_DELETE_CONFERENCE_NOT_FOUND);
2267
0
        return false;
2268
0
    }
2269
2270
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_DELETE_OK);
2271
0
    return true;
2272
0
}
2273
2274
uint32_t tox_conference_peer_count(const Tox *tox, uint32_t conference_number, Tox_Err_Conference_Peer_Query *error)
2275
0
{
2276
0
    assert(tox != nullptr);
2277
0
    tox_lock(tox);
2278
0
    const int ret = group_number_peers(tox->m->conferences_object, conference_number, false);
2279
0
    tox_unlock(tox);
2280
2281
0
    if (ret == -1) {
2282
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2283
0
        return UINT32_MAX;
2284
0
    }
2285
2286
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2287
0
    return ret;
2288
0
}
2289
2290
size_t tox_conference_peer_get_name_size(const Tox *tox, uint32_t conference_number, uint32_t peer_number,
2291
        Tox_Err_Conference_Peer_Query *error)
2292
0
{
2293
0
    assert(tox != nullptr);
2294
0
    tox_lock(tox);
2295
0
    const int ret = group_peername_size(tox->m->conferences_object, conference_number, peer_number, false);
2296
0
    tox_unlock(tox);
2297
2298
0
    switch (ret) {
2299
0
        case -1: {
2300
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2301
0
            return -1;
2302
0
        }
2303
2304
0
        case -2: {
2305
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2306
0
            return -1;
2307
0
        }
2308
0
    }
2309
2310
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2311
0
    return ret;
2312
0
}
2313
2314
bool tox_conference_peer_get_name(const Tox *tox, uint32_t conference_number, uint32_t peer_number, uint8_t *name,
2315
                                  Tox_Err_Conference_Peer_Query *error)
2316
0
{
2317
0
    assert(tox != nullptr);
2318
0
    tox_lock(tox);
2319
0
    const int ret = group_peername(tox->m->conferences_object, conference_number, peer_number, name, false);
2320
0
    tox_unlock(tox);
2321
2322
0
    switch (ret) {
2323
0
        case -1: {
2324
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2325
0
            return false;
2326
0
        }
2327
2328
0
        case -2: {
2329
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2330
0
            return false;
2331
0
        }
2332
0
    }
2333
2334
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2335
0
    return true;
2336
0
}
2337
2338
bool tox_conference_peer_get_public_key(const Tox *tox, uint32_t conference_number, uint32_t peer_number,
2339
                                        uint8_t public_key[TOX_PUBLIC_KEY_SIZE], Tox_Err_Conference_Peer_Query *error)
2340
0
{
2341
0
    assert(tox != nullptr);
2342
0
    tox_lock(tox);
2343
0
    const int ret = group_peer_pubkey(tox->m->conferences_object, conference_number, peer_number, public_key, false);
2344
0
    tox_unlock(tox);
2345
2346
0
    switch (ret) {
2347
0
        case -1: {
2348
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2349
0
            return false;
2350
0
        }
2351
2352
0
        case -2: {
2353
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2354
0
            return false;
2355
0
        }
2356
0
    }
2357
2358
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2359
0
    return true;
2360
0
}
2361
2362
bool tox_conference_peer_number_is_ours(const Tox *tox, uint32_t conference_number, uint32_t peer_number,
2363
                                        Tox_Err_Conference_Peer_Query *error)
2364
0
{
2365
0
    assert(tox != nullptr);
2366
0
    tox_lock(tox);
2367
0
    const int ret = group_peernumber_is_ours(tox->m->conferences_object, conference_number, peer_number);
2368
0
    tox_unlock(tox);
2369
2370
0
    switch (ret) {
2371
0
        case -1: {
2372
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2373
0
            return false;
2374
0
        }
2375
2376
0
        case -2: {
2377
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2378
0
            return false;
2379
0
        }
2380
2381
0
        case -3: {
2382
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_NO_CONNECTION);
2383
0
            return false;
2384
0
        }
2385
0
    }
2386
2387
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2388
0
    return ret != 0;
2389
0
}
2390
2391
uint32_t tox_conference_offline_peer_count(const Tox *tox, uint32_t conference_number,
2392
        Tox_Err_Conference_Peer_Query *error)
2393
0
{
2394
0
    assert(tox != nullptr);
2395
0
    tox_lock(tox);
2396
0
    const int ret = group_number_peers(tox->m->conferences_object, conference_number, true);
2397
0
    tox_unlock(tox);
2398
2399
0
    if (ret == -1) {
2400
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2401
0
        return UINT32_MAX;
2402
0
    }
2403
2404
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2405
0
    return ret;
2406
0
}
2407
2408
size_t tox_conference_offline_peer_get_name_size(const Tox *tox, uint32_t conference_number,
2409
        uint32_t offline_peer_number,
2410
        Tox_Err_Conference_Peer_Query *error)
2411
0
{
2412
0
    assert(tox != nullptr);
2413
0
    tox_lock(tox);
2414
0
    const int ret = group_peername_size(tox->m->conferences_object, conference_number, offline_peer_number, true);
2415
0
    tox_unlock(tox);
2416
2417
0
    switch (ret) {
2418
0
        case -1: {
2419
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2420
0
            return -1;
2421
0
        }
2422
2423
0
        case -2: {
2424
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2425
0
            return -1;
2426
0
        }
2427
0
    }
2428
2429
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2430
0
    return ret;
2431
0
}
2432
2433
bool tox_conference_offline_peer_get_name(const Tox *tox, uint32_t conference_number, uint32_t offline_peer_number,
2434
        uint8_t *name,
2435
        Tox_Err_Conference_Peer_Query *error)
2436
0
{
2437
0
    assert(tox != nullptr);
2438
0
    tox_lock(tox);
2439
0
    const int ret = group_peername(tox->m->conferences_object, conference_number, offline_peer_number, name, true);
2440
0
    tox_unlock(tox);
2441
2442
0
    switch (ret) {
2443
0
        case -1: {
2444
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2445
0
            return false;
2446
0
        }
2447
2448
0
        case -2: {
2449
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2450
0
            return false;
2451
0
        }
2452
0
    }
2453
2454
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2455
0
    return true;
2456
0
}
2457
2458
bool tox_conference_offline_peer_get_public_key(const Tox *tox, uint32_t conference_number,
2459
        uint32_t offline_peer_number,
2460
        uint8_t public_key[TOX_PUBLIC_KEY_SIZE], Tox_Err_Conference_Peer_Query *error)
2461
0
{
2462
0
    assert(tox != nullptr);
2463
0
    tox_lock(tox);
2464
0
    const int ret = group_peer_pubkey(tox->m->conferences_object, conference_number, offline_peer_number, public_key, true);
2465
0
    tox_unlock(tox);
2466
2467
0
    switch (ret) {
2468
0
        case -1: {
2469
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2470
0
            return false;
2471
0
        }
2472
2473
0
        case -2: {
2474
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2475
0
            return false;
2476
0
        }
2477
0
    }
2478
2479
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2480
0
    return true;
2481
0
}
2482
2483
uint64_t tox_conference_offline_peer_get_last_active(const Tox *tox, uint32_t conference_number,
2484
        uint32_t offline_peer_number,
2485
        Tox_Err_Conference_Peer_Query *error)
2486
0
{
2487
0
    assert(tox != nullptr);
2488
0
    uint64_t last_active = UINT64_MAX;
2489
0
    tox_lock(tox);
2490
0
    const int ret = group_frozen_last_active(tox->m->conferences_object, conference_number, offline_peer_number,
2491
0
                    &last_active);
2492
0
    tox_unlock(tox);
2493
2494
0
    switch (ret) {
2495
0
        case -1: {
2496
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_CONFERENCE_NOT_FOUND);
2497
0
            return UINT64_MAX;
2498
0
        }
2499
2500
0
        case -2: {
2501
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_PEER_NOT_FOUND);
2502
0
            return UINT64_MAX;
2503
0
        }
2504
0
    }
2505
2506
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_PEER_QUERY_OK);
2507
0
    return last_active;
2508
0
}
2509
2510
bool tox_conference_set_max_offline(Tox *tox, uint32_t conference_number,
2511
                                    uint32_t max_offline,
2512
                                    Tox_Err_Conference_Set_Max_Offline *error)
2513
0
{
2514
0
    assert(tox != nullptr);
2515
0
    tox_lock(tox);
2516
0
    const int ret = group_set_max_frozen(tox->m->conferences_object, conference_number, max_offline);
2517
0
    tox_unlock(tox);
2518
2519
0
    if (ret == -1) {
2520
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SET_MAX_OFFLINE_CONFERENCE_NOT_FOUND);
2521
0
        return false;
2522
0
    }
2523
2524
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SET_MAX_OFFLINE_OK);
2525
0
    return true;
2526
0
}
2527
2528
bool tox_conference_invite(Tox *tox, uint32_t friend_number, uint32_t conference_number,
2529
                           Tox_Err_Conference_Invite *error)
2530
0
{
2531
0
    assert(tox != nullptr);
2532
0
    tox_lock(tox);
2533
0
    const int ret = invite_friend(tox->m->conferences_object, friend_number, conference_number);
2534
0
    tox_unlock(tox);
2535
2536
0
    switch (ret) {
2537
0
        case -1: {
2538
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_INVITE_CONFERENCE_NOT_FOUND);
2539
0
            return false;
2540
0
        }
2541
2542
0
        case -2: {
2543
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_INVITE_FAIL_SEND);
2544
0
            return false;
2545
0
        }
2546
2547
0
        case -3: {
2548
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_INVITE_NO_CONNECTION);
2549
0
            return false;
2550
0
        }
2551
0
    }
2552
2553
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_INVITE_OK);
2554
0
    return true;
2555
0
}
2556
2557
uint32_t tox_conference_join(Tox *tox, uint32_t friend_number, const uint8_t *cookie, size_t length,
2558
                             Tox_Err_Conference_Join *error)
2559
0
{
2560
0
    assert(tox != nullptr);
2561
2562
0
    if (cookie == nullptr) {
2563
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_NULL);
2564
0
        return UINT32_MAX;
2565
0
    }
2566
2567
0
    tox_lock(tox);
2568
0
    const int ret = join_groupchat(tox->m->conferences_object, friend_number, GROUPCHAT_TYPE_TEXT, cookie, length);
2569
0
    tox_unlock(tox);
2570
2571
0
    switch (ret) {
2572
0
        case -1: {
2573
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_INVALID_LENGTH);
2574
0
            return UINT32_MAX;
2575
0
        }
2576
2577
0
        case -2: {
2578
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_WRONG_TYPE);
2579
0
            return UINT32_MAX;
2580
0
        }
2581
2582
0
        case -3: {
2583
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_FRIEND_NOT_FOUND);
2584
0
            return UINT32_MAX;
2585
0
        }
2586
2587
0
        case -4: {
2588
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_DUPLICATE);
2589
0
            return UINT32_MAX;
2590
0
        }
2591
2592
0
        case -5: {
2593
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_INIT_FAIL);
2594
0
            return UINT32_MAX;
2595
0
        }
2596
2597
0
        case -6: {
2598
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_FAIL_SEND);
2599
0
            return UINT32_MAX;
2600
0
        }
2601
0
    }
2602
2603
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_JOIN_OK);
2604
0
    return ret;
2605
0
}
2606
2607
bool tox_conference_send_message(Tox *tox, uint32_t conference_number, Tox_Message_Type type, const uint8_t *message,
2608
                                 size_t length, Tox_Err_Conference_Send_Message *error)
2609
0
{
2610
0
    assert(tox != nullptr);
2611
0
    tox_lock(tox);
2612
0
    int ret = 0;
2613
2614
0
    if (type == TOX_MESSAGE_TYPE_NORMAL) {
2615
0
        ret = group_message_send(tox->m->conferences_object, conference_number, message, length);
2616
0
    } else {
2617
0
        ret = group_action_send(tox->m->conferences_object, conference_number, message, length);
2618
0
    }
2619
2620
0
    tox_unlock(tox);
2621
2622
0
    switch (ret) {
2623
0
        case -1: {
2624
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SEND_MESSAGE_CONFERENCE_NOT_FOUND);
2625
0
            return false;
2626
0
        }
2627
2628
0
        case -2: {
2629
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SEND_MESSAGE_TOO_LONG);
2630
0
            return false;
2631
0
        }
2632
2633
0
        case -3: {
2634
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SEND_MESSAGE_NO_CONNECTION);
2635
0
            return false;
2636
0
        }
2637
2638
0
        case -4: {
2639
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SEND_MESSAGE_FAIL_SEND);
2640
0
            return false;
2641
0
        }
2642
0
    }
2643
2644
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_SEND_MESSAGE_OK);
2645
0
    return true;
2646
0
}
2647
2648
size_t tox_conference_get_title_size(const Tox *tox, uint32_t conference_number, Tox_Err_Conference_Title *error)
2649
0
{
2650
0
    assert(tox != nullptr);
2651
0
    tox_lock(tox);
2652
0
    const int ret = group_title_get_size(tox->m->conferences_object, conference_number);
2653
0
    tox_unlock(tox);
2654
2655
0
    switch (ret) {
2656
0
        case -1: {
2657
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_CONFERENCE_NOT_FOUND);
2658
0
            return -1;
2659
0
        }
2660
2661
0
        case -2: {
2662
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_INVALID_LENGTH);
2663
0
            return -1;
2664
0
        }
2665
0
    }
2666
2667
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_OK);
2668
0
    return ret;
2669
0
}
2670
2671
bool tox_conference_get_title(const Tox *tox, uint32_t conference_number, uint8_t *title,
2672
                              Tox_Err_Conference_Title *error)
2673
0
{
2674
0
    assert(tox != nullptr);
2675
0
    tox_lock(tox);
2676
0
    const int ret = group_title_get(tox->m->conferences_object, conference_number, title);
2677
0
    tox_unlock(tox);
2678
2679
0
    switch (ret) {
2680
0
        case -1: {
2681
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_CONFERENCE_NOT_FOUND);
2682
0
            return false;
2683
0
        }
2684
2685
0
        case -2: {
2686
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_INVALID_LENGTH);
2687
0
            return false;
2688
0
        }
2689
0
    }
2690
2691
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_OK);
2692
0
    return true;
2693
0
}
2694
2695
bool tox_conference_set_title(Tox *tox, uint32_t conference_number, const uint8_t *title, size_t length,
2696
                              Tox_Err_Conference_Title *error)
2697
0
{
2698
0
    assert(tox != nullptr);
2699
0
    tox_lock(tox);
2700
0
    const int ret = group_title_send(tox->m->conferences_object, conference_number, title, length);
2701
0
    tox_unlock(tox);
2702
2703
0
    switch (ret) {
2704
0
        case -1: {
2705
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_CONFERENCE_NOT_FOUND);
2706
0
            return false;
2707
0
        }
2708
2709
0
        case -2: {
2710
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_INVALID_LENGTH);
2711
0
            return false;
2712
0
        }
2713
2714
0
        case -3: {
2715
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_FAIL_SEND);
2716
0
            return false;
2717
0
        }
2718
0
    }
2719
2720
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_TITLE_OK);
2721
0
    return true;
2722
0
}
2723
2724
size_t tox_conference_get_chatlist_size(const Tox *tox)
2725
0
{
2726
0
    assert(tox != nullptr);
2727
0
    tox_lock(tox);
2728
0
    const size_t ret = count_chatlist(tox->m->conferences_object);
2729
0
    tox_unlock(tox);
2730
0
    return ret;
2731
0
}
2732
2733
void tox_conference_get_chatlist(const Tox *tox, uint32_t *chatlist)
2734
0
{
2735
0
    assert(tox != nullptr);
2736
0
    tox_lock(tox);
2737
0
    const size_t list_size = count_chatlist(tox->m->conferences_object);
2738
0
    copy_chatlist(tox->m->conferences_object, chatlist, list_size);
2739
0
    tox_unlock(tox);
2740
0
}
2741
2742
Tox_Conference_Type tox_conference_get_type(const Tox *tox, uint32_t conference_number,
2743
        Tox_Err_Conference_Get_Type *error)
2744
0
{
2745
0
    assert(tox != nullptr);
2746
0
    tox_lock(tox);
2747
0
    const int ret = group_get_type(tox->m->conferences_object, conference_number);
2748
0
    tox_unlock(tox);
2749
2750
0
    if (ret == -1) {
2751
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_GET_TYPE_CONFERENCE_NOT_FOUND);
2752
0
        return (Tox_Conference_Type)ret;
2753
0
    }
2754
2755
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_GET_TYPE_OK);
2756
0
    return (Tox_Conference_Type)ret;
2757
0
}
2758
2759
bool tox_conference_get_id(const Tox *tox, uint32_t conference_number, uint8_t id[TOX_CONFERENCE_ID_SIZE])
2760
0
{
2761
0
    assert(tox != nullptr);
2762
0
    tox_lock(tox);
2763
0
    const bool ret = conference_get_id(tox->m->conferences_object, conference_number, id);
2764
0
    tox_unlock(tox);
2765
0
    return ret;
2766
0
}
2767
2768
// TODO(iphydf): Delete in 0.3.0.
2769
bool tox_conference_get_uid(const Tox *tox, uint32_t conference_number, uint8_t uid[TOX_CONFERENCE_UID_SIZE])
2770
0
{
2771
0
    assert(tox != nullptr);
2772
0
    return tox_conference_get_id(tox, conference_number, uid);
2773
0
}
2774
2775
uint32_t tox_conference_by_id(const Tox *tox, const uint8_t id[TOX_CONFERENCE_ID_SIZE], Tox_Err_Conference_By_Id *error)
2776
0
{
2777
0
    assert(tox != nullptr);
2778
2779
0
    if (id == nullptr) {
2780
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_ID_NULL);
2781
0
        return UINT32_MAX;
2782
0
    }
2783
2784
0
    tox_lock(tox);
2785
0
    const int32_t ret = conference_by_id(tox->m->conferences_object, id);
2786
0
    tox_unlock(tox);
2787
2788
0
    if (ret == -1) {
2789
0
        SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_ID_NOT_FOUND);
2790
0
        return UINT32_MAX;
2791
0
    }
2792
2793
0
    SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_ID_OK);
2794
0
    assert(ret >= 0);
2795
0
    return (uint32_t)ret;
2796
0
}
2797
2798
// TODO(iphydf): Delete in 0.3.0.
2799
uint32_t tox_conference_by_uid(const Tox *tox, const uint8_t uid[TOX_CONFERENCE_UID_SIZE], Tox_Err_Conference_By_Uid *error)
2800
0
{
2801
0
    assert(tox != nullptr);
2802
0
    Tox_Err_Conference_By_Id id_error;
2803
0
    const uint32_t res = tox_conference_by_id(tox, uid, &id_error);
2804
2805
0
    switch (id_error) {
2806
0
        case TOX_ERR_CONFERENCE_BY_ID_OK: {
2807
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_UID_OK);
2808
0
            break;
2809
0
        }
2810
2811
0
        case TOX_ERR_CONFERENCE_BY_ID_NULL: {
2812
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_UID_NULL);
2813
0
            break;
2814
0
        }
2815
2816
0
        case TOX_ERR_CONFERENCE_BY_ID_NOT_FOUND: {
2817
0
            SET_ERROR_PARAMETER(error, TOX_ERR_CONFERENCE_BY_UID_NOT_FOUND);
2818
0
            break;
2819
0
        }
2820
0
    }
2821
2822
0
    return res;
2823
0
}
2824
2825
nullable(2)
2826
static void set_custom_packet_error(int ret, Tox_Err_Friend_Custom_Packet *error)
2827
0
{
2828
0
    switch (ret) {
2829
0
        case 0: {
2830
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_OK);
2831
0
            break;
2832
0
        }
2833
2834
0
        case -1: {
2835
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_FRIEND_NOT_FOUND);
2836
0
            break;
2837
0
        }
2838
2839
0
        case -2: {
2840
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_TOO_LONG);
2841
0
            break;
2842
0
        }
2843
2844
0
        case -3: {
2845
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_INVALID);
2846
0
            break;
2847
0
        }
2848
2849
0
        case -4: {
2850
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_FRIEND_NOT_CONNECTED);
2851
0
            break;
2852
0
        }
2853
2854
0
        case -5: {
2855
0
            SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_SENDQ);
2856
0
            break;
2857
0
        }
2858
0
    }
2859
0
}
2860
2861
// cppcheck-suppress constParameterPointer
2862
bool tox_friend_send_lossy_packet(Tox *tox, uint32_t friend_number, const uint8_t *data, size_t length,
2863
                                  Tox_Err_Friend_Custom_Packet *error)
2864
0
{
2865
0
    assert(tox != nullptr);
2866
2867
0
    if (data == nullptr) {
2868
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_NULL);
2869
0
        return false;
2870
0
    }
2871
2872
0
    if (length == 0) {
2873
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_EMPTY);
2874
0
        return false;
2875
0
    }
2876
2877
0
    if (data[0] < PACKET_ID_RANGE_LOSSY_START || data[0] > PACKET_ID_RANGE_LOSSY_END) {
2878
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_INVALID);
2879
0
        return false;
2880
0
    }
2881
2882
0
    tox_lock(tox);
2883
0
    const int ret = m_send_custom_lossy_packet(tox->m, friend_number, data, length);
2884
0
    tox_unlock(tox);
2885
2886
0
    set_custom_packet_error(ret, error);
2887
2888
0
    return ret == 0;
2889
0
}
2890
2891
void tox_callback_friend_lossy_packet(Tox *tox, tox_friend_lossy_packet_cb *callback)
2892
0
{
2893
0
    assert(tox != nullptr);
2894
2895
    /* start at PACKET_ID_RANGE_LOSSY_CUSTOM_START so ToxAV Packets are excluded */
2896
0
    for (uint8_t i = PACKET_ID_RANGE_LOSSY_CUSTOM_START; i <= PACKET_ID_RANGE_LOSSY_END; ++i) {
2897
0
        tox->friend_lossy_packet_callback_per_pktid[i] = callback;
2898
0
    }
2899
0
}
2900
2901
// cppcheck-suppress constParameterPointer
2902
bool tox_friend_send_lossless_packet(Tox *tox, uint32_t friend_number, const uint8_t *data, size_t length,
2903
                                     Tox_Err_Friend_Custom_Packet *error)
2904
0
{
2905
0
    assert(tox != nullptr);
2906
2907
0
    if (data == nullptr) {
2908
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_NULL);
2909
0
        return false;
2910
0
    }
2911
2912
0
    if (length == 0) {
2913
0
        SET_ERROR_PARAMETER(error, TOX_ERR_FRIEND_CUSTOM_PACKET_EMPTY);
2914
0
        return false;
2915
0
    }
2916
2917
0
    tox_lock(tox);
2918
0
    const int ret = send_custom_lossless_packet(tox->m, friend_number, data, length);
2919
0
    tox_unlock(tox);
2920
2921
0
    set_custom_packet_error(ret, error);
2922
2923
0
    return ret == 0;
2924
0
}
2925
2926
void tox_callback_friend_lossless_packet(Tox *tox, tox_friend_lossless_packet_cb *callback)
2927
0
{
2928
0
    assert(tox != nullptr);
2929
2930
0
    for (uint8_t i = PACKET_ID_RANGE_LOSSLESS_CUSTOM_START; i <= PACKET_ID_RANGE_LOSSLESS_CUSTOM_END; ++i) {
2931
0
        tox->friend_lossless_packet_callback_per_pktid[i] = callback;
2932
0
    }
2933
0
}
2934
2935
void tox_self_get_dht_id(const Tox *tox, uint8_t dht_id[TOX_PUBLIC_KEY_SIZE])
2936
0
{
2937
0
    assert(tox != nullptr);
2938
2939
0
    if (dht_id != nullptr) {
2940
0
        tox_lock(tox);
2941
0
        memcpy(dht_id, dht_get_self_public_key(tox->m->dht), CRYPTO_PUBLIC_KEY_SIZE);
2942
0
        tox_unlock(tox);
2943
0
    }
2944
0
}
2945
2946
uint16_t tox_self_get_udp_port(const Tox *tox, Tox_Err_Get_Port *error)
2947
0
{
2948
0
    assert(tox != nullptr);
2949
0
    tox_lock(tox);
2950
0
    const uint16_t port = tox->m == nullptr || tox->m->net == nullptr ? 0 : net_htons(net_port(tox->m->net));
2951
0
    tox_unlock(tox);
2952
2953
0
    if (port == 0) {
2954
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GET_PORT_NOT_BOUND);
2955
0
        return 0;
2956
0
    }
2957
2958
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GET_PORT_OK);
2959
0
    return port;
2960
0
}
2961
2962
uint16_t tox_self_get_tcp_port(const Tox *tox, Tox_Err_Get_Port *error)
2963
0
{
2964
0
    assert(tox != nullptr);
2965
0
    tox_lock(tox);
2966
2967
0
    if (tox->m != nullptr && tox->m->tcp_server != nullptr) {
2968
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GET_PORT_OK);
2969
0
        const uint16_t ret = tox->m->options.tcp_server_port;
2970
0
        tox_unlock(tox);
2971
0
        return ret;
2972
0
    }
2973
2974
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GET_PORT_NOT_BOUND);
2975
0
    tox_unlock(tox);
2976
0
    return 0;
2977
0
}
2978
2979
/* GROUPCHAT FUNCTIONS */
2980
2981
void tox_callback_group_invite(Tox *tox, tox_group_invite_cb *callback)
2982
0
{
2983
0
    assert(tox != nullptr);
2984
0
    tox->group_invite_callback = callback;
2985
0
}
2986
2987
void tox_callback_group_message(Tox *tox, tox_group_message_cb *callback)
2988
0
{
2989
0
    assert(tox != nullptr);
2990
0
    tox->group_message_callback = callback;
2991
0
}
2992
2993
void tox_callback_group_private_message(Tox *tox, tox_group_private_message_cb *callback)
2994
0
{
2995
0
    assert(tox != nullptr);
2996
0
    tox->group_private_message_callback = callback;
2997
0
}
2998
2999
void tox_callback_group_custom_packet(Tox *tox, tox_group_custom_packet_cb *callback)
3000
0
{
3001
0
    assert(tox != nullptr);
3002
0
    tox->group_custom_packet_callback = callback;
3003
0
}
3004
3005
void tox_callback_group_custom_private_packet(Tox *tox, tox_group_custom_private_packet_cb *callback)
3006
0
{
3007
0
    assert(tox != nullptr);
3008
0
    tox->group_custom_private_packet_callback = callback;
3009
0
}
3010
3011
void tox_callback_group_moderation(Tox *tox, tox_group_moderation_cb *callback)
3012
0
{
3013
0
    assert(tox != nullptr);
3014
0
    tox->group_moderation_callback = callback;
3015
0
}
3016
3017
void tox_callback_group_peer_name(Tox *tox, tox_group_peer_name_cb *callback)
3018
0
{
3019
0
    assert(tox != nullptr);
3020
0
    tox->group_peer_name_callback = callback;
3021
0
}
3022
3023
void tox_callback_group_peer_status(Tox *tox, tox_group_peer_status_cb *callback)
3024
0
{
3025
0
    assert(tox != nullptr);
3026
0
    tox->group_peer_status_callback = callback;
3027
0
}
3028
3029
void tox_callback_group_topic(Tox *tox, tox_group_topic_cb *callback)
3030
0
{
3031
0
    assert(tox != nullptr);
3032
0
    tox->group_topic_callback = callback;
3033
0
}
3034
3035
void tox_callback_group_privacy_state(Tox *tox, tox_group_privacy_state_cb *callback)
3036
0
{
3037
0
    assert(tox != nullptr);
3038
0
    tox->group_privacy_state_callback = callback;
3039
0
}
3040
3041
void tox_callback_group_topic_lock(Tox *tox, tox_group_topic_lock_cb *callback)
3042
0
{
3043
0
    assert(tox != nullptr);
3044
0
    tox->group_topic_lock_callback = callback;
3045
0
}
3046
3047
void tox_callback_group_voice_state(Tox *tox, tox_group_voice_state_cb *callback)
3048
0
{
3049
0
    assert(tox != nullptr);
3050
0
    tox->group_voice_state_callback = callback;
3051
0
}
3052
3053
void tox_callback_group_peer_limit(Tox *tox, tox_group_peer_limit_cb *callback)
3054
0
{
3055
0
    assert(tox != nullptr);
3056
0
    tox->group_peer_limit_callback = callback;
3057
0
}
3058
3059
void tox_callback_group_password(Tox *tox, tox_group_password_cb *callback)
3060
0
{
3061
0
    assert(tox != nullptr);
3062
0
    tox->group_password_callback = callback;
3063
0
}
3064
3065
void tox_callback_group_peer_join(Tox *tox, tox_group_peer_join_cb *callback)
3066
0
{
3067
0
    assert(tox != nullptr);
3068
0
    tox->group_peer_join_callback = callback;
3069
0
}
3070
3071
void tox_callback_group_peer_exit(Tox *tox, tox_group_peer_exit_cb *callback)
3072
0
{
3073
0
    assert(tox != nullptr);
3074
0
    tox->group_peer_exit_callback = callback;
3075
0
}
3076
3077
void tox_callback_group_self_join(Tox *tox, tox_group_self_join_cb *callback)
3078
0
{
3079
0
    assert(tox != nullptr);
3080
0
    tox->group_self_join_callback = callback;
3081
0
}
3082
3083
void tox_callback_group_join_fail(Tox *tox, tox_group_join_fail_cb *callback)
3084
0
{
3085
0
    assert(tox != nullptr);
3086
0
    tox->group_join_fail_callback = callback;
3087
0
}
3088
3089
uint32_t tox_group_new(Tox *tox, Tox_Group_Privacy_State privacy_state, const uint8_t *group_name,
3090
                       size_t group_name_length, const uint8_t *name, size_t name_length, Tox_Err_Group_New *error)
3091
0
{
3092
0
    assert(tox != nullptr);
3093
3094
0
    tox_lock(tox);
3095
0
    const int ret = gc_group_add(tox->m->group_handler, (Group_Privacy_State) privacy_state,
3096
0
                                 group_name, group_name_length, name, name_length);
3097
0
    tox_unlock(tox);
3098
3099
0
    if (ret >= 0) {
3100
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_OK);
3101
0
        return ret;
3102
0
    }
3103
3104
0
    switch (ret) {
3105
0
        case -1: {
3106
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_TOO_LONG);
3107
0
            return UINT32_MAX;
3108
0
        }
3109
3110
0
        case -2: {
3111
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_EMPTY);
3112
0
            return UINT32_MAX;
3113
0
        }
3114
3115
0
        case -3: {
3116
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_INIT);
3117
0
            return UINT32_MAX;
3118
0
        }
3119
3120
0
        case -4: {
3121
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_STATE);
3122
0
            return UINT32_MAX;
3123
0
        }
3124
3125
0
        case -5: {
3126
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_NEW_ANNOUNCE);
3127
0
            return UINT32_MAX;
3128
0
        }
3129
0
    }
3130
3131
    /* can't happen */
3132
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3133
3134
0
    return UINT32_MAX;
3135
0
}
3136
3137
uint32_t tox_group_join(Tox *tox, const uint8_t chat_id[TOX_GROUP_CHAT_ID_SIZE], const uint8_t *name, size_t name_length,
3138
                        const uint8_t *password, size_t password_length, Tox_Err_Group_Join *error)
3139
0
{
3140
0
    assert(tox != nullptr);
3141
3142
0
    tox_lock(tox);
3143
0
    const int ret = gc_group_join(tox->m->group_handler, chat_id, name, name_length, password, password_length);
3144
0
    tox_unlock(tox);
3145
3146
0
    if (ret >= 0) {
3147
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_OK);
3148
0
        return ret;
3149
0
    }
3150
3151
0
    switch (ret) {
3152
0
        case -1: {
3153
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_INIT);
3154
0
            return UINT32_MAX;
3155
0
        }
3156
3157
0
        case -2: {
3158
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_BAD_CHAT_ID);
3159
0
            return UINT32_MAX;
3160
0
        }
3161
3162
0
        case -3: {
3163
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_TOO_LONG);
3164
0
            return UINT32_MAX;
3165
0
        }
3166
3167
0
        case -4: {
3168
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_EMPTY);
3169
0
            return UINT32_MAX;
3170
0
        }
3171
3172
0
        case -5: {
3173
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_PASSWORD);
3174
0
            return UINT32_MAX;
3175
0
        }
3176
3177
0
        case -6: {
3178
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_JOIN_CORE);
3179
0
            return UINT32_MAX;
3180
0
        }
3181
0
    }
3182
3183
    /* can't happen */
3184
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3185
3186
0
    return UINT32_MAX;
3187
0
}
3188
3189
bool tox_group_is_connected(const Tox *tox, uint32_t group_number, Tox_Err_Group_Is_Connected *error)
3190
0
{
3191
0
    assert(tox != nullptr);
3192
3193
0
    tox_lock(tox);
3194
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3195
3196
0
    if (chat == nullptr) {
3197
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_IS_CONNECTED_GROUP_NOT_FOUND);
3198
0
        tox_unlock(tox);
3199
0
        return false;
3200
0
    }
3201
3202
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_IS_CONNECTED_OK);
3203
3204
0
    const bool ret = chat->connection_state == CS_CONNECTED || chat->connection_state == CS_CONNECTING;
3205
0
    tox_unlock(tox);
3206
3207
0
    return ret;
3208
0
}
3209
3210
bool tox_group_disconnect(const Tox *tox, uint32_t group_number, Tox_Err_Group_Disconnect *error)
3211
0
{
3212
0
    assert(tox != nullptr);
3213
3214
0
    tox_lock(tox);
3215
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3216
3217
0
    if (chat == nullptr) {
3218
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_DISCONNECT_GROUP_NOT_FOUND);
3219
0
        tox_unlock(tox);
3220
0
        return false;
3221
0
    }
3222
3223
0
    if (chat->connection_state == CS_DISCONNECTED) {
3224
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_DISCONNECT_ALREADY_DISCONNECTED);
3225
0
        tox_unlock(tox);
3226
0
        return false;
3227
0
    }
3228
3229
0
    const bool ret = gc_disconnect_from_group(tox->m->group_handler, chat);
3230
3231
0
    tox_unlock(tox);
3232
3233
0
    if (!ret) {
3234
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_DISCONNECT_GROUP_NOT_FOUND);
3235
0
        return false;
3236
0
    }
3237
3238
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_DISCONNECT_OK);
3239
3240
0
    return true;
3241
0
}
3242
3243
bool tox_group_reconnect(Tox *tox, uint32_t group_number, Tox_Err_Group_Reconnect *error)
3244
0
{
3245
0
    assert(tox != nullptr);
3246
3247
0
    tox_lock(tox);
3248
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3249
3250
0
    if (chat == nullptr) {
3251
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_RECONNECT_GROUP_NOT_FOUND);
3252
0
        tox_unlock(tox);
3253
0
        return false;
3254
0
    }
3255
3256
0
    const int ret = gc_rejoin_group(tox->m->group_handler, chat, nullptr, 0);
3257
0
    tox_unlock(tox);
3258
3259
0
    switch (ret) {
3260
0
        case 0: {
3261
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_RECONNECT_OK);
3262
0
            return true;
3263
0
        }
3264
3265
0
        case -1: {
3266
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_RECONNECT_GROUP_NOT_FOUND);
3267
0
            return false;
3268
0
        }
3269
3270
0
        case -2: {
3271
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_RECONNECT_CORE);
3272
0
            return false;
3273
0
        }
3274
0
    }
3275
3276
    /* can't happen */
3277
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3278
3279
0
    return false;
3280
0
}
3281
3282
bool tox_group_leave(Tox *tox, uint32_t group_number, const uint8_t *part_message, size_t length,
3283
                     Tox_Err_Group_Leave *error)
3284
0
{
3285
0
    assert(tox != nullptr);
3286
3287
0
    tox_lock(tox);
3288
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3289
3290
0
    if (chat == nullptr) {
3291
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_LEAVE_GROUP_NOT_FOUND);
3292
0
        tox_unlock(tox);
3293
0
        return false;
3294
0
    }
3295
3296
0
    const int ret = gc_group_exit(tox->m->group_handler, chat, part_message, length);
3297
0
    tox_unlock(tox);
3298
3299
0
    switch (ret) {
3300
0
        case 0: {
3301
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_LEAVE_OK);
3302
0
            return true;
3303
0
        }
3304
3305
0
        case -1: {
3306
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_LEAVE_TOO_LONG);
3307
0
            return false;
3308
0
        }
3309
3310
0
        case -2: {
3311
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_LEAVE_FAIL_SEND);
3312
0
            return true;   /* the group was still successfully deleted */
3313
0
        }
3314
0
    }
3315
3316
    /* can't happen */
3317
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3318
3319
0
    return false;
3320
0
}
3321
3322
bool tox_group_self_set_name(Tox *tox, uint32_t group_number, const uint8_t *name, size_t length,
3323
                             Tox_Err_Group_Self_Name_Set *error)
3324
0
{
3325
0
    assert(tox != nullptr);
3326
3327
0
    tox_lock(tox);
3328
0
    const int ret = gc_set_self_nick(tox->m, group_number, name, length);
3329
0
    tox_unlock(tox);
3330
3331
0
    switch (ret) {
3332
0
        case 0: {
3333
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_NAME_SET_OK);
3334
0
            return true;
3335
0
        }
3336
3337
0
        case -1: {
3338
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_NAME_SET_GROUP_NOT_FOUND);
3339
0
            return false;
3340
0
        }
3341
3342
0
        case -2: {
3343
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_NAME_SET_TOO_LONG);
3344
0
            return false;
3345
0
        }
3346
3347
0
        case -3: {
3348
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_NAME_SET_INVALID);
3349
0
            return false;
3350
0
        }
3351
3352
0
        case -4: {
3353
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_NAME_SET_FAIL_SEND);
3354
0
            return false;
3355
0
        }
3356
0
    }
3357
3358
    /* can't happen */
3359
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3360
3361
0
    return false;
3362
0
}
3363
3364
size_t tox_group_self_get_name_size(const Tox *tox, uint32_t group_number, Tox_Err_Group_Self_Query *error)
3365
0
{
3366
0
    assert(tox != nullptr);
3367
3368
0
    tox_lock(tox);
3369
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3370
3371
0
    if (chat == nullptr) {
3372
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3373
0
        tox_unlock(tox);
3374
0
        return -1;
3375
0
    }
3376
3377
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3378
3379
0
    const size_t ret = gc_get_self_nick_size(chat);
3380
0
    tox_unlock(tox);
3381
3382
0
    return ret;
3383
0
}
3384
3385
bool tox_group_self_get_name(const Tox *tox, uint32_t group_number, uint8_t *name, Tox_Err_Group_Self_Query *error)
3386
0
{
3387
0
    assert(tox != nullptr);
3388
3389
0
    tox_lock(tox);
3390
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3391
3392
0
    if (chat == nullptr) {
3393
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3394
0
        tox_unlock(tox);
3395
0
        return false;
3396
0
    }
3397
3398
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3399
3400
0
    gc_get_self_nick(chat, name);
3401
0
    tox_unlock(tox);
3402
3403
0
    return true;
3404
0
}
3405
3406
bool tox_group_self_set_status(Tox *tox, uint32_t group_number, Tox_User_Status status,
3407
                               Tox_Err_Group_Self_Status_Set *error)
3408
0
{
3409
0
    assert(tox != nullptr);
3410
3411
0
    tox_lock(tox);
3412
0
    const int ret = gc_set_self_status(tox->m, group_number, (Group_Peer_Status) status);
3413
0
    tox_unlock(tox);
3414
3415
0
    switch (ret) {
3416
0
        case 0: {
3417
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_STATUS_SET_OK);
3418
0
            return true;
3419
0
        }
3420
3421
0
        case -1: {
3422
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_STATUS_SET_GROUP_NOT_FOUND);
3423
0
            return false;
3424
0
        }
3425
3426
0
        case -2: {
3427
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_STATUS_SET_FAIL_SEND);
3428
0
            return false;
3429
0
        }
3430
0
    }
3431
3432
    /* can't happen */
3433
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3434
3435
0
    return false;
3436
0
}
3437
3438
Tox_User_Status tox_group_self_get_status(const Tox *tox, uint32_t group_number, Tox_Err_Group_Self_Query *error)
3439
0
{
3440
0
    assert(tox != nullptr);
3441
3442
0
    tox_lock(tox);
3443
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3444
3445
0
    if (chat == nullptr) {
3446
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3447
0
        tox_unlock(tox);
3448
0
        return (Tox_User_Status) - 1;
3449
0
    }
3450
3451
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3452
3453
0
    const uint8_t status = gc_get_self_status(chat);
3454
0
    tox_unlock(tox);
3455
3456
0
    return (Tox_User_Status)status;
3457
0
}
3458
3459
Tox_Group_Role tox_group_self_get_role(const Tox *tox, uint32_t group_number, Tox_Err_Group_Self_Query *error)
3460
0
{
3461
0
    assert(tox != nullptr);
3462
3463
0
    tox_lock(tox);
3464
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3465
3466
0
    if (chat == nullptr) {
3467
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3468
0
        tox_unlock(tox);
3469
0
        return (Tox_Group_Role) - 1;
3470
0
    }
3471
3472
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3473
3474
0
    const Group_Role role = gc_get_self_role(chat);
3475
0
    tox_unlock(tox);
3476
3477
0
    return (Tox_Group_Role)role;
3478
0
}
3479
3480
uint32_t tox_group_self_get_peer_id(const Tox *tox, uint32_t group_number, Tox_Err_Group_Self_Query *error)
3481
0
{
3482
0
    assert(tox != nullptr);
3483
3484
0
    tox_lock(tox);
3485
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3486
3487
0
    if (chat == nullptr) {
3488
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3489
0
        tox_unlock(tox);
3490
0
        return -1;
3491
0
    }
3492
3493
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3494
3495
0
    const GC_Peer_Id ret = gc_get_self_peer_id(chat);
3496
0
    tox_unlock(tox);
3497
3498
0
    return gc_peer_id_to_int(ret);
3499
0
}
3500
3501
bool tox_group_self_get_public_key(const Tox *tox, uint32_t group_number, uint8_t public_key[TOX_PUBLIC_KEY_SIZE],
3502
                                   Tox_Err_Group_Self_Query *error)
3503
0
{
3504
0
    assert(tox != nullptr);
3505
3506
0
    tox_lock(tox);
3507
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3508
3509
0
    if (chat == nullptr) {
3510
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_GROUP_NOT_FOUND);
3511
0
        tox_unlock(tox);
3512
0
        return false;
3513
0
    }
3514
3515
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SELF_QUERY_OK);
3516
3517
0
    gc_get_self_public_key(chat, public_key);
3518
0
    tox_unlock(tox);
3519
3520
0
    return true;
3521
0
}
3522
3523
size_t tox_group_peer_get_name_size(const Tox *tox, uint32_t group_number, uint32_t peer_id,
3524
                                    Tox_Err_Group_Peer_Query *error)
3525
0
{
3526
0
    assert(tox != nullptr);
3527
3528
0
    tox_lock(tox);
3529
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3530
3531
0
    if (chat == nullptr) {
3532
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3533
0
        tox_unlock(tox);
3534
0
        return -1;
3535
0
    }
3536
3537
0
    const int ret = gc_get_peer_nick_size(chat, gc_peer_id_from_int(peer_id));
3538
0
    tox_unlock(tox);
3539
3540
0
    if (ret == -1) {
3541
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3542
0
        return -1;
3543
0
    } else {
3544
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3545
0
        return ret;
3546
0
    }
3547
0
}
3548
3549
bool tox_group_peer_get_name(const Tox *tox, uint32_t group_number, uint32_t peer_id, uint8_t *name,
3550
                             Tox_Err_Group_Peer_Query *error)
3551
0
{
3552
0
    assert(tox != nullptr);
3553
3554
0
    tox_lock(tox);
3555
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3556
3557
0
    if (chat == nullptr) {
3558
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3559
0
        tox_unlock(tox);
3560
0
        return false;
3561
0
    }
3562
3563
0
    const bool ret = gc_get_peer_nick(chat, gc_peer_id_from_int(peer_id), name);
3564
0
    tox_unlock(tox);
3565
3566
0
    if (!ret) {
3567
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3568
0
        return false;
3569
0
    }
3570
3571
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3572
0
    return true;
3573
0
}
3574
3575
Tox_User_Status tox_group_peer_get_status(const Tox *tox, uint32_t group_number, uint32_t peer_id,
3576
        Tox_Err_Group_Peer_Query *error)
3577
0
{
3578
0
    assert(tox != nullptr);
3579
3580
0
    tox_lock(tox);
3581
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3582
3583
0
    if (chat == nullptr) {
3584
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3585
0
        tox_unlock(tox);
3586
0
        return (Tox_User_Status) - 1;
3587
0
    }
3588
3589
0
    const uint8_t ret = gc_get_status(chat, gc_peer_id_from_int(peer_id));
3590
0
    tox_unlock(tox);
3591
3592
0
    if (ret == UINT8_MAX) {
3593
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3594
0
        return (Tox_User_Status) - 1;
3595
0
    }
3596
3597
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3598
0
    return (Tox_User_Status)ret;
3599
0
}
3600
3601
Tox_Group_Role tox_group_peer_get_role(const Tox *tox, uint32_t group_number, uint32_t peer_id,
3602
                                       Tox_Err_Group_Peer_Query *error)
3603
0
{
3604
0
    assert(tox != nullptr);
3605
3606
0
    tox_lock(tox);
3607
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3608
3609
0
    if (chat == nullptr) {
3610
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3611
0
        tox_unlock(tox);
3612
0
        return (Tox_Group_Role) - 1;
3613
0
    }
3614
3615
0
    const uint8_t ret = gc_get_role(chat, gc_peer_id_from_int(peer_id));
3616
0
    tox_unlock(tox);
3617
3618
0
    if (ret == (uint8_t) -1) {
3619
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3620
0
        return (Tox_Group_Role) - 1;
3621
0
    }
3622
3623
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3624
0
    return (Tox_Group_Role)ret;
3625
0
}
3626
3627
bool tox_group_peer_get_public_key(const Tox *tox, uint32_t group_number, uint32_t peer_id, uint8_t public_key[TOX_PUBLIC_KEY_SIZE],
3628
                                   Tox_Err_Group_Peer_Query *error)
3629
0
{
3630
0
    assert(tox != nullptr);
3631
3632
0
    tox_lock(tox);
3633
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3634
3635
0
    if (chat == nullptr) {
3636
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3637
0
        tox_unlock(tox);
3638
0
        return false;
3639
0
    }
3640
3641
0
    const int ret = gc_get_peer_public_key_by_peer_id(chat, gc_peer_id_from_int(peer_id), public_key);
3642
0
    tox_unlock(tox);
3643
3644
0
    if (ret == -1) {
3645
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3646
0
        return false;
3647
0
    }
3648
3649
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3650
0
    return true;
3651
0
}
3652
3653
Tox_Connection tox_group_peer_get_connection_status(const Tox *tox, uint32_t group_number, uint32_t peer_id,
3654
        Tox_Err_Group_Peer_Query *error)
3655
0
{
3656
0
    assert(tox != nullptr);
3657
3658
0
    tox_lock(tox);
3659
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3660
3661
0
    if (chat == nullptr) {
3662
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_GROUP_NOT_FOUND);
3663
0
        tox_unlock(tox);
3664
0
        return TOX_CONNECTION_NONE;
3665
0
    }
3666
3667
0
    const unsigned int ret = gc_get_peer_connection_status(chat, gc_peer_id_from_int(peer_id));
3668
0
    tox_unlock(tox);
3669
3670
0
    if (ret == 0) {
3671
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_PEER_NOT_FOUND);
3672
0
        return TOX_CONNECTION_NONE;
3673
0
    }
3674
3675
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_PEER_QUERY_OK);
3676
0
    return (Tox_Connection)ret;
3677
0
}
3678
3679
bool tox_group_set_topic(Tox *tox, uint32_t group_number, const uint8_t *topic, size_t length,
3680
                         Tox_Err_Group_Topic_Set *error)
3681
0
{
3682
0
    assert(tox != nullptr);
3683
3684
0
    tox_lock(tox);
3685
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3686
3687
0
    if (chat == nullptr) {
3688
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_GROUP_NOT_FOUND);
3689
0
        tox_unlock(tox);
3690
0
        return false;
3691
0
    }
3692
3693
0
    if (chat->connection_state == CS_DISCONNECTED) {
3694
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_DISCONNECTED);
3695
0
        tox_unlock(tox);
3696
0
        return false;
3697
0
    }
3698
3699
0
    const int ret = gc_set_topic(chat, topic, length);
3700
0
    tox_unlock(tox);
3701
3702
0
    switch (ret) {
3703
0
        case 0: {
3704
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_OK);
3705
0
            return true;
3706
0
        }
3707
3708
0
        case -1: {
3709
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_TOO_LONG);
3710
0
            return false;
3711
0
        }
3712
3713
0
        case -2: {
3714
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_PERMISSIONS);
3715
0
            return false;
3716
0
        }
3717
3718
0
        case -3: {
3719
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_FAIL_CREATE);
3720
0
            return false;
3721
0
        }
3722
3723
0
        case -4: {
3724
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_TOPIC_SET_FAIL_SEND);
3725
0
            return false;
3726
0
        }
3727
0
    }
3728
3729
    /* can't happen */
3730
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
3731
3732
0
    return false;
3733
0
}
3734
3735
size_t tox_group_get_topic_size(const Tox *tox, uint32_t group_number, Tox_Err_Group_State_Query *error)
3736
0
{
3737
0
    assert(tox != nullptr);
3738
3739
0
    tox_lock(tox);
3740
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3741
3742
0
    if (chat == nullptr) {
3743
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3744
0
        tox_unlock(tox);
3745
0
        return -1;
3746
0
    }
3747
3748
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3749
3750
0
    const size_t ret = gc_get_topic_size(chat);
3751
0
    tox_unlock(tox);
3752
3753
0
    return ret;
3754
0
}
3755
3756
bool tox_group_get_topic(const Tox *tox, uint32_t group_number, uint8_t *topic, Tox_Err_Group_State_Query *error)
3757
0
{
3758
0
    assert(tox != nullptr);
3759
3760
0
    tox_lock(tox);
3761
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3762
3763
0
    if (chat == nullptr) {
3764
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3765
0
        tox_unlock(tox);
3766
0
        return false;
3767
0
    }
3768
3769
0
    gc_get_topic(chat, topic);
3770
0
    tox_unlock(tox);
3771
3772
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3773
0
    return true;
3774
0
}
3775
3776
size_t tox_group_get_name_size(const Tox *tox, uint32_t group_number, Tox_Err_Group_State_Query *error)
3777
0
{
3778
0
    assert(tox != nullptr);
3779
3780
0
    tox_lock(tox);
3781
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3782
3783
0
    if (chat == nullptr) {
3784
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3785
0
        tox_unlock(tox);
3786
0
        return -1;
3787
0
    }
3788
3789
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3790
3791
0
    const size_t ret = gc_get_group_name_size(chat);
3792
0
    tox_unlock(tox);
3793
3794
0
    return ret;
3795
0
}
3796
3797
bool tox_group_get_name(const Tox *tox, uint32_t group_number, uint8_t *name, Tox_Err_Group_State_Query *error)
3798
0
{
3799
0
    assert(tox != nullptr);
3800
3801
0
    tox_lock(tox);
3802
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3803
3804
0
    if (chat == nullptr) {
3805
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3806
0
        tox_unlock(tox);
3807
0
        return false;
3808
0
    }
3809
3810
0
    gc_get_group_name(chat, name);
3811
0
    tox_unlock(tox);
3812
3813
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3814
3815
0
    return true;
3816
0
}
3817
3818
bool tox_group_get_chat_id(const Tox *tox, uint32_t group_number, uint8_t chat_id[TOX_GROUP_CHAT_ID_SIZE], Tox_Err_Group_State_Query *error)
3819
0
{
3820
0
    assert(tox != nullptr);
3821
3822
0
    tox_lock(tox);
3823
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3824
3825
0
    if (chat == nullptr) {
3826
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3827
0
        tox_unlock(tox);
3828
0
        return false;
3829
0
    }
3830
3831
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3832
0
    gc_get_chat_id(chat, chat_id);
3833
0
    tox_unlock(tox);
3834
3835
0
    return true;
3836
0
}
3837
3838
uint32_t tox_group_get_number_groups(const Tox *tox)
3839
0
{
3840
0
    assert(tox != nullptr);
3841
3842
0
    tox_lock(tox);
3843
0
    const uint32_t ret = gc_count_groups(tox->m->group_handler);
3844
0
    tox_unlock(tox);
3845
3846
0
    return ret;
3847
0
}
3848
3849
Tox_Group_Privacy_State tox_group_get_privacy_state(const Tox *tox, uint32_t group_number,
3850
        Tox_Err_Group_State_Query *error)
3851
0
{
3852
0
    assert(tox != nullptr);
3853
3854
0
    tox_lock(tox);
3855
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3856
3857
0
    if (chat == nullptr) {
3858
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3859
0
        tox_unlock(tox);
3860
0
        return (Tox_Group_Privacy_State) - 1;
3861
0
    }
3862
3863
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3864
3865
0
    const uint8_t state = gc_get_privacy_state(chat);
3866
0
    tox_unlock(tox);
3867
3868
0
    return (Tox_Group_Privacy_State)state;
3869
0
}
3870
3871
Tox_Group_Topic_Lock tox_group_get_topic_lock(const Tox *tox, uint32_t group_number, Tox_Err_Group_State_Query *error)
3872
0
{
3873
0
    assert(tox != nullptr);
3874
3875
0
    tox_lock(tox);
3876
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3877
3878
0
    if (chat == nullptr) {
3879
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3880
0
        tox_unlock(tox);
3881
0
        return (Tox_Group_Topic_Lock) - 1;
3882
0
    }
3883
3884
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3885
3886
0
    const Group_Topic_Lock topic_lock = gc_get_topic_lock_state(chat);
3887
0
    tox_unlock(tox);
3888
3889
0
    return (Tox_Group_Topic_Lock)topic_lock;
3890
0
}
3891
3892
Tox_Group_Voice_State tox_group_get_voice_state(const Tox *tox, uint32_t group_number,
3893
        Tox_Err_Group_State_Query *error)
3894
0
{
3895
0
    assert(tox != nullptr);
3896
3897
0
    tox_lock(tox);
3898
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3899
3900
0
    if (chat == nullptr) {
3901
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3902
0
        tox_unlock(tox);
3903
0
        return (Tox_Group_Voice_State) - 1;
3904
0
    }
3905
3906
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3907
3908
0
    const Group_Voice_State voice_state = gc_get_voice_state(chat);
3909
0
    tox_unlock(tox);
3910
3911
0
    return (Tox_Group_Voice_State)voice_state;
3912
0
}
3913
3914
uint16_t tox_group_get_peer_limit(const Tox *tox, uint32_t group_number, Tox_Err_Group_State_Query *error)
3915
0
{
3916
0
    assert(tox != nullptr);
3917
3918
0
    tox_lock(tox);
3919
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3920
3921
0
    if (chat == nullptr) {
3922
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3923
0
        tox_unlock(tox);
3924
0
        return -1;
3925
0
    }
3926
3927
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3928
3929
0
    const uint16_t ret = gc_get_max_peers(chat);
3930
0
    tox_unlock(tox);
3931
3932
0
    return ret;
3933
0
}
3934
3935
size_t tox_group_get_password_size(const Tox *tox, uint32_t group_number, Tox_Err_Group_State_Query *error)
3936
0
{
3937
0
    assert(tox != nullptr);
3938
3939
0
    tox_lock(tox);
3940
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3941
3942
0
    if (chat == nullptr) {
3943
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3944
0
        tox_unlock(tox);
3945
0
        return -1;
3946
0
    }
3947
3948
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3949
3950
0
    const size_t ret = gc_get_password_size(chat);
3951
0
    tox_unlock(tox);
3952
3953
0
    return ret;
3954
0
}
3955
3956
bool tox_group_get_password(const Tox *tox, uint32_t group_number, uint8_t *password,
3957
                            Tox_Err_Group_State_Query *error)
3958
0
{
3959
0
    assert(tox != nullptr);
3960
3961
0
    tox_lock(tox);
3962
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3963
3964
0
    if (chat == nullptr) {
3965
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_GROUP_NOT_FOUND);
3966
0
        tox_unlock(tox);
3967
0
        return false;
3968
0
    }
3969
3970
0
    SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_STATE_QUERY_OK);
3971
3972
0
    gc_get_password(chat, password);
3973
0
    tox_unlock(tox);
3974
3975
0
    return true;
3976
0
}
3977
3978
Tox_Group_Message_Id tox_group_send_message(
3979
    const Tox *tox, uint32_t group_number, Tox_Message_Type message_type, const uint8_t *message,
3980
    size_t length, Tox_Err_Group_Send_Message *error)
3981
0
{
3982
0
    assert(tox != nullptr);
3983
3984
0
    tox_lock(tox);
3985
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
3986
3987
0
    if (chat == nullptr) {
3988
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_GROUP_NOT_FOUND);
3989
0
        tox_unlock(tox);
3990
0
        return -1;
3991
0
    }
3992
3993
0
    if (chat->connection_state == CS_DISCONNECTED) {
3994
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_DISCONNECTED);
3995
0
        tox_unlock(tox);
3996
0
        return -1;
3997
0
    }
3998
3999
0
    uint32_t message_id = 0;
4000
0
    const int ret = gc_send_message(chat, message, length, message_type, &message_id);
4001
0
    tox_unlock(tox);
4002
4003
0
    switch (ret) {
4004
0
        case 0: {
4005
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_OK);
4006
0
            return message_id;
4007
0
        }
4008
4009
0
        case -1: {
4010
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_TOO_LONG);
4011
0
            return -1;
4012
0
        }
4013
4014
0
        case -2: {
4015
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_EMPTY);
4016
0
            return -1;
4017
0
        }
4018
4019
0
        case -3: {
4020
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_BAD_TYPE);
4021
0
            return -1;
4022
0
        }
4023
4024
0
        case -4: {
4025
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_PERMISSIONS);
4026
0
            return -1;
4027
0
        }
4028
4029
0
        case -5: {
4030
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_MESSAGE_FAIL_SEND);
4031
0
            return -1;
4032
0
        }
4033
0
    }
4034
4035
    /* can't happen */
4036
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4037
4038
0
    return -1;
4039
0
}
4040
4041
Tox_Group_Message_Id tox_group_send_private_message(const Tox *tox, uint32_t group_number, uint32_t peer_id,
4042
        Tox_Message_Type message_type, const uint8_t *message, size_t length, Tox_Err_Group_Send_Private_Message *error)
4043
0
{
4044
0
    assert(tox != nullptr);
4045
4046
0
    tox_lock(tox);
4047
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4048
4049
0
    if (chat == nullptr) {
4050
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_GROUP_NOT_FOUND);
4051
0
        tox_unlock(tox);
4052
0
        return -1;
4053
0
    }
4054
4055
0
    if (chat->connection_state == CS_DISCONNECTED) {
4056
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_DISCONNECTED);
4057
0
        tox_unlock(tox);
4058
0
        return -1;
4059
0
    }
4060
4061
0
    uint32_t message_id = 0;
4062
0
    const int ret = gc_send_private_message(chat, gc_peer_id_from_int(peer_id), message_type, message, length, &message_id);
4063
0
    tox_unlock(tox);
4064
4065
0
    switch (ret) {
4066
0
        case 0: {
4067
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_OK);
4068
0
            return message_id;
4069
0
        }
4070
4071
0
        case -1: {
4072
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_TOO_LONG);
4073
0
            return -1;
4074
0
        }
4075
4076
0
        case -2: {
4077
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_EMPTY);
4078
0
            return -1;
4079
0
        }
4080
4081
0
        case -3: {
4082
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_PEER_NOT_FOUND);
4083
0
            return -1;
4084
0
        }
4085
4086
0
        case -4: {
4087
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_BAD_TYPE);
4088
0
            return -1;
4089
0
        }
4090
4091
0
        case -5: {
4092
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_PERMISSIONS);
4093
0
            return -1;
4094
0
        }
4095
4096
0
        case -6: {
4097
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_PRIVATE_MESSAGE_FAIL_SEND);
4098
0
            return -1;
4099
0
        }
4100
0
    }
4101
4102
    /* can't happen */
4103
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4104
4105
0
    return -1;
4106
0
}
4107
4108
bool tox_group_send_custom_packet(const Tox *tox, uint32_t group_number, bool lossless, const uint8_t *data,
4109
                                  size_t length, Tox_Err_Group_Send_Custom_Packet *error)
4110
0
{
4111
0
    assert(tox != nullptr);
4112
4113
0
    tox_lock(tox);
4114
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4115
4116
0
    if (chat == nullptr) {
4117
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_GROUP_NOT_FOUND);
4118
0
        tox_unlock(tox);
4119
0
        return false;
4120
0
    }
4121
4122
0
    if (chat->connection_state == CS_DISCONNECTED) {
4123
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_DISCONNECTED);
4124
0
        tox_unlock(tox);
4125
0
        return false;
4126
0
    }
4127
4128
0
    const int ret = gc_send_custom_packet(chat, lossless, data, length);
4129
0
    tox_unlock(tox);
4130
4131
0
    switch (ret) {
4132
0
        case 0: {
4133
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_OK);
4134
0
            return true;
4135
0
        }
4136
4137
0
        case -1: {
4138
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_TOO_LONG);
4139
0
            return false;
4140
0
        }
4141
4142
0
        case -2: {
4143
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_EMPTY);
4144
0
            return false;
4145
0
        }
4146
4147
0
        case -3: {
4148
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PACKET_FAIL_SEND);
4149
0
            return false;
4150
0
        }
4151
0
    }
4152
4153
    /* can't happen */
4154
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4155
4156
0
    return false;
4157
0
}
4158
4159
bool tox_group_send_custom_private_packet(const Tox *tox, uint32_t group_number, uint32_t peer_id, bool lossless,
4160
        const uint8_t *data, size_t length,
4161
        Tox_Err_Group_Send_Custom_Private_Packet *error)
4162
0
{
4163
0
    assert(tox != nullptr);
4164
4165
0
    tox_lock(tox);
4166
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4167
4168
0
    if (chat == nullptr) {
4169
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_GROUP_NOT_FOUND);
4170
0
        tox_unlock(tox);
4171
0
        return false;
4172
0
    }
4173
4174
0
    if (chat->connection_state == CS_DISCONNECTED) {
4175
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_DISCONNECTED);
4176
0
        tox_unlock(tox);
4177
0
        return false;
4178
0
    }
4179
4180
0
    const int ret = gc_send_custom_private_packet(chat, lossless, gc_peer_id_from_int(peer_id), data, length);
4181
0
    tox_unlock(tox);
4182
4183
0
    switch (ret) {
4184
0
        case 0: {
4185
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_OK);
4186
0
            return true;
4187
0
        }
4188
4189
0
        case -1: {
4190
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_TOO_LONG);
4191
0
            return false;
4192
0
        }
4193
4194
0
        case -2: {
4195
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_EMPTY);
4196
0
            return false;
4197
0
        }
4198
4199
0
        case -3: {
4200
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_PEER_NOT_FOUND);
4201
0
            return false;
4202
0
        }
4203
4204
0
        case -4: {
4205
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SEND_CUSTOM_PRIVATE_PACKET_FAIL_SEND);
4206
0
            return false;
4207
0
        }
4208
0
    }
4209
4210
    /* can't happen */
4211
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4212
4213
0
    return false;
4214
0
}
4215
4216
bool tox_group_invite_friend(const Tox *tox, uint32_t group_number, uint32_t friend_number,
4217
                             Tox_Err_Group_Invite_Friend *error)
4218
0
{
4219
0
    assert(tox != nullptr);
4220
4221
0
    tox_lock(tox);
4222
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4223
4224
0
    if (chat == nullptr) {
4225
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_GROUP_NOT_FOUND);
4226
0
        tox_unlock(tox);
4227
0
        return false;
4228
0
    }
4229
4230
0
    if (chat->connection_state == CS_DISCONNECTED) {
4231
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_DISCONNECTED);
4232
0
        tox_unlock(tox);
4233
0
        return false;
4234
0
    }
4235
4236
0
    if (!friend_is_valid(tox->m, friend_number)) {
4237
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_FRIEND_NOT_FOUND);
4238
0
        tox_unlock(tox);
4239
0
        return false;
4240
0
    }
4241
4242
0
    const int ret = gc_invite_friend(tox->m->group_handler, chat, friend_number, send_group_invite_packet);
4243
0
    tox_unlock(tox);
4244
4245
0
    switch (ret) {
4246
0
        case 0: {
4247
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_OK);
4248
0
            return true;
4249
0
        }
4250
4251
0
        case -1: {
4252
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_FRIEND_NOT_FOUND);
4253
0
            return false;
4254
0
        }
4255
4256
0
        case -2: {
4257
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_INVITE_FAIL);
4258
0
            return false;
4259
0
        }
4260
4261
0
        case -3: {
4262
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_FRIEND_FAIL_SEND);
4263
0
            return false;
4264
0
        }
4265
0
    }
4266
4267
    /* can't happen */
4268
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4269
4270
0
    return false;
4271
0
}
4272
4273
uint32_t tox_group_invite_accept(Tox *tox, uint32_t friend_number, const uint8_t *invite_data, size_t length,
4274
                                 const uint8_t *name, size_t name_length, const uint8_t *password,
4275
                                 size_t password_length, Tox_Err_Group_Invite_Accept *error)
4276
0
{
4277
0
    assert(tox != nullptr);
4278
4279
0
    if (invite_data == nullptr || name == nullptr) {
4280
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_NULL);
4281
0
        return UINT32_MAX;
4282
0
    }
4283
4284
0
    tox_lock(tox);
4285
0
    const int ret = gc_accept_invite(tox->m->group_handler, friend_number, invite_data, length, name, name_length, password,
4286
0
                                     password_length);
4287
0
    tox_unlock(tox);
4288
4289
0
    if (ret >= 0) {
4290
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_OK);
4291
0
        return ret;
4292
0
    }
4293
4294
0
    switch (ret) {
4295
0
        case -1: {
4296
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_BAD_INVITE);
4297
0
            return UINT32_MAX;
4298
0
        }
4299
4300
0
        case -2: {
4301
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_INIT_FAILED);
4302
0
            return UINT32_MAX;
4303
0
        }
4304
4305
0
        case -3: {
4306
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_TOO_LONG);
4307
0
            return UINT32_MAX;
4308
0
        }
4309
4310
0
        case -4: {
4311
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_EMPTY);
4312
0
            return UINT32_MAX;
4313
0
        }
4314
4315
0
        case -5: {
4316
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_PASSWORD);
4317
0
            return UINT32_MAX;
4318
0
        }
4319
4320
0
        case -6: {
4321
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_FRIEND_NOT_FOUND);
4322
0
            return UINT32_MAX;
4323
0
        }
4324
4325
0
        case -7: {
4326
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_INVITE_ACCEPT_FAIL_SEND);
4327
0
            return UINT32_MAX;
4328
0
        }
4329
0
    }
4330
4331
    /* can't happen */
4332
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4333
4334
0
    return UINT32_MAX;
4335
0
}
4336
4337
bool tox_group_set_password(Tox *tox, uint32_t group_number, const uint8_t *password, size_t length,
4338
                            Tox_Err_Group_Set_Password *error)
4339
0
{
4340
0
    assert(tox != nullptr);
4341
4342
0
    tox_lock(tox);
4343
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4344
4345
0
    if (chat == nullptr) {
4346
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_GROUP_NOT_FOUND);
4347
0
        tox_unlock(tox);
4348
0
        return false;
4349
0
    }
4350
4351
0
    if (chat->connection_state == CS_DISCONNECTED) {
4352
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_DISCONNECTED);
4353
0
        tox_unlock(tox);
4354
0
        return false;
4355
0
    }
4356
4357
0
    const int ret = gc_founder_set_password(chat, password, length);
4358
0
    tox_unlock(tox);
4359
4360
0
    switch (ret) {
4361
0
        case 0: {
4362
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_OK);
4363
0
            return true;
4364
0
        }
4365
4366
0
        case -1: {
4367
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_PERMISSIONS);
4368
0
            return false;
4369
0
        }
4370
4371
0
        case -2: {
4372
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_TOO_LONG);
4373
0
            return false;
4374
0
        }
4375
4376
0
        case -3: {
4377
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_FAIL_SEND);
4378
0
            return false;
4379
0
        }
4380
4381
0
        case -4: {
4382
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PASSWORD_MALLOC);
4383
0
            return false;
4384
0
        }
4385
0
    }
4386
4387
    /* can't happen */
4388
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4389
4390
0
    return false;
4391
0
}
4392
4393
bool tox_group_set_privacy_state(Tox *tox, uint32_t group_number, Tox_Group_Privacy_State privacy_state,
4394
                                 Tox_Err_Group_Set_Privacy_State *error)
4395
0
{
4396
0
    assert(tox != nullptr);
4397
4398
0
    tox_lock(tox);
4399
0
    const int ret = gc_founder_set_privacy_state(tox->m, group_number, (Group_Privacy_State) privacy_state);
4400
0
    tox_unlock(tox);
4401
4402
0
    switch (ret) {
4403
0
        case 0: {
4404
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_OK);
4405
0
            return true;
4406
0
        }
4407
4408
0
        case -1: {
4409
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_GROUP_NOT_FOUND);
4410
0
            return false;
4411
0
        }
4412
4413
0
        case -2: {
4414
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_PERMISSIONS);
4415
0
            return false;
4416
0
        }
4417
4418
0
        case -3: {
4419
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_DISCONNECTED);
4420
0
            return false;
4421
0
        }
4422
4423
0
        case -4: {
4424
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_FAIL_SET);
4425
0
            return false;
4426
0
        }
4427
4428
0
        case -5: {
4429
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PRIVACY_STATE_FAIL_SEND);
4430
0
            return false;
4431
0
        }
4432
0
    }
4433
4434
    /* can't happen */
4435
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4436
4437
0
    return false;
4438
0
}
4439
4440
bool tox_group_set_topic_lock(Tox *tox, uint32_t group_number, Tox_Group_Topic_Lock topic_lock,
4441
                              Tox_Err_Group_Set_Topic_Lock *error)
4442
0
{
4443
0
    assert(tox != nullptr);
4444
4445
0
    tox_lock(tox);
4446
0
    const int ret = gc_founder_set_topic_lock(tox->m, group_number, (Group_Topic_Lock) topic_lock);
4447
0
    tox_unlock(tox);
4448
4449
0
    switch (ret) {
4450
0
        case 0: {
4451
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_OK);
4452
0
            return true;
4453
0
        }
4454
4455
0
        case -1: {
4456
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_GROUP_NOT_FOUND);
4457
0
            return false;
4458
0
        }
4459
4460
0
        case -2: {
4461
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_INVALID);
4462
0
            return false;
4463
0
        }
4464
4465
0
        case -3: {
4466
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_PERMISSIONS);
4467
0
            return false;
4468
0
        }
4469
4470
0
        case -4: {
4471
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_DISCONNECTED);
4472
0
            return false;
4473
0
        }
4474
4475
0
        case -5: {
4476
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_FAIL_SET);
4477
0
            return false;
4478
0
        }
4479
4480
0
        case -6: {
4481
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_TOPIC_LOCK_FAIL_SEND);
4482
0
            return false;
4483
0
        }
4484
0
    }
4485
4486
    /* can't happen */
4487
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4488
4489
0
    return false;
4490
0
}
4491
4492
bool tox_group_set_voice_state(Tox *tox, uint32_t group_number, Tox_Group_Voice_State voice_state,
4493
                               Tox_Err_Group_Set_Voice_State *error)
4494
0
{
4495
0
    assert(tox != nullptr);
4496
4497
0
    tox_lock(tox);
4498
0
    const int ret = gc_founder_set_voice_state(tox->m, group_number, (Group_Voice_State)voice_state);
4499
0
    tox_unlock(tox);
4500
4501
0
    switch (ret) {
4502
0
        case 0: {
4503
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_OK);
4504
0
            return true;
4505
0
        }
4506
4507
0
        case -1: {
4508
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_GROUP_NOT_FOUND);
4509
0
            return false;
4510
0
        }
4511
4512
0
        case -2: {
4513
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_PERMISSIONS);
4514
0
            return false;
4515
0
        }
4516
4517
0
        case -3: {
4518
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_DISCONNECTED);
4519
0
            return false;
4520
0
        }
4521
4522
0
        case -4: {
4523
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_FAIL_SET);
4524
0
            return false;
4525
0
        }
4526
4527
0
        case -5: {
4528
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_VOICE_STATE_FAIL_SEND);
4529
0
            return false;
4530
0
        }
4531
0
    }
4532
4533
    /* can't happen */
4534
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4535
4536
0
    return false;
4537
0
}
4538
4539
bool tox_group_set_peer_limit(Tox *tox, uint32_t group_number, uint16_t peer_limit,
4540
                              Tox_Err_Group_Set_Peer_Limit *error)
4541
0
{
4542
0
    assert(tox != nullptr);
4543
4544
0
    tox_lock(tox);
4545
0
    GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4546
4547
0
    if (chat == nullptr) {
4548
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_GROUP_NOT_FOUND);
4549
0
        tox_unlock(tox);
4550
0
        return false;
4551
0
    }
4552
4553
0
    if (chat->connection_state == CS_DISCONNECTED) {
4554
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_DISCONNECTED);
4555
0
        tox_unlock(tox);
4556
0
        return false;
4557
0
    }
4558
4559
0
    const int ret = gc_founder_set_max_peers(chat, peer_limit);
4560
0
    tox_unlock(tox);
4561
4562
0
    switch (ret) {
4563
0
        case 0: {
4564
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_OK);
4565
0
            return true;
4566
0
        }
4567
4568
0
        case -1: {
4569
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_PERMISSIONS);
4570
0
            return false;
4571
0
        }
4572
4573
0
        case -2: {
4574
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_FAIL_SET);
4575
0
            return false;
4576
0
        }
4577
4578
0
        case -3: {
4579
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_PEER_LIMIT_FAIL_SEND);
4580
0
            return false;
4581
0
        }
4582
0
    }
4583
4584
    /* can't happen */
4585
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4586
4587
0
    return false;
4588
0
}
4589
4590
bool tox_group_set_ignore(Tox *tox, uint32_t group_number, uint32_t peer_id, bool ignore,
4591
                          Tox_Err_Group_Set_Ignore *error)
4592
0
{
4593
0
    assert(tox != nullptr);
4594
4595
0
    tox_lock(tox);
4596
0
    const GC_Chat *chat = gc_get_group(tox->m->group_handler, group_number);
4597
4598
0
    if (chat == nullptr) {
4599
0
        SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_IGNORE_GROUP_NOT_FOUND);
4600
0
        tox_unlock(tox);
4601
0
        return false;
4602
0
    }
4603
4604
0
    const int ret = gc_set_ignore(chat, gc_peer_id_from_int(peer_id), ignore);
4605
0
    tox_unlock(tox);
4606
4607
0
    switch (ret) {
4608
0
        case 0: {
4609
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_IGNORE_OK);
4610
0
            return true;
4611
0
        }
4612
4613
0
        case -1: {
4614
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_IGNORE_PEER_NOT_FOUND);
4615
0
            return false;
4616
0
        }
4617
4618
0
        case -2: {
4619
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_IGNORE_SELF);
4620
0
            return false;
4621
0
        }
4622
0
    }
4623
4624
    /* can't happen */
4625
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4626
4627
0
    return false;
4628
0
}
4629
4630
bool tox_group_set_role(Tox *tox, uint32_t group_number, uint32_t peer_id, Tox_Group_Role role,
4631
                        Tox_Err_Group_Set_Role *error)
4632
0
{
4633
0
    assert(tox != nullptr);
4634
4635
0
    tox_lock(tox);
4636
0
    const int ret = gc_set_peer_role(tox->m, group_number, gc_peer_id_from_int(peer_id), (Group_Role) role);
4637
0
    tox_unlock(tox);
4638
4639
0
    switch (ret) {
4640
0
        case 0: {
4641
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_OK);
4642
0
            return true;
4643
0
        }
4644
4645
0
        case -1: {
4646
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_GROUP_NOT_FOUND);
4647
0
            return false;
4648
0
        }
4649
4650
0
        case -2: {
4651
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_PEER_NOT_FOUND);
4652
0
            return false;
4653
0
        }
4654
4655
0
        case -3: {
4656
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_PERMISSIONS);
4657
0
            return false;
4658
0
        }
4659
4660
0
        case -4: {
4661
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_ASSIGNMENT);
4662
0
            return false;
4663
0
        }
4664
4665
0
        case -5: {
4666
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_FAIL_ACTION);
4667
0
            return false;
4668
0
        }
4669
4670
0
        case -6: {
4671
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_SET_ROLE_SELF);
4672
0
            return false;
4673
0
        }
4674
0
    }
4675
4676
    /* can't happen */
4677
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4678
4679
0
    return false;
4680
0
}
4681
4682
bool tox_group_kick_peer(const Tox *tox, uint32_t group_number, uint32_t peer_id,
4683
                         Tox_Err_Group_Kick_Peer *error)
4684
0
{
4685
0
    assert(tox != nullptr);
4686
4687
0
    tox_lock(tox);
4688
0
    const int ret = gc_kick_peer(tox->m, group_number, gc_peer_id_from_int(peer_id));
4689
0
    tox_unlock(tox);
4690
4691
0
    switch (ret) {
4692
0
        case 0: {
4693
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_OK);
4694
0
            return true;
4695
0
        }
4696
4697
0
        case -1: {
4698
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_GROUP_NOT_FOUND);
4699
0
            return false;
4700
0
        }
4701
4702
0
        case -2: {
4703
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_PEER_NOT_FOUND);
4704
0
            return false;
4705
0
        }
4706
4707
0
        case -3: {
4708
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_PERMISSIONS);
4709
0
            return false;
4710
0
        }
4711
4712
0
        case -4: {
4713
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_FAIL_ACTION);
4714
0
            return false;
4715
0
        }
4716
4717
0
        case -5: {
4718
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_FAIL_SEND);
4719
0
            return false;
4720
0
        }
4721
4722
0
        case -6: {
4723
0
            SET_ERROR_PARAMETER(error, TOX_ERR_GROUP_KICK_PEER_SELF);
4724
0
            return false;
4725
0
        }
4726
0
    }
4727
4728
    /* can't happen */
4729
0
    LOGGER_FATAL(tox->m->log, "impossible return value: %d", ret);
4730
4731
0
    return false;
4732
0
}
4733
4734
const Tox_System *tox_get_system(Tox *tox)
4735
0
{
4736
0
    assert(tox != nullptr);
4737
0
    return &tox->sys;
4738
0
}