/work/toxcore/tox_events.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* SPDX-License-Identifier: GPL-3.0-or-later |
2 | | * Copyright © 2022-2024 The TokTok team. |
3 | | */ |
4 | | |
5 | | #include "tox_events.h" |
6 | | |
7 | | #include <assert.h> |
8 | | #include <string.h> |
9 | | |
10 | | #include "bin_pack.h" |
11 | | #include "bin_unpack.h" |
12 | | #include "ccompat.h" |
13 | | #include "events/events_alloc.h" |
14 | | #include "logger.h" |
15 | | #include "mem.h" |
16 | | #include "tox.h" |
17 | | #include "tox_event.h" |
18 | | #include "tox_private.h" |
19 | | #include "tox_struct.h" |
20 | | |
21 | | |
22 | | /***************************************************** |
23 | | * |
24 | | * :: Set up event handlers. |
25 | | * |
26 | | *****************************************************/ |
27 | | |
28 | | |
29 | | void tox_events_init(Tox *tox) |
30 | 2.33k | { |
31 | 2.33k | tox_callback_conference_connected(tox, tox_events_handle_conference_connected); |
32 | 2.33k | tox_callback_conference_invite(tox, tox_events_handle_conference_invite); |
33 | 2.33k | tox_callback_conference_message(tox, tox_events_handle_conference_message); |
34 | 2.33k | tox_callback_conference_peer_list_changed(tox, tox_events_handle_conference_peer_list_changed); |
35 | 2.33k | tox_callback_conference_peer_name(tox, tox_events_handle_conference_peer_name); |
36 | 2.33k | tox_callback_conference_title(tox, tox_events_handle_conference_title); |
37 | 2.33k | tox_callback_file_chunk_request(tox, tox_events_handle_file_chunk_request); |
38 | 2.33k | tox_callback_file_recv_chunk(tox, tox_events_handle_file_recv_chunk); |
39 | 2.33k | tox_callback_file_recv_control(tox, tox_events_handle_file_recv_control); |
40 | 2.33k | tox_callback_file_recv(tox, tox_events_handle_file_recv); |
41 | 2.33k | tox_callback_friend_connection_status(tox, tox_events_handle_friend_connection_status); |
42 | 2.33k | tox_callback_friend_lossless_packet(tox, tox_events_handle_friend_lossless_packet); |
43 | 2.33k | tox_callback_friend_lossy_packet(tox, tox_events_handle_friend_lossy_packet); |
44 | 2.33k | tox_callback_friend_message(tox, tox_events_handle_friend_message); |
45 | 2.33k | tox_callback_friend_name(tox, tox_events_handle_friend_name); |
46 | 2.33k | tox_callback_friend_read_receipt(tox, tox_events_handle_friend_read_receipt); |
47 | 2.33k | tox_callback_friend_request(tox, tox_events_handle_friend_request); |
48 | 2.33k | tox_callback_friend_status_message(tox, tox_events_handle_friend_status_message); |
49 | 2.33k | tox_callback_friend_status(tox, tox_events_handle_friend_status); |
50 | 2.33k | tox_callback_friend_typing(tox, tox_events_handle_friend_typing); |
51 | 2.33k | tox_callback_self_connection_status(tox, tox_events_handle_self_connection_status); |
52 | 2.33k | tox_callback_group_peer_name(tox, tox_events_handle_group_peer_name); |
53 | 2.33k | tox_callback_group_peer_status(tox, tox_events_handle_group_peer_status); |
54 | 2.33k | tox_callback_group_topic(tox, tox_events_handle_group_topic); |
55 | 2.33k | tox_callback_group_privacy_state(tox, tox_events_handle_group_privacy_state); |
56 | 2.33k | tox_callback_group_voice_state(tox, tox_events_handle_group_voice_state); |
57 | 2.33k | tox_callback_group_topic_lock(tox, tox_events_handle_group_topic_lock); |
58 | 2.33k | tox_callback_group_peer_limit(tox, tox_events_handle_group_peer_limit); |
59 | 2.33k | tox_callback_group_password(tox, tox_events_handle_group_password); |
60 | 2.33k | tox_callback_group_message(tox, tox_events_handle_group_message); |
61 | 2.33k | tox_callback_group_private_message(tox, tox_events_handle_group_private_message); |
62 | 2.33k | tox_callback_group_custom_packet(tox, tox_events_handle_group_custom_packet); |
63 | 2.33k | tox_callback_group_custom_private_packet(tox, tox_events_handle_group_custom_private_packet); |
64 | 2.33k | tox_callback_group_invite(tox, tox_events_handle_group_invite); |
65 | 2.33k | tox_callback_group_peer_join(tox, tox_events_handle_group_peer_join); |
66 | 2.33k | tox_callback_group_peer_exit(tox, tox_events_handle_group_peer_exit); |
67 | 2.33k | tox_callback_group_self_join(tox, tox_events_handle_group_self_join); |
68 | 2.33k | tox_callback_group_join_fail(tox, tox_events_handle_group_join_fail); |
69 | 2.33k | tox_callback_group_moderation(tox, tox_events_handle_group_moderation); |
70 | 2.33k | tox_callback_dht_get_nodes_response(tox, tox_events_handle_dht_get_nodes_response); |
71 | 2.33k | } |
72 | | |
73 | | uint32_t tox_events_get_size(const Tox_Events *events) |
74 | 447k | { |
75 | 447k | return events == nullptr ? 0 : events->events_size; |
76 | 447k | } |
77 | | |
78 | | nullable(1) |
79 | | static const Tox_Event *tox_events_get_events(const Tox_Events *events) |
80 | 256k | { |
81 | 256k | return events == nullptr ? nullptr : events->events; |
82 | 256k | } |
83 | | |
84 | | const Tox_Event *tox_events_get(const Tox_Events *events, uint32_t index) |
85 | 4 | { |
86 | 4 | if (index >= tox_events_get_size(events)) { |
87 | 0 | return nullptr; |
88 | 0 | } |
89 | | |
90 | 4 | return &events->events[index]; |
91 | 4 | } |
92 | | |
93 | | Tox_Events *tox_events_iterate(Tox *tox, bool fail_hard, Tox_Err_Events_Iterate *error) |
94 | 193k | { |
95 | 193k | const Tox_System *sys = tox_get_system(tox); |
96 | 193k | Tox_Events_State state = {TOX_ERR_EVENTS_ITERATE_OK, sys->mem}; |
97 | 193k | tox_iterate(tox, &state); |
98 | | |
99 | 193k | if (error != nullptr) { |
100 | 191k | *error = state.error; |
101 | 191k | } |
102 | | |
103 | 193k | if (fail_hard && state.error != TOX_ERR_EVENTS_ITERATE_OK) { |
104 | 162 | tox_events_free(state.events); |
105 | 162 | return nullptr; |
106 | 162 | } |
107 | | |
108 | 193k | return state.events; |
109 | 193k | } |
110 | | |
111 | | non_null() |
112 | | static bool tox_event_pack_handler(const void *arr, uint32_t index, const Logger *logger, Bin_Pack *bp) |
113 | 1.56k | { |
114 | 1.56k | const Tox_Event *events = (const Tox_Event *)arr; |
115 | 1.56k | assert(events != nullptr); |
116 | 1.56k | return tox_event_pack(&events[index], bp); |
117 | 1.56k | } |
118 | | |
119 | | non_null(3) nullable(1, 2) |
120 | | static bool tox_events_pack_handler(const void *obj, const Logger *logger, Bin_Pack *bp) |
121 | 256k | { |
122 | 256k | const Tox_Events *events = (const Tox_Events *)obj; |
123 | 256k | return bin_pack_obj_array(bp, tox_event_pack_handler, tox_events_get_events(events), tox_events_get_size(events), logger); |
124 | 256k | } |
125 | | |
126 | | uint32_t tox_events_bytes_size(const Tox_Events *events) |
127 | 127k | { |
128 | 127k | return bin_pack_obj_size(tox_events_pack_handler, events, nullptr); |
129 | 127k | } |
130 | | |
131 | | bool tox_events_get_bytes(const Tox_Events *events, uint8_t *bytes) |
132 | 128k | { |
133 | 128k | return bin_pack_obj(tox_events_pack_handler, events, nullptr, bytes, UINT32_MAX); |
134 | 128k | } |
135 | | |
136 | | non_null() |
137 | | static bool tox_events_unpack_handler(void *obj, Bin_Unpack *bu) |
138 | 1.01k | { |
139 | 1.01k | Tox_Events *events = (Tox_Events *)obj; |
140 | | |
141 | 1.01k | uint32_t size; |
142 | 1.01k | if (!bin_unpack_array(bu, &size)) { |
143 | 65 | return false; |
144 | 65 | } |
145 | | |
146 | 2.05k | for (uint32_t i = 0; i < size; ++i) { |
147 | 1.48k | Tox_Event event = {TOX_EVENT_INVALID}; |
148 | 1.48k | if (!tox_event_unpack_into(&event, bu, events->mem)) { |
149 | 371 | tox_event_destruct(&event, events->mem); |
150 | 371 | return false; |
151 | 371 | } |
152 | | |
153 | 1.11k | if (!tox_events_add(events, &event)) { |
154 | 1 | tox_event_destruct(&event, events->mem); |
155 | 1 | return false; |
156 | 1 | } |
157 | 1.11k | } |
158 | | |
159 | | // Invariant: if all adds worked, the events size must be the input array size. |
160 | 573 | assert(tox_events_get_size(events) == size); |
161 | 573 | return true; |
162 | 573 | } |
163 | | |
164 | | Tox_Events *tox_events_load(const Tox_System *sys, const uint8_t *bytes, uint32_t bytes_size) |
165 | 1.01k | { |
166 | 1.01k | Tox_Events *events = (Tox_Events *)mem_alloc(sys->mem, sizeof(Tox_Events)); |
167 | | |
168 | 1.01k | if (events == nullptr) { |
169 | 1 | return nullptr; |
170 | 1 | } |
171 | | |
172 | 1.01k | *events = (Tox_Events) { |
173 | 1.01k | nullptr |
174 | 1.01k | }; |
175 | 1.01k | events->mem = sys->mem; |
176 | | |
177 | 1.01k | if (!bin_unpack_obj(tox_events_unpack_handler, events, bytes, bytes_size)) { |
178 | 437 | tox_events_free(events); |
179 | 437 | return nullptr; |
180 | 437 | } |
181 | | |
182 | 573 | return events; |
183 | 1.01k | } |
184 | | |
185 | | bool tox_events_equal(const Tox_System *sys, const Tox_Events *a, const Tox_Events *b) |
186 | 63.6k | { |
187 | 63.6k | assert(sys != nullptr); |
188 | 63.6k | assert(sys->mem != nullptr); |
189 | | |
190 | 63.6k | const uint32_t a_size = tox_events_bytes_size(a); |
191 | 63.6k | const uint32_t b_size = tox_events_bytes_size(b); |
192 | | |
193 | 63.6k | if (a_size != b_size) { |
194 | 0 | return false; |
195 | 0 | } |
196 | | |
197 | 63.6k | uint8_t *a_bytes = (uint8_t *)mem_balloc(sys->mem, a_size); |
198 | 63.6k | uint8_t *b_bytes = (uint8_t *)mem_balloc(sys->mem, b_size); |
199 | | |
200 | 63.6k | if (a_bytes == nullptr || b_bytes == nullptr) { |
201 | 0 | mem_delete(sys->mem, b_bytes); |
202 | 0 | mem_delete(sys->mem, a_bytes); |
203 | 0 | return false; |
204 | 0 | } |
205 | | |
206 | 63.6k | tox_events_get_bytes(a, a_bytes); |
207 | 63.6k | tox_events_get_bytes(b, b_bytes); |
208 | | |
209 | 63.6k | const bool ret = memcmp(a_bytes, b_bytes, a_size) == 0; |
210 | | |
211 | 63.6k | mem_delete(sys->mem, b_bytes); |
212 | 63.6k | mem_delete(sys->mem, a_bytes); |
213 | | |
214 | 63.6k | return ret; |
215 | 63.6k | } |