zopfli 0.0.2 → 0.0.8
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +6 -14
- data/.gitmodules +1 -1
- data/.travis.yml +23 -0
- data/Gemfile +2 -0
- data/README.md +6 -1
- data/Rakefile +8 -10
- data/ext/extconf.rb +2 -1
- data/ext/zopfli.c +39 -20
- data/lib/zopfli/version.rb +1 -1
- data/smoke.sh +9 -0
- data/{test → spec}/fixtures/alice29.txt +0 -0
- data/spec/spec_helper.rb +2 -0
- data/spec/zopfli_spec.rb +68 -0
- data/vendor/zopfli/{blocksplitter.c → src/zopfli/blocksplitter.c} +41 -53
- data/vendor/zopfli/{blocksplitter.h → src/zopfli/blocksplitter.h} +2 -6
- data/vendor/zopfli/{cache.c → src/zopfli/cache.c} +6 -0
- data/vendor/zopfli/{cache.h → src/zopfli/cache.h} +0 -0
- data/vendor/zopfli/src/zopfli/deflate.c +931 -0
- data/vendor/zopfli/{deflate.h → src/zopfli/deflate.h} +17 -2
- data/vendor/zopfli/src/zopfli/gzip_container.c +124 -0
- data/vendor/zopfli/{gzip_container.h → src/zopfli/gzip_container.h} +8 -0
- data/vendor/zopfli/{hash.c → src/zopfli/hash.c} +18 -10
- data/vendor/zopfli/{hash.h → src/zopfli/hash.h} +10 -7
- data/vendor/zopfli/{katajainen.c → src/zopfli/katajainen.c} +73 -62
- data/vendor/zopfli/{katajainen.h → src/zopfli/katajainen.h} +1 -1
- data/vendor/zopfli/{lz77.c → src/zopfli/lz77.c} +190 -42
- data/vendor/zopfli/{lz77.h → src/zopfli/lz77.h} +39 -23
- data/vendor/zopfli/{squeeze.c → src/zopfli/squeeze.c} +75 -61
- data/vendor/zopfli/{squeeze.h → src/zopfli/squeeze.h} +1 -0
- data/vendor/zopfli/{util.c → src/zopfli/symbols.h} +49 -23
- data/vendor/zopfli/{tree.c → src/zopfli/tree.c} +0 -0
- data/vendor/zopfli/{tree.h → src/zopfli/tree.h} +0 -0
- data/vendor/zopfli/src/zopfli/util.c +35 -0
- data/vendor/zopfli/{util.h → src/zopfli/util.h} +6 -23
- data/vendor/zopfli/{zlib_container.c → src/zopfli/zlib_container.c} +1 -1
- data/vendor/zopfli/{zlib_container.h → src/zopfli/zlib_container.h} +8 -0
- data/vendor/zopfli/{zopfli.h → src/zopfli/zopfli.h} +10 -4
- data/vendor/zopfli/{zopfli_bin.c → src/zopfli/zopfli_bin.c} +31 -15
- data/vendor/zopfli/{zopfli_lib.c → src/zopfli/zopfli_lib.c} +1 -2
- data/zopfli.gemspec +9 -28
- metadata +51 -50
- data/test/test_zopfli_deflate.rb +0 -47
- data/vendor/zopfli/CONTRIBUTORS +0 -6
- data/vendor/zopfli/README +0 -25
- data/vendor/zopfli/deflate.c +0 -698
- data/vendor/zopfli/gzip_container.c +0 -117
- data/vendor/zopfli/makefile +0 -5
checksums.yaml
CHANGED
@@ -1,15 +1,7 @@
|
|
1
1
|
---
|
2
|
-
|
3
|
-
metadata.gz:
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
metadata.gz: !binary |-
|
9
|
-
OTQxNThkMzU3OWRkMmE0YzU5MzE0YTFlNjBiZjJjMzg4NGVjYTAyMjNjZTY4
|
10
|
-
OTk1OGUyYTRjMzM1ODE5YjAxMjEyZTg1ZTY0MzhjYWRhYmVjNGVmNTgyZjhi
|
11
|
-
YzA2MWFiOGQyNjZmMmFiMWI4OGFiMjI5YmUxZDhiYmExYmVkNDM=
|
12
|
-
data.tar.gz: !binary |-
|
13
|
-
ZjM1N2E1MjA5NDViMmZkYmFhZWVjZWQzODE2ZDAxMmZkNDUyNWMzMGZlMTBl
|
14
|
-
MWM0OTA3OGIzOGViOGE0YTAyZGVjNTliZmVlMjJlMjBmZDhiODdiZjE1OWM3
|
15
|
-
M2VmODAzMWUyNDQ4YzJiMjcwNDk3Mzg2YjUxMzA0MTY0NzQzMWU=
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 3121a61abacb0a102a03f4ec91ac9e6ee4a6132942d6e402bcfded04a25f39bd
|
4
|
+
data.tar.gz: 544aee94aaaa325abcaee491d8b052a59dbd6809794fb2cdd4dd3fd787c466dd
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 5ff282d1b59f9bd4ab899cb1a16b63cadc64bb5a24c72b76289b136eadd8b50fcd14cb3a787169bc342ca1c6bb4ca5f405cc412eb3c13af6f4aa0c389cae2ec9
|
7
|
+
data.tar.gz: 24c803340ae499cf43da0d62b4fd916a0a05db1db07967461a7e65145810b70129a8063d4b972a1dad67ec8a6691b0ecf4d057390a78be771efae17983cae142
|
data/.gitmodules
CHANGED
data/.travis.yml
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
language: ruby
|
2
|
+
sudo: false
|
3
|
+
cache:
|
4
|
+
bundler: true
|
5
|
+
directories:
|
6
|
+
- "$HOME/.ccache"
|
7
|
+
rvm:
|
8
|
+
- 2.3.3
|
9
|
+
- 2.4.0
|
10
|
+
env:
|
11
|
+
global:
|
12
|
+
- PATH="/usr/lib/ccache:$PATH"
|
13
|
+
before_install:
|
14
|
+
- travis_retry gem install bundler --no-document
|
15
|
+
deploy:
|
16
|
+
provider: rubygems
|
17
|
+
api_key:
|
18
|
+
secure: Ix2H4oY1Ypo5mdYj9JTXufY/E7H/TWjAM54dtG2i/94yV8wMVINl+rhjnFso87LN6y0Tpt7rWy1OG1y3YPbwhBUPLbRXZ+xGjnsLA8/sUpEFODFk/QaYgOkCk2g3+xCgzzNvctd2u9fii/q9eIUts8BznegxtAYbat5a8dGvoVEkYAvHUX4lMRcEZVgMMpjT+xz3IGRe9rnW4umLd8P2mE6CEXAtrwFNSsjL1SVfKVOwwJ/UyiJfUnK6U/mDWdCDYWiPEE/CTz7PbrGGo4nAGoShLmkMUDsOamgECObT2rsNo42XzzrB4GAP+4PPQnias34BtJQLI9x2IEAUkM8vnKhSrtshp43S+ikU01W6rj4NMAvER4s9WNmRjsWgZQ/vEIHIwZXmdnkGvlYSiwRru+17YL2HCbrVp/YXi7X5HgBczuGOt3TZLB0xAKJG21qz+HdtP1nZTZe3EONDvWx9bM3XF2YjJ5gt3UNno8GsefTl71PFDubs1/LrVbkGgGwO6VTb4OgS2bEzntk17488BLTBo3l7bke5FTu8xQUbCawX2dqtOiukP45Ke8hRmgwWSMK+EH8YYcRcuZSbOXYJtGfpfC6nHtlArW+EjJ2Og+nWX68n7NFOIfGfCSX+V7cNgnKsqMjaesF+jEP+53CfVa2xnYxkZx7TgGrFyQEKr9s=
|
19
|
+
gem: zopfli
|
20
|
+
on:
|
21
|
+
tags: true
|
22
|
+
repo: miyucy/zopfli
|
23
|
+
script: bundle exec rake && ./smoke.sh
|
data/Gemfile
CHANGED
data/README.md
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
# Zopfli
|
2
2
|
|
3
|
-
see https://
|
3
|
+
see https://github.com/google/zopfli
|
4
4
|
|
5
5
|
## Installation
|
6
6
|
|
@@ -30,6 +30,11 @@ compressed_data = Zopfli.deflate string
|
|
30
30
|
uncompressed_data = Zlib::Inflate.inflate compressed_data
|
31
31
|
uncompressed_data == string
|
32
32
|
# => true
|
33
|
+
|
34
|
+
Zlib.deflate(File.read('LICENSE.txt'), Zlib::BEST_COMPRESSION).bytesize
|
35
|
+
# => 628
|
36
|
+
Zopfli.deflate(File.read('LICENSE.txt')).bytesize
|
37
|
+
# => 601
|
33
38
|
```
|
34
39
|
|
35
40
|
## Contributing
|
data/Rakefile
CHANGED
@@ -1,25 +1,23 @@
|
|
1
1
|
require "bundler/gem_tasks"
|
2
|
-
require "
|
2
|
+
require "rspec/core/rake_task"
|
3
3
|
require "rbconfig"
|
4
4
|
|
5
5
|
DLEXT = RbConfig::CONFIG["DLEXT"]
|
6
6
|
|
7
|
-
file "ext/zopfli.#{DLEXT}" => Dir.glob("ext/*{.rb,.c}") do
|
7
|
+
file "ext/zopfli.#{DLEXT}" => Dir.glob("ext/*{.rb,.c,.h}") do
|
8
8
|
Dir.chdir("ext") do
|
9
|
-
|
10
|
-
|
9
|
+
ruby "extconf.rb"
|
10
|
+
sh "make"
|
11
11
|
end
|
12
12
|
cp "ext/zopfli.#{DLEXT}", "lib"
|
13
13
|
end
|
14
14
|
|
15
15
|
task :clean do
|
16
16
|
files = Dir["ext/*"] - ["ext/extconf.rb", "ext/zopfli.c"]
|
17
|
-
files+= ["ext/zopfli.#{DLEXT}", "lib/zopfli.#{DLEXT}"]
|
17
|
+
files += ["ext/zopfli.#{DLEXT}", "lib/zopfli.#{DLEXT}"]
|
18
18
|
rm_rf(files) unless files.empty?
|
19
19
|
end
|
20
20
|
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
end
|
25
|
-
task :test => "ext/zopfli.#{DLEXT}"
|
21
|
+
RSpec::Core::RakeTask.new(:spec)
|
22
|
+
task :spec => "ext/zopfli.#{DLEXT}"
|
23
|
+
task :default => :spec
|
data/ext/extconf.rb
CHANGED
@@ -9,7 +9,7 @@
|
|
9
9
|
require "mkmf"
|
10
10
|
|
11
11
|
dst = File.dirname File.expand_path __FILE__
|
12
|
-
src = File.join dst, "..", "vendor", "zopfli"
|
12
|
+
src = File.join dst, "..", "vendor", "zopfli", "src", "zopfli"
|
13
13
|
|
14
14
|
%w(
|
15
15
|
blocksplitter.c
|
@@ -28,6 +28,7 @@ lz77.c
|
|
28
28
|
lz77.h
|
29
29
|
squeeze.c
|
30
30
|
squeeze.h
|
31
|
+
symbols.h
|
31
32
|
tree.c
|
32
33
|
tree.h
|
33
34
|
util.c
|
data/ext/zopfli.c
CHANGED
@@ -1,12 +1,11 @@
|
|
1
1
|
#include "ruby.h"
|
2
|
+
#include "ruby/thread.h"
|
2
3
|
#include "zopfli.h"
|
3
4
|
|
4
5
|
#define CSTR2SYM(x) ID2SYM(rb_intern(x))
|
5
6
|
#define DEFAULT_FORMAT ZOPFLI_FORMAT_ZLIB
|
6
7
|
|
7
|
-
static
|
8
|
-
|
9
|
-
ZopfliFormat
|
8
|
+
static ZopfliFormat
|
10
9
|
zopfli_deflate_parse_options(ZopfliOptions *options, VALUE opts)
|
11
10
|
{
|
12
11
|
ZopfliFormat format;
|
@@ -50,35 +49,56 @@ zopfli_deflate_parse_options(ZopfliOptions *options, VALUE opts)
|
|
50
49
|
return format;
|
51
50
|
}
|
52
51
|
|
53
|
-
|
54
|
-
zopfli_deflate(int argc, VALUE *argv, VALUE self)
|
55
|
-
{
|
56
|
-
VALUE in, out, opts;
|
52
|
+
typedef struct {
|
57
53
|
ZopfliOptions options;
|
58
54
|
ZopfliFormat format;
|
59
|
-
unsigned char
|
60
|
-
size_t
|
55
|
+
unsigned char *in;
|
56
|
+
size_t insize;
|
57
|
+
unsigned char *out;
|
58
|
+
size_t outsize;
|
59
|
+
} zopfli_deflate_args_t;
|
60
|
+
|
61
|
+
static void*
|
62
|
+
zopfli_deflate_no_gvl(void* arg)
|
63
|
+
{
|
64
|
+
zopfli_deflate_args_t *args = (zopfli_deflate_args_t*)arg;
|
61
65
|
|
62
|
-
|
66
|
+
ZopfliCompress(&args->options, args->format,
|
67
|
+
args->in, args->insize,
|
68
|
+
&args->out, &args->outsize);
|
69
|
+
|
70
|
+
return arg;
|
71
|
+
}
|
72
|
+
|
73
|
+
static VALUE
|
74
|
+
zopfli_deflate(int argc, VALUE *argv, VALUE self)
|
75
|
+
{
|
76
|
+
zopfli_deflate_args_t args;
|
77
|
+
VALUE in, out, opts;
|
78
|
+
|
79
|
+
ZopfliInitOptions(&args.options);
|
63
80
|
|
64
81
|
rb_scan_args(argc, argv, "11", &in, &opts);
|
65
82
|
|
66
83
|
if (!NIL_P(opts)) {
|
67
|
-
format = zopfli_deflate_parse_options(&options, opts);
|
84
|
+
args.format = zopfli_deflate_parse_options(&args.options, opts);
|
68
85
|
} else {
|
69
|
-
format = DEFAULT_FORMAT;
|
86
|
+
args.format = DEFAULT_FORMAT;
|
70
87
|
}
|
71
88
|
|
72
89
|
StringValue(in);
|
73
90
|
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
91
|
+
args.in = (unsigned char*)RSTRING_PTR(in);
|
92
|
+
args.insize = RSTRING_LEN(in);
|
93
|
+
|
94
|
+
args.out = NULL;
|
95
|
+
args.outsize = 0;
|
96
|
+
|
97
|
+
rb_thread_call_without_gvl(zopfli_deflate_no_gvl, (void *)&args, NULL, NULL);
|
78
98
|
|
79
|
-
out = rb_str_new(
|
99
|
+
out = rb_str_new((const char*)args.out, args.outsize);
|
80
100
|
|
81
|
-
free(
|
101
|
+
free(args.out);
|
82
102
|
|
83
103
|
return out;
|
84
104
|
}
|
@@ -86,7 +106,6 @@ zopfli_deflate(int argc, VALUE *argv, VALUE self)
|
|
86
106
|
void
|
87
107
|
Init_zopfli()
|
88
108
|
{
|
89
|
-
rb_mZopfli = rb_define_module("Zopfli");
|
109
|
+
VALUE rb_mZopfli = rb_define_module("Zopfli");
|
90
110
|
rb_define_singleton_method(rb_mZopfli, "deflate", zopfli_deflate, -1);
|
91
|
-
rb_require("zopfli/version");
|
92
111
|
}
|
data/lib/zopfli/version.rb
CHANGED
data/smoke.sh
ADDED
@@ -0,0 +1,9 @@
|
|
1
|
+
#!/bin/bash
|
2
|
+
gem list | grep zopfli && gem uninstall --force zopfli
|
3
|
+
bundle exec rake clean build
|
4
|
+
gem install --force --local --no-document "$(ls pkg/zopfli-*.gem)"
|
5
|
+
cat <<EOF | ruby
|
6
|
+
require 'zopfli'
|
7
|
+
require 'zlib'
|
8
|
+
abort if Zlib::Inflate.inflate(Zopfli.deflate(File.read('smoke.sh'))) != File.read('smoke.sh')
|
9
|
+
EOF
|
File without changes
|
data/spec/spec_helper.rb
ADDED
data/spec/zopfli_spec.rb
ADDED
@@ -0,0 +1,68 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
require 'zlib'
|
3
|
+
require 'stringio'
|
4
|
+
require 'benchmark'
|
5
|
+
require 'thread'
|
6
|
+
|
7
|
+
RSpec.describe Zopfli do
|
8
|
+
let(:fixture) { File.read(File.expand_path('fixtures/alice29.txt', File.dirname(__FILE__))) }
|
9
|
+
|
10
|
+
it 'works' do
|
11
|
+
expect(Zopfli.deflate(fixture)).to eq Zopfli.deflate(fixture, format: :zlib)
|
12
|
+
end
|
13
|
+
|
14
|
+
it 'works (zlib format)' do
|
15
|
+
deflated = Zopfli.deflate fixture, format: :zlib
|
16
|
+
|
17
|
+
uncompressed = Zlib::Inflate.inflate deflated
|
18
|
+
|
19
|
+
expect(uncompressed).to eq fixture
|
20
|
+
end
|
21
|
+
|
22
|
+
it 'works (gzip format)' do
|
23
|
+
deflated = Zopfli.deflate fixture, format: :gzip
|
24
|
+
|
25
|
+
uncompressed = Zlib::GzipReader.wrap(StringIO.new(deflated), &:read)
|
26
|
+
|
27
|
+
expect(uncompressed).to eq fixture
|
28
|
+
end
|
29
|
+
|
30
|
+
it 'works (deflate format)' do
|
31
|
+
deflated = Zopfli.deflate fixture, format: :deflate
|
32
|
+
p fixture.bytesize, deflated.bytesize
|
33
|
+
end
|
34
|
+
|
35
|
+
context 'benchmark' do
|
36
|
+
it 'seq' do
|
37
|
+
data = 10.times.map { fixture.dup }
|
38
|
+
deflates = nil
|
39
|
+
t = Benchmark.realtime { deflates = data.map { |datum| Zopfli.deflate datum } }
|
40
|
+
deflates.each { |deflate| expect(Zlib::Inflate.inflate(deflate)).to eq fixture }
|
41
|
+
puts t
|
42
|
+
# => 4.483513999963179
|
43
|
+
end
|
44
|
+
|
45
|
+
it 'thread' do
|
46
|
+
q = Queue.new
|
47
|
+
10.times { q.push fixture.dup }
|
48
|
+
10.times { q.push nil }
|
49
|
+
w = 10.times.map do
|
50
|
+
Thread.new do
|
51
|
+
deflates = []
|
52
|
+
loop do
|
53
|
+
datum = q.pop
|
54
|
+
break if datum.nil?
|
55
|
+
deflates << Zopfli.deflate(datum)
|
56
|
+
end
|
57
|
+
deflates
|
58
|
+
end
|
59
|
+
end
|
60
|
+
deflates = nil
|
61
|
+
t = Benchmark.realtime { deflates = w.map(&:value) }
|
62
|
+
deflates.flatten.each { |deflate| expect(Zlib::Inflate.inflate(deflate)).to eq fixture }
|
63
|
+
puts t
|
64
|
+
# => 2.6099140000296757
|
65
|
+
# => 0.9405940000433475 (w/o gvl)
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -24,7 +24,6 @@ Author: jyrki.alakuijala@gmail.com (Jyrki Alakuijala)
|
|
24
24
|
#include <stdlib.h>
|
25
25
|
|
26
26
|
#include "deflate.h"
|
27
|
-
#include "lz77.h"
|
28
27
|
#include "squeeze.h"
|
29
28
|
#include "tree.h"
|
30
29
|
#include "util.h"
|
@@ -39,9 +38,10 @@ typedef double FindMinimumFun(size_t i, void* context);
|
|
39
38
|
/*
|
40
39
|
Finds minimum of function f(i) where is is of type size_t, f(i) is of type
|
41
40
|
double, i is in range start-end (excluding end).
|
41
|
+
Outputs the minimum value in *smallest and returns the index of this value.
|
42
42
|
*/
|
43
43
|
static size_t FindMinimum(FindMinimumFun f, void* context,
|
44
|
-
size_t start, size_t end) {
|
44
|
+
size_t start, size_t end, double* smallest) {
|
45
45
|
if (end - start < 1024) {
|
46
46
|
double best = ZOPFLI_LARGE_FLOAT;
|
47
47
|
size_t result = start;
|
@@ -53,6 +53,7 @@ static size_t FindMinimum(FindMinimumFun f, void* context,
|
|
53
53
|
result = i;
|
54
54
|
}
|
55
55
|
}
|
56
|
+
*smallest = best;
|
56
57
|
return result;
|
57
58
|
} else {
|
58
59
|
/* Try to find minimum faster by recursively checking multiple points. */
|
@@ -88,6 +89,7 @@ static size_t FindMinimum(FindMinimumFun f, void* context,
|
|
88
89
|
pos = p[besti];
|
89
90
|
lastbest = best;
|
90
91
|
}
|
92
|
+
*smallest = lastbest;
|
91
93
|
return pos;
|
92
94
|
#undef NUM
|
93
95
|
}
|
@@ -103,16 +105,13 @@ dists: ll77 distances
|
|
103
105
|
lstart: start of block
|
104
106
|
lend: end of block (not inclusive)
|
105
107
|
*/
|
106
|
-
static double EstimateCost(const
|
107
|
-
const unsigned short* dists,
|
108
|
+
static double EstimateCost(const ZopfliLZ77Store* lz77,
|
108
109
|
size_t lstart, size_t lend) {
|
109
|
-
return
|
110
|
+
return ZopfliCalculateBlockSizeAutoType(lz77, lstart, lend);
|
110
111
|
}
|
111
112
|
|
112
113
|
typedef struct SplitCostContext {
|
113
|
-
const
|
114
|
-
const unsigned short* dists;
|
115
|
-
size_t llsize;
|
114
|
+
const ZopfliLZ77Store* lz77;
|
116
115
|
size_t start;
|
117
116
|
size_t end;
|
118
117
|
} SplitCostContext;
|
@@ -125,23 +124,20 @@ type: FindMinimumFun
|
|
125
124
|
*/
|
126
125
|
static double SplitCost(size_t i, void* context) {
|
127
126
|
SplitCostContext* c = (SplitCostContext*)context;
|
128
|
-
return EstimateCost(c->
|
129
|
-
EstimateCost(c->litlens, c->dists, i, c->end);
|
127
|
+
return EstimateCost(c->lz77, c->start, i) + EstimateCost(c->lz77, i, c->end);
|
130
128
|
}
|
131
129
|
|
132
130
|
static void AddSorted(size_t value, size_t** out, size_t* outsize) {
|
133
131
|
size_t i;
|
134
132
|
ZOPFLI_APPEND_DATA(value, out, outsize);
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
(*out)[j] = (*out)[j - 1];
|
141
|
-
}
|
142
|
-
(*out)[i] = value;
|
143
|
-
break;
|
133
|
+
for (i = 0; i + 1 < *outsize; i++) {
|
134
|
+
if ((*out)[i] > value) {
|
135
|
+
size_t j;
|
136
|
+
for (j = *outsize - 1; j > i; j--) {
|
137
|
+
(*out)[j] = (*out)[j - 1];
|
144
138
|
}
|
139
|
+
(*out)[i] = value;
|
140
|
+
break;
|
145
141
|
}
|
146
142
|
}
|
147
143
|
}
|
@@ -149,9 +145,8 @@ static void AddSorted(size_t value, size_t** out, size_t* outsize) {
|
|
149
145
|
/*
|
150
146
|
Prints the block split points as decimal and hex values in the terminal.
|
151
147
|
*/
|
152
|
-
static void PrintBlockSplitPoints(const
|
153
|
-
const
|
154
|
-
size_t llsize, const size_t* lz77splitpoints,
|
148
|
+
static void PrintBlockSplitPoints(const ZopfliLZ77Store* lz77,
|
149
|
+
const size_t* lz77splitpoints,
|
155
150
|
size_t nlz77points) {
|
156
151
|
size_t* splitpoints = 0;
|
157
152
|
size_t npoints = 0;
|
@@ -160,8 +155,8 @@ static void PrintBlockSplitPoints(const unsigned short* litlens,
|
|
160
155
|
index values. */
|
161
156
|
size_t pos = 0;
|
162
157
|
if (nlz77points > 0) {
|
163
|
-
for (i = 0; i <
|
164
|
-
size_t length = dists[i] == 0 ? 1 : litlens[i];
|
158
|
+
for (i = 0; i < lz77->size; i++) {
|
159
|
+
size_t length = lz77->dists[i] == 0 ? 1 : lz77->litlens[i];
|
165
160
|
if (lz77splitpoints[npoints] == i) {
|
166
161
|
ZOPFLI_APPEND_DATA(pos, &splitpoints, &npoints);
|
167
162
|
if (npoints == nlz77points) break;
|
@@ -188,7 +183,7 @@ static void PrintBlockSplitPoints(const unsigned short* litlens,
|
|
188
183
|
Finds next block to try to split, the largest of the available ones.
|
189
184
|
The largest is chosen to make sure that if only a limited amount of blocks is
|
190
185
|
requested, their sizes are spread evenly.
|
191
|
-
|
186
|
+
lz77size: the size of the LL77 data, which is the size of the done array here.
|
192
187
|
done: array indicating which blocks starting at that position are no longer
|
193
188
|
splittable (splitting them increases rather than decreases cost).
|
194
189
|
splitpoints: the splitpoints found so far.
|
@@ -198,7 +193,7 @@ lend: output variable, giving end of block.
|
|
198
193
|
returns 1 if a block was found, 0 if no block found (all are done).
|
199
194
|
*/
|
200
195
|
static int FindLargestSplittableBlock(
|
201
|
-
size_t
|
196
|
+
size_t lz77size, const unsigned char* done,
|
202
197
|
const size_t* splitpoints, size_t npoints,
|
203
198
|
size_t* lstart, size_t* lend) {
|
204
199
|
size_t longest = 0;
|
@@ -206,7 +201,7 @@ static int FindLargestSplittableBlock(
|
|
206
201
|
size_t i;
|
207
202
|
for (i = 0; i <= npoints; i++) {
|
208
203
|
size_t start = i == 0 ? 0 : splitpoints[i - 1];
|
209
|
-
size_t end = i == npoints ?
|
204
|
+
size_t end = i == npoints ? lz77size - 1 : splitpoints[i];
|
210
205
|
if (!done[start] && end - start > longest) {
|
211
206
|
*lstart = start;
|
212
207
|
*lend = end;
|
@@ -218,9 +213,7 @@ static int FindLargestSplittableBlock(
|
|
218
213
|
}
|
219
214
|
|
220
215
|
void ZopfliBlockSplitLZ77(const ZopfliOptions* options,
|
221
|
-
const
|
222
|
-
const unsigned short* dists,
|
223
|
-
size_t llsize, size_t maxblocks,
|
216
|
+
const ZopfliLZ77Store* lz77, size_t maxblocks,
|
224
217
|
size_t** splitpoints, size_t* npoints) {
|
225
218
|
size_t lstart, lend;
|
226
219
|
size_t i;
|
@@ -229,14 +222,14 @@ void ZopfliBlockSplitLZ77(const ZopfliOptions* options,
|
|
229
222
|
unsigned char* done;
|
230
223
|
double splitcost, origcost;
|
231
224
|
|
232
|
-
if (
|
225
|
+
if (lz77->size < 10) return; /* This code fails on tiny files. */
|
233
226
|
|
234
|
-
done = (unsigned char*)malloc(
|
227
|
+
done = (unsigned char*)malloc(lz77->size);
|
235
228
|
if (!done) exit(-1); /* Allocation failed. */
|
236
|
-
for (i = 0; i <
|
229
|
+
for (i = 0; i < lz77->size; i++) done[i] = 0;
|
237
230
|
|
238
231
|
lstart = 0;
|
239
|
-
lend =
|
232
|
+
lend = lz77->size;
|
240
233
|
for (;;) {
|
241
234
|
SplitCostContext c;
|
242
235
|
|
@@ -244,20 +237,16 @@ void ZopfliBlockSplitLZ77(const ZopfliOptions* options,
|
|
244
237
|
break;
|
245
238
|
}
|
246
239
|
|
247
|
-
c.
|
248
|
-
c.dists = dists;
|
249
|
-
c.llsize = llsize;
|
240
|
+
c.lz77 = lz77;
|
250
241
|
c.start = lstart;
|
251
242
|
c.end = lend;
|
252
243
|
assert(lstart < lend);
|
253
|
-
llpos = FindMinimum(SplitCost, &c, lstart + 1, lend);
|
244
|
+
llpos = FindMinimum(SplitCost, &c, lstart + 1, lend, &splitcost);
|
254
245
|
|
255
246
|
assert(llpos > lstart);
|
256
247
|
assert(llpos < lend);
|
257
248
|
|
258
|
-
|
259
|
-
EstimateCost(litlens, dists, llpos, lend);
|
260
|
-
origcost = EstimateCost(litlens, dists, lstart, lend);
|
249
|
+
origcost = EstimateCost(lz77, lstart, lend);
|
261
250
|
|
262
251
|
if (splitcost > origcost || llpos == lstart + 1 || llpos == lend) {
|
263
252
|
done[lstart] = 1;
|
@@ -267,7 +256,7 @@ void ZopfliBlockSplitLZ77(const ZopfliOptions* options,
|
|
267
256
|
}
|
268
257
|
|
269
258
|
if (!FindLargestSplittableBlock(
|
270
|
-
|
259
|
+
lz77->size, done, *splitpoints, *npoints, &lstart, &lend)) {
|
271
260
|
break; /* No further split will probably reduce compression. */
|
272
261
|
}
|
273
262
|
|
@@ -277,7 +266,7 @@ void ZopfliBlockSplitLZ77(const ZopfliOptions* options,
|
|
277
266
|
}
|
278
267
|
|
279
268
|
if (options->verbose) {
|
280
|
-
PrintBlockSplitPoints(
|
269
|
+
PrintBlockSplitPoints(lz77, *splitpoints, *npoints);
|
281
270
|
}
|
282
271
|
|
283
272
|
free(done);
|
@@ -292,25 +281,22 @@ void ZopfliBlockSplit(const ZopfliOptions* options,
|
|
292
281
|
size_t* lz77splitpoints = 0;
|
293
282
|
size_t nlz77points = 0;
|
294
283
|
ZopfliLZ77Store store;
|
284
|
+
ZopfliHash hash;
|
285
|
+
ZopfliHash* h = &hash;
|
295
286
|
|
296
|
-
ZopfliInitLZ77Store(&store);
|
297
|
-
|
298
|
-
|
299
|
-
s.blockstart = instart;
|
300
|
-
s.blockend = inend;
|
301
|
-
#ifdef ZOPFLI_LONGEST_MATCH_CACHE
|
302
|
-
s.lmc = 0;
|
303
|
-
#endif
|
287
|
+
ZopfliInitLZ77Store(in, &store);
|
288
|
+
ZopfliInitBlockState(options, instart, inend, 0, &s);
|
289
|
+
ZopfliAllocHash(ZOPFLI_WINDOW_SIZE, h);
|
304
290
|
|
305
291
|
*npoints = 0;
|
306
292
|
*splitpoints = 0;
|
307
293
|
|
308
294
|
/* Unintuitively, Using a simple LZ77 method here instead of ZopfliLZ77Optimal
|
309
295
|
results in better blocks. */
|
310
|
-
ZopfliLZ77Greedy(&s, in, instart, inend, &store);
|
296
|
+
ZopfliLZ77Greedy(&s, in, instart, inend, &store, h);
|
311
297
|
|
312
298
|
ZopfliBlockSplitLZ77(options,
|
313
|
-
store
|
299
|
+
&store, maxblocks,
|
314
300
|
&lz77splitpoints, &nlz77points);
|
315
301
|
|
316
302
|
/* Convert LZ77 positions to positions in the uncompressed input. */
|
@@ -328,7 +314,9 @@ void ZopfliBlockSplit(const ZopfliOptions* options,
|
|
328
314
|
assert(*npoints == nlz77points);
|
329
315
|
|
330
316
|
free(lz77splitpoints);
|
317
|
+
ZopfliCleanBlockState(&s);
|
331
318
|
ZopfliCleanLZ77Store(&store);
|
319
|
+
ZopfliCleanHash(h);
|
332
320
|
}
|
333
321
|
|
334
322
|
void ZopfliBlockSplitSimple(const unsigned char* in,
|