File: compression_algorithm_zlib.h

package info (click to toggle)
mysql-8.0 8.0.45-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 1,273,048 kB
  • sloc: cpp: 4,685,434; ansic: 412,712; pascal: 108,396; java: 83,641; perl: 30,221; cs: 27,067; sql: 26,594; python: 21,816; sh: 17,285; yacc: 17,169; php: 11,522; xml: 7,388; javascript: 7,083; makefile: 1,793; lex: 1,075; awk: 670; asm: 520; objc: 183; ruby: 97; lisp: 86
file content (126 lines) | stat: -rw-r--r-- 4,241 bytes parent folder | download
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
/*
 * Copyright (c) 2019, 2025, Oracle and/or its affiliates.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License, version 2.0,
 * as published by the Free Software Foundation.
 *
 * This program is designed to work with certain software (including
 * but not limited to OpenSSL) that is licensed under separate terms,
 * as designated in a particular file or component or in included license
 * documentation.  The authors of MySQL hereby grant you an additional
 * permission to link the program and your derivative works with the
 * separately licensed software that they have either included with
 * the program or referenced in the documentation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License, version 2.0, for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA
 */
#ifndef PLUGIN_X_PROTOCOL_STREAM_COMPRESSION_COMPRESSION_ALGORITHM_ZLIB_H_
#define PLUGIN_X_PROTOCOL_STREAM_COMPRESSION_COMPRESSION_ALGORITHM_ZLIB_H_

#include <memory>
#include <utility>

#include "my_dbug.h"  // NOLINT(build/include_subdir)
#include "zlib.h"     // NOLINT(build/include_subdir)

#include "plugin/x/protocol/stream/compression/compression_algorithm_interface.h"

namespace protocol {

class Compression_algorithm_zlib : public Compression_algorithm_interface {
 public:
  explicit Compression_algorithm_zlib(const int32_t level) {
    m_stream.zalloc = Z_NULL;
    m_stream.zfree = Z_NULL;
    m_stream.opaque = Z_NULL;
    m_stream.avail_in = 0;
    m_stream.avail_out = 0;
    m_stream.next_in = Z_NULL;
    m_stream.next_out = Z_NULL;
    deflateInit(&m_stream, level);
  }

  ~Compression_algorithm_zlib() override { deflateEnd(&m_stream); }

  void set_pledged_source_size(const int /*src_size*/) override {}

  void set_input(uint8_t *in_ptr, const int in_size) override {
    m_stream.avail_in = in_size;
    m_stream.next_in = in_ptr;
    m_flush_finished = false;
  }

  bool compress(uint8_t *out_ptr, int *out_size) override {
    DBUG_TRACE;

    if (m_stream.avail_in) return compress_impl(out_ptr, out_size, false);

    *out_size = 0;

    return true;
  }

  bool flush(uint8_t *out_ptr, int *out_size) override {
    DBUG_TRACE;
    if (m_flush_finished) {
      *out_size = 0;
      return true;
    }

    if (m_stream.avail_in) return compress(out_ptr, out_size);

    const auto result = compress_impl(out_ptr, out_size, true);

    if (0 != m_stream.avail_out) m_flush_finished = true;

    return result;
  }

  static int32_t get_level_min() { return Z_BEST_SPEED; }
  static int32_t get_level_max() { return Z_BEST_COMPRESSION; }

 private:
  bool compress_impl(uint8_t *out_ptr, int *out_size, const bool flush) {
    DBUG_TRACE;

    const int backup_out_size = *out_size;
    m_stream.avail_out = backup_out_size;
    m_stream.next_out = out_ptr;

    do {
      DBUG_LOG("debug", "deflate(in_size:" << m_stream.avail_in << ", out_size:"
                                           << m_stream.avail_out << ")");
      const auto result = deflate(&m_stream, flush ? Z_SYNC_FLUSH : Z_NO_FLUSH);
      if (Z_OK != result) {
        DBUG_LOG("debug",
                 "deflate(out_size:"
                     << m_stream.avail_out << ", in_size:" << m_stream.avail_in
                     << ") returned an error, executed with " << result);
        return false;
      }

      DBUG_LOG("debug", "should retry deflate(in_size:"
                            << m_stream.avail_in
                            << ", out_size:" << m_stream.avail_out << ")");
    } while (!flush && (0 != m_stream.avail_out && 0 != m_stream.avail_in));

    *out_size = backup_out_size - static_cast<int>(m_stream.avail_out);

    return true;
  }

  bool m_flush_finished = false;
  z_stream m_stream;
};

}  // namespace protocol

#endif  // PLUGIN_X_PROTOCOL_STREAM_COMPRESSION_COMPRESSION_ALGORITHM_ZLIB_H_