Coverage Report

Created: 2026-02-02 10:56

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/c-toxcore/toxcore/state.c
Line
Count
Source
1
/* SPDX-License-Identifier: GPL-3.0-or-later
2
 * Copyright © 2016-2020 The TokTok team.
3
 * Copyright © 2014 Tox project.
4
 */
5
#include "state.h"
6
7
#include <string.h>
8
9
#include "ccompat.h"
10
#include "logger.h"
11
12
/** state load/save */
13
int state_load(const Logger *_Nonnull log, state_load_cb *_Nonnull state_load_callback, void *_Nonnull outer,
14
               const uint8_t *_Nonnull data, uint32_t length, uint16_t cookie_inner)
15
2.01k
{
16
2.01k
    if (state_load_callback == nullptr || data == nullptr) {
17
0
        LOGGER_ERROR(log, "state_load() called with invalid args.");
18
0
        return -1;
19
0
    }
20
21
2.01k
    const uint32_t size_head = sizeof(uint32_t) * 2;
22
23
33.9k
    while (length >= size_head) {
24
32.2k
        uint32_t length_sub;
25
32.2k
        lendian_bytes_to_host32(&length_sub, data);
26
27
32.2k
        uint32_t cookie_type;
28
32.2k
        lendian_bytes_to_host32(&cookie_type, data + sizeof(uint32_t));
29
30
32.2k
        data += size_head;
31
32.2k
        length -= size_head;
32
33
32.2k
        if (length < length_sub) {
34
            /* file truncated */
35
245
            LOGGER_ERROR(log, "state file too short: %u < %u", length, length_sub);
36
245
            return -1;
37
245
        }
38
39
32.0k
        if (lendian_to_host16(cookie_type >> 16) != cookie_inner) {
40
            /* something is not matching up in a bad way, give up */
41
41
            LOGGER_ERROR(log, "state file garbled: %04x != %04x", cookie_type >> 16, cookie_inner);
42
41
            return -1;
43
41
        }
44
45
32.0k
        const uint16_t type = lendian_to_host16(cookie_type & 0xFFFF);
46
47
32.0k
        switch (state_load_callback(outer, data, length_sub, type)) {
48
31.8k
            case STATE_LOAD_STATUS_CONTINUE: {
49
31.8k
                data += length_sub;
50
31.8k
                length -= length_sub;
51
31.8k
                break;
52
0
            }
53
54
113
            case STATE_LOAD_STATUS_ERROR: {
55
113
                LOGGER_ERROR(log, "Error occcured in state file (type: 0x%02x).", type);
56
113
                return -1;
57
0
            }
58
59
6
            case STATE_LOAD_STATUS_END: {
60
6
                return 0;
61
0
            }
62
32.0k
        }
63
32.0k
    }
64
65
1.61k
    if (length != 0) {
66
363
        LOGGER_ERROR(log, "unparsed data in state file of length %u", length);
67
363
        return -1;
68
363
    }
69
70
1.24k
    return 0;
71
1.61k
}
72
73
uint8_t *_Nonnull state_write_section_header(uint8_t *_Nonnull data, uint16_t cookie_type, uint32_t len, uint32_t section_type)
74
12.0k
{
75
12.0k
    host_to_lendian_bytes32(data, len);
76
12.0k
    data += sizeof(uint32_t);
77
12.0k
    host_to_lendian_bytes32(data, (host_to_lendian16(cookie_type) << 16) | host_to_lendian16(section_type));
78
12.0k
    data += sizeof(uint32_t);
79
12.0k
    return data;
80
12.0k
}
81
82
uint16_t lendian_to_host16(uint16_t lendian)
83
88.0k
{
84
#ifdef WORDS_BIGENDIAN
85
    return (lendian << 8) | (lendian >> 8);
86
#else
87
88.0k
    return lendian;
88
88.0k
#endif /* WORDS_BIGENDIAN */
89
88.0k
}
90
91
uint16_t host_to_lendian16(uint16_t host)
92
24.0k
{
93
24.0k
    return lendian_to_host16(host);
94
24.0k
}
95
96
void host_to_lendian_bytes64(uint8_t *_Nonnull dest, uint64_t num)
97
152
{
98
#ifdef WORDS_BIGENDIAN
99
    num = ((num << 8) & 0xFF00FF00FF00FF00) | ((num >> 8) & 0xFF00FF00FF00FF);
100
    num = ((num << 16) & 0xFFFF0000FFFF0000) | ((num >> 16) & 0xFFFF0000FFFF);
101
    num = (num << 32) | (num >> 32);
102
#endif /* WORDS_BIGENDIAN */
103
152
    memcpy(dest, &num, sizeof(uint64_t));
104
152
}
105
106
void lendian_bytes_to_host64(uint64_t *_Nonnull dest, const uint8_t *_Nonnull lendian)
107
400
{
108
400
    uint64_t d;
109
400
    memcpy(&d, lendian, sizeof(uint64_t));
110
#ifdef WORDS_BIGENDIAN
111
    d = ((d << 8) & 0xFF00FF00FF00FF00) | ((d >> 8) & 0xFF00FF00FF00FF);
112
    d = ((d << 16) & 0xFFFF0000FFFF0000) | ((d >> 16) & 0xFFFF0000FFFF);
113
    d = (d << 32) | (d >> 32);
114
#endif /* WORDS_BIGENDIAN */
115
400
    *dest = d;
116
400
}
117
118
void host_to_lendian_bytes32(uint8_t *_Nonnull dest, uint32_t num)
119
29.9k
{
120
#ifdef WORDS_BIGENDIAN
121
    num = ((num << 8) & 0xFF00FF00) | ((num >> 8) & 0xFF00FF);
122
    num = (num << 16) | (num >> 16);
123
#endif /* WORDS_BIGENDIAN */
124
29.9k
    memcpy(dest, &num, sizeof(uint32_t));
125
29.9k
}
126
127
void lendian_bytes_to_host32(uint32_t *_Nonnull dest, const uint8_t *_Nonnull lendian)
128
82.1k
{
129
82.1k
    uint32_t d;
130
82.1k
    memcpy(&d, lendian, sizeof(uint32_t));
131
#ifdef WORDS_BIGENDIAN
132
    d = ((d << 8) & 0xFF00FF00) | ((d >> 8) & 0xFF00FF);
133
    d = (d << 16) | (d >> 16);
134
#endif /* WORDS_BIGENDIAN */
135
82.1k
    *dest = d;
136
82.1k
}
137
138
void host_to_lendian_bytes16(uint8_t *_Nonnull dest, uint16_t num)
139
3.17k
{
140
#ifdef WORDS_BIGENDIAN
141
    num = (num << 8) | (num >> 8);
142
#endif /* WORDS_BIGENDIAN */
143
3.17k
    memcpy(dest, &num, sizeof(uint16_t));
144
3.17k
}
145
146
void lendian_bytes_to_host16(uint16_t *_Nonnull dest, const uint8_t *_Nonnull lendian)
147
15.3k
{
148
15.3k
    uint16_t d;
149
15.3k
    memcpy(&d, lendian, sizeof(uint16_t));
150
#ifdef WORDS_BIGENDIAN
151
    d = (d << 8) | (d >> 8);
152
#endif /* WORDS_BIGENDIAN */
153
15.3k
    *dest = d;
154
15.3k
}