summaryrefslogtreecommitdiffstats
path: root/src/database/engine/dbengine-compression.c
blob: a703d1dc9e4b80fac58b05dea286818b4d8bb743 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
// SPDX-License-Identifier: GPL-3.0-or-later

#include "rrdengine.h"
#include "dbengine-compression.h"

#ifdef ENABLE_LZ4
#include <lz4.h>
#endif

#ifdef ENABLE_ZSTD
#include <zstd.h>
#define DBENGINE_ZSTD_DEFAULT_COMPRESSION_LEVEL 3
#endif

uint8_t dbengine_default_compression(void) {

#ifdef ENABLE_ZSTD
    return RRDENG_COMPRESSION_ZSTD;
#endif

#ifdef ENABLE_LZ4
    return RRDENG_COMPRESSION_LZ4;
#endif

    return RRDENG_COMPRESSION_NONE;
}

bool dbengine_valid_compression_algorithm(uint8_t algorithm) {
    switch(algorithm) {
        case RRDENG_COMPRESSION_NONE:

#ifdef ENABLE_LZ4
        case RRDENG_COMPRESSION_LZ4:
#endif

#ifdef ENABLE_ZSTD
        case RRDENG_COMPRESSION_ZSTD:
#endif

            return true;

        default:
            return false;
    }
}

size_t dbengine_max_compressed_size(size_t uncompressed_size, uint8_t algorithm) {
    switch(algorithm) {
#ifdef ENABLE_LZ4
        case RRDENG_COMPRESSION_LZ4:
            fatal_assert(uncompressed_size < LZ4_MAX_INPUT_SIZE);
            return LZ4_compressBound((int)uncompressed_size);
#endif

#ifdef ENABLE_ZSTD
        case RRDENG_COMPRESSION_ZSTD:
            return ZSTD_compressBound(uncompressed_size);
#endif

        case RRDENG_COMPRESSION_NONE:
            return uncompressed_size;

        default: {
            fatal("DBENGINE: unknown compression algorithm %u", algorithm);
            //we will never reach this point, but we have warnings from compiler
            return 0;
        }
    }
}

size_t dbengine_compress(void *payload, size_t uncompressed_size, uint8_t algorithm) {
    // the result should be stored in the payload
    // the caller must have called dbengine_max_compressed_size() to make sure the
    // payload is big enough to fit the max size needed.

    switch(algorithm) {
#ifdef ENABLE_LZ4
        case RRDENG_COMPRESSION_LZ4: {
            size_t max_compressed_size = dbengine_max_compressed_size(uncompressed_size, algorithm);
            struct extent_buffer *eb = extent_buffer_get(max_compressed_size);
            void *compressed_buf = eb->data;

            size_t compressed_size =
                LZ4_compress_default(payload, compressed_buf, (int)uncompressed_size, (int)max_compressed_size);

            if(compressed_size > 0 && compressed_size < uncompressed_size)
                memcpy(payload, compressed_buf, compressed_size);
            else
                compressed_size = 0;

            extent_buffer_release(eb);
            return compressed_size;
        }
#endif

#ifdef ENABLE_ZSTD
        case RRDENG_COMPRESSION_ZSTD: {
            size_t max_compressed_size = dbengine_max_compressed_size(uncompressed_size, algorithm);
            struct extent_buffer *eb = extent_buffer_get(max_compressed_size);
            void *compressed_buf = eb->data;

            size_t compressed_size = ZSTD_compress(compressed_buf, max_compressed_size, payload, uncompressed_size,
                                                   DBENGINE_ZSTD_DEFAULT_COMPRESSION_LEVEL);

            if (ZSTD_isError(compressed_size)) {
                internal_fatal(true, "DBENGINE: ZSTD compression error %s", ZSTD_getErrorName(compressed_size));
                compressed_size = 0;
            }

            if(compressed_size > 0 && compressed_size < uncompressed_size)
                memcpy(payload, compressed_buf, compressed_size);
            else
                compressed_size = 0;

            extent_buffer_release(eb);
            return compressed_size;
        }
#endif

        case RRDENG_COMPRESSION_NONE:
            return 0;

        default: {
            fatal("DBENGINE: unknown compression algorithm %u", algorithm);
            //we will never reach this point, but we have warnings from compiler
            return 0;
        }
    }
}

size_t dbengine_decompress(void *dst, void *src, size_t dst_size, size_t src_size, uint8_t algorithm) {
    switch(algorithm) {

#ifdef ENABLE_LZ4
        case RRDENG_COMPRESSION_LZ4: {
            int rc = LZ4_decompress_safe(src, dst, (int)src_size, (int)dst_size);
            if(rc < 0) {
                nd_log(NDLS_DAEMON, NDLP_ERR, "DBENGINE: ZSTD decompression error %d", rc);
                rc = 0;
            }

            return rc;
        }
#endif

#ifdef ENABLE_ZSTD
        case RRDENG_COMPRESSION_ZSTD: {
            size_t decompressed_size = ZSTD_decompress(dst, dst_size, src, src_size);

            if (ZSTD_isError(decompressed_size)) {
                nd_log(NDLS_DAEMON, NDLP_ERR, "DBENGINE: ZSTD decompression error %s",
                       ZSTD_getErrorName(decompressed_size));

                decompressed_size = 0;
            }

            return decompressed_size;
        }
#endif

        case RRDENG_COMPRESSION_NONE:
            internal_fatal(true, "DBENGINE: %s() should not be called for uncompressed pages", __FUNCTION__ );
            return 0;

        default:
            internal_fatal(true, "DBENGINE: unknown compression algorithm %u", algorithm);
            return 0;
    }
}