summaryrefslogtreecommitdiffstats
path: root/vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java
diff options
context:
space:
mode:
authorBjørn Christian Seime <bjorncs@verizonmedia.com>2021-01-19 18:06:57 +0100
committerGitHub <noreply@github.com>2021-01-19 18:06:57 +0100
commit6f326f6cc056dc46f5538a8a186c8420b0379edc (patch)
tree7efbbf5b4f585edca3c135be7b8b8f59fa22f1e7 /vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java
parent4e6d9b184867553f740a65124fdb2d9c380caf22 (diff)
parentecd1ac5e05d4e7b2b059af3ca01084c3a3783148 (diff)
Merge pull request #16102 from vespa-engine/bjorncs/zstd-java
Bjorncs/zstd java
Diffstat (limited to 'vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java')
-rw-r--r--vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java51
1 files changed, 51 insertions, 0 deletions
diff --git a/vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java b/vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java
new file mode 100644
index 00000000000..72ccb730db7
--- /dev/null
+++ b/vespajlib/src/main/java/com/yahoo/compress/ZstdCompressor.java
@@ -0,0 +1,51 @@
+// Copyright Verizon Media. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
+package com.yahoo.compress;
+
+import java.util.Arrays;
+
+/**
+ * Frame based Zstd compressor (https://github.com/facebook/zstd)
+ * Implemented based on https://github.com/airlift/aircompressor - a pure Java implementation (no JNI).
+ *
+ * @author bjorncs
+ */
+public class ZstdCompressor {
+
+ private static final io.airlift.compress.zstd.ZstdCompressor compressor = new io.airlift.compress.zstd.ZstdCompressor();
+ private static final io.airlift.compress.zstd.ZstdDecompressor decompressor = new io.airlift.compress.zstd.ZstdDecompressor();
+
+ public byte[] compress(byte[] input, int inputOffset, int inputLength) {
+ int maxCompressedLength = getMaxCompressedLength(inputLength);
+ byte[] output = new byte[maxCompressedLength];
+ int compressedLength = compress(input, inputOffset, inputLength, output, 0, maxCompressedLength);
+ return Arrays.copyOf(output, compressedLength);
+ }
+
+ public int compress(byte[] input, int inputOffset, int inputLength, byte[] output, int outputOffset, int maxOutputLength) {
+ return compressor.compress(input, inputOffset, inputLength, output, outputOffset, maxOutputLength);
+ }
+
+ /**
+ * Note:
+ * Implementation assumes single frame (since {@link #getDecompressedLength(byte[], int, int)} only includes the first frame)
+ * The {@link #decompress(byte[], int, int, byte[], int, int)} overload will try to decompress all frame, causing the output buffer to overflow.
+ */
+ public byte[] decompress(byte[] input, int inputOffset, int inputLength) {
+ int decompressedLength = getDecompressedLength(input, inputOffset, inputLength);
+ byte[] output = new byte[decompressedLength];
+ decompress(input, inputOffset, inputLength, output, 0, decompressedLength);
+ return output;
+ }
+
+ public int decompress(byte[] input, int inputOffset, int inputLength, byte[] output, int outputOffset, int maxOutputLength) {
+ return decompressor.decompress(input, inputOffset, inputLength, output, outputOffset, maxOutputLength);
+ }
+
+ public static int getMaxCompressedLength(int uncompressedLength) {
+ return compressor.maxCompressedLength(uncompressedLength);
+ }
+
+ public static int getDecompressedLength(byte[] input, int inputOffset, int inputLength) {
+ return (int) io.airlift.compress.zstd.ZstdDecompressor.getDecompressedSize(input, inputOffset, inputLength);
+ }
+}